The communications platform that puts data protection first.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
Rocket.Chat/apps/meteor/client/hooks/useInfiniteMessageQueryUpda...

108 lines
3.2 KiB

import type { IMessage, IRoom } from '@rocket.chat/core-typings';
import { useEffectEvent } from '@rocket.chat/fuselage-hooks';
import { useStream, useUserId } from '@rocket.chat/ui-contexts';
import type { InfiniteData } from '@tanstack/react-query';
import { useQueryClient } from '@tanstack/react-query';
import { useEffect } from 'react';
export const useInfiniteMessageQueryUpdates = <T extends IMessage, TQueryKey extends readonly unknown[]>({
queryKey,
roomId,
filter,
compare = (a, b) => b.ts.getTime() - a.ts.getTime(),
}: {
queryKey: TQueryKey;
roomId: IRoom['_id'];
filter: (message: IMessage) => message is T;
compare?: (a: T, b: T) => number;
}) => {
const queryClient = useQueryClient();
const subscribeToRoomMessages = useStream('room-messages');
const subscribeToNotifyRoom = useStream('notify-room');
const getQueryKey = useEffectEvent(() => queryKey);
const doFilter = useEffectEvent(filter);
const doCompare = useEffectEvent(compare);
const mutateQueryData = useEffectEvent((mutation: (items: T[]) => void) => {
const queryData = queryClient.getQueryData<
InfiniteData<
{
items: T[];
itemCount: number;
},
number
>
>(queryKey);
const items = queryData?.pages.flatMap((page) => page.items) ?? [];
const lastPage = queryData?.pages.at(-1) ?? { items: [], itemCount: 0 };
const beforeMutationItemsLength = items.length;
mutation(items);
const afterMutationItemsLength = items.length;
const pageSize = lastPage.items.length || items.length;
const newPageCount = items.length > 0 ? Math.ceil(items.length / pageSize) : 0;
const newQueryData: typeof queryData = {
pages: Array.from({ length: newPageCount }, (_, pageIndex) => ({
items: items.slice(pageIndex * pageSize, (pageIndex + 1) * pageSize),
itemCount: lastPage.itemCount - (beforeMutationItemsLength - afterMutationItemsLength),
})),
pageParams: Array.from({ length: newPageCount }, (_, pageIndex) => pageIndex * pageSize),
};
queryClient.setQueryData<
InfiniteData<
{
items: T[];
itemCount: number;
},
number
>
>(queryKey, newQueryData);
});
const userId = useUserId();
useEffect(() => {
if (!userId || !roomId) {
return;
}
const unsubscribeFromRoomMessages = subscribeToRoomMessages(roomId, (message) => {
if (!doFilter(message)) return;
mutateQueryData((items) => {
const index = items.findIndex((i) => i._id === message._id);
if (index !== -1) {
items[index] = message as T;
} else {
items.push(message as T);
items.sort(doCompare);
}
});
});
const unsubscribeFromDeleteMessage = subscribeToNotifyRoom(`${roomId}/deleteMessage`, ({ _id }) => {
mutateQueryData((items) => {
const index = items.findIndex((i) => i._id === _id);
if (index !== -1) {
items.splice(index, 1);
}
});
});
const unsubscribeFromDeleteMessageBulk = subscribeToNotifyRoom(`${roomId}/deleteMessageBulk`, () => {
queryClient.invalidateQueries({ queryKey: getQueryKey() });
});
return () => {
unsubscribeFromRoomMessages();
unsubscribeFromDeleteMessage();
unsubscribeFromDeleteMessageBulk();
};
}, [subscribeToRoomMessages, subscribeToNotifyRoom, userId, roomId, queryClient, getQueryKey, doFilter, doCompare, mutateQueryData]);
};