@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,182 +0,0 @@
1
- import { useEffect, useId, useRef, useState } from "react";
2
- import type { ReactNode } from "react";
3
- import { Check, ChevronDown, Columns2, File, ZoomIn } from "lucide-react";
4
- import {
5
- PAGE_VIEWPORT_SCALE_OPTIONS,
6
- type PageLayoutMode,
7
- type PageViewportScaleMode,
8
- } from "../../reader";
9
-
10
- export function PageZoomControl({
11
- scaleMode,
12
- scaleLabel,
13
- pageLayoutMode,
14
- onScaleModeChange,
15
- onPageLayoutModeChange,
16
- }: {
17
- scaleMode: PageViewportScaleMode;
18
- scaleLabel: string;
19
- pageLayoutMode: PageLayoutMode;
20
- onScaleModeChange: (mode: PageViewportScaleMode) => void;
21
- onPageLayoutModeChange: (mode: PageLayoutMode) => void;
22
- }) {
23
- const menuId = useId();
24
- const rootRef = useRef<HTMLDivElement | null>(null);
25
- const [open, setOpen] = useState(false);
26
- const fixedOptions = PAGE_VIEWPORT_SCALE_OPTIONS.filter((option) => option.value.startsWith("scale-"));
27
- const fitOptions = PAGE_VIEWPORT_SCALE_OPTIONS.filter((option) => option.value.startsWith("fit-"));
28
-
29
- useEffect(() => {
30
- if (!open) return undefined;
31
- const handlePointerDown = (event: PointerEvent) => {
32
- if (event.target instanceof Node && rootRef.current?.contains(event.target)) return;
33
- setOpen(false);
34
- };
35
- const handleKeyDown = (event: KeyboardEvent) => {
36
- if (event.key === "Escape") setOpen(false);
37
- };
38
- window.addEventListener("pointerdown", handlePointerDown);
39
- window.addEventListener("keydown", handleKeyDown);
40
- return () => {
41
- window.removeEventListener("pointerdown", handlePointerDown);
42
- window.removeEventListener("keydown", handleKeyDown);
43
- };
44
- }, [open]);
45
-
46
- const selectScale = (mode: PageViewportScaleMode) => {
47
- onScaleModeChange(mode);
48
- setOpen(false);
49
- };
50
- const selectLayout = (mode: PageLayoutMode) => {
51
- onPageLayoutModeChange(mode);
52
- setOpen(false);
53
- };
54
-
55
- return (
56
- <div className="openpress-workbench-zoom-control-wrap" ref={rootRef} data-openpress-page-zoom-control>
57
- <button
58
- type="button"
59
- className="openpress-workbench-zoom-control"
60
- data-openpress-page-zoom
61
- data-openpress-scale-mode={scaleMode}
62
- data-openpress-toolbar-active={scaleMode === "fit-width" ? "false" : "true"}
63
- title={`頁面縮放 ${scaleLabel}`}
64
- aria-label={`頁面縮放 ${scaleLabel}`}
65
- aria-haspopup="menu"
66
- aria-expanded={open}
67
- aria-controls={open ? menuId : undefined}
68
- onClick={() => setOpen((value) => !value)}
69
- >
70
- <ZoomIn aria-hidden="true" />
71
- <span>{scaleLabel}</span>
72
- <ChevronDown className="openpress-workbench-zoom-control__chevron" aria-hidden="true" />
73
- </button>
74
- {open ? (
75
- <div
76
- id={menuId}
77
- className="openpress-workbench-zoom-menu"
78
- data-openpress-page-zoom-menu
79
- role="menu"
80
- aria-label="頁面顯示與縮放"
81
- >
82
- <div className="openpress-workbench-zoom-menu__section" role="group" aria-label="頁面模式">
83
- <PageLayoutOption
84
- mode="single"
85
- active={pageLayoutMode === "single"}
86
- icon={<File aria-hidden="true" />}
87
- label="一頁"
88
- onSelect={selectLayout}
89
- />
90
- <PageLayoutOption
91
- mode="spread"
92
- active={pageLayoutMode === "spread"}
93
- icon={<Columns2 aria-hidden="true" />}
94
- label="雙頁"
95
- onSelect={selectLayout}
96
- />
97
- </div>
98
- <div className="openpress-workbench-zoom-menu__divider" role="presentation" />
99
- <div className="openpress-workbench-zoom-menu__section" role="group" aria-label="固定縮放">
100
- {fixedOptions.map((option) => (
101
- <ZoomOption
102
- key={option.value}
103
- mode={option.value}
104
- active={scaleMode === option.value}
105
- label={option.label}
106
- onSelect={selectScale}
107
- />
108
- ))}
109
- </div>
110
- <div className="openpress-workbench-zoom-menu__divider" role="presentation" />
111
- <div className="openpress-workbench-zoom-menu__section" role="group" aria-label="符合顯示">
112
- {fitOptions.map((option) => (
113
- <ZoomOption
114
- key={option.value}
115
- mode={option.value}
116
- active={scaleMode === option.value}
117
- label={option.label}
118
- onSelect={selectScale}
119
- />
120
- ))}
121
- </div>
122
- </div>
123
- ) : null}
124
- </div>
125
- );
126
- }
127
-
128
- function PageLayoutOption({
129
- mode,
130
- active,
131
- icon,
132
- label,
133
- onSelect,
134
- }: {
135
- mode: PageLayoutMode;
136
- active: boolean;
137
- icon: ReactNode;
138
- label: string;
139
- onSelect: (mode: PageLayoutMode) => void;
140
- }) {
141
- return (
142
- <button
143
- type="button"
144
- className="openpress-workbench-zoom-menu__item"
145
- data-openpress-page-layout-option={mode}
146
- role="menuitemcheckbox"
147
- aria-checked={active}
148
- onClick={() => onSelect(mode)}
149
- >
150
- <span className="openpress-workbench-zoom-menu__check">{active ? <Check aria-hidden="true" /> : null}</span>
151
- {icon}
152
- <span>{label}</span>
153
- </button>
154
- );
155
- }
156
-
157
- function ZoomOption({
158
- mode,
159
- active,
160
- label,
161
- onSelect,
162
- }: {
163
- mode: PageViewportScaleMode;
164
- active: boolean;
165
- label: string;
166
- onSelect: (mode: PageViewportScaleMode) => void;
167
- }) {
168
- return (
169
- <button
170
- type="button"
171
- className="openpress-workbench-zoom-menu__item"
172
- data-openpress-zoom-option={mode}
173
- role="menuitemradio"
174
- aria-checked={active}
175
- onClick={() => onSelect(mode)}
176
- >
177
- <span className="openpress-workbench-zoom-menu__check">{active ? <Check aria-hidden="true" /> : null}</span>
178
- <span className="openpress-workbench-zoom-menu__spacer" aria-hidden="true" />
179
- <span>{label}</span>
180
- </button>
181
- );
182
- }
@@ -1,345 +0,0 @@
1
- import { useCallback, useEffect, useId, useMemo, useRef, useState, type FormEvent } from "react";
2
- import { FileText, Loader2, Search } from "lucide-react";
3
- import type { SourceBlock } from "../../document-model";
4
- import { WorkbenchDialog } from "../dialog";
5
-
6
- type SearchScope = "content" | "all";
7
- type SearchStatus = "idle" | "loading" | "success" | "error";
8
- const SEARCH_SCOPE: SearchScope = "all";
9
- const LIVE_SEARCH_DEBOUNCE_MS = 280;
10
-
11
- type SearchFile = {
12
- scope: string;
13
- file: string;
14
- path: string;
15
- matchCount: number;
16
- };
17
-
18
- type SearchMatch = {
19
- id: string;
20
- scope: string;
21
- file: string;
22
- path: string;
23
- line: number;
24
- column: number;
25
- index: number;
26
- text: string;
27
- preview: string;
28
- };
29
-
30
- type SearchReport = {
31
- ok?: boolean;
32
- kind: "search";
33
- query: string;
34
- scope: SearchScope;
35
- caseSensitive: boolean;
36
- matchCount: number;
37
- files: Array<SearchFile>;
38
- matches: Array<SearchMatch>;
39
- message?: string;
40
- };
41
-
42
- type SearchJumpTarget = {
43
- blockId: string;
44
- pageIndex: number;
45
- pageNumber: number;
46
- };
47
-
48
- export function SearchControl({
49
- sourceBlocksByPath = {},
50
- onSelectPage,
51
- }: {
52
- sourceBlocksByPath?: Record<string, SourceBlock[]>;
53
- onSelectPage?: (pageIndex: number, options?: { behavior?: ScrollBehavior }) => void;
54
- }) {
55
- const titleId = useId();
56
- const [open, setOpen] = useState(false);
57
- const [query, setQuery] = useState("");
58
- const [status, setStatus] = useState<SearchStatus>("idle");
59
- const [report, setReport] = useState<SearchReport | null>(null);
60
- const [error, setError] = useState("");
61
- const activeSearchControllerRef = useRef<AbortController | null>(null);
62
- const searchRequestIdRef = useRef(0);
63
- const submittedQueryRef = useRef<string | null>(null);
64
- const matchesByPath = useMemo(() => groupMatchesByPath(report?.matches ?? []), [report]);
65
-
66
- const jumpToMatch = (match: SearchMatch) => {
67
- const target = resolveSearchJumpTarget(match, sourceBlocksByPath);
68
- if (!target || !onSelectPage) return;
69
- onSelectPage(target.pageIndex, { behavior: "smooth" });
70
- setOpen(false);
71
- };
72
-
73
- useEffect(() => {
74
- if (!open) return;
75
- const handleKeyDown = (event: KeyboardEvent) => {
76
- if (event.key === "Escape") setOpen(false);
77
- };
78
- window.addEventListener("keydown", handleKeyDown);
79
- return () => window.removeEventListener("keydown", handleKeyDown);
80
- }, [open]);
81
-
82
- const runSearch = useCallback(async (rawQuery: string, controller: AbortController) => {
83
- const trimmedQuery = rawQuery.trim();
84
- if (!trimmedQuery) {
85
- activeSearchControllerRef.current?.abort();
86
- activeSearchControllerRef.current = null;
87
- setReport(null);
88
- setError("");
89
- setStatus("idle");
90
- return;
91
- }
92
-
93
- activeSearchControllerRef.current?.abort();
94
- activeSearchControllerRef.current = controller;
95
- const requestId = searchRequestIdRef.current + 1;
96
- searchRequestIdRef.current = requestId;
97
- setStatus("loading");
98
- setError("");
99
-
100
- try {
101
- const params = new URLSearchParams();
102
- params.set("q", trimmedQuery);
103
- params.set("scope", SEARCH_SCOPE);
104
- const response = await fetch(`/__openpress/search?${params.toString()}`, {
105
- cache: "no-store",
106
- signal: controller.signal,
107
- });
108
- const data = await response.json().catch(() => null) as (Partial<SearchReport> & { message?: string }) | null;
109
- if (controller.signal.aborted || requestId !== searchRequestIdRef.current) return;
110
- if (!response.ok || data?.ok === false || !isSearchReport(data)) {
111
- throw new Error(data?.message ?? "搜尋失敗。");
112
- }
113
- setReport(data);
114
- setStatus("success");
115
- } catch (searchError) {
116
- if (controller.signal.aborted || requestId !== searchRequestIdRef.current) return;
117
- setError(searchError instanceof Error ? searchError.message : String(searchError));
118
- setStatus("error");
119
- }
120
- }, []);
121
-
122
- useEffect(() => {
123
- if (!open) return undefined;
124
- const trimmedQuery = query.trim();
125
- if (!trimmedQuery) {
126
- activeSearchControllerRef.current?.abort();
127
- activeSearchControllerRef.current = null;
128
- setReport(null);
129
- setError("");
130
- setStatus("idle");
131
- return undefined;
132
- }
133
-
134
- const controller = new AbortController();
135
- const timer = window.setTimeout(() => {
136
- if (submittedQueryRef.current === trimmedQuery) {
137
- submittedQueryRef.current = null;
138
- return;
139
- }
140
- void runSearch(trimmedQuery, controller);
141
- }, LIVE_SEARCH_DEBOUNCE_MS);
142
-
143
- return () => {
144
- window.clearTimeout(timer);
145
- controller.abort();
146
- };
147
- }, [open, query, runSearch]);
148
-
149
- const submitSearch = async (event: FormEvent<HTMLFormElement>) => {
150
- event.preventDefault();
151
- const trimmedQuery = query.trim();
152
- submittedQueryRef.current = trimmedQuery;
153
- await runSearch(trimmedQuery, new AbortController());
154
- };
155
-
156
- const dialog = open ? (
157
- <WorkbenchDialog
158
- titleId={titleId}
159
- title="搜尋文件"
160
- eyebrow="Search"
161
- className="openpress-search-dialog"
162
- backdropClassName="openpress-search-dialog-backdrop"
163
- headerClassName="openpress-search-dialog__header"
164
- closeLabel="關閉搜尋"
165
- onClose={() => setOpen(false)}
166
- >
167
- <form className="openpress-search-dialog__form" role="search" aria-label="文件搜尋" onSubmit={submitSearch}>
168
- <div className="openpress-search-dialog__input-row">
169
- <Search aria-hidden="true" />
170
- <input
171
- type="search"
172
- value={query}
173
- autoFocus
174
- aria-label="搜尋文字"
175
- placeholder="搜尋所有來源"
176
- onChange={(event) => setQuery(event.currentTarget.value)}
177
- />
178
- <button type="submit" disabled={status === "loading"}>
179
- {status === "loading" ? <Loader2 aria-hidden="true" /> : <Search aria-hidden="true" />}
180
- <span>搜尋</span>
181
- </button>
182
- </div>
183
- </form>
184
- <SearchResults
185
- status={status}
186
- report={report}
187
- error={error}
188
- matchesByPath={matchesByPath}
189
- sourceBlocksByPath={sourceBlocksByPath}
190
- onJumpToMatch={jumpToMatch}
191
- />
192
- </WorkbenchDialog>
193
- ) : null;
194
-
195
- return (
196
- <>
197
- <button
198
- type="button"
199
- className="openpress-workbench-toolbar-action"
200
- data-openpress-search
201
- data-openpress-toolbar-expanded="false"
202
- data-openpress-toolbar-active={open ? "true" : "false"}
203
- aria-label="搜尋文件"
204
- title="搜尋文件"
205
- onClick={() => setOpen(true)}
206
- >
207
- <Search aria-hidden="true" />
208
- </button>
209
- {dialog}
210
- </>
211
- );
212
- }
213
-
214
- function SearchResults({
215
- status,
216
- report,
217
- error,
218
- matchesByPath,
219
- sourceBlocksByPath,
220
- onJumpToMatch,
221
- }: {
222
- status: SearchStatus;
223
- report: SearchReport | null;
224
- error: string;
225
- matchesByPath: Map<string, Array<SearchMatch>>;
226
- sourceBlocksByPath: Record<string, SourceBlock[]>;
227
- onJumpToMatch: (match: SearchMatch) => void;
228
- }) {
229
- if (status === "idle") {
230
- return <p className="openpress-search-dialog__empty">輸入關鍵字即可搜尋文件內容、元件與設定來源。</p>;
231
- }
232
-
233
- if (status === "loading") {
234
- return (
235
- <p className="openpress-search-dialog__empty" role="status">
236
- <Loader2 aria-hidden="true" />
237
- <span>搜尋中</span>
238
- </p>
239
- );
240
- }
241
-
242
- if (status === "error") {
243
- return <p className="openpress-search-dialog__error" role="alert">{error || "搜尋失敗。"}</p>;
244
- }
245
-
246
- if (!report || report.matchCount === 0) {
247
- return <p className="openpress-search-dialog__empty">沒有符合的來源文字。</p>;
248
- }
249
-
250
- return (
251
- <div className="openpress-search-dialog__results" aria-live="polite">
252
- <p className="openpress-search-dialog__summary">{report.matchCount} 個符合結果</p>
253
- {report.files.map((file) => (
254
- <section className="openpress-search-dialog__file" key={file.path}>
255
- <h3>
256
- <FileText aria-hidden="true" />
257
- <span>{file.path}</span>
258
- <small>{file.matchCount}</small>
259
- </h3>
260
- <ol>
261
- {(matchesByPath.get(file.path) ?? []).map((match) => {
262
- const jumpTarget = resolveSearchJumpTarget(match, sourceBlocksByPath);
263
- return (
264
- <li key={match.id}>
265
- <button
266
- type="button"
267
- className="openpress-search-dialog__result"
268
- data-openpress-search-result-jump={jumpTarget ? "true" : "false"}
269
- disabled={!jumpTarget}
270
- onClick={() => onJumpToMatch(match)}
271
- >
272
- <span className="openpress-search-dialog__line">{match.line}:{match.column}</span>
273
- <span className="openpress-search-dialog__preview">{match.preview}</span>
274
- <span className="openpress-search-dialog__page">
275
- {jumpTarget ? `P${String(jumpTarget.pageNumber).padStart(2, "0")}` : "source"}
276
- </span>
277
- </button>
278
- </li>
279
- );
280
- })}
281
- </ol>
282
- </section>
283
- ))}
284
- </div>
285
- );
286
- }
287
-
288
- function groupMatchesByPath(matches: Array<SearchMatch>) {
289
- const grouped = new Map<string, Array<SearchMatch>>();
290
- for (const match of matches) {
291
- const existing = grouped.get(match.path) ?? [];
292
- existing.push(match);
293
- grouped.set(match.path, existing);
294
- }
295
- return grouped;
296
- }
297
-
298
- function isSearchReport(value: unknown): value is SearchReport {
299
- if (!value || typeof value !== "object") return false;
300
- const report = value as Partial<SearchReport>;
301
- return report.kind === "search"
302
- && typeof report.query === "string"
303
- && typeof report.matchCount === "number"
304
- && Array.isArray(report.files)
305
- && Array.isArray(report.matches);
306
- }
307
-
308
- function resolveSearchJumpTarget(
309
- match: SearchMatch,
310
- sourceBlocksByPath: Record<string, SourceBlock[]>,
311
- ): SearchJumpTarget | null {
312
- const blocks = sourcePathKeys(match.path)
313
- .flatMap((key) => sourceBlocksByPath[key] ?? []);
314
- if (!blocks.length) return null;
315
-
316
- let selectedBlock: SourceBlock | null = null;
317
- for (const block of blocks) {
318
- const line = block.source?.line;
319
- if (typeof line !== "number") continue;
320
- if (line <= match.line) {
321
- selectedBlock = block;
322
- continue;
323
- }
324
- break;
325
- }
326
-
327
- const targetBlock = selectedBlock ?? blocks[0] ?? null;
328
- if (!targetBlock || typeof targetBlock.pageIndex !== "number") return null;
329
- return {
330
- blockId: targetBlock.id,
331
- pageIndex: targetBlock.pageIndex,
332
- pageNumber: targetBlock.pageNumber ?? targetBlock.pageIndex + 1,
333
- };
334
- }
335
-
336
- function sourcePathKeys(value: string) {
337
- const normalized = value.trim().replaceAll("\\", "/").replace(/^\.\//, "");
338
- const keys = [normalized];
339
- if (normalized.startsWith("document/")) {
340
- keys.push(normalized.replace(/^document\//, ""));
341
- } else {
342
- keys.push(`document/${normalized}`);
343
- }
344
- return Array.from(new Set(keys));
345
- }
@@ -1,112 +0,0 @@
1
- import type { DeploymentInfo } from "../../document-model";
2
- import type { DeployStatus, PdfActionStatus } from "../workbenchTypes";
3
-
4
- export function deployButtonText(info: DeploymentInfo, status: DeployStatus) {
5
- if (info.configured === false || status === "setup") return "設定部署";
6
- if (status === "deploying") return "部署中";
7
- if (status === "failed") return "重試部署";
8
- if (status === "unavailable") return "本機限定";
9
- if (isDeploymentDirty(info, status)) return "重新部署";
10
- return "部署";
11
- }
12
-
13
- export function workbenchPdfButtonText(localPdfEnabled: boolean, status: PdfActionStatus, staticPdfHref?: string) {
14
- if (localPdfEnabled) {
15
- if (status === "generating") return "產生中";
16
- if (status === "opening") return "正在開啟";
17
- if (status === "failed") return "重試 PDF";
18
- return "產生 PDF";
19
- }
20
- return !staticPdfHref ? "PDF 未部署" : "開啟 PDF";
21
- }
22
-
23
- export function workbenchPdfStatusMessage(localPdfEnabled: boolean, status: PdfActionStatus) {
24
- if (!localPdfEnabled) return null;
25
- if (status === "generating") return "正在產生 PDF";
26
- if (status === "opening") return "PDF 已完成,正在開啟";
27
- if (status === "failed") return "PDF 產生失敗,請重試";
28
- return null;
29
- }
30
-
31
- export function deploymentStatusKind(info: DeploymentInfo, status: DeployStatus) {
32
- if (info.configured === false || status === "setup") return "failed";
33
- if (status === "deploying") return "deploying";
34
- if (status === "failed") return "failed";
35
- if (status === "unavailable") return "unavailable";
36
- if (isDeploymentDirty(info, status)) return "dirty";
37
- if (status === "deployed" || hasOnlineDeployment(info)) return "online";
38
- return "offline";
39
- }
40
-
41
- export function deploymentStatusSummary(info: DeploymentInfo, status: DeployStatus) {
42
- const label = deploymentStatusLabel(info, status);
43
- if ((status === "deployed" || hasOnlineDeployment(info)) && info.deployedAt) {
44
- return `${label} · ${formatDeployTime(info.deployedAt)}`;
45
- }
46
- return label;
47
- }
48
-
49
- export function deploymentStatusText(info: DeploymentInfo, status: DeployStatus) {
50
- if (info.configured === false || status === "setup") {
51
- return info.setupMessage ?? "部署設定尚未完成,請先設定 deploy.projectName";
52
- }
53
- if (status === "deploying") return "部署中";
54
- if (status === "failed") return "部署失敗,請查看終端機";
55
- if (status === "unavailable") return "目前環境沒有本地部署服務";
56
- if (isDeploymentDirty(info, status)) return "已上線但內容有更動,點擊重新部署";
57
- if (status === "deployed" || hasOnlineDeployment(info)) {
58
- return `已上線${info.deployedAt ? `,更新:${formatDeployTime(info.deployedAt)}` : ""}`;
59
- }
60
- return "未上線";
61
- }
62
-
63
- export function parseDeployError(text: string): {
64
- message?: string;
65
- deploy_configured?: boolean;
66
- deploy_adapter?: string;
67
- deploy_source?: string;
68
- deploy_project_name?: string;
69
- } | null {
70
- try {
71
- return JSON.parse(text) as {
72
- message?: string;
73
- deploy_configured?: boolean;
74
- deploy_adapter?: string;
75
- deploy_source?: string;
76
- deploy_project_name?: string;
77
- };
78
- } catch {
79
- return null;
80
- }
81
- }
82
-
83
- function deploymentStatusLabel(info: DeploymentInfo, status: DeployStatus) {
84
- if (info.configured === false || status === "setup") return "缺少設定";
85
- if (status === "deploying") return "正在部署";
86
- if (status === "failed") return "部署失敗";
87
- if (status === "unavailable") return "本機限定";
88
- if (isDeploymentDirty(info, status)) return "有更新";
89
- if (status === "deployed" || hasOnlineDeployment(info)) return "已上線";
90
- return "未上線";
91
- }
92
-
93
- function hasOnlineDeployment(info: DeploymentInfo) {
94
- if (info.configured === false) return false;
95
- return Boolean(info.online || info.deployedAt || info.publicUrl || (info.pdf && /^https?:\/\//i.test(info.pdf)));
96
- }
97
-
98
- function isDeploymentDirty(info: DeploymentInfo, status: DeployStatus) {
99
- return status === "idle" && hasOnlineDeployment(info) && info.dirty === true;
100
- }
101
-
102
- function formatDeployTime(value: string) {
103
- const date = new Date(value);
104
- if (Number.isNaN(date.getTime())) return "時間未知";
105
- return new Intl.DateTimeFormat("zh-TW", {
106
- month: "2-digit",
107
- day: "2-digit",
108
- hour: "2-digit",
109
- minute: "2-digit",
110
- hour12: false,
111
- }).format(date);
112
- }
@@ -1,5 +0,0 @@
1
- export * from "./deploymentStatusModel";
2
- export * from "./DeploymentControl";
3
- export * from "./PageZoomControl";
4
- export * from "./SearchControl";
5
- export * from "./useDeploymentWorkbench";