61 lines
1.6 KiB
TypeScript
61 lines
1.6 KiB
TypeScript
import { useEffect, useMemo, useState } from "react"
|
|
import {
|
|
AwarenessList,
|
|
TypedLiveblocksProvider,
|
|
UserAwareness,
|
|
useSelf,
|
|
} from "@/liveblocks.config"
|
|
import { colors } from "@/lib/colors"
|
|
|
|
export function Cursors({ yProvider }: { yProvider: TypedLiveblocksProvider }) {
|
|
// Get user info from Liveblocks authentication endpoint
|
|
const userInfo = useSelf((me) => me.info)
|
|
|
|
if (!userInfo) return null
|
|
|
|
const [awarenessUsers, setAwarenessUsers] = useState<AwarenessList>([])
|
|
|
|
useEffect(() => {
|
|
// Add user info to Yjs awareness
|
|
const localUser: UserAwareness["user"] = userInfo
|
|
yProvider.awareness.setLocalStateField("user", localUser)
|
|
|
|
// On changes, update `awarenessUsers`
|
|
function setUsers() {
|
|
setAwarenessUsers(
|
|
Array.from(yProvider.awareness.getStates()) as AwarenessList
|
|
)
|
|
}
|
|
yProvider.awareness.on("change", setUsers)
|
|
setUsers()
|
|
|
|
return () => {
|
|
yProvider.awareness.off("change", setUsers)
|
|
}
|
|
}, [yProvider])
|
|
|
|
// Insert awareness info into cursors with styles
|
|
const styleSheet = useMemo(() => {
|
|
let cursorStyles = ""
|
|
|
|
for (const [clientId, client] of awarenessUsers) {
|
|
if (client?.user) {
|
|
cursorStyles += `
|
|
.yRemoteSelection-${clientId},
|
|
.yRemoteSelectionHead-${clientId} {
|
|
--user-color: ${colors[client.user.color]};
|
|
}
|
|
|
|
.yRemoteSelectionHead-${clientId}::after {
|
|
content: "${client.user.name}";
|
|
}
|
|
`
|
|
}
|
|
}
|
|
|
|
return { __html: cursorStyles }
|
|
}, [awarenessUsers])
|
|
|
|
return <style dangerouslySetInnerHTML={styleSheet} />
|
|
}
|