@open-press/cli 0.8.0 → 1.1.0

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 (250) hide show
  1. package/README.md +33 -23
  2. package/dist/cli.js +320 -252
  3. package/package.json +9 -8
  4. package/template/core/AGENTS.md +0 -126
  5. package/template/core/CHANGELOG.md +0 -215
  6. package/template/core/README.md +0 -40
  7. package/template/core/engine/cli.mjs +0 -96
  8. package/template/core/engine/commands/_shared.mjs +0 -177
  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/init.mjs +0 -24
  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/typecheck.mjs +0 -5
  21. package/template/core/engine/commands/upgrade.mjs +0 -159
  22. package/template/core/engine/commands/validate.mjs +0 -17
  23. package/template/core/engine/document-export.mjs +0 -15
  24. package/template/core/engine/init.mjs +0 -90
  25. package/template/core/engine/output/chrome-pdf.d.mts +0 -34
  26. package/template/core/engine/output/chrome-pdf.mjs +0 -358
  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 -532
  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 -324
  39. package/template/core/engine/react/document-export.mjs +0 -373
  40. package/template/core/engine/react/http-json.mjs +0 -24
  41. package/template/core/engine/react/mdx-compile.mjs +0 -599
  42. package/template/core/engine/react/measurement-css.mjs +0 -136
  43. package/template/core/engine/react/object-entities.mjs +0 -119
  44. package/template/core/engine/react/pagination/allocator.mjs +0 -122
  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 -251
  49. package/template/core/engine/react/pipeline/final-render.mjs +0 -94
  50. package/template/core/engine/react/pipeline/frame-measurement.mjs +0 -302
  51. package/template/core/engine/react/pipeline/press-tree.mjs +0 -135
  52. package/template/core/engine/react/project-asset-endpoint.d.mts +0 -10
  53. package/template/core/engine/react/project-asset-endpoint.mjs +0 -361
  54. package/template/core/engine/react/section-css.mjs +0 -56
  55. package/template/core/engine/react/source-edit-endpoint.d.mts +0 -10
  56. package/template/core/engine/react/source-edit-endpoint.mjs +0 -75
  57. package/template/core/engine/react/sources/heading-numbering.mjs +0 -132
  58. package/template/core/engine/react/sources/mdx-resolver.mjs +0 -439
  59. package/template/core/engine/react/style-discovery.mjs +0 -142
  60. package/template/core/engine/runtime/config.d.mts +0 -40
  61. package/template/core/engine/runtime/config.mjs +0 -175
  62. package/template/core/engine/runtime/file-utils.mjs +0 -106
  63. package/template/core/engine/runtime/file-walk.mjs +0 -22
  64. package/template/core/engine/runtime/inspection.mjs +0 -328
  65. package/template/core/engine/runtime/issue-report.mjs +0 -44
  66. package/template/core/engine/runtime/path-utils.mjs +0 -20
  67. package/template/core/engine/runtime/source-text-tools.d.mts +0 -102
  68. package/template/core/engine/runtime/source-text-tools.mjs +0 -832
  69. package/template/core/engine/runtime/source-workspace.mjs +0 -159
  70. package/template/core/engine/runtime/validation.mjs +0 -174
  71. package/template/core/index.html +0 -13
  72. package/template/core/openpress.config.mjs +0 -12
  73. package/template/core/package.json +0 -91
  74. package/template/core/src/main.tsx +0 -16
  75. package/template/core/src/openpress/app/OpenPressApp.tsx +0 -140
  76. package/template/core/src/openpress/app/OpenPressRuntime.tsx +0 -94
  77. package/template/core/src/openpress/app/index.ts +0 -2
  78. package/template/core/src/openpress/core/Frame.tsx +0 -78
  79. package/template/core/src/openpress/core/FrameContext.tsx +0 -24
  80. package/template/core/src/openpress/core/MdxArea.tsx +0 -34
  81. package/template/core/src/openpress/core/Press.tsx +0 -34
  82. package/template/core/src/openpress/core/cn.ts +0 -4
  83. package/template/core/src/openpress/core/index.tsx +0 -40
  84. package/template/core/src/openpress/core/primitives.tsx +0 -44
  85. package/template/core/src/openpress/core/types.ts +0 -191
  86. package/template/core/src/openpress/core/useSource.ts +0 -28
  87. package/template/core/src/openpress/document-model/anchorMapModel.ts +0 -27
  88. package/template/core/src/openpress/document-model/documentIndexes.ts +0 -329
  89. package/template/core/src/openpress/document-model/documentTypes.ts +0 -138
  90. package/template/core/src/openpress/document-model/index.ts +0 -6
  91. package/template/core/src/openpress/document-model/objectEntityModel.ts +0 -51
  92. package/template/core/src/openpress/document-model/projectIdentityModel.ts +0 -15
  93. package/template/core/src/openpress/document-model/reactDocumentMetadataModel.ts +0 -27
  94. package/template/core/src/openpress/manuscript/index.tsx +0 -238
  95. package/template/core/src/openpress/mdx/index.ts +0 -88
  96. package/template/core/src/openpress/numbering/index.ts +0 -294
  97. package/template/core/src/openpress/reader/PublicReaderPage.tsx +0 -267
  98. package/template/core/src/openpress/reader/ReaderNavigationPanel.tsx +0 -123
  99. package/template/core/src/openpress/reader/index.ts +0 -10
  100. package/template/core/src/openpress/reader/pageViewportScaleModel.ts +0 -73
  101. package/template/core/src/openpress/reader/readerPageRegistry.ts +0 -41
  102. package/template/core/src/openpress/reader/readerPageRoute.ts +0 -21
  103. package/template/core/src/openpress/reader/readerScroll.ts +0 -92
  104. package/template/core/src/openpress/reader/readerStateModel.ts +0 -15
  105. package/template/core/src/openpress/reader/readerTypes.ts +0 -4
  106. package/template/core/src/openpress/reader/usePageViewportScale.ts +0 -119
  107. package/template/core/src/openpress/reader/usePanelState.ts +0 -56
  108. package/template/core/src/openpress/reader/useReaderHashSync.ts +0 -61
  109. package/template/core/src/openpress/reader/useReaderKeyboardNav.ts +0 -48
  110. package/template/core/src/openpress/reader/useReaderRuntime.ts +0 -146
  111. package/template/core/src/openpress/reader/useReaderScrollAnchor.ts +0 -64
  112. package/template/core/src/openpress/shared/Panel.tsx +0 -77
  113. package/template/core/src/openpress/shared/frameScheduler.ts +0 -32
  114. package/template/core/src/openpress/shared/index.ts +0 -4
  115. package/template/core/src/openpress/shared/numberUtils.ts +0 -3
  116. package/template/core/src/openpress/shared/runtimeMode.ts +0 -11
  117. package/template/core/src/openpress/workbench/Workbench.tsx +0 -407
  118. package/template/core/src/openpress/workbench/actions/DeploymentControl.tsx +0 -157
  119. package/template/core/src/openpress/workbench/actions/PageZoomControl.tsx +0 -182
  120. package/template/core/src/openpress/workbench/actions/SearchControl.tsx +0 -345
  121. package/template/core/src/openpress/workbench/actions/deploymentStatusModel.ts +0 -112
  122. package/template/core/src/openpress/workbench/actions/index.ts +0 -5
  123. package/template/core/src/openpress/workbench/actions/useDeploymentWorkbench.ts +0 -136
  124. package/template/core/src/openpress/workbench/dialog/WorkbenchDialog.tsx +0 -72
  125. package/template/core/src/openpress/workbench/dialog/index.ts +0 -1
  126. package/template/core/src/openpress/workbench/document/components/DocumentPanel.tsx +0 -127
  127. package/template/core/src/openpress/workbench/document/components/InlineSourceEditorLayer.tsx +0 -207
  128. package/template/core/src/openpress/workbench/document/components/ReaderStage.tsx +0 -9
  129. package/template/core/src/openpress/workbench/document/hooks/useDocumentWorkbenchModel.ts +0 -34
  130. package/template/core/src/openpress/workbench/document/hooks/useInlineDocumentEditor.ts +0 -525
  131. package/template/core/src/openpress/workbench/document/index.ts +0 -10
  132. package/template/core/src/openpress/workbench/index.ts +0 -2
  133. package/template/core/src/openpress/workbench/inspector/InlineInspectorLayer.tsx +0 -459
  134. package/template/core/src/openpress/workbench/inspector/index.ts +0 -5
  135. package/template/core/src/openpress/workbench/inspector/inlineCommentModel.ts +0 -125
  136. package/template/core/src/openpress/workbench/inspector/inspectorGeometryModel.ts +0 -160
  137. package/template/core/src/openpress/workbench/inspector/inspectorModel.ts +0 -408
  138. package/template/core/src/openpress/workbench/inspector/useInspectorComments.ts +0 -248
  139. package/template/core/src/openpress/workbench/mentions/MentionSuggestionList.tsx +0 -41
  140. package/template/core/src/openpress/workbench/mentions/index.ts +0 -2
  141. package/template/core/src/openpress/workbench/mentions/useComposerMentions.ts +0 -185
  142. package/template/core/src/openpress/workbench/panels/Panel.tsx +0 -1
  143. package/template/core/src/openpress/workbench/panels/PendingCommentsPanel.tsx +0 -76
  144. package/template/core/src/openpress/workbench/panels/WorkbenchControlPanel.tsx +0 -29
  145. package/template/core/src/openpress/workbench/panels/index.ts +0 -3
  146. package/template/core/src/openpress/workbench/project/ProjectEntryPanel.tsx +0 -523
  147. package/template/core/src/openpress/workbench/project/ProjectPreviewDialog.tsx +0 -35
  148. package/template/core/src/openpress/workbench/project/index.ts +0 -2
  149. package/template/core/src/openpress/workbench/project/projectPreviewTypes.ts +0 -11
  150. package/template/core/src/openpress/workbench/project/projectSourceModel.ts +0 -24
  151. package/template/core/src/openpress/workbench/shell/WorkbenchShell.tsx +0 -167
  152. package/template/core/src/openpress/workbench/shell/index.ts +0 -1
  153. package/template/core/src/openpress/workbench/workbenchFormatters.ts +0 -120
  154. package/template/core/src/openpress/workbench/workbenchTypes.ts +0 -35
  155. package/template/core/src/styles/openpress/app-shell.css +0 -251
  156. package/template/core/src/styles/openpress/media-workspace.css +0 -230
  157. package/template/core/src/styles/openpress/print-route.css +0 -184
  158. package/template/core/src/styles/openpress/project-preview-panel.css +0 -924
  159. package/template/core/src/styles/openpress/public-viewer.css +0 -688
  160. package/template/core/src/styles/openpress/reader-runtime.css +0 -980
  161. package/template/core/src/styles/openpress/responsive.css +0 -245
  162. package/template/core/src/styles/openpress/workbench-panels.css +0 -594
  163. package/template/core/src/styles/openpress/workbench.css +0 -1255
  164. package/template/core/src/styles/openpress.css +0 -14
  165. package/template/core/src/vite-env.d.ts +0 -9
  166. package/template/core/tsconfig.json +0 -40
  167. package/template/core/vite.config.ts +0 -584
  168. package/template/packs/academic-paper/document/chapters/01-introduction/content/01-introduction.mdx +0 -35
  169. package/template/packs/academic-paper/document/chapters/02-methods/content/01-methods.mdx +0 -50
  170. package/template/packs/academic-paper/document/chapters/03-results-and-discussion/content/01-results.mdx +0 -47
  171. package/template/packs/academic-paper/document/chapters/04-acknowledgment/content/01-acknowledgment.mdx +0 -26
  172. package/template/packs/academic-paper/document/chapters/05-references/content/01-references.mdx +0 -32
  173. package/template/packs/academic-paper/document/components/ChapterOpenerVisual/index.tsx +0 -76
  174. package/template/packs/academic-paper/document/components/Page.tsx +0 -60
  175. package/template/packs/academic-paper/document/components/TokenSwatchGrid/index.tsx +0 -46
  176. package/template/packs/academic-paper/document/components/TokenSwatchGrid/style.css +0 -63
  177. package/template/packs/academic-paper/document/components/TypeSpecimen/index.tsx +0 -38
  178. package/template/packs/academic-paper/document/components/TypeSpecimen/style.css +0 -111
  179. package/template/packs/academic-paper/document/design.md +0 -279
  180. package/template/packs/academic-paper/document/index.tsx +0 -123
  181. package/template/packs/academic-paper/document/media/README.md +0 -13
  182. package/template/packs/academic-paper/document/media/figure-placeholder.svg +0 -9
  183. package/template/packs/academic-paper/document/openpress.config.mjs +0 -26
  184. package/template/packs/academic-paper/document/theme/README.md +0 -11
  185. package/template/packs/academic-paper/document/theme/base/page-contract.css +0 -522
  186. package/template/packs/academic-paper/document/theme/base/print.css +0 -93
  187. package/template/packs/academic-paper/document/theme/base/typography.css +0 -333
  188. package/template/packs/academic-paper/document/theme/fonts.css +0 -3
  189. package/template/packs/academic-paper/document/theme/page-surfaces/back-cover.css +0 -43
  190. package/template/packs/academic-paper/document/theme/page-surfaces/chapter-opener.css +0 -205
  191. package/template/packs/academic-paper/document/theme/page-surfaces/cover.css +0 -294
  192. package/template/packs/academic-paper/document/theme/page-surfaces/toc.css +0 -149
  193. package/template/packs/academic-paper/document/theme/patterns/_chart-frame.css +0 -49
  194. package/template/packs/academic-paper/document/theme/patterns/figure-grid.css +0 -68
  195. package/template/packs/academic-paper/document/theme/patterns/table-utilities.css +0 -66
  196. package/template/packs/academic-paper/document/theme/shell/reader-controls.css +0 -761
  197. package/template/packs/academic-paper/document/theme/tokens.css +0 -80
  198. package/template/packs/academic-paper/openpress.config.mjs +0 -5
  199. package/template/packs/claude-document/document/chapters/01-document-shape/content/01-document-shape.mdx +0 -51
  200. package/template/packs/claude-document/document/chapters/02-review-loop/content/01-review-loop.mdx +0 -31
  201. package/template/packs/claude-document/document/components/ChapterOpenerVisual.tsx +0 -96
  202. package/template/packs/claude-document/document/components/Page.tsx +0 -37
  203. package/template/packs/claude-document/document/design.md +0 -142
  204. package/template/packs/claude-document/document/index.tsx +0 -94
  205. package/template/packs/claude-document/document/media/README.md +0 -13
  206. package/template/packs/claude-document/document/openpress.config.mjs +0 -26
  207. package/template/packs/claude-document/document/theme/README.md +0 -15
  208. package/template/packs/claude-document/document/theme/base/page-contract.css +0 -525
  209. package/template/packs/claude-document/document/theme/base/print.css +0 -93
  210. package/template/packs/claude-document/document/theme/base/typography.css +0 -612
  211. package/template/packs/claude-document/document/theme/fonts.css +0 -4
  212. package/template/packs/claude-document/document/theme/page-surfaces/back-cover.css +0 -72
  213. package/template/packs/claude-document/document/theme/page-surfaces/chapter-opener.css +0 -236
  214. package/template/packs/claude-document/document/theme/page-surfaces/cover.css +0 -309
  215. package/template/packs/claude-document/document/theme/page-surfaces/toc.css +0 -225
  216. package/template/packs/claude-document/document/theme/patterns/_chart-frame.css +0 -53
  217. package/template/packs/claude-document/document/theme/patterns/figure-grid.css +0 -68
  218. package/template/packs/claude-document/document/theme/patterns/table-utilities.css +0 -66
  219. package/template/packs/claude-document/document/theme/shell/reader-controls.css +0 -789
  220. package/template/packs/claude-document/document/theme/tokens.css +0 -89
  221. package/template/packs/claude-document/openpress.config.mjs +0 -5
  222. package/template/packs/editorial-monograph/document/chapters/01-product-and-use-cases/content/01-product-and-use-cases.mdx +0 -31
  223. package/template/packs/editorial-monograph/document/chapters/02-workflow/content/01-workflow.mdx +0 -89
  224. package/template/packs/editorial-monograph/document/chapters/03-agent-skills-contributors/content/01-agent-skills-contributors.mdx +0 -51
  225. package/template/packs/editorial-monograph/document/chapters/04-validation-deploy/content/01-validation-deploy.mdx +0 -39
  226. package/template/packs/editorial-monograph/document/components/ChapterOpenerVisual/index.tsx +0 -76
  227. package/template/packs/editorial-monograph/document/components/Page.tsx +0 -37
  228. package/template/packs/editorial-monograph/document/components/TokenSwatchGrid/index.tsx +0 -46
  229. package/template/packs/editorial-monograph/document/components/TokenSwatchGrid/style.css +0 -63
  230. package/template/packs/editorial-monograph/document/components/TypeSpecimen/index.tsx +0 -38
  231. package/template/packs/editorial-monograph/document/components/TypeSpecimen/style.css +0 -111
  232. package/template/packs/editorial-monograph/document/design.md +0 -279
  233. package/template/packs/editorial-monograph/document/index.tsx +0 -97
  234. package/template/packs/editorial-monograph/document/media/README.md +0 -13
  235. package/template/packs/editorial-monograph/document/openpress.config.mjs +0 -26
  236. package/template/packs/editorial-monograph/document/theme/README.md +0 -11
  237. package/template/packs/editorial-monograph/document/theme/base/page-contract.css +0 -505
  238. package/template/packs/editorial-monograph/document/theme/base/print.css +0 -93
  239. package/template/packs/editorial-monograph/document/theme/base/typography.css +0 -336
  240. package/template/packs/editorial-monograph/document/theme/fonts.css +0 -3
  241. package/template/packs/editorial-monograph/document/theme/page-surfaces/back-cover.css +0 -43
  242. package/template/packs/editorial-monograph/document/theme/page-surfaces/chapter-opener.css +0 -205
  243. package/template/packs/editorial-monograph/document/theme/page-surfaces/cover.css +0 -147
  244. package/template/packs/editorial-monograph/document/theme/page-surfaces/toc.css +0 -149
  245. package/template/packs/editorial-monograph/document/theme/patterns/_chart-frame.css +0 -49
  246. package/template/packs/editorial-monograph/document/theme/patterns/figure-grid.css +0 -68
  247. package/template/packs/editorial-monograph/document/theme/patterns/table-utilities.css +0 -66
  248. package/template/packs/editorial-monograph/document/theme/shell/reader-controls.css +0 -761
  249. package/template/packs/editorial-monograph/document/theme/tokens.css +0 -80
  250. package/template/packs/editorial-monograph/openpress.config.mjs +0 -5
@@ -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";