import { EdgelessLegacySlotIdentifier } from '@blocksuite/affine-block-surface'; import { NoteBlockModel, NoteDisplayMode } from '@blocksuite/affine-model'; import { NotificationProvider, SidebarExtensionIdentifier, type ToolbarAction, type ToolbarContext, type ToolbarModuleConfig, ToolbarModuleExtension, } from '@blocksuite/affine-shared/services'; import { Bound } from '@blocksuite/global/gfx'; import { AutoHeightIcon, CustomizedHeightIcon, InsertIntoPageIcon, ScissorsIcon, } from '@blocksuite/icons/lit'; import { BlockFlavourIdentifier } from '@blocksuite/std'; import type { ExtensionType } from '@blocksuite/store'; import { computed } from '@preact/signals-core'; import { html } from 'lit'; import { changeNoteDisplayMode } from '../commands'; import { NoteConfigExtension } from '../config'; const trackBaseProps = { category: 'note', }; const builtinSurfaceToolbarConfig = { actions: [ { id: 'a.show-in', when(ctx) { return ( ctx.getSurfaceModelsByType(NoteBlockModel).length === 1 && ctx.features.getFlag('enable_advanced_block_visibility') ); }, content(ctx) { const models = ctx.getSurfaceModelsByType(NoteBlockModel); if (!models.length) return null; const firstModel = models[0]; const { displayMode } = firstModel.props; const onSelect = (e: CustomEvent) => { e.stopPropagation(); const newMode = e.detail; setDisplayMode(ctx, firstModel, newMode); }; return html` `; }, }, { id: 'b.display-in-page', when(ctx) { const elements = ctx.getSurfaceModelsByType(NoteBlockModel); return ( elements.length === 1 && !elements[0].isPageBlock() && !ctx.features.getFlag('enable_advanced_block_visibility') ); }, generate(ctx) { const models = ctx.getSurfaceModelsByType(NoteBlockModel); if (!models.length) return null; const firstModel = models[0]; const shouldShowTooltip$ = computed( () => firstModel.props.displayMode$.value === NoteDisplayMode.DocAndEdgeless ); const label$ = computed(() => firstModel.props.displayMode$.value === NoteDisplayMode.EdgelessOnly ? 'Display in Page' : 'Displayed in Page' ); const onSelect = () => { const newMode = firstModel.props.displayMode === NoteDisplayMode.EdgelessOnly ? NoteDisplayMode.DocAndEdgeless : NoteDisplayMode.EdgelessOnly; setDisplayMode(ctx, firstModel, newMode); ctx.track('BlockCreated', { page: 'whiteboard editor', module: 'toolbar', segment: 'toolbar', blockType: 'affine:note', control: 'toolbar:general', other: `display in page: ${newMode === NoteDisplayMode.EdgelessOnly ? 'off' : 'on'}`, }); }; return { content: html` onSelect()} > ${InsertIntoPageIcon()} ${label$.value} `, }; }, }, { id: 'd.style', when(ctx) { const elements = ctx.getSurfaceModelsByType(NoteBlockModel); return ( elements.length > 0 && elements[0].props.displayMode !== NoteDisplayMode.DocOnly ); }, actions: [ { id: 'b.style', when: ctx => { const models = ctx.getSurfaceModels(); return ( models.length > 0 && models.every(model => model instanceof NoteBlockModel) ); }, content(ctx) { const notes = ctx.getSurfaceModelsByType(NoteBlockModel); return html``; }, } satisfies ToolbarAction, ], }, { id: 'e.slicer', label: 'Slicer', icon: ScissorsIcon(), tooltip: html``, active: false, when(ctx) { return ( ctx.getSurfaceModelsByType(NoteBlockModel).length === 1 && ctx.features.getFlag('enable_advanced_block_visibility') ); }, run(ctx) { ctx.std.get(EdgelessLegacySlotIdentifier).toggleNoteSlicer.next(); }, }, { id: 'f.auto-height', label: 'Size', when(ctx) { const elements = ctx.getSurfaceModelsByType(NoteBlockModel); return ( elements.length > 0 && (!elements[0].isPageBlock() || !ctx.std.getOptional(NoteConfigExtension.identifier) ?.edgelessNoteHeader) ); }, generate(ctx) { const models = ctx.getSurfaceModelsByType(NoteBlockModel); if (!models.length) return null; const firstModel = models[0]; const { collapse } = firstModel.props.edgeless$.value; const options: Pick = collapse ? { tooltip: 'Auto height', icon: AutoHeightIcon(), } : { tooltip: 'Customized height', icon: CustomizedHeightIcon(), }; return { ...options, run(ctx) { ctx.store.captureSync(); for (const model of models) { const edgeless = model.props.edgeless; if (edgeless.collapse) { ctx.store.updateBlock(model, () => { model.props.edgeless.collapse = false; }); continue; } if (edgeless.collapsedHeight) { const bounds = Bound.deserialize(model.xywh); bounds.h = edgeless.collapsedHeight * (edgeless.scale ?? 1); const xywh = bounds.serialize(); ctx.store.updateBlock(model, () => { model.xywh = xywh; model.props.edgeless.collapse = true; }); } } }, }; }, }, { id: 'g.scale', content(ctx) { const models = ctx.getSurfaceModelsByType(NoteBlockModel); if (!models.length) return null; const firstModel = models[0]; const scale$ = computed(() => { const scale = firstModel.props.edgeless$.value.scale ?? 1; return Math.round(100 * scale); }); const onSelect = (e: CustomEvent) => { e.stopPropagation(); const scale = e.detail / 100; models.forEach(model => { const bounds = Bound.deserialize(model.xywh); const oldScale = model.props.edgeless.scale ?? 1; const ratio = scale / oldScale; bounds.w *= ratio; bounds.h *= ratio; const xywh = bounds.serialize(); ctx.store.updateBlock(model, () => { model.xywh = xywh; model.props.edgeless.scale = scale; }); }); ctx.track('SelectedCardScale', { ...trackBaseProps, control: 'select card scale', }); }; const onToggle = (e: CustomEvent) => { e.stopPropagation(); const opened = e.detail; if (!opened) return; ctx.track('OpenedCardScaleSelector', { ...trackBaseProps, control: 'switch card scale', }); }; const format = (value: number) => `${value}%`; return html``; }, }, ], when: ctx => ctx.getSurfaceModelsByType(NoteBlockModel).length > 0, } as const satisfies ToolbarModuleConfig; function setDisplayMode( ctx: ToolbarContext, model: NoteBlockModel, newMode: NoteDisplayMode ) { const displayMode = model.props.displayMode; ctx.command.exec(changeNoteDisplayMode, { noteId: model.id, mode: newMode, stopCapture: true, }); // if change note to page only, should clear the selection if (newMode === NoteDisplayMode.DocOnly) { ctx.selection.clear(); } const data = newMode === NoteDisplayMode.EdgelessOnly ? { title: 'Note removed from Page Mode', message: 'Content removed from your page.', } : { title: 'Note displayed in Page Mode', message: 'Content added to your page.', }; const notification = ctx.std.getOptional(NotificationProvider); notification?.notifyWithUndoAction({ title: data.title, message: `${data.message} Find it in the TOC for quick navigation.`, accent: 'success', duration: 5 * 1000, actions: [ { key: 'view-in-toc', label: 'View in Toc', onClick: () => { const sidebar = ctx.std.getOptional(SidebarExtensionIdentifier); sidebar?.open('outline'); }, }, ], }); ctx.track('NoteDisplayModeChanged', { ...trackBaseProps, control: 'display mode', other: `from ${displayMode} to ${newMode}`, }); } export const createBuiltinToolbarConfigExtension = ( flavour: string ): ExtensionType[] => { const name = flavour.split(':').pop(); return [ ToolbarModuleExtension({ id: BlockFlavourIdentifier(`affine:surface:${name}`), config: builtinSurfaceToolbarConfig, }), ]; };