import React, { createContext, useContext, useEffect, useMemo, useState } from 'react'; import { useHistory, useParams } from 'react-router-dom'; import { toggleMainWindow } from 'soapbox/actions/chats'; import { useAppDispatch, useOwnAccount, useSettings } from 'soapbox/hooks'; import { IChat, useChat } from 'soapbox/queries/chats'; type WindowState = 'open' | 'minimized'; const ChatContext = createContext({ isOpen: false, needsAcceptance: false, }); enum ChatWidgetScreens { INBOX = 'INBOX', SEARCH = 'SEARCH', CHAT = 'CHAT', CHAT_SETTINGS = 'CHAT_SETTINGS' } interface IChatProvider { children: React.ReactNode } const ChatProvider: React.FC = ({ children }) => { const history = useHistory(); const dispatch = useAppDispatch(); const settings = useSettings(); const account = useOwnAccount(); const path = history.location.pathname; const isUsingMainChatPage = Boolean(path.match(/^\/chats/)); const { chatId } = useParams<{ chatId: string }>(); const [screen, setScreen] = useState(ChatWidgetScreens.INBOX); const [currentChatId, setCurrentChatId] = useState(chatId); const { data: chat } = useChat(currentChatId as string); const mainWindowState = settings.getIn(['chats', 'mainWindow']) as WindowState; const needsAcceptance = !chat?.accepted && chat?.created_by_account !== account?.id; const isOpen = mainWindowState === 'open'; const changeScreen = (screen: ChatWidgetScreens, currentChatId?: string | null) => { setCurrentChatId(currentChatId || null); setScreen(screen); }; const toggleChatPane = () => dispatch(toggleMainWindow()); const value = useMemo(() => ({ chat, needsAcceptance, isOpen, isUsingMainChatPage, toggleChatPane, screen, changeScreen, currentChatId, }), [chat, currentChatId, needsAcceptance, isUsingMainChatPage, isOpen, screen, changeScreen]); useEffect(() => { if (chatId) { setCurrentChatId(chatId); } else { setCurrentChatId(null); } }, [chatId]); return ( {children} ); }; interface IChatContext { chat: IChat | null isOpen: boolean isUsingMainChatPage?: boolean needsAcceptance: boolean toggleChatPane(): void screen: ChatWidgetScreens currentChatId: string | null changeScreen(screen: ChatWidgetScreens, currentChatId?: string | null): void } const useChatContext = (): IChatContext => useContext(ChatContext); export { ChatContext, ChatProvider, useChatContext, ChatWidgetScreens };