diff --git a/app/soapbox/components/hover_ref_wrapper.tsx b/app/soapbox/components/hover_ref_wrapper.tsx index bf4e253f21..2090543cca 100644 --- a/app/soapbox/components/hover_ref_wrapper.tsx +++ b/app/soapbox/components/hover_ref_wrapper.tsx @@ -15,7 +15,7 @@ const showProfileHoverCard = debounce((dispatch, ref, accountId) => { interface IHoverRefWrapper { accountId: string, - inline: boolean, + inline?: boolean, className?: string, } diff --git a/app/soapbox/features/chats/components/chat_window.js b/app/soapbox/features/chats/components/chat_window.js deleted file mode 100644 index e525e34323..0000000000 Binary files a/app/soapbox/features/chats/components/chat_window.js and /dev/null differ diff --git a/app/soapbox/features/chats/components/chat_window.tsx b/app/soapbox/features/chats/components/chat_window.tsx new file mode 100644 index 0000000000..aecdca1801 --- /dev/null +++ b/app/soapbox/features/chats/components/chat_window.tsx @@ -0,0 +1,113 @@ +import React, { useEffect, useRef } from 'react'; +import { Link } from 'react-router-dom'; + +import { + closeChat, + toggleChat, +} from 'soapbox/actions/chats'; +import Avatar from 'soapbox/components/avatar'; +import HoverRefWrapper from 'soapbox/components/hover_ref_wrapper'; +import IconButton from 'soapbox/components/icon_button'; +import { Counter } from 'soapbox/components/ui'; +import { useAppSelector, useAppDispatch } from 'soapbox/hooks'; +import { makeGetChat } from 'soapbox/selectors'; +import { getAcct } from 'soapbox/utils/accounts'; +import { displayFqn as getDisplayFqn } from 'soapbox/utils/state'; + +import ChatBox from './chat_box'; + +import type { Account as AccountEntity } from 'soapbox/types/entities'; + +type WindowState = 'open' | 'minimized'; + +const getChat = makeGetChat(); + +interface IChatWindow { + /** Position of the chat window on the screen, where 0 is rightmost. */ + idx: number, + /** ID of the chat entity. */ + chatId: string, + /** Whether the window is open or minimized. */ + windowState: WindowState, +} + +/** Floating desktop chat window. */ +const ChatWindow: React.FC = ({ idx, chatId, windowState }) => { + const dispatch = useAppDispatch(); + + const displayFqn = useAppSelector(getDisplayFqn); + + const chat = useAppSelector(state => { + const chat = state.chats.items.get(chatId); + return chat ? getChat(state, chat.toJS() as any) : undefined; + }); + + const inputElem = useRef(null); + + const handleChatClose = (chatId: string) => { + return () => { + dispatch(closeChat(chatId)); + }; + }; + + const handleChatToggle = (chatId: string) => { + return () => { + dispatch(toggleChat(chatId)); + }; + }; + + const handleInputRef = (el: HTMLTextAreaElement) => { + inputElem.current = el; + focusInput(); + }; + + const focusInput = () => { + inputElem.current?.focus(); + }; + + useEffect(() => { + focusInput(); + }, [windowState === 'open']); + + if (!chat) return null; + const account = chat.account as unknown as AccountEntity; + + const right = (285 * (idx + 1)) + 20; + const unreadCount = chat.unread; + + const unreadIcon = ( +
+ +
+ ); + + const avatar = ( + + + + + + ); + + return ( +
+
+ {unreadCount > 0 ? unreadIcon : avatar } + +
+ +
+
+
+ +
+
+ ); +}; + +export default ChatWindow;