@open-press/cli 1.0.0 → 1.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (175) hide show
  1. package/README.md +11 -12
  2. package/dist/cli.js +298 -79
  3. package/package.json +9 -7
  4. package/template/core/AGENTS.md +0 -130
  5. package/template/core/CHANGELOG.md +0 -218
  6. package/template/core/README.md +0 -43
  7. package/template/core/engine/cli.mjs +0 -96
  8. package/template/core/engine/commands/_shared.mjs +0 -199
  9. package/template/core/engine/commands/deploy.mjs +0 -31
  10. package/template/core/engine/commands/dev.mjs +0 -49
  11. package/template/core/engine/commands/doctor.mjs +0 -229
  12. package/template/core/engine/commands/export.mjs +0 -8
  13. package/template/core/engine/commands/image.mjs +0 -29
  14. package/template/core/engine/commands/inspect.mjs +0 -35
  15. package/template/core/engine/commands/pdf.mjs +0 -26
  16. package/template/core/engine/commands/preview.mjs +0 -26
  17. package/template/core/engine/commands/render.mjs +0 -17
  18. package/template/core/engine/commands/replace.mjs +0 -41
  19. package/template/core/engine/commands/search.mjs +0 -33
  20. package/template/core/engine/commands/skills-sync.mjs +0 -71
  21. package/template/core/engine/commands/typecheck.mjs +0 -67
  22. package/template/core/engine/commands/upgrade.mjs +0 -159
  23. package/template/core/engine/commands/validate.mjs +0 -17
  24. package/template/core/engine/document-export.mjs +0 -15
  25. package/template/core/engine/output/chrome-pdf.d.mts +0 -34
  26. package/template/core/engine/output/chrome-pdf.mjs +0 -450
  27. package/template/core/engine/output/deploy-sync.mjs +0 -15
  28. package/template/core/engine/output/fonts.mjs +0 -62
  29. package/template/core/engine/output/katex-assets.mjs +0 -45
  30. package/template/core/engine/output/page-block.mjs +0 -30
  31. package/template/core/engine/output/pdf-media.mjs +0 -45
  32. package/template/core/engine/output/public-assets.mjs +0 -19
  33. package/template/core/engine/output/static-server.mjs +0 -571
  34. package/template/core/engine/react/caption-numbering.mjs +0 -73
  35. package/template/core/engine/react/comment-endpoint.d.mts +0 -11
  36. package/template/core/engine/react/comment-endpoint.mjs +0 -102
  37. package/template/core/engine/react/comment-marker.mjs +0 -374
  38. package/template/core/engine/react/document-entry.mjs +0 -331
  39. package/template/core/engine/react/document-export.mjs +0 -512
  40. package/template/core/engine/react/http-json.mjs +0 -24
  41. package/template/core/engine/react/mdx-compile.mjs +0 -629
  42. package/template/core/engine/react/measurement-css.mjs +0 -157
  43. package/template/core/engine/react/object-entities.mjs +0 -204
  44. package/template/core/engine/react/pagination/allocator.mjs +0 -167
  45. package/template/core/engine/react/pagination/regions.mjs +0 -81
  46. package/template/core/engine/react/pagination-constants.mjs +0 -3
  47. package/template/core/engine/react/pagination.mjs +0 -9
  48. package/template/core/engine/react/pipeline/allocate.mjs +0 -217
  49. package/template/core/engine/react/pipeline/final-render.mjs +0 -94
  50. package/template/core/engine/react/pipeline/frame-measurement.mjs +0 -306
  51. package/template/core/engine/react/pipeline/press-tree.mjs +0 -135
  52. package/template/core/engine/react/press-tree-inspection.mjs +0 -172
  53. package/template/core/engine/react/project-asset-endpoint.d.mts +0 -10
  54. package/template/core/engine/react/project-asset-endpoint.mjs +0 -361
  55. package/template/core/engine/react/section-css.mjs +0 -56
  56. package/template/core/engine/react/source-edit-endpoint.d.mts +0 -10
  57. package/template/core/engine/react/source-edit-endpoint.mjs +0 -75
  58. package/template/core/engine/react/sources/heading-numbering.mjs +0 -132
  59. package/template/core/engine/react/sources/mdx-resolver.mjs +0 -439
  60. package/template/core/engine/react/style-discovery.mjs +0 -160
  61. package/template/core/engine/runtime/config.d.mts +0 -48
  62. package/template/core/engine/runtime/config.mjs +0 -172
  63. package/template/core/engine/runtime/file-utils.mjs +0 -114
  64. package/template/core/engine/runtime/file-walk.mjs +0 -22
  65. package/template/core/engine/runtime/inspection.mjs +0 -328
  66. package/template/core/engine/runtime/issue-report.mjs +0 -44
  67. package/template/core/engine/runtime/page-geometry.mjs +0 -131
  68. package/template/core/engine/runtime/path-utils.mjs +0 -20
  69. package/template/core/engine/runtime/source-text-tools.d.mts +0 -102
  70. package/template/core/engine/runtime/source-text-tools.mjs +0 -832
  71. package/template/core/engine/runtime/source-workspace.mjs +0 -168
  72. package/template/core/engine/runtime/validation.mjs +0 -183
  73. package/template/core/index.html +0 -13
  74. package/template/core/openpress.config.mjs +0 -8
  75. package/template/core/package.json +0 -89
  76. package/template/core/src/main.tsx +0 -16
  77. package/template/core/src/openpress/app/OpenPressApp.tsx +0 -296
  78. package/template/core/src/openpress/app/OpenPressRuntime.tsx +0 -102
  79. package/template/core/src/openpress/app/WorkspaceGalleryPage.tsx +0 -219
  80. package/template/core/src/openpress/app/index.ts +0 -2
  81. package/template/core/src/openpress/core/Frame.tsx +0 -91
  82. package/template/core/src/openpress/core/FrameContext.tsx +0 -26
  83. package/template/core/src/openpress/core/MdxArea.tsx +0 -34
  84. package/template/core/src/openpress/core/Press.tsx +0 -55
  85. package/template/core/src/openpress/core/Workspace.tsx +0 -36
  86. package/template/core/src/openpress/core/cn.ts +0 -4
  87. package/template/core/src/openpress/core/index.tsx +0 -47
  88. package/template/core/src/openpress/core/primitives.tsx +0 -91
  89. package/template/core/src/openpress/core/types.ts +0 -236
  90. package/template/core/src/openpress/core/useSource.ts +0 -28
  91. package/template/core/src/openpress/document-model/anchorMapModel.ts +0 -27
  92. package/template/core/src/openpress/document-model/documentIndexes.ts +0 -329
  93. package/template/core/src/openpress/document-model/documentTypes.ts +0 -147
  94. package/template/core/src/openpress/document-model/index.ts +0 -7
  95. package/template/core/src/openpress/document-model/objectEntityModel.ts +0 -55
  96. package/template/core/src/openpress/document-model/projectIdentityModel.ts +0 -15
  97. package/template/core/src/openpress/document-model/reactDocumentMetadataModel.ts +0 -27
  98. package/template/core/src/openpress/document-model/workspaceManifestModel.ts +0 -57
  99. package/template/core/src/openpress/manuscript/index.tsx +0 -238
  100. package/template/core/src/openpress/mdx/index.ts +0 -96
  101. package/template/core/src/openpress/numbering/index.ts +0 -294
  102. package/template/core/src/openpress/reader/PageThumbnailsPanel.tsx +0 -168
  103. package/template/core/src/openpress/reader/PublicReaderPage.tsx +0 -267
  104. package/template/core/src/openpress/reader/ReaderNavigationPanel.tsx +0 -123
  105. package/template/core/src/openpress/reader/index.ts +0 -11
  106. package/template/core/src/openpress/reader/pageViewportScaleModel.ts +0 -73
  107. package/template/core/src/openpress/reader/readerPageRegistry.ts +0 -41
  108. package/template/core/src/openpress/reader/readerPageRoute.ts +0 -21
  109. package/template/core/src/openpress/reader/readerScroll.ts +0 -92
  110. package/template/core/src/openpress/reader/readerStateModel.ts +0 -15
  111. package/template/core/src/openpress/reader/readerTypes.ts +0 -4
  112. package/template/core/src/openpress/reader/usePageViewportScale.ts +0 -119
  113. package/template/core/src/openpress/reader/usePanelState.ts +0 -56
  114. package/template/core/src/openpress/reader/useReaderHashSync.ts +0 -61
  115. package/template/core/src/openpress/reader/useReaderKeyboardNav.ts +0 -48
  116. package/template/core/src/openpress/reader/useReaderRuntime.ts +0 -146
  117. package/template/core/src/openpress/reader/useReaderScrollAnchor.ts +0 -64
  118. package/template/core/src/openpress/shared/Panel.tsx +0 -77
  119. package/template/core/src/openpress/shared/frameScheduler.ts +0 -32
  120. package/template/core/src/openpress/shared/index.ts +0 -4
  121. package/template/core/src/openpress/shared/numberUtils.ts +0 -3
  122. package/template/core/src/openpress/shared/runtimeMode.ts +0 -11
  123. package/template/core/src/openpress/workbench/Workbench.tsx +0 -506
  124. package/template/core/src/openpress/workbench/actions/DeploymentControl.tsx +0 -157
  125. package/template/core/src/openpress/workbench/actions/ExportImageControl.tsx +0 -96
  126. package/template/core/src/openpress/workbench/actions/PageZoomControl.tsx +0 -182
  127. package/template/core/src/openpress/workbench/actions/SearchControl.tsx +0 -345
  128. package/template/core/src/openpress/workbench/actions/deploymentStatusModel.ts +0 -112
  129. package/template/core/src/openpress/workbench/actions/index.ts +0 -6
  130. package/template/core/src/openpress/workbench/actions/useDeploymentWorkbench.ts +0 -136
  131. package/template/core/src/openpress/workbench/dialog/WorkbenchDialog.tsx +0 -72
  132. package/template/core/src/openpress/workbench/dialog/index.ts +0 -1
  133. package/template/core/src/openpress/workbench/document/components/DocumentPanel.tsx +0 -127
  134. package/template/core/src/openpress/workbench/document/components/InlineSourceEditorLayer.tsx +0 -207
  135. package/template/core/src/openpress/workbench/document/components/ReaderStage.tsx +0 -9
  136. package/template/core/src/openpress/workbench/document/hooks/useDocumentWorkbenchModel.ts +0 -34
  137. package/template/core/src/openpress/workbench/document/hooks/useInlineDocumentEditor.ts +0 -525
  138. package/template/core/src/openpress/workbench/document/index.ts +0 -10
  139. package/template/core/src/openpress/workbench/index.ts +0 -2
  140. package/template/core/src/openpress/workbench/inspector/InlineInspectorLayer.tsx +0 -459
  141. package/template/core/src/openpress/workbench/inspector/index.ts +0 -5
  142. package/template/core/src/openpress/workbench/inspector/inlineCommentModel.ts +0 -125
  143. package/template/core/src/openpress/workbench/inspector/inspectorGeometryModel.ts +0 -160
  144. package/template/core/src/openpress/workbench/inspector/inspectorModel.ts +0 -408
  145. package/template/core/src/openpress/workbench/inspector/useInspectorComments.ts +0 -254
  146. package/template/core/src/openpress/workbench/mentions/MentionSuggestionList.tsx +0 -41
  147. package/template/core/src/openpress/workbench/mentions/index.ts +0 -2
  148. package/template/core/src/openpress/workbench/mentions/useComposerMentions.ts +0 -185
  149. package/template/core/src/openpress/workbench/panels/Panel.tsx +0 -1
  150. package/template/core/src/openpress/workbench/panels/PendingCommentsPanel.tsx +0 -80
  151. package/template/core/src/openpress/workbench/panels/WorkbenchControlPanel.tsx +0 -29
  152. package/template/core/src/openpress/workbench/panels/index.ts +0 -3
  153. package/template/core/src/openpress/workbench/project/ProjectEntryPanel.tsx +0 -525
  154. package/template/core/src/openpress/workbench/project/ProjectPreviewDialog.tsx +0 -35
  155. package/template/core/src/openpress/workbench/project/index.ts +0 -2
  156. package/template/core/src/openpress/workbench/project/projectPreviewTypes.ts +0 -11
  157. package/template/core/src/openpress/workbench/project/projectSourceModel.ts +0 -24
  158. package/template/core/src/openpress/workbench/shell/WorkbenchShell.tsx +0 -167
  159. package/template/core/src/openpress/workbench/shell/index.ts +0 -1
  160. package/template/core/src/openpress/workbench/workbenchFormatters.ts +0 -120
  161. package/template/core/src/openpress/workbench/workbenchTypes.ts +0 -35
  162. package/template/core/src/styles/openpress/app-shell.css +0 -251
  163. package/template/core/src/styles/openpress/media-workspace.css +0 -230
  164. package/template/core/src/styles/openpress/print-route.css +0 -184
  165. package/template/core/src/styles/openpress/project-preview-panel.css +0 -924
  166. package/template/core/src/styles/openpress/public-viewer.css +0 -688
  167. package/template/core/src/styles/openpress/reader-runtime.css +0 -989
  168. package/template/core/src/styles/openpress/responsive.css +0 -245
  169. package/template/core/src/styles/openpress/workbench-panels.css +0 -707
  170. package/template/core/src/styles/openpress/workbench.css +0 -1255
  171. package/template/core/src/styles/openpress/workspace-gallery.css +0 -300
  172. package/template/core/src/styles/openpress.css +0 -15
  173. package/template/core/src/vite-env.d.ts +0 -9
  174. package/template/core/tsconfig.json +0 -40
  175. package/template/core/vite.config.ts +0 -584
