mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-02-13 21:05:19 +00:00
65 lines
1.6 KiB
TypeScript
65 lines
1.6 KiB
TypeScript
import { assertExists, DisposableGroup } from '@blocksuite/global/utils';
|
|
import type { Page, Workspace } from '@blocksuite/store';
|
|
import type { Atom } from 'jotai';
|
|
import { atom, useAtomValue } from 'jotai';
|
|
import { useEffect } from 'react';
|
|
|
|
const weakMap = new WeakMap<Workspace, Map<string, Atom<Page | null>>>();
|
|
|
|
const emptyAtom = atom<Page | null>(null);
|
|
|
|
function getAtom(w: Workspace, pageId: string | null): Atom<Page | null> {
|
|
if (!pageId) {
|
|
return emptyAtom;
|
|
}
|
|
if (!weakMap.has(w)) {
|
|
weakMap.set(w, new Map());
|
|
}
|
|
const map = weakMap.get(w);
|
|
assertExists(map);
|
|
if (!map.has(pageId)) {
|
|
const baseAtom = atom(w.getPage(pageId));
|
|
baseAtom.onMount = set => {
|
|
const group = new DisposableGroup();
|
|
group.add(
|
|
w.slots.pageAdded.on(id => {
|
|
if (pageId === id) {
|
|
set(w.getPage(id));
|
|
}
|
|
})
|
|
);
|
|
group.add(
|
|
w.slots.pageRemoved.on(id => {
|
|
if (pageId === id) {
|
|
set(null);
|
|
}
|
|
})
|
|
);
|
|
return () => {
|
|
group.dispose();
|
|
};
|
|
};
|
|
map.set(pageId, baseAtom);
|
|
return baseAtom;
|
|
} else {
|
|
return map.get(pageId) as Atom<Page | null>;
|
|
}
|
|
}
|
|
|
|
export function useBlockSuiteWorkspacePage(
|
|
blockSuiteWorkspace: Workspace,
|
|
pageId: string | null
|
|
): Page | null {
|
|
const pageAtom = getAtom(blockSuiteWorkspace, pageId);
|
|
assertExists(pageAtom);
|
|
const page = useAtomValue(pageAtom);
|
|
|
|
useEffect(() => {
|
|
if (!page?.loaded) {
|
|
page?.waitForLoaded().catch(console.error);
|
|
}
|
|
}, [page]);
|
|
|
|
return page;
|
|
}
|