mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-02-13 12:55:00 +00:00
110 lines
3.2 KiB
TypeScript
110 lines
3.2 KiB
TypeScript
import type { DocMode } from '@blocksuite/affine-model';
|
|
import type { Container } from '@blocksuite/global/di';
|
|
import { createIdentifier } from '@blocksuite/global/di';
|
|
import { noop } from '@blocksuite/global/utils';
|
|
import type { ExtensionType } from '@blocksuite/store';
|
|
import { Extension } from '@blocksuite/store';
|
|
import { Subject, type Subscription } from 'rxjs';
|
|
|
|
const DEFAULT_MODE: DocMode = 'page';
|
|
|
|
export interface DocModeProvider {
|
|
/**
|
|
* Set the primary mode of the doc.
|
|
* This would not affect the current editor mode.
|
|
* If you want to switch the editor mode, use `setEditorMode` instead.
|
|
* @param mode - The mode to set.
|
|
* @param docId - The id of the doc.
|
|
*/
|
|
setPrimaryMode: (mode: DocMode, docId: string) => void;
|
|
/**
|
|
* Get the primary mode of the doc.
|
|
* Normally, it would be used to query the mode of other doc.
|
|
* @param docId - The id of the doc.
|
|
* @returns The primary mode of the document.
|
|
*/
|
|
getPrimaryMode: (docId: string) => DocMode;
|
|
/**
|
|
* Toggle the primary mode of the doc.
|
|
* @param docId - The id of the doc.
|
|
* @returns The new primary mode of the doc.
|
|
*/
|
|
togglePrimaryMode: (docId: string) => DocMode;
|
|
/**
|
|
* Subscribe to changes in the primary mode of the doc.
|
|
* For example:
|
|
* Embed-linked-doc-block will subscribe to the primary mode of the linked doc,
|
|
* and will display different UI according to the primary mode of the linked doc.
|
|
* @param handler - The handler to call when the primary mode of certain doc changes.
|
|
* @param docId - The id of the doc.
|
|
* @returns A disposable to stop the subscription.
|
|
*/
|
|
onPrimaryModeChange: (
|
|
handler: (mode: DocMode) => void,
|
|
docId: string
|
|
) => Subscription;
|
|
/**
|
|
* Set the editor mode. Normally, it would be used to set the mode of the current editor.
|
|
* When patch or override the doc mode service, can pass a callback to set the editor mode.
|
|
* @param mode - The mode to set.
|
|
*/
|
|
setEditorMode: (mode: DocMode) => void;
|
|
/**
|
|
* Get current editor mode.
|
|
* @returns The editor mode.
|
|
*/
|
|
getEditorMode: () => DocMode | null;
|
|
}
|
|
|
|
export const DocModeProvider = createIdentifier<DocModeProvider>(
|
|
'AffineDocModeService'
|
|
);
|
|
|
|
const modeMap = new Map<string, DocMode>();
|
|
const slotMap = new Map<string, Subject<DocMode>>();
|
|
|
|
export class DocModeService extends Extension implements DocModeProvider {
|
|
static override setup(di: Container) {
|
|
di.addImpl(DocModeProvider, DocModeService);
|
|
}
|
|
|
|
getEditorMode(): DocMode | null {
|
|
return null;
|
|
}
|
|
|
|
getPrimaryMode(id: string) {
|
|
return modeMap.get(id) ?? DEFAULT_MODE;
|
|
}
|
|
|
|
onPrimaryModeChange(handler: (mode: DocMode) => void, id: string) {
|
|
if (!slotMap.get(id)) {
|
|
slotMap.set(id, new Subject());
|
|
}
|
|
return slotMap.get(id)!.subscribe(handler);
|
|
}
|
|
|
|
setEditorMode(mode: DocMode) {
|
|
noop(mode);
|
|
}
|
|
|
|
setPrimaryMode(mode: DocMode, id: string) {
|
|
modeMap.set(id, mode);
|
|
slotMap.get(id)?.next(mode);
|
|
}
|
|
|
|
togglePrimaryMode(id: string) {
|
|
const mode = this.getPrimaryMode(id) === 'page' ? 'edgeless' : 'page';
|
|
this.setPrimaryMode(mode, id);
|
|
|
|
return mode;
|
|
}
|
|
}
|
|
|
|
export function DocModeExtension(service: DocModeProvider): ExtensionType {
|
|
return {
|
|
setup: di => {
|
|
di.override(DocModeProvider, () => service);
|
|
},
|
|
};
|
|
}
|