@@ -1,525 +0,0 @@
1
- import { useLayoutEffect, type RefObject } from "react";
2
- import type { SourceBlock } from "../../../document-model";
3
-
4
- export type InlineDocumentEditState = "idle" | "editing" | "saving" | "saved" | "failed";
5
-
6
- export type InlineDocumentEditStatus = {
7
- state: InlineDocumentEditState;
8
- blockId?: string;
9
- message?: string;
10
- };
11
-
12
- export type InlineDocumentEditorOptions = {
13
- enabled: boolean;
14
- sourceContainerRef: RefObject<HTMLElement | null>;
15
- sourceBlockMap: Record<string, SourceBlock>;
16
- fetchImpl?: typeof fetch;
17
- onStatusChange?: (status: InlineDocumentEditStatus) => void;
18
- onOpenSourceBlock?: (target: InlineDocumentSourceTarget) => void;
19
- onDocumentEdited?: () => void | Promise<void>;
20
- };
21
-
22
- export type InlineDocumentSourceTarget = {
23
- block: SourceBlock;
24
- element: HTMLElement;
25
- rect: DOMRect;
26
- };
27
-
28
- type DocumentWithCaretFromPoint = Document & {
29
- caretPositionFromPoint?: (x: number, y: number) => { offsetNode: Node; offset: number } | null;
30
- caretRangeFromPoint?: (x: number, y: number) => Range | null;
31
- };
32
-
33
- const EDITABLE_SELECTOR = "[data-openpress-editable-block='true']";
34
- const SOURCE_SELECTOR = "[data-openpress-source-editable-block='true']";
35
- const SAVED_EDIT_STATE_RESET_DELAY_MS = 900;
36
- const UNSAFE_EDITABLE_CHILDREN = [
37
- "a",
38
- "button",
39
- "canvas",
40
- "figure",
41
- "form",
42
- "img",
43
- "input",
44
- "ol",
45
- "picture",
46
- "select",
47
- "svg",
48
- "table",
49
- "textarea",
50
- "ul",
51
- "video",
52
- ].join(",");
53
- const EDITABLE_COMPONENT_CAPTION_NAMES = new Set(["MediaFigure", "ImageFigure"]);
54
-
55
- export function useInlineDocumentEditor({
56
- enabled,
57
- sourceContainerRef,
58
- sourceBlockMap,
59
- fetchImpl,
60
- onStatusChange,
61
- onOpenSourceBlock,
62
- onDocumentEdited,
63
- }: InlineDocumentEditorOptions) {
64
- useLayoutEffect(() => {
65
- const root = sourceContainerRef.current;
66
- if (!root) return undefined;
67
- const ownerDocument = root.ownerDocument;
68
-
69
- if (!enabled) {
70
- onStatusChange?.({ state: "idle" });
71
- return undefined;
72
- }
73
- const markedElements = new Set<HTMLElement>();
74
- markEditableElements(root, sourceBlockMap, markedElements);
75
- const mutationObserver = typeof MutationObserver === "undefined"
76
- ? null
77
- : new MutationObserver(() => markEditableElements(root, sourceBlockMap, markedElements));
78
- mutationObserver?.observe(root, { childList: true, subtree: true });
79
- let activeEditableElement: HTMLElement | null = null;
80
-
81
- const finishElementEdit = (element: HTMLElement) => {
82
- if (element.dataset.openpressEditing !== "true") return;
83
- if (activeEditableElement === element) activeEditableElement = null;
84
- void persistElementEdit(
85
- element,
86
- sourceBlockMap,
87
- fetchImpl ?? globalThis.fetch?.bind(globalThis),
88
- onStatusChange,
89
- onDocumentEdited,
90
- );
91
- };
92
-
93
- const focusEditableElement = (element: HTMLElement, event?: MouseEvent) => {
94
- beginElementEdit(element);
95
- activeEditableElement = element;
96
- element.focus({ preventScroll: true });
97
- if (event) placeCaretFromMouseEvent(element, event);
98
- };
99
-
100
- const handleFocusIn = (event: FocusEvent) => {
101
- const element = editableElementFromEvent(event, root);
102
- if (!element) return;
103
- focusEditableElement(element);
104
- };
105
-
106
- const handleKeyDown = (event: KeyboardEvent) => {
107
- const element = editableElementFromEvent(event, root);
108
- if (!element) return;
109
- event.stopPropagation();
110
- if (event.key === "Escape") {
111
- event.preventDefault();
112
- element.dataset.openpressEditCanceled = "true";
113
- element.textContent = element.dataset.openpressOriginalText ?? "";
114
- finishElementEdit(element);
115
- element.blur();
116
- return;
117
- }
118
- if (event.key === "Enter") {
119
- if (element.dataset.openpressPreserveLineBreaks === "true") return;
120
- event.preventDefault();
121
- finishElementEdit(element);
122
- element.blur();
123
- }
124
- };
125
-
126
- const handleFocusOut = (event: FocusEvent) => {
127
- const element = editableElementFromEvent(event, root);
128
- if (!element) return;
129
- finishElementEdit(element);
130
- };
131
-
132
- const handleEditablePointerDown = (event: MouseEvent) => {
133
- const element = editableElementFromEvent(event, root);
134
- const target = eventTargetElement(event);
135
- if (activeEditableElement && (!element || element !== activeEditableElement) && target && !activeEditableElement.contains(target)) {
136
- finishElementEdit(activeEditableElement);
137
- }
138
- if (!element) return;
139
- focusEditableElement(element, event);
140
- };
141
-
142
- const handleClick = (event: MouseEvent) => {
143
- const editableElement = editableElementFromEvent(event, root);
144
- if (editableElement) {
145
- focusEditableElement(editableElement, event);
146
- return;
147
- }
148
- const element = sourceElementFromEvent(event, root);
149
- if (!element) return;
150
- const block = blockFromElement(element, sourceBlockMap);
151
- if (!block) return;
152
- event.preventDefault();
153
- event.stopPropagation();
154
- onOpenSourceBlock?.({ block, element, rect: element.getBoundingClientRect() });
155
- };
156
-
157
- const handleSourceKeyDown = (event: KeyboardEvent) => {
158
- if (event.key !== "Enter" && event.key !== " ") return;
159
- const element = sourceElementFromEvent(event, root);
160
- if (!element) return;
161
- const block = blockFromElement(element, sourceBlockMap);
162
- if (!block) return;
163
- event.preventDefault();
164
- event.stopPropagation();
165
- onOpenSourceBlock?.({ block, element, rect: element.getBoundingClientRect() });
166
- };
167
-
168
- ownerDocument.addEventListener("focusin", handleFocusIn, true);
169
- ownerDocument.addEventListener("mousedown", handleEditablePointerDown, true);
170
- ownerDocument.addEventListener("keydown", handleKeyDown, true);
171
- ownerDocument.addEventListener("focusout", handleFocusOut, true);
172
- root.addEventListener("keydown", handleSourceKeyDown);
173
- root.addEventListener("click", handleClick);
174
-
175
- return () => {
176
- ownerDocument.removeEventListener("focusin", handleFocusIn, true);
177
- ownerDocument.removeEventListener("mousedown", handleEditablePointerDown, true);
178
- ownerDocument.removeEventListener("keydown", handleKeyDown, true);
179
- ownerDocument.removeEventListener("focusout", handleFocusOut, true);
180
- root.removeEventListener("keydown", handleSourceKeyDown);
181
- root.removeEventListener("click", handleClick);
182
- mutationObserver?.disconnect();
183
- for (const element of markedElements) clearEditableElement(element);
184
- };
185
- }, [enabled, fetchImpl, onDocumentEdited, onOpenSourceBlock, onStatusChange, sourceBlockMap, sourceContainerRef]);
186
- }
187
-
188
- function beginElementEdit(element: HTMLElement) {
189
- if (element.dataset.openpressEditing === "true") return;
190
- clearElementEditState(element);
191
- element.dataset.openpressOriginalText = element.dataset.openpressOriginalText ?? readableElementText(element);
192
- element.dataset.openpressEditing = "true";
193
- }
194
-
195
- function placeCaretFromMouseEvent(element: HTMLElement, event: MouseEvent) {
196
- const selection = element.ownerDocument.getSelection?.();
197
- if (!selection) return;
198
-
199
- const range = createCaretRangeFromPoint(element.ownerDocument, event.clientX, event.clientY);
200
- if (range && element.contains(range.startContainer)) {
201
- selection.removeAllRanges();
202
- selection.addRange(range);
203
- return;
204
- }
205
-
206
- placeCaretAtEnd(element, selection);
207
- }
208
-
209
- function createCaretRangeFromPoint(document: Document, x: number, y: number) {
210
- const documentWithCaret = document as DocumentWithCaretFromPoint;
211
- const rangeFromPoint = documentWithCaret.caretRangeFromPoint?.(x, y);
212
- if (rangeFromPoint) return rangeFromPoint;
213
-
214
- const caretPosition = documentWithCaret.caretPositionFromPoint?.(x, y);
215
- if (!caretPosition) return null;
216
-
217
- const range = document.createRange();
218
- range.setStart(caretPosition.offsetNode, caretPosition.offset);
219
- range.collapse(true);
220
- return range;
221
- }
222
-
223
- function placeCaretAtEnd(element: HTMLElement, selection: Selection) {
224
- const range = element.ownerDocument.createRange();
225
- range.selectNodeContents(element);
226
- range.collapse(false);
227
- selection.removeAllRanges();
228
- selection.addRange(range);
229
- }
230
-
231
- function markEditableElements(
232
- root: HTMLElement,
233
- sourceBlockMap: Record<string, SourceBlock>,
234
- markedElements: Set<HTMLElement>,
235
- ) {
236
- root.querySelectorAll<HTMLElement>("[data-openpress-block-id]").forEach((element) => {
237
- const sourceBlock = blockFromElement(element, sourceBlockMap);
238
- if (sourceBlock?.kind === "table-row") {
239
- markEditableTableCells(element, sourceBlock, markedElements);
240
- return;
241
- }
242
-
243
- if (sourceBlock && markEditableComponentCaption(element, sourceBlock, markedElements)) {
244
- return;
245
- }
246
-
247
- if (isEditableTextBlockElement(element, sourceBlockMap)) {
248
- markEditableTextElement(element, markedElements, {
249
- label: sourceBlock?.name === "pre" ? "編輯程式碼文字" : "編輯文字",
250
- preserveLineBreaks: sourceBlock?.name === "pre",
251
- });
252
- return;
253
- }
254
-
255
- if (!isSourceEditableBlockElement(element, sourceBlockMap)) return;
256
- element.setAttribute("tabindex", "0");
257
- element.setAttribute("role", "button");
258
- element.setAttribute("aria-label", "編輯 source");
259
- element.dataset.openpressSourceEditableBlock = "true";
260
- markedElements.add(element);
261
- });
262
- }
263
-
264
- function markEditableComponentCaption(
265
- componentElement: HTMLElement,
266
- sourceBlock: SourceBlock,
267
- markedElements: Set<HTMLElement>,
268
- ) {
269
- if (sourceBlock.kind !== "component") return false;
270
- if (!EDITABLE_COMPONENT_CAPTION_NAMES.has(String(sourceBlock.name))) return false;
271
- if (!sourceBlock.path || !sourceBlock.source?.line) return false;
272
-
273
- const caption = componentElement.querySelector<HTMLElement>("figcaption");
274
- if (!caption) return false;
275
- if (caption.matches(UNSAFE_EDITABLE_CHILDREN) || caption.querySelector(UNSAFE_EDITABLE_CHILDREN)) return false;
276
- if (!readableElementText(caption).trim()) return false;
277
-
278
- caption.dataset.openpressBlockId = sourceBlock.id;
279
- caption.dataset.openpressInheritedBlockId = "true";
280
- if (!caption.dataset.openpressObjectId) {
281
- caption.dataset.openpressObjectId = createInlineEditableObjectId(sourceBlock.id, "caption");
282
- caption.dataset.openpressInheritedObjectId = "true";
283
- }
284
- caption.dataset.openpressEditKind = "component-caption";
285
- caption.dataset.openpressEditName = String(sourceBlock.name);
286
- markEditableTextElement(caption, markedElements, { label: "編輯圖說文字" });
287
- return true;
288
- }
289
-
290
- function markEditableTableCells(row: HTMLElement, sourceBlock: SourceBlock, markedElements: Set<HTMLElement>) {
291
- Array.from(row.children).forEach((child, cellIndex) => {
292
- if (!(child instanceof HTMLElement)) return;
293
- if (child.tagName !== "TD" && child.tagName !== "TH") return;
294
- if (child.matches(UNSAFE_EDITABLE_CHILDREN) || child.querySelector(UNSAFE_EDITABLE_CHILDREN)) return;
295
- if (!readableElementText(child).trim()) return;
296
-
297
- child.dataset.openpressBlockId = sourceBlock.id;
298
- child.dataset.openpressInheritedBlockId = "true";
299
- if (!child.dataset.openpressObjectId) {
300
- child.dataset.openpressObjectId = createInlineEditableObjectId(sourceBlock.id, "cell", cellIndex);
301
- child.dataset.openpressInheritedObjectId = "true";
302
- }
303
- child.dataset.openpressEditKind = "table-cell";
304
- child.dataset.openpressEditName = child.tagName.toLowerCase();
305
- child.dataset.openpressTableCellIndex = String(cellIndex);
306
- markEditableTextElement(child, markedElements, { label: "編輯表格文字" });
307
- });
308
- }
309
-
310
- function markEditableTextElement(
311
- element: HTMLElement,
312
- markedElements: Set<HTMLElement>,
313
- { label, preserveLineBreaks = false }: { label: string; preserveLineBreaks?: boolean },
314
- ) {
315
- element.setAttribute("contenteditable", "true");
316
- element.setAttribute("spellcheck", "false");
317
- element.setAttribute("tabindex", "0");
318
- element.setAttribute("role", "textbox");
319
- element.setAttribute("aria-label", label);
320
- element.dataset.openpressEditableBlock = "true";
321
- if (preserveLineBreaks) {
322
- element.dataset.openpressPreserveLineBreaks = "true";
323
- element.setAttribute("aria-multiline", "true");
324
- } else {
325
- delete element.dataset.openpressPreserveLineBreaks;
326
- element.removeAttribute("aria-multiline");
327
- }
328
- element.querySelectorAll<HTMLElement>("[data-openpress-caption-label]").forEach((labelElement) => {
329
- labelElement.setAttribute("contenteditable", "false");
330
- });
331
- if (!element.dataset.openpressOriginalText) {
332
- element.dataset.openpressOriginalText = readableElementText(element);
333
- }
334
- markedElements.add(element);
335
- }
336
-
337
- function clearEditableElement(element: HTMLElement) {
338
- element.removeAttribute("contenteditable");
339
- element.removeAttribute("spellcheck");
340
- element.removeAttribute("tabindex");
341
- element.removeAttribute("role");
342
- element.removeAttribute("aria-label");
343
- delete element.dataset.openpressEditableBlock;
344
- delete element.dataset.openpressSourceEditableBlock;
345
- delete element.dataset.openpressEditing;
346
- delete element.dataset.openpressOriginalText;
347
- delete element.dataset.openpressEditCanceled;
348
- delete element.dataset.openpressEditState;
349
- delete element.dataset.openpressEditStateToken;
350
- delete element.dataset.openpressEditKind;
351
- delete element.dataset.openpressEditName;
352
- delete element.dataset.openpressTableCellIndex;
353
- delete element.dataset.openpressPreserveLineBreaks;
354
- element.removeAttribute("aria-busy");
355
- element.removeAttribute("aria-multiline");
356
- if (element.dataset.openpressInheritedBlockId === "true") {
357
- delete element.dataset.openpressBlockId;
358
- }
359
- delete element.dataset.openpressInheritedBlockId;
360
- if (element.dataset.openpressInheritedObjectId === "true") {
361
- delete element.dataset.openpressObjectId;
362
- }
363
- delete element.dataset.openpressInheritedObjectId;
364
- }
365
-
366
- function isEditableTextBlockElement(element: HTMLElement, sourceBlockMap: Record<string, SourceBlock>) {
367
- const blockId = element.dataset.openpressBlockId;
368
- const sourceBlock = blockId ? sourceBlockMap[blockId] : undefined;
369
- if (!sourceBlock?.path || !sourceBlock.source?.line) return false;
370
- if (!isEditableSourceBlock(sourceBlock)) return false;
371
- if (element.matches(UNSAFE_EDITABLE_CHILDREN) || element.querySelector(UNSAFE_EDITABLE_CHILDREN)) return false;
372
- return true;
373
- }
374
-
375
- function isSourceEditableBlockElement(element: HTMLElement, sourceBlockMap: Record<string, SourceBlock>) {
376
- if (element.dataset.openpressTableCellIndex) return false;
377
- const sourceBlock = blockFromElement(element, sourceBlockMap);
378
- if (!sourceBlock?.path || !sourceBlock.source?.line) return false;
379
- return false;
380
- }
381
-
382
- function isEditableSourceBlock(sourceBlock: SourceBlock) {
383
- if (sourceBlock.kind === "list-item") return true;
384
- if (sourceBlock.kind !== "element") return false;
385
- return typeof sourceBlock.name === "string" && /^(h[1-6]|p|blockquote|pre|caption|figcaption)$/.test(sourceBlock.name);
386
- }
387
-
388
- function editableElementFromEvent(event: Event, root?: HTMLElement) {
389
- const target = eventTargetElement(event);
390
- const element = target?.closest<HTMLElement>(EDITABLE_SELECTOR) ?? null;
391
- if (!element || (root && !root.contains(element))) return null;
392
- return element;
393
- }
394
-
395
- function sourceElementFromEvent(event: Event, root?: HTMLElement) {
396
- const target = eventTargetElement(event);
397
- const element = target?.closest<HTMLElement>(SOURCE_SELECTOR) ?? null;
398
- if (!element || (root && !root.contains(element))) return null;
399
- return element;
400
- }
401
-
402
- function eventTargetElement(event: Event) {
403
- if (event.target instanceof HTMLElement) return event.target;
404
- if (event.target instanceof Node && event.target.parentElement instanceof HTMLElement) return event.target.parentElement;
405
- return null;
406
- }
407
-
408
- function blockFromElement(element: HTMLElement, sourceBlockMap: Record<string, SourceBlock>) {
409
- const blockId = element.dataset.openpressBlockId;
410
- return blockId ? sourceBlockMap[blockId] : undefined;
411
- }
412
-
413
- async function persistElementEdit(
414
- element: HTMLElement,
415
- sourceBlockMap: Record<string, SourceBlock>,
416
- fetchImpl: typeof fetch | undefined,
417
- onStatusChange: InlineDocumentEditorOptions["onStatusChange"],
418
- onDocumentEdited: InlineDocumentEditorOptions["onDocumentEdited"],
419
- ) {
420
- const blockId = element.dataset.openpressBlockId;
421
- const sourceBlock = blockId ? sourceBlockMap[blockId] : undefined;
422
- const preserveLineBreaks = element.dataset.openpressPreserveLineBreaks === "true";
423
- const originalText = normalizeEditableText(element.dataset.openpressOriginalText ?? "", { preserveLineBreaks });
424
- const nextText = normalizeEditableText(readableElementText(element), { preserveLineBreaks });
425
- const canceled = element.dataset.openpressEditCanceled === "true";
426
- delete element.dataset.openpressEditing;
427
- delete element.dataset.openpressEditCanceled;
428
-
429
- if (!sourceBlock || canceled || nextText === originalText) {
430
- delete element.dataset.openpressOriginalText;
431
- clearElementEditState(element);
432
- return;
433
- }
434
- if (!fetchImpl) {
435
- element.textContent = originalText;
436
- setElementEditState(element, "failed");
437
- onStatusChange?.({ state: "failed", blockId, message: "Source edit endpoint is unavailable." });
438
- return;
439
- }
440
-
441
- setElementEditState(element, "saving");
442
- let sourceSaved = false;
443
- try {
444
- const editKind = element.dataset.openpressEditKind || sourceBlock.kind;
445
- const editName = element.dataset.openpressEditName || sourceBlock.name;
446
- const tableCellIndex = element.dataset.openpressTableCellIndex;
447
- const response = await fetchImpl("/__openpress/source-edit", {
448
- method: "POST",
449
- headers: { "Content-Type": "application/json" },
450
- body: JSON.stringify({
451
- blockId,
452
- path: sourceBlock.path,
453
- kind: editKind,
454
- name: editName,
455
- source: sourceBlock.source,
456
- text: nextText,
457
- cellIndex: tableCellIndex ? Number(tableCellIndex) : undefined,
458
- }),
459
- });
460
- if (!response.ok) {
461
- const text = await response.text().catch(() => "");
462
- throw new Error(text || `Source edit failed with status ${response.status}`);
463
- }
464
- sourceSaved = true;
465
- element.dataset.openpressOriginalText = nextText;
466
- await onDocumentEdited?.();
467
- setElementEditState(element, "saved");
468
- scheduleClearElementEditState(element, "saved");
469
- } catch (error) {
470
- if (!sourceSaved) {
471
- element.textContent = originalText;
472
- }
473
- setElementEditState(element, "failed");
474
- onStatusChange?.({
475
- state: "failed",
476
- blockId,
477
- message: error instanceof Error ? error.message : String(error),
478
- });
479
- }
480
- }
481
-
482
- function setElementEditState(element: HTMLElement, state: "saving" | "saved" | "failed") {
483
- element.dataset.openpressEditState = state;
484
- delete element.dataset.openpressEditStateToken;
485
- if (state === "saving") {
486
- element.setAttribute("aria-busy", "true");
487
- return;
488
- }
489
- element.removeAttribute("aria-busy");
490
- }
491
-
492
- function clearElementEditState(element: HTMLElement) {
493
- delete element.dataset.openpressEditState;
494
- delete element.dataset.openpressEditStateToken;
495
- element.removeAttribute("aria-busy");
496
- }
497
-
498
- function scheduleClearElementEditState(element: HTMLElement, state: "saved" | "failed") {
499
- const token = `${Date.now()}-${Math.random()}`;
500
- element.dataset.openpressEditStateToken = token;
501
- window.setTimeout(() => {
502
- if (element.dataset.openpressEditStateToken !== token) return;
503
- if (element.dataset.openpressEditState !== state) return;
504
- clearElementEditState(element);
505
- }, SAVED_EDIT_STATE_RESET_DELAY_MS);
506
- }
507
-
508
- function readableElementText(element: HTMLElement) {
509
- const captionLabel = element.querySelector("[data-openpress-caption-label]");
510
- if (!captionLabel) return typeof element.innerText === "string" ? element.innerText : (element.textContent ?? "");
511
- const clone = element.cloneNode(true) as HTMLElement;
512
- clone.querySelectorAll("[data-openpress-caption-label]").forEach((node) => node.remove());
513
- return typeof clone.innerText === "string" ? clone.innerText : (clone.textContent ?? "");
514
- }
515
-
516
- function normalizeEditableText(value: string, { preserveLineBreaks = false }: { preserveLineBreaks?: boolean } = {}) {
517
- if (preserveLineBreaks) return value.replace(/\r\n?/g, "\n").replace(/^\n+|\n+$/g, "");
518
- return value.replace(/\s*\r?\n\s*/g, " ").trim();
519
- }
520
-
521
- function createInlineEditableObjectId(blockId: string, kind: "caption" | "cell", index?: number) {
522
- const parts = ["mdx-block", blockId, kind];
523
- if (typeof index === "number") parts.push(String(index));
524
- return parts.join(":");
525
- }
@@ -1,10 +0,0 @@
1
- export { DocumentPanel } from "./components/DocumentPanel";
2
- export { InlineSourceEditorLayer } from "./components/InlineSourceEditorLayer";
3
- export { ReaderStage } from "./components/ReaderStage";
4
- export { useDocumentWorkbenchModel } from "./hooks/useDocumentWorkbenchModel";
5
- export {
6
- useInlineDocumentEditor,
7
- type InlineDocumentEditStatus,
8
- type InlineDocumentEditState,
9
- type InlineDocumentSourceTarget,
10
- } from "./hooks/useInlineDocumentEditor";
@@ -1,2 +0,0 @@
1
- export { HtmlWorkbench } from "./Workbench";
2
- export type { WorkbenchPanel } from "./panels";