import { InfiniteData } from '@tanstack/react-query'; import { getSettings } from 'soapbox/actions/settings'; import messages from 'soapbox/locales/messages'; import { normalizeChatMessage } from 'soapbox/normalizers'; import { ChatKeys, IChat, isLastMessage } from 'soapbox/queries/chats'; import { queryClient } from 'soapbox/queries/client'; import { updatePageItem, appendPageItem, removePageItem, flattenPages, PaginatedResult, sortQueryData } from 'soapbox/utils/queries'; // import { play, soundCache } from 'soapbox/utils/sounds'; import { connectStream } from '../stream'; import { deleteAnnouncement, fetchAnnouncements, updateAnnouncements, updateReaction as updateAnnouncementsReaction, } from './announcements'; import { updateConversations } from './conversations'; import { fetchFilters } from './filters'; import { MARKER_FETCH_SUCCESS } from './markers'; import { updateNotificationsQueue, expandNotifications } from './notifications'; import { updateStatus } from './statuses'; import { // deleteFromTimelines, expandHomeTimeline, connectTimeline, disconnectTimeline, processTimelineUpdate, } from './timelines'; import type { AppDispatch, RootState } from 'soapbox/store'; import type { APIEntity, Chat, ChatMessage } from 'soapbox/types/entities'; const STREAMING_CHAT_UPDATE = 'STREAMING_CHAT_UPDATE'; const STREAMING_FOLLOW_RELATIONSHIPS_UPDATE = 'STREAMING_FOLLOW_RELATIONSHIPS_UPDATE'; const validLocale = (locale: string) => Object.keys(messages).includes(locale); const getLocale = (state: RootState) => { const locale = getSettings(state).get('locale') as string; return validLocale(locale) ? locale : 'en'; }; const updateFollowRelationships = (relationships: APIEntity) => (dispatch: AppDispatch, getState: () => RootState) => { const me = getState().me; return dispatch({ type: STREAMING_FOLLOW_RELATIONSHIPS_UPDATE, me, ...relationships, }); }; interface ChatPayload extends Omit { last_message: ChatMessage | null, } const dateComparator = (chatA: IChat, chatB: IChat): number => { const chatADate = new Date(chatA.last_message?.created_at as string); const chatBDate = new Date(chatB.last_message?.created_at as string); if (chatBDate < chatADate) return -1; if (chatBDate > chatADate) return 1; return 0; }; const updateChat = (payload: ChatPayload) => { const { id: chatId, last_message: lastMessage } = payload; const currentChats = flattenPages( queryClient.getQueryData>>(ChatKeys.chatSearch()), ); if (currentChats?.find((chat: any) => chat.id === chatId)) { // If the chat exists in the client, let's update it. updatePageItem(ChatKeys.chatSearch(), payload as any, (o, n) => o.id === n.id); // Now that we have the new chat loaded, let's re-sort to put // the most recent on top. sortQueryData(ChatKeys.chatSearch(), dateComparator); } else { // If this is a brand-new chat, let's invalid the queries. queryClient.invalidateQueries(ChatKeys.chatSearch()); } if (lastMessage) { // Update the Chat Messages query data. appendPageItem(ChatKeys.chatMessages(payload.id), normalizeChatMessage(lastMessage)); } }; const removeChatMessage = (payload: string) => { const data = JSON.parse(payload); const chatId = data.chat_id; const chatMessageId = data.deleted_message_id; // If the user just deleted the "last_message", then let's invalidate // the Chat Search query so the Chat List will show the new "last_message". if (isLastMessage(chatMessageId)) { queryClient.invalidateQueries(ChatKeys.chatSearch()); } removePageItem(ChatKeys.chatMessages(chatId), chatMessageId, (o: any, n: any) => String(o.id) === String(n)); }; // Update the specific Chat query data. const updateChatQuery = (chat: IChat) => { const cachedChat = queryClient.getQueryData(ChatKeys.chat(chat.id)); if (!cachedChat) { return; } const newChat = { ...cachedChat, latest_read_message_by_account: chat.latest_read_message_by_account, latest_read_message_created_at: chat.latest_read_message_created_at, }; queryClient.setQueryData(ChatKeys.chat(chat.id), newChat as any); }; const connectTimelineStream = ( timelineId: string, path: string, pollingRefresh: ((dispatch: AppDispatch, done?: () => void) => void) | null = null, accept: ((status: APIEntity) => boolean) | null = null, ) => connectStream(path, pollingRefresh, (dispatch: AppDispatch, getState: () => RootState) => { const locale = getLocale(getState()); return { onConnect() { dispatch(connectTimeline(timelineId)); }, onDisconnect() { dispatch(disconnectTimeline(timelineId)); }, onReceive(data: any) { switch (data.event) { case 'update': dispatch(processTimelineUpdate(timelineId, JSON.parse(data.payload), accept)); break; case 'status.update': dispatch(updateStatus(JSON.parse(data.payload))); break; // FIXME: We think delete & redraft is causing jumpy timelines. // Fix that in ScrollableList then re-enable this! // // case 'delete': // dispatch(deleteFromTimelines(data.payload)); // break; case 'notification': messages[locale]().then(messages => { dispatch( updateNotificationsQueue( JSON.parse(data.payload), messages, locale, window.location.pathname, ), ); }).catch(error => { console.error(error); }); break; case 'conversation': dispatch(updateConversations(JSON.parse(data.payload))); break; case 'filters_changed': dispatch(fetchFilters()); break; case 'pleroma:chat_update': case 'chat_message.created': // TruthSocial dispatch((_dispatch: AppDispatch, getState: () => RootState) => { const chat = JSON.parse(data.payload); const me = getState().me; const messageOwned = chat.last_message?.account_id === me; // Don't update own messages from streaming if (!messageOwned) { updateChat(chat); // Temp disable until we support disabling/enabling. // play(soundCache.chat); } }); break; case 'chat_message.deleted': // TruthSocial removeChatMessage(data.payload); break; case 'chat_message.read': // TruthSocial dispatch((_dispatch: AppDispatch, getState: () => RootState) => { const chat = JSON.parse(data.payload); const me = getState().me; const isFromOtherUser = chat.account.id !== me; if (isFromOtherUser) { updateChatQuery(JSON.parse(data.payload)); } }); break; case 'pleroma:follow_relationships_update': dispatch(updateFollowRelationships(JSON.parse(data.payload))); break; case 'announcement': dispatch(updateAnnouncements(JSON.parse(data.payload))); break; case 'announcement.reaction': dispatch(updateAnnouncementsReaction(JSON.parse(data.payload))); break; case 'announcement.delete': dispatch(deleteAnnouncement(data.payload)); break; case 'marker': dispatch({ type: MARKER_FETCH_SUCCESS, marker: JSON.parse(data.payload) }); break; } }, }; }); const refreshHomeTimelineAndNotification = (dispatch: AppDispatch, done?: () => void) => dispatch(expandHomeTimeline({}, () => dispatch(expandNotifications({}, () => dispatch(fetchAnnouncements(done)))))); const connectUserStream = () => connectTimelineStream('home', 'user', refreshHomeTimelineAndNotification); const connectCommunityStream = ({ onlyMedia }: Record = {}) => connectTimelineStream(`community${onlyMedia ? ':media' : ''}`, `public:local${onlyMedia ? ':media' : ''}`); const connectPublicStream = ({ onlyMedia }: Record = {}) => connectTimelineStream(`public${onlyMedia ? ':media' : ''}`, `public${onlyMedia ? ':media' : ''}`); const connectRemoteStream = (instance: string, { onlyMedia }: Record = {}) => connectTimelineStream(`remote${onlyMedia ? ':media' : ''}:${instance}`, `public:remote${onlyMedia ? ':media' : ''}&instance=${instance}`); const connectHashtagStream = (id: string, tag: string, accept: (status: APIEntity) => boolean) => connectTimelineStream(`hashtag:${id}`, `hashtag&tag=${tag}`, null, accept); const connectDirectStream = () => connectTimelineStream('direct', 'direct'); const connectListStream = (id: string) => connectTimelineStream(`list:${id}`, `list&list=${id}`); const connectGroupStream = (id: string) => connectTimelineStream(`group:${id}`, `group&group=${id}`); export { STREAMING_CHAT_UPDATE, STREAMING_FOLLOW_RELATIONSHIPS_UPDATE, connectTimelineStream, connectUserStream, connectCommunityStream, connectPublicStream, connectRemoteStream, connectHashtagStream, connectDirectStream, connectListStream, connectGroupStream, };