Files
AFFiNE-Mirror/blocksuite/playground/apps/_common/mock-services.ts
2024-12-20 15:38:06 +08:00

207 lines
5.7 KiB
TypeScript

/* eslint-disable @typescript-eslint/no-non-null-assertion */
import type {
PeekOptions,
PeekViewService,
} from '@blocksuite/affine-components/peek';
import { PeekViewExtension } from '@blocksuite/affine-components/peek';
import { BlockComponent } from '@blocksuite/block-std';
import {
ColorScheme,
type DocMode,
type DocModeProvider,
type GenerateDocUrlService,
matchFlavours,
type NotificationService,
type ParseDocUrlService,
type ReferenceParams,
type ThemeExtension,
toast,
} from '@blocksuite/blocks';
import type { AffineEditorContainer } from '@blocksuite/presets';
import { type DocCollection, Slot } from '@blocksuite/store';
import { signal } from '@preact/signals-core';
import type { TemplateResult } from 'lit';
import type { AttachmentViewerPanel } from './components/attachment-viewer-panel.js';
function getModeFromStorage() {
const mapJson = localStorage.getItem('playground:docMode');
const mapArray = mapJson ? JSON.parse(mapJson) : [];
return new Map<string, DocMode>(mapArray);
}
function saveModeToStorage(map: Map<string, DocMode>) {
const mapArray = Array.from(map);
const mapJson = JSON.stringify(mapArray);
localStorage.setItem('playground:docMode', mapJson);
}
export function removeModeFromStorage(docId: string) {
const modeMap = getModeFromStorage();
modeMap.delete(docId);
saveModeToStorage(modeMap);
}
const DEFAULT_MODE: DocMode = 'page';
const slotMap = new Map<string, Slot<DocMode>>();
export function mockDocModeService(
getEditorModeCallback: () => DocMode,
setEditorModeCallback: (mode: DocMode) => void
) {
const docModeService: DocModeProvider = {
getPrimaryMode: (docId: string) => {
try {
const modeMap = getModeFromStorage();
return modeMap.get(docId) ?? DEFAULT_MODE;
} catch {
return DEFAULT_MODE;
}
},
onPrimaryModeChange: (handler: (mode: DocMode) => void, docId: string) => {
if (!slotMap.get(docId)) {
slotMap.set(docId, new Slot());
}
return slotMap.get(docId)!.on(handler);
},
getEditorMode: () => {
return getEditorModeCallback();
},
setEditorMode: (mode: DocMode) => {
setEditorModeCallback(mode);
},
setPrimaryMode: (mode: DocMode, docId: string) => {
const modeMap = getModeFromStorage();
modeMap.set(docId, mode);
saveModeToStorage(modeMap);
slotMap.get(docId)?.emit(mode);
},
togglePrimaryMode: (docId: string) => {
const mode =
docModeService.getPrimaryMode(docId) === 'page' ? 'edgeless' : 'page';
docModeService.setPrimaryMode(mode, docId);
return mode;
},
};
return docModeService;
}
export function mockNotificationService(editor: AffineEditorContainer) {
const notificationService: NotificationService = {
toast: (message, options) => {
toast(editor.host!, message, options?.duration);
},
confirm: notification => {
return Promise.resolve(confirm(notification.title.toString()));
},
prompt: notification => {
return Promise.resolve(
prompt(notification.title.toString(), notification.autofill?.toString())
);
},
notify: notification => {
// todo: implement in playground
console.log(notification);
},
};
return notificationService;
}
export function mockParseDocUrlService(collection: DocCollection) {
const parseDocUrlService: ParseDocUrlService = {
parseDocUrl: (url: string) => {
if (url && URL.canParse(url)) {
const path = decodeURIComponent(new URL(url).hash.slice(1));
const item =
path.length > 0
? [...collection.docs.values()].find(doc => doc.id === path)
: null;
if (item) {
return {
docId: item.id,
};
}
}
return;
},
};
return parseDocUrlService;
}
export class MockEdgelessTheme {
theme$ = signal(ColorScheme.Light);
setTheme(theme: ColorScheme) {
this.theme$.value = theme;
}
toggleTheme() {
const theme =
this.theme$.value === ColorScheme.Dark
? ColorScheme.Light
: ColorScheme.Dark;
this.theme$.value = theme;
}
}
export const mockEdgelessTheme = new MockEdgelessTheme();
export const themeExtension: ThemeExtension = {
getEdgelessTheme() {
return mockEdgelessTheme.theme$;
},
};
export function mockPeekViewExtension(
attachmentViewerPanel: AttachmentViewerPanel
) {
return PeekViewExtension({
peek(
element: {
target: HTMLElement;
docId: string;
blockIds?: string[];
template?: TemplateResult;
},
options?: PeekOptions
) {
const { target } = element;
if (
target instanceof BlockComponent &&
matchFlavours(target.model, ['affine:attachment'])
) {
attachmentViewerPanel.open(target.model);
return Promise.resolve();
}
alert('Peek view not implemented in playground');
console.log('peek', element, options);
return Promise.resolve();
},
} satisfies PeekViewService);
}
export function mockGenerateDocUrlService(collection: DocCollection) {
const generateDocUrlService: GenerateDocUrlService = {
generateDocUrl: (docId: string, params?: ReferenceParams) => {
const doc = collection.getDoc(docId);
if (!doc) return;
const url = new URL(location.pathname, location.origin);
url.search = location.search;
if (params) {
const search = url.searchParams;
for (const [key, value] of Object.entries(params)) {
search.set(key, Array.isArray(value) ? value.join(',') : value);
}
}
url.hash = encodeURIComponent(docId);
return url.toString();
},
};
return generateDocUrlService;
}