mirror of
https://github.com/docmost/docmost.git
synced 2026-05-07 22:53:08 +08:00
bd68e47e03
* feat: page verification workflow * feat: refactor page-verification * sync * fix type * fix * fix * notification icon * use full word * accept .license file * - update templates - update migration and notification * fix copy * update audit labels * sync * add space name
169 lines
5.4 KiB
TypeScript
169 lines
5.4 KiB
TypeScript
import React from "react";
|
|
import { socketAtom } from "@/features/websocket/atoms/socket-atom.ts";
|
|
import { useAtom } from "jotai";
|
|
import { InfiniteData, useQueryClient } from "@tanstack/react-query";
|
|
import { WebSocketEvent } from "@/features/websocket/types";
|
|
import { IPage } from "../page/types/page.types";
|
|
import { IPagination } from "@/lib/types";
|
|
import {
|
|
invalidateOnCreatePage,
|
|
invalidateOnDeletePage,
|
|
updateCacheOnMovePage,
|
|
invalidateOnUpdatePage,
|
|
} from "../page/queries/page-query";
|
|
import { RQ_KEY } from "../comment/queries/comment-query";
|
|
import { IComment } from "@/features/comment/types/comment.types";
|
|
|
|
export const useQuerySubscription = () => {
|
|
const queryClient = useQueryClient();
|
|
const [socket] = useAtom(socketAtom);
|
|
|
|
React.useEffect(() => {
|
|
socket?.on("message", (event) => {
|
|
const data: WebSocketEvent = event;
|
|
|
|
let entity = null;
|
|
let queryKeyId = null;
|
|
|
|
switch (data.operation) {
|
|
case "invalidate":
|
|
queryClient.invalidateQueries({
|
|
queryKey: [...data.entity, data.id].filter(Boolean),
|
|
});
|
|
break;
|
|
case "commentCreated": {
|
|
const createCache = queryClient.getQueryData(
|
|
RQ_KEY(data.pageId),
|
|
) as InfiniteData<IPagination<IComment>> | undefined;
|
|
|
|
if (createCache && createCache.pages.length > 0) {
|
|
const alreadyExists = createCache.pages.some((page) =>
|
|
page.items.some((c) => c.id === data.comment.id),
|
|
);
|
|
if (alreadyExists) break;
|
|
|
|
const lastIdx = createCache.pages.length - 1;
|
|
queryClient.setQueryData(RQ_KEY(data.pageId), {
|
|
...createCache,
|
|
pages: createCache.pages.map((page, i) =>
|
|
i === lastIdx
|
|
? { ...page, items: [...page.items, data.comment] }
|
|
: page,
|
|
),
|
|
});
|
|
}
|
|
break;
|
|
}
|
|
case "commentUpdated":
|
|
case "commentResolved": {
|
|
const updateCache = queryClient.getQueryData(
|
|
RQ_KEY(data.pageId),
|
|
) as InfiniteData<IPagination<IComment>> | undefined;
|
|
|
|
if (updateCache) {
|
|
queryClient.setQueryData(RQ_KEY(data.pageId), {
|
|
...updateCache,
|
|
pages: updateCache.pages.map((page) => ({
|
|
...page,
|
|
items: page.items.map((comment) =>
|
|
comment.id === data.comment.id ? data.comment : comment,
|
|
),
|
|
})),
|
|
});
|
|
}
|
|
break;
|
|
}
|
|
case "commentDeleted": {
|
|
const deleteCache = queryClient.getQueryData(
|
|
RQ_KEY(data.pageId),
|
|
) as InfiniteData<IPagination<IComment>> | undefined;
|
|
|
|
if (deleteCache) {
|
|
queryClient.setQueryData(RQ_KEY(data.pageId), {
|
|
...deleteCache,
|
|
pages: deleteCache.pages.map((page) => ({
|
|
...page,
|
|
items: page.items.filter(
|
|
(comment) => comment.id !== data.commentId,
|
|
),
|
|
})),
|
|
});
|
|
}
|
|
break;
|
|
}
|
|
case "addTreeNode":
|
|
invalidateOnCreatePage(data.payload.data);
|
|
break;
|
|
case "moveTreeNode":
|
|
updateCacheOnMovePage(
|
|
data.spaceId,
|
|
data.payload.id,
|
|
data.payload.oldParentId,
|
|
data.payload.parentId,
|
|
data.payload.pageData,
|
|
);
|
|
break;
|
|
case "deleteTreeNode":
|
|
invalidateOnDeletePage(data.payload.node.id);
|
|
break;
|
|
case "updateOne":
|
|
entity = data.entity[0];
|
|
if (entity === "pages") {
|
|
// we have to do this because the usePageQuery cache key is the slugId.
|
|
queryKeyId = data.payload.slugId;
|
|
} else {
|
|
queryKeyId = data.id;
|
|
}
|
|
|
|
// only update if data was already in cache
|
|
if (queryClient.getQueryData([...data.entity, queryKeyId])) {
|
|
queryClient.setQueryData([...data.entity, queryKeyId], {
|
|
...queryClient.getQueryData([...data.entity, queryKeyId]),
|
|
...data.payload,
|
|
});
|
|
}
|
|
|
|
if (entity === "pages") {
|
|
invalidateOnUpdatePage(
|
|
data.spaceId,
|
|
data.payload.parentPageId,
|
|
data.id,
|
|
data.payload.title,
|
|
data.payload.icon,
|
|
);
|
|
}
|
|
|
|
/*
|
|
queryClient.setQueriesData(
|
|
{ queryKey: [data.entity, data.id] },
|
|
(oldData: any) => {
|
|
const update = (entity: Record<string, unknown>) =>
|
|
entity.id === data.id ? { ...entity, ...data.payload } : entity;
|
|
return Array.isArray(oldData)
|
|
? oldData.map(update)
|
|
: update(oldData as Record<string, unknown>);
|
|
},
|
|
);
|
|
*/
|
|
break;
|
|
case "refetchRootTreeNodeEvent": {
|
|
const spaceId = data.spaceId;
|
|
queryClient.refetchQueries({
|
|
queryKey: ["root-sidebar-pages", spaceId],
|
|
});
|
|
|
|
queryClient.invalidateQueries({
|
|
queryKey: ["recent-changes", spaceId],
|
|
});
|
|
break;
|
|
}
|
|
case "verificationUpdated":
|
|
queryClient.invalidateQueries({
|
|
queryKey: ["page-verification-info", data.pageId],
|
|
});
|
|
break;
|
|
}
|
|
});
|
|
}, [queryClient, socket]);
|
|
};
|