buster/web/src/context/Chats/ChatProvider.tsx

148 lines
3.7 KiB
TypeScript
Raw Normal View History

import React, { useEffect, useRef, useTransition } from 'react';
2025-01-28 05:20:26 +08:00
import {
createContext,
ContextSelector,
useContextSelector
} from '@fluentui/react-context-selector';
import { useBusterWebSocket } from '../BusterWebSocket';
2025-01-29 01:04:54 +08:00
import type { BusterChatAsset, BusterChat } from '@/api/buster_socket/chats';
import { useMemoizedFn, useUnmount } from 'ahooks';
2025-01-28 05:20:26 +08:00
import type { FileType } from '@/api/buster_socket/chats';
2025-01-29 01:04:54 +08:00
import { MOCK_CHAT } from './MOCK_CHAT';
import { IBusterChat } from './interfaces';
import { chatUpgrader } from './helpers';
2025-01-28 05:20:26 +08:00
export const useBusterChat = () => {
const busterSocket = useBusterWebSocket();
const [isPending, startTransition] = useTransition();
const chatsRef = useRef<Record<string, IBusterChat>>({});
// GETTERS
// SETTERS
// LISTENERS
2025-01-29 01:04:54 +08:00
const _onGetChat = useMemoizedFn((chat: BusterChat): IBusterChat => {
const upgradedChat = chatUpgrader(chat);
chatsRef.current[chat.id] = upgradedChat;
2025-01-28 05:20:26 +08:00
startTransition(() => {
//just used to trigger UI update
});
2025-01-29 01:04:54 +08:00
return upgradedChat;
2025-01-28 05:20:26 +08:00
});
const _onGetChatAsset = useMemoizedFn((asset: BusterChatAsset) => {
const { id, type } = asset;
console.log('TODO: handle this. Put the asset in their respective chat');
return asset;
});
// EMITTERS
const unsubscribeFromChat = useMemoizedFn(({ chatId }: { chatId: string }) => {
return busterSocket.emit({
route: '/chats/unsubscribe',
payload: {
id: chatId
}
});
});
const subscribeToChat = useMemoizedFn(({ chatId }: { chatId: string }) => {
2025-01-29 01:04:54 +08:00
_onGetChat(MOCK_CHAT);
// return busterSocket.emitAndOnce({
// emitEvent: {
// route: '/chats/get',
// payload: { id: chatId }
// },
// responseEvent: {
// route: '/chats/get:getChat',
// callback: _onGetChat
// }
// });
2025-01-28 05:20:26 +08:00
});
const getChatAsset = useMemoizedFn(
({
chatId: chat_id,
assetId: asset_id,
type,
versionId: version_id
}: {
chatId?: string;
assetId: string;
type: FileType;
versionId?: string;
}) => {
return busterSocket.emitAndOnce({
emitEvent: {
route: '/chats/get/asset',
payload: {
type,
chat_id,
asset_id,
version_id
}
},
responseEvent: {
route: '/chats/get:getChatAsset',
callback: _onGetChatAsset
}
});
}
);
return {
chats: chatsRef.current,
unsubscribeFromChat,
subscribeToChat,
getChatAsset
2025-01-28 05:20:26 +08:00
};
};
const BusterChat = createContext<ReturnType<typeof useBusterChat>>(
{} as ReturnType<typeof useBusterChat>
);
export const BusterChatProvider: React.FC<{
children: React.ReactNode;
}> = ({ children }) => {
const value = useBusterChat();
return <BusterChat.Provider value={value}>{children}</BusterChat.Provider>;
};
export const useBusterChatContextSelector = <T,>(
selector: ContextSelector<ReturnType<typeof useBusterChat>, T>
) => useContextSelector(BusterChat, selector);
export const useBusterChatIndividual = ({ chatId: chatIdProp }: { chatId?: string }) => {
const chatId = chatIdProp || '';
2025-01-28 05:20:26 +08:00
const chat = useBusterChatContextSelector((x) => x.chats[chatId]);
const subscribeToChat = useBusterChatContextSelector((x) => x.subscribeToChat);
const unsubscribeFromChat = useBusterChatContextSelector((x) => x.unsubscribeFromChat);
useEffect(() => {
if (chatId) subscribeToChat({ chatId });
}, [chatId]);
2025-01-28 05:20:26 +08:00
useUnmount(() => {
if (chatId) unsubscribeFromChat({ chatId });
2025-01-28 05:20:26 +08:00
});
return {
chat
2025-01-28 05:20:26 +08:00
};
};
2025-01-28 06:48:29 +08:00
export const useBusterChatAssetIndividual = ({
chatId,
assetId,
type
}: {
chatId: string;
assetId: string;
type: FileType;
}) => {};