-
Notifications
You must be signed in to change notification settings - Fork 4
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Resolves #629 What has been done: - Added `reflect` library - Current users' cursors are visible on the island Live preview: https://reflect--taho-development.netlify.app/
- Loading branch information
Showing
18 changed files
with
841 additions
and
14 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
{ | ||
"server": "./src/shared/utils/reflect.ts", | ||
"apps": { | ||
"default": { | ||
"appID": "loztnkex" | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,7 +1,8 @@ | ||
export * from "./wallets" | ||
export * from "./assistant" | ||
export * from "./helpers" | ||
export * from "./island" | ||
export * from "./transactions" | ||
export * from "./assistant" | ||
export * from "./reflect" | ||
export * from "./tenderly" | ||
export * from "./transactions" | ||
export * from "./wallets" | ||
export * from "./population" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,137 @@ | ||
import { useEffect, useState } from "react" | ||
import { | ||
selectDisplayedRealmId, | ||
selectStakingRealmId, | ||
selectWalletName, | ||
useDappSelector, | ||
} from "redux-state" | ||
import { usePresence, useSubscribe } from "@rocicorp/reflect/react" | ||
import { getClientState, reflectInstance } from "shared/utils" | ||
import { getRealmMapData } from "shared/constants" | ||
|
||
export function useReflect() { | ||
const name = useDappSelector(selectWalletName) | ||
const stakingRealmId = useDappSelector(selectStakingRealmId) | ||
const realmMapData = stakingRealmId ? getRealmMapData(stakingRealmId) : null | ||
|
||
const [reflectInitialized, setReflectInitialized] = useState(false) | ||
|
||
const realmIcon = realmMapData?.partnerIcons.default ?? null | ||
const stakingRealmColor = realmMapData?.color ?? "#2C2C2C" | ||
const cursorTextColor = realmMapData?.cursorText ?? "#FFF" | ||
|
||
useEffect(() => { | ||
const initReflect = async () => { | ||
if (reflectInitialized || !reflectInstance) return | ||
|
||
await reflectInstance.mutate.initClientState({ | ||
id: reflectInstance.clientID, | ||
cursor: null, | ||
userInfo: { | ||
name, | ||
realmIcon, | ||
stakingRealmColor, | ||
cursorTextColor, | ||
}, | ||
isPresent: true, | ||
}) | ||
|
||
setReflectInitialized(true) | ||
} | ||
|
||
const updateUserInfo = async () => { | ||
if (!reflectInstance) return | ||
|
||
await reflectInstance.mutate.setUserInfo({ | ||
name, | ||
realmIcon, | ||
stakingRealmColor, | ||
cursorTextColor, | ||
}) | ||
} | ||
|
||
initReflect() | ||
updateUserInfo() | ||
}, [reflectInitialized, name, realmIcon, stakingRealmColor, cursorTextColor]) | ||
|
||
useEffect(() => { | ||
const handleReflectCursor = async (e: MouseEvent) => { | ||
if (!reflectInstance) return | ||
|
||
await reflectInstance.mutate.setCursor({ x: e.clientX, y: e.clientY }) | ||
} | ||
|
||
window.addEventListener("mousemove", handleReflectCursor) | ||
return () => window.removeEventListener("mousemove", handleReflectCursor) | ||
}, []) | ||
} | ||
|
||
export function useReflectPresence() { | ||
const presentClientsdIds = usePresence(reflectInstance) | ||
const presentClients = useSubscribe( | ||
reflectInstance, | ||
async (tx) => { | ||
const clients = await Promise.all( | ||
presentClientsdIds.map(async (clientID) => { | ||
const presentClient = await getClientState(tx, clientID) | ||
return presentClient ? [presentClient] : [] | ||
}) | ||
) | ||
|
||
return clients | ||
.flatMap((client) => client) | ||
.filter((client) => client.isPresent) | ||
}, | ||
[], | ||
[presentClientsdIds] | ||
) | ||
|
||
return presentClients | ||
} | ||
|
||
export function useReflectCurrentUser() { | ||
return useSubscribe( | ||
reflectInstance, | ||
async (tx) => { | ||
const currentUser = await getClientState(tx, tx.clientID) | ||
return currentUser | ||
}, | ||
null | ||
) | ||
} | ||
|
||
export function useReflectCursors() { | ||
useReflect() | ||
|
||
const reflectClients = useReflectPresence() | ||
const currentUser = useReflectCurrentUser() | ||
const realmModalOpened = useDappSelector(selectDisplayedRealmId) | ||
|
||
// Find index of current user to determine the "room" placement | ||
const currentUserIndex = reflectClients.findIndex( | ||
(client) => client.id === currentUser?.id | ||
) | ||
|
||
// Set max number of visible cursors in .env (or default to 5) | ||
const maxNumberOfVisibleCursors = | ||
Number(process.env.REFLECT_MAX_CAPACITY) || 5 | ||
|
||
const currentUserRoom = Math.floor( | ||
currentUserIndex / maxNumberOfVisibleCursors | ||
) | ||
|
||
const currentRoomValue = currentUserRoom * maxNumberOfVisibleCursors | ||
|
||
// We want users to always have a chance to interact with each other so we split them | ||
// into "rooms" based on their index in the reflectClients array. This way map stays interactive | ||
// while still being not overcrowded. | ||
const visibleClients = reflectClients.slice( | ||
currentRoomValue, | ||
currentRoomValue + maxNumberOfVisibleCursors | ||
) | ||
|
||
// Hide current user cursor when the realm modal is opened | ||
return !realmModalOpened | ||
? visibleClients | ||
: visibleClients.filter((client) => client.id !== currentUser?.id) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,2 @@ | ||
// eslint-disable-next-line import/prefer-default-export | ||
export { default as TransactionService } from "./transaction" | ||
export { default as StorageService } from "./storage" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
export type ReflectCursor = { | ||
x: number | ||
y: number | ||
} | ||
|
||
export type ReflectUserInfo = { | ||
name: string | ||
realmIcon: string | null | ||
stakingRealmColor: string | ||
cursorTextColor: string | ||
} | ||
|
||
export type ReflectClient = { | ||
id: string | ||
cursor: ReflectCursor | null | ||
userInfo: ReflectUserInfo | ||
isPresent: boolean | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,67 @@ | ||
import type { ReflectServerOptions } from "@rocicorp/reflect/server" | ||
import { WriteTransaction } from "@rocicorp/reflect" | ||
import { generate } from "@rocicorp/rails" | ||
import { Reflect } from "@rocicorp/reflect/client" | ||
import { ReflectClient, ReflectCursor, ReflectUserInfo } from "shared/types" | ||
import { nanoid } from "@reduxjs/toolkit" | ||
|
||
export const { | ||
init: initClientState, | ||
get: getClientState, | ||
update: updateClientState, | ||
} = generate<ReflectClient>("client-state") | ||
|
||
async function setCursor( | ||
tx: WriteTransaction, | ||
coordinates: ReflectCursor | ||
): Promise<void> { | ||
await updateClientState(tx, { id: tx.clientID, cursor: coordinates }) | ||
} | ||
|
||
async function setUserInfo( | ||
tx: WriteTransaction, | ||
userInfo: ReflectUserInfo | ||
): Promise<void> { | ||
await updateClientState(tx, { | ||
id: tx.clientID, | ||
userInfo, | ||
}) | ||
} | ||
|
||
async function setUserPresence(tx: WriteTransaction, isPresent: boolean) { | ||
const clientState = await getClientState(tx, tx.clientID) | ||
await updateClientState(tx, { | ||
id: tx.clientID, | ||
...clientState, | ||
isPresent, | ||
}) | ||
} | ||
|
||
export const mutators = { | ||
initClientState, | ||
getClientState, | ||
updateClientState, | ||
setCursor, | ||
setUserInfo, | ||
setUserPresence, | ||
} | ||
|
||
export type ReflectMutators = typeof mutators | ||
export type ReflectInstance = Reflect<ReflectMutators> | ||
|
||
export const reflectInstance = | ||
process.env.DISABLE_REFLECT === "true" | ||
? null | ||
: new Reflect<ReflectMutators>({ | ||
userID: nanoid(), | ||
roomID: "/", | ||
server: process.env.REFLECT_SERVER ?? "", | ||
mutators, | ||
}) | ||
|
||
const makeOptions = (): ReflectServerOptions<ReflectMutators> => ({ | ||
mutators, | ||
logLevel: "debug", | ||
}) | ||
|
||
export default makeOptions |
Oops, something went wrong.