@templatical/core
Advanced tools
+2
-9
@@ -1,2 +0,2 @@ | ||
| import { createBlock, createDefaultTemplateContent, generateId } from "@templatical/types"; | ||
| import { createBlock, createDefaultTemplateContent, generateId, safeClone } from "@templatical/types"; | ||
| import { computed, reactive, readonly, ref, watch } from "@vue/reactivity"; | ||
@@ -197,10 +197,3 @@ //#region src/editor.ts | ||
| function cloneContent() { | ||
| const seen = /* @__PURE__ */ new WeakSet(); | ||
| return JSON.parse(JSON.stringify(content.value, (_key, value) => { | ||
| if (typeof value === "object" && value !== null) { | ||
| if (seen.has(value)) return void 0; | ||
| seen.add(value); | ||
| } | ||
| return value; | ||
| })); | ||
| return safeClone(content.value); | ||
| } | ||
@@ -207,0 +200,0 @@ function pushToUndoStack(snapshot) { |
@@ -1,1 +0,1 @@ | ||
| {"version":3,"file":"index.js","names":[],"sources":["../src/editor.ts","../src/history.ts","../src/block-actions.ts","../src/auto-save.ts","../src/condition-preview.ts","../src/data-source-fetch.ts","../src/history-interceptor.ts"],"sourcesContent":["import type {\n Block,\n ColumnLayout,\n TemplateContent,\n TemplateDefaults,\n TemplateSettings,\n UiTheme,\n ViewportSize,\n} from \"@templatical/types\";\nimport { createDefaultTemplateContent } from \"@templatical/types\";\n\nfunction getColumnCount(layout: ColumnLayout): number {\n if (layout === \"1\") return 1;\n if (layout === \"3\") return 3;\n return 2;\n}\nimport {\n computed,\n reactive,\n readonly,\n type DeepReadonly,\n type Ref,\n} from \"@vue/reactivity\";\n\nexport interface EditorState {\n content: TemplateContent;\n selectedBlockId: string | null;\n viewport: ViewportSize;\n darkMode: boolean;\n previewMode: boolean;\n isDirty: boolean;\n uiTheme: UiTheme;\n}\n\nexport interface UseEditorOptions {\n content: TemplateContent;\n defaultFontFamily?: string;\n templateDefaults?: TemplateDefaults;\n lockedBlocks?: Ref<Map<string, unknown>>;\n}\n\nexport interface UseEditorReturn {\n state: DeepReadonly<EditorState>;\n content: Ref<TemplateContent>;\n selectedBlock: Ref<Block | null>;\n setContent: (content: TemplateContent, markDirty?: boolean) => void;\n selectBlock: (blockId: string | null) => void;\n setViewport: (viewport: ViewportSize) => void;\n setDarkMode: (darkMode: boolean) => void;\n setPreviewMode: (previewMode: boolean) => void;\n setUiTheme: (theme: UiTheme) => void;\n updateBlock: (blockId: string, updates: Partial<Block>) => void;\n updateSettings: (updates: Partial<TemplateSettings>) => void;\n addBlock: (\n block: Block,\n targetSectionId?: string,\n columnIndex?: number,\n index?: number,\n ) => void;\n removeBlock: (blockId: string) => void;\n moveBlock: (\n blockId: string,\n newIndex: number,\n targetSectionId?: string,\n columnIndex?: number,\n ) => void;\n isBlockLocked: (blockId: string) => boolean;\n markDirty: () => void;\n findBlockLocation: (blockId: string) => {\n targetSectionId?: string;\n columnIndex?: number;\n index: number;\n } | null;\n}\n\nexport function useEditor(options: UseEditorOptions): UseEditorReturn {\n const state = reactive<EditorState>({\n content:\n options.content ??\n createDefaultTemplateContent(\n options.defaultFontFamily,\n options.templateDefaults,\n ),\n selectedBlockId: null,\n viewport: \"desktop\",\n darkMode: false,\n previewMode: false,\n isDirty: false,\n uiTheme: \"auto\",\n });\n\n const content = computed({\n get: () => state.content,\n set: (value: TemplateContent) => {\n state.content = value;\n state.isDirty = true;\n },\n });\n\n const selectedBlock = computed(() => {\n if (!state.selectedBlockId) return null;\n return findBlockById(state.content.blocks, state.selectedBlockId);\n });\n\n function findBlockById(blocks: Block[], id: string): Block | null {\n for (const block of blocks) {\n if (block.id === id) return block;\n if (block.type === \"section\") {\n for (const column of block.children) {\n const found = findBlockById(column, id);\n if (found) return found;\n }\n }\n }\n return null;\n }\n\n function collectBlockIds(block: Block, ids: Set<string>): void {\n ids.add(block.id);\n if (block.type === \"section\") {\n for (const column of block.children) {\n for (const child of column) {\n collectBlockIds(child, ids);\n }\n }\n }\n }\n\n function findBlockParent(\n blocks: Block[],\n id: string,\n parent: {\n blocks: Block[];\n sectionId?: string;\n columnIndex?: number;\n } = { blocks },\n ): { blocks: Block[]; sectionId?: string; columnIndex?: number } | null {\n for (let i = 0; i < blocks.length; i++) {\n const block = blocks[i];\n if (block.id === id) return parent;\n if (block.type === \"section\") {\n for (let colIdx = 0; colIdx < block.children.length; colIdx++) {\n const result = findBlockParent(block.children[colIdx], id, {\n blocks: block.children[colIdx],\n sectionId: block.id,\n columnIndex: colIdx,\n });\n if (result) return result;\n }\n }\n }\n return null;\n }\n\n function isBlockLocked(blockId: string): boolean {\n return options.lockedBlocks?.value.has(blockId) ?? false;\n }\n\n function findBlockLocation(blockId: string): {\n targetSectionId?: string;\n columnIndex?: number;\n index: number;\n } | null {\n const parent = findBlockParent(state.content.blocks, blockId);\n if (!parent) return null;\n const index = parent.blocks.findIndex((b) => b.id === blockId);\n if (index === -1) return null;\n return {\n targetSectionId: parent.sectionId,\n columnIndex: parent.columnIndex,\n index,\n };\n }\n\n // TODO(collab): the lock checks in addBlock/moveBlock/removeBlock/updateBlock\n // are shallow — they only consider the directly-targeted block id. A section\n // can still be removed, moved, or have its `children` array rewritten while a\n // peer is editing one of its descendants, which silently disrupts that peer's\n // edit. Add a `hasLockedDescendant(blockId)` helper and gate section-level\n // operations on it (and on the parent of each affected child) so cascades\n // through the tree are also blocked.\n\n function setContent(newContent: TemplateContent, markDirty = true): void {\n state.content = newContent;\n if (markDirty) {\n state.isDirty = true;\n }\n }\n\n function selectBlock(blockId: string | null): void {\n if (blockId && isBlockLocked(blockId)) {\n return;\n }\n state.selectedBlockId = blockId;\n }\n\n function setViewport(viewport: ViewportSize): void {\n state.viewport = viewport;\n }\n\n function setDarkMode(darkMode: boolean): void {\n state.darkMode = darkMode;\n }\n\n function setUiTheme(theme: UiTheme): void {\n state.uiTheme = theme;\n }\n\n function setPreviewMode(previewMode: boolean): void {\n state.previewMode = previewMode;\n if (previewMode) {\n state.selectedBlockId = null;\n }\n }\n\n function updateBlock(blockId: string, updates: Partial<Block>): void {\n if (isBlockLocked(blockId)) {\n return;\n }\n const block = findBlockById(state.content.blocks, blockId);\n if (block) {\n Object.assign(block, updates);\n state.isDirty = true;\n }\n }\n\n function updateSettings(updates: Partial<TemplateSettings>): void {\n state.content.settings = { ...state.content.settings, ...updates };\n state.isDirty = true;\n }\n\n function addBlock(\n block: Block,\n targetSectionId?: string,\n columnIndex = 0,\n index?: number,\n ): void {\n if (targetSectionId) {\n if (isBlockLocked(targetSectionId)) {\n return;\n }\n const section = findBlockById(state.content.blocks, targetSectionId);\n if (section && section.type === \"section\") {\n if (columnIndex < 0 || columnIndex >= getColumnCount(section.columns)) {\n return;\n }\n section.children[columnIndex] = section.children[columnIndex] || [];\n const targetArray = section.children[columnIndex];\n if (index !== undefined && index < targetArray.length) {\n targetArray.splice(index, 0, block);\n } else {\n targetArray.push(block);\n }\n }\n } else {\n if (index !== undefined && index < state.content.blocks.length) {\n state.content.blocks.splice(index, 0, block);\n } else {\n state.content.blocks.push(block);\n }\n }\n state.isDirty = true;\n }\n\n function removeBlock(blockId: string): void {\n if (isBlockLocked(blockId)) {\n return;\n }\n const parent = findBlockParent(state.content.blocks, blockId);\n if (parent) {\n const index = parent.blocks.findIndex((b) => b.id === blockId);\n if (index !== -1) {\n const [removed] = parent.blocks.splice(index, 1);\n if (state.selectedBlockId) {\n const removedIds = new Set<string>();\n collectBlockIds(removed, removedIds);\n if (removedIds.has(state.selectedBlockId)) {\n state.selectedBlockId = null;\n }\n }\n state.isDirty = true;\n }\n }\n }\n\n function moveBlock(\n blockId: string,\n newIndex: number,\n targetSectionId?: string,\n columnIndex = 0,\n ): void {\n if (isBlockLocked(blockId)) {\n return;\n }\n if (targetSectionId && isBlockLocked(targetSectionId)) {\n return;\n }\n\n const parent = findBlockParent(state.content.blocks, blockId);\n if (!parent) return;\n\n const oldIndex = parent.blocks.findIndex((b) => b.id === blockId);\n if (oldIndex === -1) return;\n\n // Resolve target before mutating the source — otherwise an invalid\n // targetSectionId leaves the block spliced-out and unrecoverable.\n let targetArray: Block[];\n if (targetSectionId) {\n const section = findBlockById(state.content.blocks, targetSectionId);\n if (!section || section.type !== \"section\") return;\n if (columnIndex < 0 || columnIndex >= getColumnCount(section.columns)) {\n return;\n }\n section.children[columnIndex] = section.children[columnIndex] || [];\n targetArray = section.children[columnIndex];\n } else {\n targetArray = state.content.blocks;\n }\n\n const [block] = parent.blocks.splice(oldIndex, 1);\n targetArray.splice(newIndex, 0, block);\n\n state.isDirty = true;\n }\n\n function markDirty(): void {\n state.isDirty = true;\n }\n\n return {\n state: readonly(state),\n content,\n selectedBlock,\n isBlockLocked,\n setContent,\n selectBlock,\n setViewport,\n setDarkMode,\n setUiTheme,\n setPreviewMode,\n updateBlock,\n updateSettings,\n addBlock,\n removeBlock,\n moveBlock,\n markDirty,\n findBlockLocation,\n };\n}\n","import type { TemplateContent } from \"@templatical/types\";\nimport { computed, ref, type ComputedRef, type Ref } from \"@vue/reactivity\";\n\nexport interface UseHistoryOptions {\n content: Ref<TemplateContent>;\n setContent: (content: TemplateContent, markDirty?: boolean) => void;\n isRemoteOperation?: () => boolean;\n maxSize?: number;\n}\n\nexport interface UseHistoryReturn {\n canUndo: ComputedRef<boolean>;\n canRedo: ComputedRef<boolean>;\n isNavigating: Ref<boolean>;\n undo: () => void;\n redo: () => void;\n record: () => void;\n recordDebounced: (blockId: string) => void;\n clear: () => void;\n destroy: () => void;\n}\n\ninterface DebouncedSnapshot {\n blockId: string;\n timeoutId: ReturnType<typeof setTimeout>;\n}\n\nconst MAX_STACK_SIZE = 50;\nconst DEBOUNCE_MS = 300;\nconst NAVIGATE_IDLE_MS = 1500;\n\nexport function useHistory(options: UseHistoryOptions): UseHistoryReturn {\n const {\n content,\n setContent,\n isRemoteOperation,\n maxSize = MAX_STACK_SIZE,\n } = options;\n\n const undoStack = ref<TemplateContent[]>([]);\n const redoStack = ref<TemplateContent[]>([]);\n const isNavigating = ref(false);\n let navigatingTimeoutId: ReturnType<typeof setTimeout> | null = null;\n let pendingDebounce: DebouncedSnapshot | null = null;\n\n const canUndo = computed(() => undoStack.value.length > 0);\n const canRedo = computed(() => redoStack.value.length > 0);\n\n function cloneContent(): TemplateContent {\n // Cycle-safe deep clone. A naked JSON.stringify throws\n // `Converting circular structure to JSON` if anything in the tree\n // is self-referencing — e.g. a DOM element with a Sortable expando\n // back-ref ever sneaks into block data via a drag handler. We'd\n // rather drop the back-ref from the snapshot than freeze the undo\n // stack (the visible symptom: any subsequent mutation that calls\n // `record()` throws, breaking clone/move/etc.).\n const seen = new WeakSet<object>();\n return JSON.parse(\n JSON.stringify(content.value, (_key, value) => {\n if (typeof value === \"object\" && value !== null) {\n if (seen.has(value)) return undefined;\n seen.add(value);\n }\n return value;\n }),\n ) as TemplateContent;\n }\n\n function pushToUndoStack(snapshot: TemplateContent): void {\n undoStack.value.push(snapshot);\n if (undoStack.value.length > maxSize) {\n undoStack.value.splice(0, undoStack.value.length - maxSize);\n }\n }\n\n function flushPendingDebounce(): void {\n if (pendingDebounce) {\n clearTimeout(pendingDebounce.timeoutId);\n pendingDebounce = null;\n }\n }\n\n function record(): void {\n if (isRemoteOperation?.()) {\n return;\n }\n\n flushPendingDebounce();\n pushToUndoStack(cloneContent());\n redoStack.value = [];\n }\n\n function recordDebounced(blockId: string): void {\n if (isRemoteOperation?.()) {\n return;\n }\n\n if (pendingDebounce && pendingDebounce.blockId === blockId) {\n clearTimeout(pendingDebounce.timeoutId);\n pendingDebounce.timeoutId = setTimeout(() => {\n pendingDebounce = null;\n }, DEBOUNCE_MS);\n return;\n }\n\n flushPendingDebounce();\n\n pushToUndoStack(cloneContent());\n redoStack.value = [];\n\n pendingDebounce = {\n blockId,\n timeoutId: setTimeout(() => {\n pendingDebounce = null;\n }, DEBOUNCE_MS),\n };\n }\n\n function setNavigating(): void {\n isNavigating.value = true;\n if (navigatingTimeoutId) {\n clearTimeout(navigatingTimeoutId);\n }\n navigatingTimeoutId = setTimeout(() => {\n isNavigating.value = false;\n navigatingTimeoutId = null;\n }, NAVIGATE_IDLE_MS);\n }\n\n function undo(): void {\n if (undoStack.value.length === 0) {\n return;\n }\n\n flushPendingDebounce();\n\n const snapshot = undoStack.value.pop()!;\n redoStack.value.push(cloneContent());\n setContent(snapshot, true);\n setNavigating();\n }\n\n function redo(): void {\n if (redoStack.value.length === 0) {\n return;\n }\n\n flushPendingDebounce();\n\n const snapshot = redoStack.value.pop()!;\n undoStack.value.push(cloneContent());\n setContent(snapshot, true);\n setNavigating();\n }\n\n function clear(): void {\n undoStack.value = [];\n redoStack.value = [];\n flushPendingDebounce();\n }\n\n function destroy(): void {\n clear();\n if (navigatingTimeoutId) {\n clearTimeout(navigatingTimeoutId);\n navigatingTimeoutId = null;\n }\n }\n\n return {\n canUndo,\n canRedo,\n isNavigating,\n undo,\n redo,\n record,\n recordDebounced,\n clear,\n destroy,\n };\n}\n","import type { Block, BlockDefaults, BlockType } from \"@templatical/types\";\nimport { createBlock, generateId } from \"@templatical/types\";\n\nfunction regenerateNestedIds(block: Block): void {\n if (block.type === \"table\") {\n block.rows = block.rows.map((row) => ({\n ...row,\n id: generateId(),\n cells: row.cells.map((cell) => ({ ...cell, id: generateId() })),\n }));\n } else if (block.type === \"social\") {\n block.icons = block.icons.map((icon) => ({ ...icon, id: generateId() }));\n } else if (block.type === \"menu\") {\n block.items = block.items.map((item) => ({ ...item, id: generateId() }));\n }\n}\n\nexport interface UseBlockActionsOptions {\n addBlock: (\n block: Block,\n targetSectionId?: string,\n columnIndex?: number,\n index?: number,\n ) => void;\n removeBlock: (blockId: string) => void;\n updateBlock: (blockId: string, updates: Partial<Block>) => void;\n selectBlock: (blockId: string | null) => void;\n /** Locate a block in the tree — used by `duplicateBlock` to insert the\n * clone right after the source instead of appending to the end. */\n findBlockLocation?: (blockId: string) => {\n targetSectionId?: string;\n columnIndex?: number;\n index: number;\n } | null;\n blockDefaults?: BlockDefaults;\n}\n\nexport interface UseBlockActionsReturn {\n createAndAddBlock: (\n type: BlockType,\n targetSectionId?: string,\n columnIndex?: number,\n ) => Block;\n duplicateBlock: (\n block: Block,\n targetSectionId?: string,\n columnIndex?: number,\n ) => Block;\n deleteBlock: (blockId: string) => void;\n updateBlockProperty: <K extends keyof Block>(\n blockId: string,\n key: K,\n value: Block[K],\n ) => void;\n}\n\nexport function useBlockActions(\n options: UseBlockActionsOptions,\n): UseBlockActionsReturn {\n const { addBlock, removeBlock, updateBlock, selectBlock, findBlockLocation } =\n options;\n\n function createAndAddBlock(\n type: BlockType,\n targetSectionId?: string,\n columnIndex?: number,\n ): Block {\n const block = createBlock(type, options.blockDefaults);\n addBlock(block, targetSectionId, columnIndex);\n selectBlock(block.id);\n return block;\n }\n\n function duplicateBlock(\n block: Block,\n targetSectionId?: string,\n columnIndex?: number,\n ): Block {\n const cloned = JSON.parse(JSON.stringify(block)) as Block;\n cloned.id = generateId();\n regenerateNestedIds(cloned);\n\n if (cloned.type === \"section\") {\n cloned.children = cloned.children.map((column) =>\n column.map((child) => {\n const clonedChild = JSON.parse(JSON.stringify(child)) as Block;\n clonedChild.id = generateId();\n regenerateNestedIds(clonedChild);\n return clonedChild;\n }),\n );\n }\n\n // Insert directly after the source block. Explicit target args win;\n // otherwise, resolve the source's location and bump index by 1. Falls\n // back to appending at the end if location is unknown.\n if (targetSectionId !== undefined || columnIndex !== undefined) {\n addBlock(cloned, targetSectionId, columnIndex);\n } else {\n const sourceLocation = findBlockLocation?.(block.id) ?? null;\n if (sourceLocation) {\n addBlock(\n cloned,\n sourceLocation.targetSectionId,\n sourceLocation.columnIndex,\n sourceLocation.index + 1,\n );\n } else {\n addBlock(cloned, targetSectionId, columnIndex);\n }\n }\n selectBlock(cloned.id);\n return cloned;\n }\n\n function deleteBlock(blockId: string): void {\n removeBlock(blockId);\n }\n\n function updateBlockProperty<K extends keyof Block>(\n blockId: string,\n key: K,\n value: Block[K],\n ): void {\n updateBlock(blockId, { [key]: value } as Partial<Block>);\n }\n\n return {\n createAndAddBlock,\n duplicateBlock,\n deleteBlock,\n updateBlockProperty,\n };\n}\n","import type { TemplateContent } from \"@templatical/types\";\nimport { watch, type Ref } from \"@vue/reactivity\";\n\nexport interface UseAutoSaveOptions {\n content: Ref<TemplateContent>;\n isDirty: () => boolean;\n onChange: (content: TemplateContent) => void;\n debounce?: number;\n enabled?: boolean | (() => boolean);\n}\n\nexport interface UseAutoSaveReturn {\n flush: () => void;\n cancel: () => void;\n pause: () => void;\n resume: () => void;\n destroy: () => void;\n}\n\nexport function useAutoSave(options: UseAutoSaveOptions): UseAutoSaveReturn {\n const {\n content,\n isDirty,\n onChange,\n debounce = 1000,\n enabled = true,\n } = options;\n\n let timeoutId: ReturnType<typeof setTimeout> | null = null;\n let paused = false;\n\n function isEnabled(): boolean {\n return typeof enabled === \"function\" ? enabled() : enabled;\n }\n\n function pause(): void {\n paused = true;\n cancel();\n }\n\n function resume(): void {\n paused = false;\n }\n\n function cancel(): void {\n if (timeoutId) {\n clearTimeout(timeoutId);\n timeoutId = null;\n }\n }\n\n function flush(): void {\n cancel();\n if (isDirty()) {\n onChange(JSON.parse(JSON.stringify(content.value)));\n }\n }\n\n function scheduleOnChange(): void {\n if (!isEnabled() || paused) return;\n\n cancel();\n timeoutId = setTimeout(() => {\n timeoutId = null;\n if (isEnabled() && !paused && isDirty()) {\n onChange(JSON.parse(JSON.stringify(content.value)));\n }\n }, debounce);\n }\n\n const stopWatch = watch(\n content,\n () => {\n if (isEnabled() && !paused && isDirty()) {\n scheduleOnChange();\n }\n },\n { deep: true },\n );\n\n function destroy(): void {\n stopWatch();\n cancel();\n }\n\n return {\n flush,\n cancel,\n pause,\n resume,\n destroy,\n };\n}\n","import type { UseEditorReturn } from \"./editor\";\nimport { computed, reactive, type ComputedRef } from \"@vue/reactivity\";\n\nexport interface UseConditionPreviewReturn {\n isHidden: (blockId: string) => boolean;\n toggleBlock: (blockId: string) => void;\n reset: () => void;\n hasHiddenBlocks: ComputedRef<boolean>;\n}\n\nexport function useConditionPreview(\n editor: UseEditorReturn,\n): UseConditionPreviewReturn {\n const hiddenBlockIds = reactive(new Set<string>());\n\n const hasHiddenBlocks = computed(() => hiddenBlockIds.size > 0);\n\n function isHidden(blockId: string): boolean {\n return hiddenBlockIds.has(blockId);\n }\n\n function toggleBlock(blockId: string): void {\n if (hiddenBlockIds.has(blockId)) {\n hiddenBlockIds.delete(blockId);\n } else {\n hiddenBlockIds.add(blockId);\n\n if (editor.state.selectedBlockId === blockId) {\n editor.selectBlock(null);\n }\n }\n }\n\n function reset(): void {\n hiddenBlockIds.clear();\n }\n\n return {\n isHidden,\n toggleBlock,\n reset,\n hasHiddenBlocks,\n };\n}\n","import type { CustomBlock, CustomBlockDefinition } from \"@templatical/types\";\nimport type { ComputedRef, Ref } from \"@vue/reactivity\";\nimport { computed, ref } from \"@vue/reactivity\";\n\nexport function useDataSourceFetch(options: {\n definition: ComputedRef<CustomBlockDefinition | undefined>;\n block: ComputedRef<CustomBlock>;\n onUpdate: (fieldValues: Record<string, unknown>, fetched: boolean) => void;\n}): {\n isFetching: Ref<boolean>;\n fetchError: Ref<boolean>;\n fetch: () => Promise<void>;\n hasDataSource: ComputedRef<boolean>;\n needsFetch: ComputedRef<boolean>;\n} {\n const isFetching = ref(false);\n const fetchError = ref(false);\n\n const hasDataSource = computed(() => !!options.definition.value?.dataSource);\n\n const needsFetch = computed(\n () => hasDataSource.value && !options.block.value.dataSourceFetched,\n );\n\n async function fetch(): Promise<void> {\n const def = options.definition.value;\n if (!def?.dataSource) {\n return;\n }\n\n isFetching.value = true;\n fetchError.value = false;\n\n try {\n const result = await def.dataSource.onFetch({\n fieldValues: { ...options.block.value.fieldValues },\n blockId: options.block.value.id,\n });\n\n if (result == null) {\n return;\n }\n\n const merged = { ...options.block.value.fieldValues };\n for (const key of Object.keys(merged)) {\n if (key in result) {\n merged[key] = result[key];\n }\n }\n\n options.onUpdate(merged, true);\n } catch (error) {\n console.warn(\"[Templatical] Data source fetch error:\", error);\n fetchError.value = true;\n } finally {\n isFetching.value = false;\n }\n }\n\n return {\n isFetching,\n fetchError,\n fetch,\n hasDataSource,\n needsFetch,\n };\n}\n","import type { UseEditorReturn } from \"./editor\";\nimport type { UseHistoryReturn } from \"./history\";\n\n/**\n * Wraps editor mutation methods to record history snapshots before each\n * operation. Mutates the editor object in place.\n *\n * Must be applied **after** any collaboration broadcast wrapping so the\n * call chain is: history.record() → broadcast → original mutation.\n */\nexport function useHistoryInterceptor(\n editor: UseEditorReturn,\n history: UseHistoryReturn,\n): void {\n const originalAddBlock = editor.addBlock;\n const originalRemoveBlock = editor.removeBlock;\n const originalMoveBlock = editor.moveBlock;\n const originalUpdateBlock = editor.updateBlock;\n const originalUpdateSettings = editor.updateSettings;\n\n // Skip recording when the underlying op is a no-op (e.g., a peer-locked\n // block or section), otherwise the undo stack fills with snapshots that\n // are identical to current state and undo silently does nothing.\n editor.addBlock = (block, targetSectionId?, columnIndex?, index?) => {\n if (targetSectionId && editor.isBlockLocked(targetSectionId)) {\n return;\n }\n history.record();\n originalAddBlock(block, targetSectionId, columnIndex, index);\n };\n\n editor.removeBlock = (blockId) => {\n if (editor.isBlockLocked(blockId)) {\n return;\n }\n history.record();\n originalRemoveBlock(blockId);\n };\n\n editor.moveBlock = (blockId, newIndex, targetSectionId?, columnIndex?) => {\n if (editor.isBlockLocked(blockId)) {\n return;\n }\n if (targetSectionId && editor.isBlockLocked(targetSectionId)) {\n return;\n }\n history.record();\n originalMoveBlock(blockId, newIndex, targetSectionId, columnIndex);\n };\n\n editor.updateBlock = (blockId, updates) => {\n if (editor.isBlockLocked(blockId)) {\n return;\n }\n history.recordDebounced(blockId);\n originalUpdateBlock(blockId, updates);\n };\n\n editor.updateSettings = (updates) => {\n history.record();\n originalUpdateSettings(updates);\n };\n}\n"],"mappings":"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"} | ||
| {"version":3,"file":"index.js","names":[],"sources":["../src/editor.ts","../src/history.ts","../src/block-actions.ts","../src/auto-save.ts","../src/condition-preview.ts","../src/data-source-fetch.ts","../src/history-interceptor.ts"],"sourcesContent":["import type {\n Block,\n ColumnLayout,\n TemplateContent,\n TemplateDefaults,\n TemplateSettings,\n UiTheme,\n ViewportSize,\n} from \"@templatical/types\";\nimport { createDefaultTemplateContent } from \"@templatical/types\";\n\nfunction getColumnCount(layout: ColumnLayout): number {\n if (layout === \"1\") return 1;\n if (layout === \"3\") return 3;\n return 2;\n}\nimport {\n computed,\n reactive,\n readonly,\n type DeepReadonly,\n type Ref,\n} from \"@vue/reactivity\";\n\nexport interface EditorState {\n content: TemplateContent;\n selectedBlockId: string | null;\n viewport: ViewportSize;\n darkMode: boolean;\n previewMode: boolean;\n isDirty: boolean;\n uiTheme: UiTheme;\n}\n\nexport interface UseEditorOptions {\n content: TemplateContent;\n defaultFontFamily?: string;\n templateDefaults?: TemplateDefaults;\n lockedBlocks?: Ref<Map<string, unknown>>;\n}\n\nexport interface UseEditorReturn {\n state: DeepReadonly<EditorState>;\n content: Ref<TemplateContent>;\n selectedBlock: Ref<Block | null>;\n setContent: (content: TemplateContent, markDirty?: boolean) => void;\n selectBlock: (blockId: string | null) => void;\n setViewport: (viewport: ViewportSize) => void;\n setDarkMode: (darkMode: boolean) => void;\n setPreviewMode: (previewMode: boolean) => void;\n setUiTheme: (theme: UiTheme) => void;\n updateBlock: (blockId: string, updates: Partial<Block>) => void;\n updateSettings: (updates: Partial<TemplateSettings>) => void;\n addBlock: (\n block: Block,\n targetSectionId?: string,\n columnIndex?: number,\n index?: number,\n ) => void;\n removeBlock: (blockId: string) => void;\n moveBlock: (\n blockId: string,\n newIndex: number,\n targetSectionId?: string,\n columnIndex?: number,\n ) => void;\n isBlockLocked: (blockId: string) => boolean;\n markDirty: () => void;\n findBlockLocation: (blockId: string) => {\n targetSectionId?: string;\n columnIndex?: number;\n index: number;\n } | null;\n}\n\nexport function useEditor(options: UseEditorOptions): UseEditorReturn {\n const state = reactive<EditorState>({\n content:\n options.content ??\n createDefaultTemplateContent(\n options.defaultFontFamily,\n options.templateDefaults,\n ),\n selectedBlockId: null,\n viewport: \"desktop\",\n darkMode: false,\n previewMode: false,\n isDirty: false,\n uiTheme: \"auto\",\n });\n\n const content = computed({\n get: () => state.content,\n set: (value: TemplateContent) => {\n state.content = value;\n state.isDirty = true;\n },\n });\n\n const selectedBlock = computed(() => {\n if (!state.selectedBlockId) return null;\n return findBlockById(state.content.blocks, state.selectedBlockId);\n });\n\n function findBlockById(blocks: Block[], id: string): Block | null {\n for (const block of blocks) {\n if (block.id === id) return block;\n if (block.type === \"section\") {\n for (const column of block.children) {\n const found = findBlockById(column, id);\n if (found) return found;\n }\n }\n }\n return null;\n }\n\n function collectBlockIds(block: Block, ids: Set<string>): void {\n ids.add(block.id);\n if (block.type === \"section\") {\n for (const column of block.children) {\n for (const child of column) {\n collectBlockIds(child, ids);\n }\n }\n }\n }\n\n function findBlockParent(\n blocks: Block[],\n id: string,\n parent: {\n blocks: Block[];\n sectionId?: string;\n columnIndex?: number;\n } = { blocks },\n ): { blocks: Block[]; sectionId?: string; columnIndex?: number } | null {\n for (let i = 0; i < blocks.length; i++) {\n const block = blocks[i];\n if (block.id === id) return parent;\n if (block.type === \"section\") {\n for (let colIdx = 0; colIdx < block.children.length; colIdx++) {\n const result = findBlockParent(block.children[colIdx], id, {\n blocks: block.children[colIdx],\n sectionId: block.id,\n columnIndex: colIdx,\n });\n if (result) return result;\n }\n }\n }\n return null;\n }\n\n function isBlockLocked(blockId: string): boolean {\n return options.lockedBlocks?.value.has(blockId) ?? false;\n }\n\n function findBlockLocation(blockId: string): {\n targetSectionId?: string;\n columnIndex?: number;\n index: number;\n } | null {\n const parent = findBlockParent(state.content.blocks, blockId);\n if (!parent) return null;\n const index = parent.blocks.findIndex((b) => b.id === blockId);\n if (index === -1) return null;\n return {\n targetSectionId: parent.sectionId,\n columnIndex: parent.columnIndex,\n index,\n };\n }\n\n // TODO(collab): the lock checks in addBlock/moveBlock/removeBlock/updateBlock\n // are shallow — they only consider the directly-targeted block id. A section\n // can still be removed, moved, or have its `children` array rewritten while a\n // peer is editing one of its descendants, which silently disrupts that peer's\n // edit. Add a `hasLockedDescendant(blockId)` helper and gate section-level\n // operations on it (and on the parent of each affected child) so cascades\n // through the tree are also blocked.\n\n function setContent(newContent: TemplateContent, markDirty = true): void {\n state.content = newContent;\n if (markDirty) {\n state.isDirty = true;\n }\n }\n\n function selectBlock(blockId: string | null): void {\n if (blockId && isBlockLocked(blockId)) {\n return;\n }\n state.selectedBlockId = blockId;\n }\n\n function setViewport(viewport: ViewportSize): void {\n state.viewport = viewport;\n }\n\n function setDarkMode(darkMode: boolean): void {\n state.darkMode = darkMode;\n }\n\n function setUiTheme(theme: UiTheme): void {\n state.uiTheme = theme;\n }\n\n function setPreviewMode(previewMode: boolean): void {\n state.previewMode = previewMode;\n if (previewMode) {\n state.selectedBlockId = null;\n }\n }\n\n function updateBlock(blockId: string, updates: Partial<Block>): void {\n if (isBlockLocked(blockId)) {\n return;\n }\n const block = findBlockById(state.content.blocks, blockId);\n if (block) {\n Object.assign(block, updates);\n state.isDirty = true;\n }\n }\n\n function updateSettings(updates: Partial<TemplateSettings>): void {\n state.content.settings = { ...state.content.settings, ...updates };\n state.isDirty = true;\n }\n\n function addBlock(\n block: Block,\n targetSectionId?: string,\n columnIndex = 0,\n index?: number,\n ): void {\n if (targetSectionId) {\n if (isBlockLocked(targetSectionId)) {\n return;\n }\n const section = findBlockById(state.content.blocks, targetSectionId);\n if (section && section.type === \"section\") {\n if (columnIndex < 0 || columnIndex >= getColumnCount(section.columns)) {\n return;\n }\n section.children[columnIndex] = section.children[columnIndex] || [];\n const targetArray = section.children[columnIndex];\n if (index !== undefined && index < targetArray.length) {\n targetArray.splice(index, 0, block);\n } else {\n targetArray.push(block);\n }\n }\n } else {\n if (index !== undefined && index < state.content.blocks.length) {\n state.content.blocks.splice(index, 0, block);\n } else {\n state.content.blocks.push(block);\n }\n }\n state.isDirty = true;\n }\n\n function removeBlock(blockId: string): void {\n if (isBlockLocked(blockId)) {\n return;\n }\n const parent = findBlockParent(state.content.blocks, blockId);\n if (parent) {\n const index = parent.blocks.findIndex((b) => b.id === blockId);\n if (index !== -1) {\n const [removed] = parent.blocks.splice(index, 1);\n if (state.selectedBlockId) {\n const removedIds = new Set<string>();\n collectBlockIds(removed, removedIds);\n if (removedIds.has(state.selectedBlockId)) {\n state.selectedBlockId = null;\n }\n }\n state.isDirty = true;\n }\n }\n }\n\n function moveBlock(\n blockId: string,\n newIndex: number,\n targetSectionId?: string,\n columnIndex = 0,\n ): void {\n if (isBlockLocked(blockId)) {\n return;\n }\n if (targetSectionId && isBlockLocked(targetSectionId)) {\n return;\n }\n\n const parent = findBlockParent(state.content.blocks, blockId);\n if (!parent) return;\n\n const oldIndex = parent.blocks.findIndex((b) => b.id === blockId);\n if (oldIndex === -1) return;\n\n // Resolve target before mutating the source — otherwise an invalid\n // targetSectionId leaves the block spliced-out and unrecoverable.\n let targetArray: Block[];\n if (targetSectionId) {\n const section = findBlockById(state.content.blocks, targetSectionId);\n if (!section || section.type !== \"section\") return;\n if (columnIndex < 0 || columnIndex >= getColumnCount(section.columns)) {\n return;\n }\n section.children[columnIndex] = section.children[columnIndex] || [];\n targetArray = section.children[columnIndex];\n } else {\n targetArray = state.content.blocks;\n }\n\n const [block] = parent.blocks.splice(oldIndex, 1);\n targetArray.splice(newIndex, 0, block);\n\n state.isDirty = true;\n }\n\n function markDirty(): void {\n state.isDirty = true;\n }\n\n return {\n state: readonly(state),\n content,\n selectedBlock,\n isBlockLocked,\n setContent,\n selectBlock,\n setViewport,\n setDarkMode,\n setUiTheme,\n setPreviewMode,\n updateBlock,\n updateSettings,\n addBlock,\n removeBlock,\n moveBlock,\n markDirty,\n findBlockLocation,\n };\n}\n","import { safeClone, type TemplateContent } from \"@templatical/types\";\nimport { computed, ref, type ComputedRef, type Ref } from \"@vue/reactivity\";\n\nexport interface UseHistoryOptions {\n content: Ref<TemplateContent>;\n setContent: (content: TemplateContent, markDirty?: boolean) => void;\n isRemoteOperation?: () => boolean;\n maxSize?: number;\n}\n\nexport interface UseHistoryReturn {\n canUndo: ComputedRef<boolean>;\n canRedo: ComputedRef<boolean>;\n isNavigating: Ref<boolean>;\n undo: () => void;\n redo: () => void;\n record: () => void;\n recordDebounced: (blockId: string) => void;\n clear: () => void;\n destroy: () => void;\n}\n\ninterface DebouncedSnapshot {\n blockId: string;\n timeoutId: ReturnType<typeof setTimeout>;\n}\n\nconst MAX_STACK_SIZE = 50;\nconst DEBOUNCE_MS = 300;\nconst NAVIGATE_IDLE_MS = 1500;\n\nexport function useHistory(options: UseHistoryOptions): UseHistoryReturn {\n const {\n content,\n setContent,\n isRemoteOperation,\n maxSize = MAX_STACK_SIZE,\n } = options;\n\n const undoStack = ref<TemplateContent[]>([]);\n const redoStack = ref<TemplateContent[]>([]);\n const isNavigating = ref(false);\n let navigatingTimeoutId: ReturnType<typeof setTimeout> | null = null;\n let pendingDebounce: DebouncedSnapshot | null = null;\n\n const canUndo = computed(() => undoStack.value.length > 0);\n const canRedo = computed(() => redoStack.value.length > 0);\n\n function cloneContent(): TemplateContent {\n // Cycle-safe deep clone (see `safeClone`). A naked JSON.stringify throws\n // `Converting circular structure to JSON` if a DOM element with a Sortable\n // expando back-ref ever sneaks into block data via a drag handler — which\n // would freeze the undo stack, since every subsequent mutation calls\n // `record()`. safeClone drops the back-ref instead of throwing.\n return safeClone(content.value);\n }\n\n function pushToUndoStack(snapshot: TemplateContent): void {\n undoStack.value.push(snapshot);\n if (undoStack.value.length > maxSize) {\n undoStack.value.splice(0, undoStack.value.length - maxSize);\n }\n }\n\n function flushPendingDebounce(): void {\n if (pendingDebounce) {\n clearTimeout(pendingDebounce.timeoutId);\n pendingDebounce = null;\n }\n }\n\n function record(): void {\n if (isRemoteOperation?.()) {\n return;\n }\n\n flushPendingDebounce();\n pushToUndoStack(cloneContent());\n redoStack.value = [];\n }\n\n function recordDebounced(blockId: string): void {\n if (isRemoteOperation?.()) {\n return;\n }\n\n if (pendingDebounce && pendingDebounce.blockId === blockId) {\n clearTimeout(pendingDebounce.timeoutId);\n pendingDebounce.timeoutId = setTimeout(() => {\n pendingDebounce = null;\n }, DEBOUNCE_MS);\n return;\n }\n\n flushPendingDebounce();\n\n pushToUndoStack(cloneContent());\n redoStack.value = [];\n\n pendingDebounce = {\n blockId,\n timeoutId: setTimeout(() => {\n pendingDebounce = null;\n }, DEBOUNCE_MS),\n };\n }\n\n function setNavigating(): void {\n isNavigating.value = true;\n if (navigatingTimeoutId) {\n clearTimeout(navigatingTimeoutId);\n }\n navigatingTimeoutId = setTimeout(() => {\n isNavigating.value = false;\n navigatingTimeoutId = null;\n }, NAVIGATE_IDLE_MS);\n }\n\n function undo(): void {\n if (undoStack.value.length === 0) {\n return;\n }\n\n flushPendingDebounce();\n\n const snapshot = undoStack.value.pop()!;\n redoStack.value.push(cloneContent());\n setContent(snapshot, true);\n setNavigating();\n }\n\n function redo(): void {\n if (redoStack.value.length === 0) {\n return;\n }\n\n flushPendingDebounce();\n\n const snapshot = redoStack.value.pop()!;\n undoStack.value.push(cloneContent());\n setContent(snapshot, true);\n setNavigating();\n }\n\n function clear(): void {\n undoStack.value = [];\n redoStack.value = [];\n flushPendingDebounce();\n }\n\n function destroy(): void {\n clear();\n if (navigatingTimeoutId) {\n clearTimeout(navigatingTimeoutId);\n navigatingTimeoutId = null;\n }\n }\n\n return {\n canUndo,\n canRedo,\n isNavigating,\n undo,\n redo,\n record,\n recordDebounced,\n clear,\n destroy,\n };\n}\n","import type { Block, BlockDefaults, BlockType } from \"@templatical/types\";\nimport { createBlock, generateId } from \"@templatical/types\";\n\nfunction regenerateNestedIds(block: Block): void {\n if (block.type === \"table\") {\n block.rows = block.rows.map((row) => ({\n ...row,\n id: generateId(),\n cells: row.cells.map((cell) => ({ ...cell, id: generateId() })),\n }));\n } else if (block.type === \"social\") {\n block.icons = block.icons.map((icon) => ({ ...icon, id: generateId() }));\n } else if (block.type === \"menu\") {\n block.items = block.items.map((item) => ({ ...item, id: generateId() }));\n }\n}\n\nexport interface UseBlockActionsOptions {\n addBlock: (\n block: Block,\n targetSectionId?: string,\n columnIndex?: number,\n index?: number,\n ) => void;\n removeBlock: (blockId: string) => void;\n updateBlock: (blockId: string, updates: Partial<Block>) => void;\n selectBlock: (blockId: string | null) => void;\n /** Locate a block in the tree — used by `duplicateBlock` to insert the\n * clone right after the source instead of appending to the end. */\n findBlockLocation?: (blockId: string) => {\n targetSectionId?: string;\n columnIndex?: number;\n index: number;\n } | null;\n blockDefaults?: BlockDefaults;\n}\n\nexport interface UseBlockActionsReturn {\n createAndAddBlock: (\n type: BlockType,\n targetSectionId?: string,\n columnIndex?: number,\n ) => Block;\n duplicateBlock: (\n block: Block,\n targetSectionId?: string,\n columnIndex?: number,\n ) => Block;\n deleteBlock: (blockId: string) => void;\n updateBlockProperty: <K extends keyof Block>(\n blockId: string,\n key: K,\n value: Block[K],\n ) => void;\n}\n\nexport function useBlockActions(\n options: UseBlockActionsOptions,\n): UseBlockActionsReturn {\n const { addBlock, removeBlock, updateBlock, selectBlock, findBlockLocation } =\n options;\n\n function createAndAddBlock(\n type: BlockType,\n targetSectionId?: string,\n columnIndex?: number,\n ): Block {\n const block = createBlock(type, options.blockDefaults);\n addBlock(block, targetSectionId, columnIndex);\n selectBlock(block.id);\n return block;\n }\n\n function duplicateBlock(\n block: Block,\n targetSectionId?: string,\n columnIndex?: number,\n ): Block {\n const cloned = JSON.parse(JSON.stringify(block)) as Block;\n cloned.id = generateId();\n regenerateNestedIds(cloned);\n\n if (cloned.type === \"section\") {\n cloned.children = cloned.children.map((column) =>\n column.map((child) => {\n const clonedChild = JSON.parse(JSON.stringify(child)) as Block;\n clonedChild.id = generateId();\n regenerateNestedIds(clonedChild);\n return clonedChild;\n }),\n );\n }\n\n // Insert directly after the source block. Explicit target args win;\n // otherwise, resolve the source's location and bump index by 1. Falls\n // back to appending at the end if location is unknown.\n if (targetSectionId !== undefined || columnIndex !== undefined) {\n addBlock(cloned, targetSectionId, columnIndex);\n } else {\n const sourceLocation = findBlockLocation?.(block.id) ?? null;\n if (sourceLocation) {\n addBlock(\n cloned,\n sourceLocation.targetSectionId,\n sourceLocation.columnIndex,\n sourceLocation.index + 1,\n );\n } else {\n addBlock(cloned, targetSectionId, columnIndex);\n }\n }\n selectBlock(cloned.id);\n return cloned;\n }\n\n function deleteBlock(blockId: string): void {\n removeBlock(blockId);\n }\n\n function updateBlockProperty<K extends keyof Block>(\n blockId: string,\n key: K,\n value: Block[K],\n ): void {\n updateBlock(blockId, { [key]: value } as Partial<Block>);\n }\n\n return {\n createAndAddBlock,\n duplicateBlock,\n deleteBlock,\n updateBlockProperty,\n };\n}\n","import type { TemplateContent } from \"@templatical/types\";\nimport { watch, type Ref } from \"@vue/reactivity\";\n\nexport interface UseAutoSaveOptions {\n content: Ref<TemplateContent>;\n isDirty: () => boolean;\n onChange: (content: TemplateContent) => void;\n debounce?: number;\n enabled?: boolean | (() => boolean);\n}\n\nexport interface UseAutoSaveReturn {\n flush: () => void;\n cancel: () => void;\n pause: () => void;\n resume: () => void;\n destroy: () => void;\n}\n\nexport function useAutoSave(options: UseAutoSaveOptions): UseAutoSaveReturn {\n const {\n content,\n isDirty,\n onChange,\n debounce = 1000,\n enabled = true,\n } = options;\n\n let timeoutId: ReturnType<typeof setTimeout> | null = null;\n let paused = false;\n\n function isEnabled(): boolean {\n return typeof enabled === \"function\" ? enabled() : enabled;\n }\n\n function pause(): void {\n paused = true;\n cancel();\n }\n\n function resume(): void {\n paused = false;\n }\n\n function cancel(): void {\n if (timeoutId) {\n clearTimeout(timeoutId);\n timeoutId = null;\n }\n }\n\n function flush(): void {\n cancel();\n if (isDirty()) {\n onChange(JSON.parse(JSON.stringify(content.value)));\n }\n }\n\n function scheduleOnChange(): void {\n if (!isEnabled() || paused) return;\n\n cancel();\n timeoutId = setTimeout(() => {\n timeoutId = null;\n if (isEnabled() && !paused && isDirty()) {\n onChange(JSON.parse(JSON.stringify(content.value)));\n }\n }, debounce);\n }\n\n const stopWatch = watch(\n content,\n () => {\n if (isEnabled() && !paused && isDirty()) {\n scheduleOnChange();\n }\n },\n { deep: true },\n );\n\n function destroy(): void {\n stopWatch();\n cancel();\n }\n\n return {\n flush,\n cancel,\n pause,\n resume,\n destroy,\n };\n}\n","import type { UseEditorReturn } from \"./editor\";\nimport { computed, reactive, type ComputedRef } from \"@vue/reactivity\";\n\nexport interface UseConditionPreviewReturn {\n isHidden: (blockId: string) => boolean;\n toggleBlock: (blockId: string) => void;\n reset: () => void;\n hasHiddenBlocks: ComputedRef<boolean>;\n}\n\nexport function useConditionPreview(\n editor: UseEditorReturn,\n): UseConditionPreviewReturn {\n const hiddenBlockIds = reactive(new Set<string>());\n\n const hasHiddenBlocks = computed(() => hiddenBlockIds.size > 0);\n\n function isHidden(blockId: string): boolean {\n return hiddenBlockIds.has(blockId);\n }\n\n function toggleBlock(blockId: string): void {\n if (hiddenBlockIds.has(blockId)) {\n hiddenBlockIds.delete(blockId);\n } else {\n hiddenBlockIds.add(blockId);\n\n if (editor.state.selectedBlockId === blockId) {\n editor.selectBlock(null);\n }\n }\n }\n\n function reset(): void {\n hiddenBlockIds.clear();\n }\n\n return {\n isHidden,\n toggleBlock,\n reset,\n hasHiddenBlocks,\n };\n}\n","import type { CustomBlock, CustomBlockDefinition } from \"@templatical/types\";\nimport type { ComputedRef, Ref } from \"@vue/reactivity\";\nimport { computed, ref } from \"@vue/reactivity\";\n\nexport function useDataSourceFetch(options: {\n definition: ComputedRef<CustomBlockDefinition | undefined>;\n block: ComputedRef<CustomBlock>;\n onUpdate: (fieldValues: Record<string, unknown>, fetched: boolean) => void;\n}): {\n isFetching: Ref<boolean>;\n fetchError: Ref<boolean>;\n fetch: () => Promise<void>;\n hasDataSource: ComputedRef<boolean>;\n needsFetch: ComputedRef<boolean>;\n} {\n const isFetching = ref(false);\n const fetchError = ref(false);\n\n const hasDataSource = computed(() => !!options.definition.value?.dataSource);\n\n const needsFetch = computed(\n () => hasDataSource.value && !options.block.value.dataSourceFetched,\n );\n\n async function fetch(): Promise<void> {\n const def = options.definition.value;\n if (!def?.dataSource) {\n return;\n }\n\n isFetching.value = true;\n fetchError.value = false;\n\n try {\n const result = await def.dataSource.onFetch({\n fieldValues: { ...options.block.value.fieldValues },\n blockId: options.block.value.id,\n });\n\n if (result == null) {\n return;\n }\n\n const merged = { ...options.block.value.fieldValues };\n for (const key of Object.keys(merged)) {\n if (key in result) {\n merged[key] = result[key];\n }\n }\n\n options.onUpdate(merged, true);\n } catch (error) {\n console.warn(\"[Templatical] Data source fetch error:\", error);\n fetchError.value = true;\n } finally {\n isFetching.value = false;\n }\n }\n\n return {\n isFetching,\n fetchError,\n fetch,\n hasDataSource,\n needsFetch,\n };\n}\n","import type { UseEditorReturn } from \"./editor\";\nimport type { UseHistoryReturn } from \"./history\";\n\n/**\n * Wraps editor mutation methods to record history snapshots before each\n * operation. Mutates the editor object in place.\n *\n * Must be applied **after** any collaboration broadcast wrapping so the\n * call chain is: history.record() → broadcast → original mutation.\n */\nexport function useHistoryInterceptor(\n editor: UseEditorReturn,\n history: UseHistoryReturn,\n): void {\n const originalAddBlock = editor.addBlock;\n const originalRemoveBlock = editor.removeBlock;\n const originalMoveBlock = editor.moveBlock;\n const originalUpdateBlock = editor.updateBlock;\n const originalUpdateSettings = editor.updateSettings;\n\n // Skip recording when the underlying op is a no-op (e.g., a peer-locked\n // block or section), otherwise the undo stack fills with snapshots that\n // are identical to current state and undo silently does nothing.\n editor.addBlock = (block, targetSectionId?, columnIndex?, index?) => {\n if (targetSectionId && editor.isBlockLocked(targetSectionId)) {\n return;\n }\n history.record();\n originalAddBlock(block, targetSectionId, columnIndex, index);\n };\n\n editor.removeBlock = (blockId) => {\n if (editor.isBlockLocked(blockId)) {\n return;\n }\n history.record();\n originalRemoveBlock(blockId);\n };\n\n editor.moveBlock = (blockId, newIndex, targetSectionId?, columnIndex?) => {\n if (editor.isBlockLocked(blockId)) {\n return;\n }\n if (targetSectionId && editor.isBlockLocked(targetSectionId)) {\n return;\n }\n history.record();\n originalMoveBlock(blockId, newIndex, targetSectionId, columnIndex);\n };\n\n editor.updateBlock = (blockId, updates) => {\n if (editor.isBlockLocked(blockId)) {\n return;\n }\n history.recordDebounced(blockId);\n originalUpdateBlock(blockId, updates);\n };\n\n editor.updateSettings = (updates) => {\n history.record();\n originalUpdateSettings(updates);\n };\n}\n"],"mappings":"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"} |
+3
-3
| { | ||
| "name": "@templatical/core", | ||
| "description": "Framework-agnostic editor logic for Templatical email editor", | ||
| "version": "0.10.1", | ||
| "version": "0.10.2", | ||
| "bugs": "https://github.com/templatical/sdk/issues", | ||
| "dependencies": { | ||
| "@vue/reactivity": "^3.5.35", | ||
| "@templatical/types": "0.10.1" | ||
| "@templatical/types": "0.10.2" | ||
| }, | ||
@@ -13,3 +13,3 @@ "devDependencies": { | ||
| "typescript": "^6.0.3", | ||
| "vitest": "^4.1.7", | ||
| "vitest": "^4.1.8", | ||
| "vue": "^3.5.35" | ||
@@ -16,0 +16,0 @@ }, |
Sorry, the diff of this file is too big to display
300891
-0.21%3389
-0.21%+ Added
- Removed
Updated