From 1eba39ceabe4e5b373123c41acd4a6956146a317 Mon Sep 17 00:00:00 2001 From: Andy Ford Date: Fri, 8 Nov 2024 13:40:30 +0000 Subject: [PATCH] refactor: combine room provider useEffects Makes it a bit easier to follow, also better caters for the "changing room options" use-case. --- demo/src/containers/Chat/Chat.tsx | 1 + src/react/contexts/chat-room-context.tsx | 7 +- src/react/providers/chat-room-provider.tsx | 79 +++++++++------------- 3 files changed, 38 insertions(+), 49 deletions(-) diff --git a/demo/src/containers/Chat/Chat.tsx b/demo/src/containers/Chat/Chat.tsx index f8049009..cd31652f 100644 --- a/demo/src/containers/Chat/Chat.tsx +++ b/demo/src/containers/Chat/Chat.tsx @@ -20,6 +20,7 @@ export const Chat = (props: { roomId: string; setRoomId: (roomId: string) => voi if (getPreviousMessages) { getPreviousMessages({ limit: 50 }) .then((result: PaginatedResult) => { + chatClient.logger.debug('backfilled messages', result); setMessages(result.items.filter((m) => !m.isDeleted).reverse()); setLoading(false); }) diff --git a/src/react/contexts/chat-room-context.tsx b/src/react/contexts/chat-room-context.tsx index a56d6813..1205b1db 100644 --- a/src/react/contexts/chat-room-context.tsx +++ b/src/react/contexts/chat-room-context.tsx @@ -1,4 +1,4 @@ -import { Room, RoomOptions } from '@ably/chat'; +import { ChatClient, Room, RoomOptions } from '@ably/chat'; import { createContext } from 'react'; /** @@ -19,6 +19,11 @@ export interface ChatRoomContextType { * Options used to create the room. */ options: RoomOptions; + + /** + * The chat client used to create the room. + */ + client: ChatClient; } /** diff --git a/src/react/providers/chat-room-provider.tsx b/src/react/providers/chat-room-provider.tsx index 8e1514b2..30f40cc5 100644 --- a/src/react/providers/chat-room-provider.tsx +++ b/src/react/providers/chat-room-provider.tsx @@ -135,13 +135,10 @@ export const ChatRoomProvider: React.FC = ({ // In StrictMode this will be called twice one after the other, but that's ok const [value, setValue] = useState(() => { logger.debug(`ChatRoomProvider(); initializing value`, { roomId, options }); - return { room: client.rooms.get(roomId, options).catch(), roomId: roomId, options: options }; + return { room: client.rooms.get(roomId, options).catch(), roomId: roomId, options: options, client: client }; }); - // Create an effect that changes the room when the id or options change - const prevId = useRef(roomId); - const prevOptions = useRef(options); - const prevClient = useRef(client); + // Create a queue to manage release ops const roomReleaseQueue = useRef(new RoomReleaseQueue(logger)); // update the release queue if the logger changes - as it means we have a new client @@ -156,62 +153,46 @@ export const ChatRoomProvider: React.FC = ({ roomReleaseQueue.current = new RoomReleaseQueue(logger); }, [logger]); + // Create an effect that manages the room state, handles attaching and releasing useEffect(() => { - logger.debug(`ChatRoomProvider(); running room options change effect`, { roomId, options }); - if (client === prevClient.current && prevId.current === roomId && prevOptions.current === options) { - return; - } + logger.debug(`ChatRoomProvider(); running lifecycle useEffect`, { roomId: roomId }); + let unmounted = false; + let resolvedRoom: Room | undefined; + const currentReleaseQueue = roomReleaseQueue.current; + + // If there was a previous release queued for this room (same id and options), abort it + currentReleaseQueue.abort(roomId, options); - prevId.current = roomId; - prevOptions.current = options; - prevClient.current = client; + // Get the room promise + const room = client.rooms.get(roomId, options).catch(); - // The room options or id is changing in some way, so we need to release the old room - // before we create a new one + // If we've had a change in the room id or options, update the value in the state setValue((prev: ChatRoomContextType) => { - logger.debug(`ChatRoomProvider(); running options change setValue`, { roomId, options }); - // If we're releasing, release the room - it's guaranteed to be changing - if (release) { - logger.debug(`ChatRoomProvider(); releasing value`, { roomId }); - void client.rooms.release(prev.roomId).catch(); - } else if (attach) { - // If we're not releasing, but we're attaching, detach the room - void prev.room.then((room) => { - logger.debug(`ChatRoomProvider(); detaching value`, { roomId }); - void room.detach().catch(); - }); + // If the room id and options haven't changed, then we don't need to do anything + if (prev.client === client && prev.roomId === roomId && prev.options === options) { + logger.debug(`ChatRoomProvider(); no change in room id or options`, { roomId, options }); + return prev; } logger.debug(`ChatRoomProvider(); updating value`, { roomId, options }); - return { room: client.rooms.get(roomId, options).catch(), roomId, options }; + return { room: room, roomId, options, client }; }); - }, [client, roomId, options, logger, attach, release]); - // Create an effect that attaches and detaches the room, or releases it when the component unmounts - useEffect(() => { - let unmounted = false; - let resolvedRoom: Room | undefined; - const currentReleaseQueue = roomReleaseQueue.current; - logger.debug(`ChatRoomProvider(); running attachment and release effect`, { roomId: value.roomId }); - - // If there was a previous release queued for this room, abort it - currentReleaseQueue.abort(value.roomId, value.options); - - // Get the latest value of the room promise to ensure we're not using stale values - void value.room + // Use the room promise to attach + void room .then((room: Room) => { if (unmounted) { - logger.debug(`ChatRoomProvider(); unmounted before room resolved`, { roomId: value.roomId }); + logger.debug(`ChatRoomProvider(); unmounted before room resolved`, { roomId: roomId }); + return; } - logger.debug(`ChatRoomProvider(); room resolved`, { roomId: value.roomId }); - + logger.debug(`ChatRoomProvider(); room resolved`, { roomId: roomId }); resolvedRoom = room; if (attach) { // attachment error and/or room status is available via useRoom // or room.status, no need to do anything with the promise here - logger.debug(`ChatRoomProvider(); attaching room`, { roomId: value.roomId }); + logger.debug(`ChatRoomProvider(); attaching room`, { roomId: roomId }); void room.attach().catch(() => { // Ignore, the error will be available via various room status properties }); @@ -222,22 +203,24 @@ export const ChatRoomProvider: React.FC = ({ // Cleanup function return () => { unmounted = true; - logger.debug(`ChatRoomProvider(); cleaning up`, { roomId: value.roomId }); + logger.debug(`ChatRoomProvider(); cleaning up lifecycle useEffect`, { roomId: roomId }); // If we're releasing, release the room. We'll do this in an abortable way so that we don't kill off the value // when using StrictMode if (release) { - logger.debug(`ChatRoomProvider(); releasing room`, { roomId: value.roomId }); - currentReleaseQueue.enqueue(client, value.roomId, value.options); + logger.debug(`ChatRoomProvider(); releasing room`, { roomId: roomId }); + currentReleaseQueue.enqueue(client, roomId, options); return; } else if (resolvedRoom && attach) { - logger.debug(`ChatRoomProvider(); detaching room`, { roomId: value.roomId }); + // If we're not releasing, but we are attaching, then we should detach the room, but only iff the room + // was resolved in time + logger.debug(`ChatRoomProvider(); detaching room`, { roomId: roomId }); void resolvedRoom.detach().catch(() => { // Ignore, the error will be available via various room status properties }); } }; - }, [value, logger, attach, release, client]); + }, [roomId, options, logger, attach, release, client]); return {children}; };