Compare commits

..

No commits in common. "3f39a3d05a644d6ac31823652bdc20a8c7a0159d" and "76fa04d1295831f02968764cdeeb7b4c57686df2" have entirely different histories.

3 changed files with 117 additions and 260 deletions

Binary file not shown.

View File

@ -57,11 +57,6 @@ export default function AdminChatPage() {
const typingTimeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null); const typingTimeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
const socketRef = useRef<Socket | null>(null); const socketRef = useRef<Socket | null>(null);
const getTokenRef = useRef(getToken);
const activeThreadIdRef = useRef<string | null>(null);
const subscribedThreadIdsRef = useRef<Set<string>>(new Set());
const loadingThreadsRef = useRef(false);
const loadingMessagesRef = useRef<Record<string, boolean>>({});
const activeMessages = useMemo( const activeMessages = useMemo(
() => (activeThreadId ? (messagesByThreadId[activeThreadId] ?? []) : []), () => (activeThreadId ? (messagesByThreadId[activeThreadId] ?? []) : []),
@ -74,28 +69,14 @@ export default function AdminChatPage() {
); );
useEffect(() => { useEffect(() => {
getTokenRef.current = getToken; const loadThreads = async () => {
}, [getToken]); const token = await getToken();
useEffect(() => {
activeThreadIdRef.current = activeThreadId;
}, [activeThreadId]);
const loadThreads = async (showLoader = true) => {
if (loadingThreadsRef.current) {
return;
}
const token = await getTokenRef.current();
if (!token) { if (!token) {
return; return;
} }
loadingThreadsRef.current = true;
try { try {
if (showLoader) {
setLoadingThreads(true); setLoadingThreads(true);
}
const response = await axios.get<{ threads: ChatThread[] }>( const response = await axios.get<{ threads: ChatThread[] }>(
"/api/chat/threads", "/api/chat/threads",
{ {
@ -104,33 +85,33 @@ export default function AdminChatPage() {
); );
setThreads(response.data.threads); setThreads(response.data.threads);
setActiveThreadId((prev) => prev ?? response.data.threads[0]?.id ?? null); setActiveThreadId(
(prev) => prev ?? response.data.threads[0]?.id ?? null,
);
setReadByThreadId({}); setReadByThreadId({});
} finally { } finally {
if (showLoader) {
setLoadingThreads(false); setLoadingThreads(false);
} }
loadingThreadsRef.current = false;
}
}; };
const loadMessages = async (threadId: string, showLoader = true) => { void loadThreads();
if (loadingMessagesRef.current[threadId]) { }, [getToken]);
useEffect(() => {
if (!activeThreadId) {
return; return;
} }
const token = await getTokenRef.current(); const loadMessages = async () => {
const token = await getToken();
if (!token) { if (!token) {
return; return;
} }
loadingMessagesRef.current[threadId] = true;
try { try {
if (showLoader) {
setLoadingMessages(true); setLoadingMessages(true);
}
const response = await axios.get<{ messages: ChatMessage[] }>( const response = await axios.get<{ messages: ChatMessage[] }>(
`/api/chat/threads/${threadId}/messages`, `/api/chat/threads/${activeThreadId}/messages`,
{ {
headers: { Authorization: `Bearer ${token}` }, headers: { Authorization: `Bearer ${token}` },
}, },
@ -138,33 +119,21 @@ export default function AdminChatPage() {
setMessagesByThreadId((prev) => ({ setMessagesByThreadId((prev) => ({
...prev, ...prev,
[threadId]: response.data.messages, [activeThreadId]: response.data.messages,
})); }));
} finally { } finally {
if (showLoader) {
setLoadingMessages(false); setLoadingMessages(false);
} }
loadingMessagesRef.current[threadId] = false;
}
}; };
useEffect(() => { void loadMessages();
void loadThreads(true); }, [activeThreadId, getToken]);
}, []);
useEffect(() => {
if (!activeThreadId) {
return;
}
void loadMessages(activeThreadId, true);
}, [activeThreadId]);
useEffect(() => { useEffect(() => {
let mounted = true; let mounted = true;
const setupSocket = async () => { const setupSocket = async () => {
const token = await getTokenRef.current(); const token = await getToken();
if (!token || !mounted) { if (!token || !mounted) {
return; return;
} }
@ -182,14 +151,14 @@ export default function AdminChatPage() {
socket.on("connect", () => { socket.on("connect", () => {
setSocketConnected(true); setSocketConnected(true);
subscribedThreadIdsRef.current.clear(); if (activeThreadId) {
void loadThreads(false); socket.emit("chat:subscribe", { threadId: activeThreadId });
}
}); });
socket.on("disconnect", () => { socket.on("disconnect", () => {
setSocketConnected(false); setSocketConnected(false);
setTypingByThreadId({}); setTypingByThreadId({});
subscribedThreadIdsRef.current.clear();
}); });
socket.on( socket.on(
@ -216,7 +185,7 @@ export default function AdminChatPage() {
lastMessageAt: event.message.createdAt, lastMessageAt: event.message.createdAt,
lastMessageBody: event.message.body, lastMessageBody: event.message.body,
unreadCount: unreadCount:
event.threadId === activeThreadIdRef.current || event.threadId === activeThreadId ||
event.message.senderUserId === userId event.message.senderUserId === userId
? thread.unreadCount ? thread.unreadCount
: thread.unreadCount + 1, : thread.unreadCount + 1,
@ -294,8 +263,6 @@ export default function AdminChatPage() {
[event.userId!]: event.lastReadMessageId!, [event.userId!]: event.lastReadMessageId!,
}, },
})); }));
void loadThreads(false);
}); });
socket.on("chat:error", (event: { code?: string; message?: string }) => { socket.on("chat:error", (event: { code?: string; message?: string }) => {
@ -317,57 +284,28 @@ export default function AdminChatPage() {
socketRef.current = null; socketRef.current = null;
} }
}; };
}, [userId]); }, [activeThreadId, getToken, userId]);
useEffect(() => { useEffect(() => {
if (!socketRef.current || !socketConnected) { if (!activeThreadId || !socketRef.current || !socketConnected) {
return;
}
const socket = socketRef.current;
const nextIds = new Set(threads.map((thread) => thread.id));
threads.forEach((thread) => {
if (!subscribedThreadIdsRef.current.has(thread.id)) {
socket.emit("chat:subscribe", { threadId: thread.id });
subscribedThreadIdsRef.current.add(thread.id);
}
});
Array.from(subscribedThreadIdsRef.current).forEach((threadId) => {
if (!nextIds.has(threadId)) {
socket.emit("chat:unsubscribe", { threadId });
subscribedThreadIdsRef.current.delete(threadId);
}
});
}, [socketConnected, threads]);
useEffect(() => {
if (!activeThreadId) {
return; return;
} }
socketRef.current.emit("chat:subscribe", { threadId: activeThreadId });
setThreads((prev) => setThreads((prev) =>
prev.map((thread) => prev.map((thread) =>
thread.id === activeThreadId ? { ...thread, unreadCount: 0 } : thread, thread.id === activeThreadId ? { ...thread, unreadCount: 0 } : thread,
), ),
); );
}, [activeThreadId]);
useEffect(() => {
if (!socketConnected) {
const interval = setInterval(() => {
void loadThreads(false);
if (activeThreadIdRef.current) {
void loadMessages(activeThreadIdRef.current, false);
}
}, 5000);
return () => { return () => {
clearInterval(interval); if (socketRef.current) {
}; socketRef.current.emit("chat:unsubscribe", {
threadId: activeThreadId,
});
} }
}, [socketConnected]); };
}, [activeThreadId, socketConnected]);
useEffect(() => { useEffect(() => {
if (!activeThreadId) { if (!activeThreadId) {
@ -381,7 +319,7 @@ export default function AdminChatPage() {
} }
const markRead = async () => { const markRead = async () => {
const token = await getTokenRef.current(); const token = await getToken();
if (!token) { if (!token) {
return; return;
} }
@ -413,7 +351,7 @@ export default function AdminChatPage() {
}; };
void markRead(); void markRead();
}, [activeThreadId, messagesByThreadId, socketConnected, userId]); }, [activeThreadId, getToken, messagesByThreadId, socketConnected, userId]);
const onChangeDraft = (value: string) => { const onChangeDraft = (value: string) => {
setDraft(value); setDraft(value);
@ -481,7 +419,7 @@ export default function AdminChatPage() {
return; return;
} }
const token = await getTokenRef.current(); const token = await getToken();
if (!token) { if (!token) {
return; return;
} }

View File

@ -74,26 +74,16 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
>({}); >({});
const socketRef = useRef<Socket | null>(null); const socketRef = useRef<Socket | null>(null);
const getTokenRef = useRef(getToken);
const lastMarkedReadByThreadRef = useRef<Record<string, string>>({}); const lastMarkedReadByThreadRef = useRef<Record<string, string>>({});
const activeThreadIdRef = useRef<string | null>(null); const activeThreadIdRef = useRef<string | null>(null);
const currentUserIdRef = useRef<string | undefined>(undefined); const currentUserIdRef = useRef<string | undefined>(undefined);
const subscribedThreadIdsRef = useRef<Set<string>>(new Set()); const refreshThreadsRef = useRef<() => Promise<void>>(async () => {});
const refreshThreadsRef = useRef<(showLoader?: boolean) => Promise<void>>( const refreshMessagesRef = useRef<(threadId: string) => Promise<void>>(
async () => {}, async () => {},
); );
const refreshMessagesRef = useRef<
(threadId: string, showLoader?: boolean) => Promise<void>
>(async () => {});
const markThreadReadRef = useRef< const markThreadReadRef = useRef<
(threadId: string, lastReadMessageId?: string) => Promise<void> (threadId: string, lastReadMessageId?: string) => Promise<void>
>(async () => {}); >(async () => {});
const refreshingThreadsRef = useRef(false);
const refreshingMessagesRef = useRef<Record<string, boolean>>({});
useEffect(() => {
getTokenRef.current = getToken;
}, [getToken]);
const clearAll = useCallback(() => { const clearAll = useCallback(() => {
setThreads([]); setThreads([]);
@ -107,7 +97,6 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
setSocketConnected(false); setSocketConnected(false);
setTypingByThreadId({}); setTypingByThreadId({});
lastMarkedReadByThreadRef.current = {}; lastMarkedReadByThreadRef.current = {};
subscribedThreadIdsRef.current.clear();
if (socketRef.current) { if (socketRef.current) {
socketRef.current.disconnect(); socketRef.current.disconnect();
@ -115,28 +104,18 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
} }
}, []); }, []);
const refreshThreads = useCallback( const refreshThreads = useCallback(async () => {
async (showLoader = true) => {
if (!isSignedIn) { if (!isSignedIn) {
return; return;
} }
if (refreshingThreadsRef.current) { const token = await getToken();
return;
}
refreshingThreadsRef.current = true;
const token = await getTokenRef.current();
if (!token) { if (!token) {
refreshingThreadsRef.current = false;
return; return;
} }
try { try {
if (showLoader) {
setLoadingThreads(true); setLoadingThreads(true);
}
if (isClientUser) { if (isClientUser) {
try { try {
await chatApi.getMyDmThread(token); await chatApi.getMyDmThread(token);
@ -158,37 +137,23 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
} catch (error) { } catch (error) {
log.warn("Failed to refresh chat threads", { error }); log.warn("Failed to refresh chat threads", { error });
} finally { } finally {
if (showLoader) {
setLoadingThreads(false); setLoadingThreads(false);
} }
refreshingThreadsRef.current = false; }, [getToken, isSignedIn, isClientUser]);
}
},
[isSignedIn, isClientUser],
);
const refreshMessages = useCallback( const refreshMessages = useCallback(
async (threadId: string, showLoader = true) => { async (threadId: string) => {
if (!isSignedIn) { if (!isSignedIn) {
return; return;
} }
if (refreshingMessagesRef.current[threadId]) { const token = await getToken();
return;
}
refreshingMessagesRef.current[threadId] = true;
const token = await getTokenRef.current();
if (!token) { if (!token) {
refreshingMessagesRef.current[threadId] = false;
return; return;
} }
try { try {
if (showLoader) {
setLoadingMessages(true); setLoadingMessages(true);
}
const response = await chatApi.getThreadMessages(threadId, token); const response = await chatApi.getThreadMessages(threadId, token);
setMessagesByThreadId((prev) => ({ setMessagesByThreadId((prev) => ({
...prev, ...prev,
@ -207,13 +172,10 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
} catch (error) { } catch (error) {
log.warn("Failed to refresh chat messages", { threadId, error }); log.warn("Failed to refresh chat messages", { threadId, error });
} finally { } finally {
if (showLoader) {
setLoadingMessages(false); setLoadingMessages(false);
} }
refreshingMessagesRef.current[threadId] = false;
}
}, },
[isSignedIn], [getToken, isSignedIn],
); );
const loadOlderMessages = useCallback( const loadOlderMessages = useCallback(
@ -231,7 +193,7 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
return; return;
} }
const token = await getTokenRef.current(); const token = await getToken();
if (!token) { if (!token) {
return; return;
} }
@ -273,7 +235,7 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
setLoadingOlderByThreadId((prev) => ({ ...prev, [threadId]: false })); setLoadingOlderByThreadId((prev) => ({ ...prev, [threadId]: false }));
} }
}, },
[isSignedIn, loadingOlderByThreadId, nextCursorByThreadId], [getToken, isSignedIn, loadingOlderByThreadId, nextCursorByThreadId],
); );
const markThreadRead = useCallback( const markThreadRead = useCallback(
@ -282,7 +244,7 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
return; return;
} }
const token = await getTokenRef.current(); const token = await getToken();
if (!token) { if (!token) {
return; return;
} }
@ -309,7 +271,7 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
log.warn("Failed to mark thread read", { threadId, error }); log.warn("Failed to mark thread read", { threadId, error });
} }
}, },
[isSignedIn, user?.id], [getToken, isSignedIn, user?.id],
); );
useEffect(() => { useEffect(() => {
@ -339,7 +301,7 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
return; return;
} }
const token = await getTokenRef.current(); const token = await getToken();
if (!token) { if (!token) {
return; return;
} }
@ -398,7 +360,7 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
throw error; throw error;
} }
}, },
[isSignedIn, socketConnected, user?.id], [getToken, isSignedIn, socketConnected, user?.id],
); );
useEffect(() => { useEffect(() => {
@ -410,7 +372,7 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
let mounted = true; let mounted = true;
const setupSocket = async () => { const setupSocket = async () => {
const token = await getTokenRef.current(); const token = await getToken();
if (!token || !mounted) { if (!token || !mounted) {
return; return;
} }
@ -420,7 +382,7 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
socket.on("connect", () => { socket.on("connect", () => {
setSocketConnected(true); setSocketConnected(true);
void refreshThreadsRef.current(false); void refreshThreadsRef.current();
if (activeThreadIdRef.current) { if (activeThreadIdRef.current) {
socket.emit("chat:subscribe", { socket.emit("chat:subscribe", {
@ -432,7 +394,6 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
socket.on("disconnect", () => { socket.on("disconnect", () => {
setSocketConnected(false); setSocketConnected(false);
setTypingByThreadId({}); setTypingByThreadId({});
subscribedThreadIdsRef.current.clear();
}); });
socket.on( socket.on(
@ -499,8 +460,6 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
: thread, : thread,
), ),
); );
} else if (event.threadId !== activeThreadIdRef.current) {
void refreshThreadsRef.current(false);
} }
}, },
); );
@ -564,7 +523,7 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
}, },
})); }));
void refreshThreadsRef.current(false); void refreshThreadsRef.current();
}, },
); );
@ -616,14 +575,14 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
socketRef.current = null; socketRef.current = null;
} }
}; };
}, [clearAll, isSignedIn]); }, [clearAll, getToken, isSignedIn]);
useEffect(() => { useEffect(() => {
if (!isSignedIn) { if (!isSignedIn) {
return; return;
} }
void refreshThreadsRef.current(true); void refreshThreadsRef.current();
}, [isSignedIn, userRole]); }, [isSignedIn, userRole]);
useEffect(() => { useEffect(() => {
@ -635,7 +594,7 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
socketRef.current.emit("chat:subscribe", { threadId: activeThreadId }); socketRef.current.emit("chat:subscribe", { threadId: activeThreadId });
} }
void refreshMessagesRef.current(activeThreadId, true); void refreshMessagesRef.current(activeThreadId);
setThreads((prev) => setThreads((prev) =>
prev.map((thread) => prev.map((thread) =>
@ -652,46 +611,6 @@ export function ChatProvider({ children }: { children: React.ReactNode }) {
}; };
}, [activeThreadId, socketConnected]); }, [activeThreadId, socketConnected]);
useEffect(() => {
if (!socketRef.current || !socketConnected) {
return;
}
const socket = socketRef.current;
const nextIds = new Set(threads.map((thread) => thread.id));
threads.forEach((thread) => {
if (!subscribedThreadIdsRef.current.has(thread.id)) {
socket.emit("chat:subscribe", { threadId: thread.id });
subscribedThreadIdsRef.current.add(thread.id);
}
});
Array.from(subscribedThreadIdsRef.current).forEach((threadId) => {
if (!nextIds.has(threadId)) {
socket.emit("chat:unsubscribe", { threadId });
subscribedThreadIdsRef.current.delete(threadId);
}
});
}, [socketConnected, threads]);
useEffect(() => {
if (!isSignedIn || socketConnected) {
return;
}
const interval = setInterval(() => {
void refreshThreadsRef.current(false);
if (activeThreadIdRef.current) {
void refreshMessagesRef.current(activeThreadIdRef.current, false);
}
}, 5000);
return () => {
clearInterval(interval);
};
}, [isSignedIn, socketConnected]);
useEffect(() => { useEffect(() => {
if (!activeThreadId) { if (!activeThreadId) {
return; return;