import classNames from 'clsx'; import { Map as ImmutableMap, List as ImmutableList, OrderedSet as ImmutableOrderedSet, } from 'immutable'; import escape from 'lodash/escape'; import throttle from 'lodash/throttle'; import React, { useState, useEffect, useRef, useLayoutEffect, useMemo } from 'react'; import { useIntl, defineMessages } from 'react-intl'; import { createSelector } from 'reselect'; import { fetchChatMessages, deleteChatMessage } from 'soapbox/actions/chats'; import { openModal } from 'soapbox/actions/modals'; import { initReportById } from 'soapbox/actions/reports'; import { Avatar, HStack, IconButton, Spinner, Stack, Text } from 'soapbox/components/ui'; import DropdownMenuContainer from 'soapbox/containers/dropdown_menu_container'; import emojify from 'soapbox/features/emoji/emoji'; import Bundle from 'soapbox/features/ui/components/bundle'; import { MediaGallery } from 'soapbox/features/ui/util/async-components'; import { useAppSelector, useAppDispatch, useRefEventHandler, useOwnAccount } from 'soapbox/hooks'; import { IChat, IChatMessage, useChatMessages } from 'soapbox/queries/chats'; import { onlyEmoji } from 'soapbox/utils/rich_content'; import type { Menu } from 'soapbox/components/dropdown_menu'; import type { ChatMessage as ChatMessageEntity } from 'soapbox/types/entities'; const BIG_EMOJI_LIMIT = 1; const messages = defineMessages({ today: { id: 'chats.dividers.today', defaultMessage: 'Today' }, more: { id: 'chats.actions.more', defaultMessage: 'More' }, delete: { id: 'chats.actions.delete', defaultMessage: 'Delete message' }, report: { id: 'chats.actions.report', defaultMessage: 'Report user' }, }); type TimeFormat = 'today' | 'date'; const timeChange = (prev: IChatMessage, curr: IChatMessage): TimeFormat | null => { const prevDate = new Date(prev.created_at).getDate(); const currDate = new Date(curr.created_at).getDate(); const nowDate = new Date().getDate(); if (prevDate !== currDate) { return currDate === nowDate ? 'today' : 'date'; } return null; }; // const makeEmojiMap = (record: any) => record.get('emojis', ImmutableList()).reduce((map: ImmutableMap, emoji: ImmutableMap) => { // return map.set(`:${emoji.get('shortcode')}:`, emoji); // }, ImmutableMap()); const getChatMessages = createSelector( [(chatMessages: ImmutableMap, chatMessageIds: ImmutableOrderedSet) => ( chatMessageIds.reduce((acc, curr) => { const chatMessage = chatMessages.get(curr); return chatMessage ? acc.push(chatMessage) : acc; }, ImmutableList()) )], chatMessages => chatMessages, ); interface IChatMessageList { /** Chat the messages are being rendered from. */ chat: IChat, /** Message IDs to render. */ chatMessageIds: ImmutableOrderedSet, /** Whether to make the chatbox fill the height of the screen. */ autosize?: boolean, } /** Scrollable list of chat messages. */ const ChatMessageList: React.FC = ({ chat, chatMessageIds, autosize }) => { const intl = useIntl(); const dispatch = useAppDispatch(); const account = useOwnAccount(); const [initialLoad, setInitialLoad] = useState(true); const [scrollPosition, setScrollPosition] = useState(0); const { data: chatMessages, isFetching, isFetched, fetchNextPage, isFetchingNextPage, isPlaceholderData } = useChatMessages(chat.id); const formattedChatMessages = chatMessages || []; const me = useAppSelector(state => state.me); const node = useRef(null); const messagesEnd = useRef(null); const lastComputedScroll = useRef(undefined); const scrollBottom = useRef(undefined); const initialCount = useMemo(() => formattedChatMessages.length, []); const scrollToBottom = () => { messagesEnd.current?.scrollIntoView(false); }; const getFormattedTimestamp = (chatMessage: ChatMessageEntity) => { return intl.formatDate( new Date(chatMessage.created_at), { hour12: false, year: 'numeric', month: 'short', day: '2-digit', hour: '2-digit', minute: '2-digit', }, ); }; const setBubbleRef = (c: HTMLDivElement) => { if (!c) return; const links = c.querySelectorAll('a[rel="ugc"]'); links.forEach(link => { link.classList.add('chat-link'); link.setAttribute('rel', 'ugc nofollow noopener'); link.setAttribute('target', '_blank'); }); if (onlyEmoji(c, BIG_EMOJI_LIMIT, false)) { c.classList.add('chat-message__bubble--onlyEmoji'); } else { c.classList.remove('chat-message__bubble--onlyEmoji'); } }; const isNearBottom = (): boolean => { const elem = node.current; if (!elem) return false; const scrollBottom = elem.scrollHeight - elem.offsetHeight - elem.scrollTop; return scrollBottom < elem.offsetHeight * 1.5; }; const restoreScrollPosition = () => { if (node.current && scrollBottom.current) { console.log('bottom', scrollBottom.current); lastComputedScroll.current = node.current.scrollHeight - scrollBottom.current; node.current.scrollTop = lastComputedScroll.current; } }; const handleLoadMore = () => { // const maxId = chatMessages.getIn([0, 'id']) as string; // dispatch(fetchChatMessages(chat.id, maxId as any)); // setIsLoading(true); if (!isFetching) { // setMaxId(formattedChatMessages[0].id); fetchNextPage() .then(() => { if (node.current) { setScrollPosition(node.current.scrollHeight - node.current.scrollTop); } }) .catch(() => null); } }; const handleScroll = throttle(() => { if (node.current) { const { scrollTop, offsetHeight } = node.current; const computedScroll = lastComputedScroll.current === scrollTop; const nearTop = scrollTop < offsetHeight; setScrollPosition(node.current.scrollHeight - node.current.scrollTop); if (nearTop && !isFetching && !initialLoad && !computedScroll) { handleLoadMore(); } } }, 150, { trailing: true, }); const onOpenMedia = (media: any, index: number) => { dispatch(openModal('MEDIA', { media, index })); }; const maybeRenderMedia = (chatMessage: ChatMessageEntity) => { const { attachment } = chatMessage; if (!attachment) return null; return (
{(Component: any) => ( )}
); }; const parsePendingContent = (content: string) => { return escape(content).replace(/(?:\r\n|\r|\n)/g, '
'); }; const parseContent = (chatMessage: ChatMessageEntity) => { const content = chatMessage.content || ''; const pending = chatMessage.pending; const deleting = chatMessage.deleting; const formatted = (pending && !deleting) ? parsePendingContent(content) : content; return formatted; // const emojiMap = makeEmojiMap(chatMessage); // return emojify(formatted, emojiMap.toJS()); }; const renderDivider = (key: React.Key, text: string) => (
); const handleDeleteMessage = (chatId: string, messageId: string) => { return () => { dispatch(deleteChatMessage(chatId, messageId)); }; }; const handleReportUser = (userId: string) => { return () => { dispatch(initReportById(userId)); }; }; const renderMessage = (chatMessage: any) => { const isMyMessage = chatMessage.account_id === me; const menu: Menu = [ { text: intl.formatMessage(messages.delete), action: handleDeleteMessage(chatMessage.chat_id, chatMessage.id), icon: require('@tabler/icons/trash.svg'), destructive: true, }, ]; if (chatMessage.account_id !== me) { menu.push({ text: intl.formatMessage(messages.report), action: handleReportUser(chatMessage.account_id), icon: require('@tabler/icons/flag.svg'), }); } return ( {isMyMessage ? ( //
) : null}
{maybeRenderMedia(chatMessage)}
{intl.formatTime(chatMessage.created_at)}
); }; useEffect(() => { if (isFetched) { setInitialLoad(false); scrollToBottom(); } }, [isFetched]); // Store the scroll position. useLayoutEffect(() => { if (node.current) { const { scrollHeight, scrollTop } = node.current; scrollBottom.current = scrollHeight - scrollTop; } }); // Stick scrollbar to bottom. // useEffect(() => { // if (isNearBottom()) { // scrollToBottom(); // } // // First load. // // if (chatMessages.count() !== initialCount) { // // setInitialLoad(false); // // setIsLoading(false); // // scrollToBottom(); // // } // }, [formattedChatMessages.length]); // useEffect(() => { // scrollToBottom(); // }, [messagesEnd.current]); // History added. const lastChatId = Number(chatMessages && chatMessages[0]?.id); useEffect(() => { // Restore scroll bar position when loading old messages. console.log('hii'); if (!initialLoad) { restoreScrollPosition(); } }, [formattedChatMessages.length, initialLoad]); if (isPlaceholderData) { return ( ); } return (
{/* style={{ height: autosize ? 'calc(100vh - 16rem)' : undefined }} */} {formattedChatMessages.reduce((acc: any, curr: any, idx: number) => { const lastMessage = formattedChatMessages[idx - 1]; if (lastMessage) { const key = `${curr.id}_divider`; switch (timeChange(lastMessage, curr)) { case 'today': acc.push(renderDivider(key, intl.formatMessage(messages.today))); break; case 'date': acc.push(renderDivider(key, intl.formatDate(new Date(curr.created_at), { weekday: 'short', hour: 'numeric', minute: '2-digit', month: 'short', day: 'numeric' }))); break; } } acc.push(renderMessage(curr)); return acc; }, [] as React.ReactNode[])}
); }; export default ChatMessageList;