Skip to content

Commit

Permalink
Add Reflect (#635)
Browse files Browse the repository at this point in the history
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
jagodarybacka authored Nov 21, 2023
2 parents f4bd074 + 3ad9ebd commit b258790
Show file tree
Hide file tree
Showing 18 changed files with 841 additions and 14 deletions.
6 changes: 5 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ node_modules
yarn-error.log
.pnp/
.pnp.js

# Yarn Integrity file
.yarn-integrity

Expand All @@ -33,4 +34,7 @@ yarn-error.log
.idea

# Dist directory
dist
dist

# Reflect files
.reflect
3 changes: 3 additions & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,8 @@
"@react-spring/konva": "^9.7.3",
"@react-spring/web": "^9.7.3",
"@reduxjs/toolkit": "^1.9.5",
"@rocicorp/rails": "^0.8.0",
"@rocicorp/reflect": "0.38.202311200859",
"@web3-onboard/core": "^2.21.0",
"@web3-onboard/react": "^2.8.11",
"@web3-onboard/taho": "^2.0.5",
Expand Down Expand Up @@ -71,6 +73,7 @@
"@typescript-eslint/eslint-plugin": "^6.0.0",
"@typescript-eslint/parser": "^6.0.0",
"babel-loader": "^9.1.3",
"concurrently": "^8.2.2",
"copy-webpack-plugin": "^11.0.0",
"dotenv-defaults": "^5.0.2",
"dotenv-webpack": "^8.0.1",
Expand Down
8 changes: 8 additions & 0 deletions reflect.config.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
{
"server": "./src/shared/utils/reflect.ts",
"apps": {
"default": {
"appID": "loztnkex"
}
}
}
5 changes: 5 additions & 0 deletions src/shared/constants/realms-data.ts
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,7 @@ export const realm4 = {
labelY: 216.5,
partnerIcons: partners.gitcoin,
partnerColor: "#1D4E56",
cursorText: "var(--primary-p1-100)",
}

export const realm5 = {
Expand Down Expand Up @@ -128,6 +129,7 @@ export const realm7 = {
labelX: 540,
labelY: 400,
partnerIcons: partners.cyberconnect,
cursorText: "var(--primary-p1-100)",
partnerColor: "#fff",
}

Expand Down Expand Up @@ -166,6 +168,7 @@ export const realm9 = {
labelX: 700,
labelY: 350,
partnerIcons: partners.arbitrum,
cursorText: "#FFF",
}

export const realm10 = {
Expand Down Expand Up @@ -359,6 +362,7 @@ export const realm19 = {
labelX: 264,
labelY: 409,
partnerIcons: partners.galxe,
cursorText: "#FFF",
}

export const realm20 = {
Expand Down Expand Up @@ -413,6 +417,7 @@ export const realm22 = {
labelX: 337,
labelY: 388,
partnerIcons: partners.frax,
cursorText: "var(--primary-p1-100)",
}

export const realm23 = {
Expand Down
7 changes: 4 additions & 3 deletions src/shared/hooks/index.ts
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"
137 changes: 137 additions & 0 deletions src/shared/hooks/reflect.ts
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)
}
11 changes: 11 additions & 0 deletions src/shared/hooks/wallets.ts
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ import {
} from "shared/constants"
import { Network } from "@ethersproject/networks"
import { Logger, defineReadOnly } from "ethers/lib/utils"
import { reflectInstance } from "shared/utils"
import { useAssistant } from "./assistant"
import { useInterval, useLocalStorageChange } from "./helpers"

Expand Down Expand Up @@ -168,6 +169,16 @@ export function useWalletOnboarding(): {
const { value, updateStorage } =
useLocalStorageChange<string>(LOCAL_STORAGE_WALLET)

useEffect(() => {
const updateReflectPresence = async () => {
if (!reflectInstance) return

await reflectInstance.mutate.setUserPresence(!!value)
}

updateReflectPresence()
}, [value])

return { walletOnboarded: value, updateWalletOnboarding: updateStorage }
}

Expand Down
1 change: 0 additions & 1 deletion src/shared/services/index.ts
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"
1 change: 1 addition & 0 deletions src/shared/types/index.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
export * from "./claim"
export * from "./pool"
export * from "./reflect"
export * from "./transaction"
export * from "./island"
export * from "./wallet"
Expand Down
1 change: 1 addition & 0 deletions src/shared/types/island.ts
Original file line number Diff line number Diff line change
Expand Up @@ -83,4 +83,5 @@ export type RealmMapData = {
labelY: number
partnerIcons: { default: string; shadow: string; population: string }
partnerColor?: string
cursorText: string
}
18 changes: 18 additions & 0 deletions src/shared/types/reflect.ts
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
}
1 change: 1 addition & 0 deletions src/shared/utils/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ export * from "./claim"
export * from "./pool"
export * from "./population"
export * from "./providers"
export * from "./reflect"
export * from "./island"
export * from "./timers"
export * from "./numbers"
Expand Down
67 changes: 67 additions & 0 deletions src/shared/utils/reflect.ts
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
Loading

0 comments on commit b258790

Please sign in to comment.