@beyondwork/docx-react-component 1.0.29 → 1.0.30
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.
- package/package.json +65 -96
- package/src/README.md +85 -0
- package/src/api/README.md +26 -0
- package/src/api/public-types.ts +1952 -0
- package/src/api/session-state.ts +62 -0
- package/src/compare/diff-engine.ts +623 -0
- package/src/compare/export-redlines.ts +280 -0
- package/src/compare/index.ts +25 -0
- package/src/compare/snapshot.ts +97 -0
- package/src/component-inventory.md +99 -0
- package/src/core/README.md +10 -0
- package/src/core/commands/README.md +3 -0
- package/{dist/chunk-TJBP2K4T.js → src/core/commands/formatting-commands.ts} +536 -196
- package/src/core/commands/image-commands.ts +373 -0
- package/src/core/commands/index.ts +1879 -0
- package/src/core/commands/list-commands.ts +565 -0
- package/src/core/commands/paragraph-layout-commands.ts +339 -0
- package/src/core/commands/review-commands.ts +108 -0
- package/{dist/core/commands/section-layout-commands.cjs → src/core/commands/section-layout-commands.ts} +340 -137
- package/src/core/commands/structural-helpers.ts +309 -0
- package/{dist/core/commands/style-commands.cjs → src/core/commands/style-commands.ts} +113 -65
- package/src/core/commands/table-structure-commands.ts +854 -0
- package/{dist/chunk-UZXBISGO.js → src/core/commands/text-commands.ts} +142 -86
- package/src/core/schema/README.md +3 -0
- package/src/core/schema/text-schema.ts +516 -0
- package/src/core/search/search-text.ts +357 -0
- package/src/core/selection/README.md +3 -0
- package/src/core/selection/mapping.ts +289 -0
- package/src/core/selection/review-anchors.ts +183 -0
- package/src/core/state/README.md +3 -0
- package/src/core/state/editor-state.ts +892 -0
- package/src/core/state/text-transaction.ts +869 -0
- package/src/formats/xlsx/io/parse-shared-strings.ts +41 -0
- package/src/formats/xlsx/io/parse-sheet.ts +459 -0
- package/src/formats/xlsx/io/parse-styles.ts +59 -0
- package/src/formats/xlsx/io/parse-workbook.ts +75 -0
- package/src/formats/xlsx/io/serialize-shared-strings.ts +72 -0
- package/src/formats/xlsx/io/serialize-sheet.ts +333 -0
- package/src/formats/xlsx/io/serialize-styles.ts +98 -0
- package/src/formats/xlsx/io/serialize-workbook.ts +429 -0
- package/src/formats/xlsx/io/xlsx-session.ts +314 -0
- package/src/formats/xlsx/model/cell.ts +189 -0
- package/src/formats/xlsx/model/sheet.ts +326 -0
- package/src/formats/xlsx/model/styles.ts +118 -0
- package/src/formats/xlsx/model/workbook.ts +453 -0
- package/src/formats/xlsx/runtime/cell-commands.ts +567 -0
- package/src/formats/xlsx/runtime/sheet-commands.ts +206 -0
- package/src/formats/xlsx/runtime/workbook-runtime.ts +177 -0
- package/src/formats/xlsx/runtime/workbook-transaction.ts +822 -0
- package/src/index.ts +142 -0
- package/src/io/README.md +10 -0
- package/src/io/docx-session.ts +3175 -0
- package/src/io/export/README.md +3 -0
- package/src/io/export/export-session.ts +220 -0
- package/src/io/export/minimal-docx.ts +115 -0
- package/src/io/export/reattach-preserved-parts.ts +54 -0
- package/src/io/export/serialize-comments.ts +947 -0
- package/src/io/export/serialize-footnotes.ts +394 -0
- package/src/io/export/serialize-headers-footers.ts +368 -0
- package/src/io/export/serialize-main-document.ts +1342 -0
- package/src/io/export/serialize-numbering.ts +218 -0
- package/src/io/export/serialize-revisions.ts +389 -0
- package/src/io/export/serialize-runtime-revisions.ts +463 -0
- package/src/io/export/serialize-tables.ts +174 -0
- package/src/io/export/split-review-boundaries.ts +356 -0
- package/src/io/export/split-story-blocks-for-runtime-revisions.ts +252 -0
- package/src/io/export/table-properties-xml.ts +318 -0
- package/src/io/normalize/README.md +3 -0
- package/src/io/normalize/normalize-text.ts +670 -0
- package/src/io/ooxml/README.md +3 -0
- package/src/io/ooxml/highlight-colors.ts +39 -0
- package/src/io/ooxml/numbering-sentinels.ts +44 -0
- package/src/io/ooxml/parse-comments.ts +852 -0
- package/src/io/ooxml/parse-complex-content.ts +287 -0
- package/src/io/ooxml/parse-fields.ts +834 -0
- package/src/io/ooxml/parse-footnotes.ts +952 -0
- package/src/io/ooxml/parse-headers-footers.ts +1212 -0
- package/src/io/ooxml/parse-inline-media.ts +461 -0
- package/src/io/ooxml/parse-main-document.ts +2947 -0
- package/src/io/ooxml/parse-numbering.ts +747 -0
- package/src/io/ooxml/parse-revisions.ts +1045 -0
- package/src/io/ooxml/parse-settings.ts +184 -0
- package/src/io/ooxml/parse-shapes.ts +296 -0
- package/src/io/ooxml/parse-styles.ts +639 -0
- package/src/io/ooxml/parse-tables.ts +627 -0
- package/src/io/ooxml/parse-theme.ts +346 -0
- package/src/io/ooxml/part-manifest.ts +136 -0
- package/src/io/ooxml/revision-boundaries.ts +475 -0
- package/src/io/ooxml/workflow-payload.ts +544 -0
- package/src/io/opc/README.md +3 -0
- package/src/io/opc/corrupt-package.ts +166 -0
- package/src/io/opc/docx-package.ts +74 -0
- package/src/io/opc/package-reader.ts +325 -0
- package/src/io/opc/package-writer.ts +273 -0
- package/src/io/source-package-provenance.ts +241 -0
- package/{dist/chunk-RMH72RZI.js → src/legal/bookmarks.ts} +130 -44
- package/src/legal/cross-references.ts +414 -0
- package/src/legal/defined-terms.ts +203 -0
- package/src/legal/index.ts +32 -0
- package/src/legal/signature-blocks.ts +259 -0
- package/src/model/README.md +3 -0
- package/src/model/canonical-document.ts +2722 -0
- package/src/model/cds-1.0.0.ts +212 -0
- package/src/model/snapshot.ts +760 -0
- package/src/preservation/README.md +3 -0
- package/src/preservation/markup-compatibility.ts +48 -0
- package/src/preservation/opaque-fragment-store.ts +89 -0
- package/src/preservation/opaque-region.ts +233 -0
- package/src/preservation/package-preservation.ts +113 -0
- package/src/preservation/preserved-part-manifest.ts +56 -0
- package/src/preservation/relationship-retention.ts +57 -0
- package/src/preservation/store.ts +255 -0
- package/src/review/README.md +16 -0
- package/src/review/store/README.md +3 -0
- package/src/review/store/comment-anchors.ts +70 -0
- package/src/review/store/comment-remapping.ts +154 -0
- package/src/review/store/comment-store.ts +349 -0
- package/src/review/store/comment-thread.ts +109 -0
- package/src/review/store/revision-actions.ts +423 -0
- package/src/review/store/revision-store.ts +323 -0
- package/src/review/store/revision-types.ts +182 -0
- package/src/review/store/runtime-comment-store.ts +43 -0
- package/src/runtime/README.md +3 -0
- package/src/runtime/ai-action-policy.ts +764 -0
- package/src/runtime/context-analytics.ts +824 -0
- package/src/runtime/document-layout.ts +332 -0
- package/src/runtime/document-locations.ts +521 -0
- package/src/runtime/document-navigation.ts +616 -0
- package/src/runtime/document-outline.ts +440 -0
- package/src/runtime/document-runtime.ts +4055 -0
- package/src/runtime/document-search.ts +145 -0
- package/src/runtime/event-refresh-hints.ts +137 -0
- package/src/runtime/numbering-prefix.ts +244 -0
- package/src/runtime/page-layout-estimation.ts +305 -0
- package/src/runtime/read-only-diagnostics-runtime.ts +241 -0
- package/src/runtime/resolved-numbering-geometry.ts +293 -0
- package/src/runtime/review-runtime.ts +44 -0
- package/src/runtime/revision-runtime.ts +107 -0
- package/src/runtime/session-capabilities.ts +192 -0
- package/src/runtime/story-context.ts +164 -0
- package/src/runtime/story-targeting.ts +162 -0
- package/src/runtime/suggestions-snapshot.ts +137 -0
- package/src/runtime/surface-projection.ts +1553 -0
- package/src/runtime/table-commands.ts +173 -0
- package/src/runtime/table-schema.ts +309 -0
- package/src/runtime/table-style-resolver.ts +409 -0
- package/src/runtime/view-state.ts +493 -0
- package/src/runtime/virtualized-rendering.ts +258 -0
- package/src/runtime/workflow-markup.ts +393 -0
- package/src/ui/README.md +30 -0
- package/src/ui/WordReviewEditor.tsx +5268 -0
- package/src/ui/browser-export.ts +52 -0
- package/src/ui/comments/README.md +3 -0
- package/src/ui/compatibility/README.md +3 -0
- package/src/ui/editor-command-bag.ts +127 -0
- package/src/ui/editor-runtime-boundary.ts +1558 -0
- package/src/ui/editor-shell-view.tsx +144 -0
- package/src/ui/editor-surface/README.md +3 -0
- package/src/ui/editor-surface-controller.tsx +66 -0
- package/src/ui/headless/comment-decoration-model.ts +124 -0
- package/src/ui/headless/preserve-editor-selection.ts +5 -0
- package/src/ui/headless/revision-decoration-model.ts +128 -0
- package/src/ui/headless/selection-helpers.ts +54 -0
- package/src/ui/headless/selection-tool-context.ts +19 -0
- package/src/ui/headless/selection-tool-resolver.ts +752 -0
- package/src/ui/headless/selection-tool-types.ts +129 -0
- package/src/ui/headless/selection-toolbar-model.ts +11 -0
- package/src/ui/headless/use-editor-keyboard.ts +103 -0
- package/src/ui/review/README.md +3 -0
- package/src/ui/runtime-shortcut-dispatch.ts +365 -0
- package/src/ui/runtime-snapshot-selectors.ts +197 -0
- package/src/ui/shared/revision-filters.ts +31 -0
- package/src/ui/status/README.md +3 -0
- package/src/ui/theme/README.md +3 -0
- package/src/ui/toolbar/README.md +3 -0
- package/src/ui/workflow-surface-blocked-rails.ts +94 -0
- package/src/ui-tailwind/chrome/chrome-preset-model.ts +107 -0
- package/src/ui-tailwind/chrome/chrome-preset-toolbar.tsx +15 -0
- package/src/ui-tailwind/chrome/review-queue-bar.tsx +97 -0
- package/src/ui-tailwind/chrome/tw-alert-banner.tsx +64 -0
- package/src/ui-tailwind/chrome/tw-context-analytics-summary.tsx +122 -0
- package/src/ui-tailwind/chrome/tw-image-context-toolbar.tsx +121 -0
- package/src/ui-tailwind/chrome/tw-layout-panel.tsx +114 -0
- package/src/ui-tailwind/chrome/tw-object-context-toolbar.tsx +30 -0
- package/src/ui-tailwind/chrome/tw-page-ruler.tsx +365 -0
- package/src/ui-tailwind/chrome/tw-selection-tool-blocked.tsx +23 -0
- package/src/ui-tailwind/chrome/tw-selection-tool-comment.tsx +35 -0
- package/src/ui-tailwind/chrome/tw-selection-tool-formatting.tsx +37 -0
- package/src/ui-tailwind/chrome/tw-selection-tool-host.tsx +298 -0
- package/src/ui-tailwind/chrome/tw-selection-tool-structure.tsx +116 -0
- package/src/ui-tailwind/chrome/tw-selection-tool-suggestion.tsx +29 -0
- package/src/ui-tailwind/chrome/tw-selection-tool-workflow.tsx +27 -0
- package/src/ui-tailwind/chrome/tw-selection-toolbar.tsx +186 -0
- package/src/ui-tailwind/chrome/tw-suggestion-card.tsx +139 -0
- package/src/ui-tailwind/chrome/tw-table-context-toolbar.tsx +200 -0
- package/src/ui-tailwind/chrome/tw-unsaved-modal.tsx +58 -0
- package/src/ui-tailwind/chrome/use-before-unload.ts +20 -0
- package/src/ui-tailwind/editor-surface/perf-probe.ts +179 -0
- package/src/ui-tailwind/editor-surface/pm-command-bridge.ts +189 -0
- package/src/ui-tailwind/editor-surface/pm-contextual-ui.ts +31 -0
- package/src/ui-tailwind/editor-surface/pm-decorations.ts +411 -0
- package/src/ui-tailwind/editor-surface/pm-position-map.ts +123 -0
- package/src/ui-tailwind/editor-surface/pm-schema.ts +927 -0
- package/src/ui-tailwind/editor-surface/pm-state-from-snapshot.ts +567 -0
- package/src/ui-tailwind/editor-surface/search-plugin.ts +168 -0
- package/src/ui-tailwind/editor-surface/surface-build-keys.ts +65 -0
- package/src/ui-tailwind/editor-surface/tw-caret.tsx +12 -0
- package/src/ui-tailwind/editor-surface/tw-editor-surface.tsx +150 -0
- package/src/ui-tailwind/editor-surface/tw-inline-token.tsx +129 -0
- package/src/ui-tailwind/editor-surface/tw-opaque-block.tsx +58 -0
- package/src/ui-tailwind/editor-surface/tw-paragraph-block.tsx +151 -0
- package/src/ui-tailwind/editor-surface/tw-prosemirror-surface.tsx +1047 -0
- package/src/ui-tailwind/editor-surface/tw-segment-view.tsx +111 -0
- package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +503 -0
- package/src/ui-tailwind/index.ts +62 -0
- package/src/ui-tailwind/page-chrome-model.ts +27 -0
- package/src/ui-tailwind/review/tw-comment-sidebar.tsx +406 -0
- package/src/ui-tailwind/review/tw-health-panel.tsx +149 -0
- package/src/ui-tailwind/review/tw-review-rail.tsx +122 -0
- package/src/ui-tailwind/review/tw-revision-sidebar.tsx +164 -0
- package/src/ui-tailwind/status/tw-status-bar.tsx +65 -0
- package/{dist → src}/ui-tailwind/theme/editor-theme.css +58 -40
- package/src/ui-tailwind/toolbar/tw-toolbar-icon-button.tsx +52 -0
- package/src/ui-tailwind/toolbar/tw-toolbar.tsx +1133 -0
- package/src/ui-tailwind/tw-review-workspace.tsx +1460 -0
- package/src/validation/README.md +3 -0
- package/src/validation/compatibility-engine.ts +878 -0
- package/src/validation/compatibility-report.ts +161 -0
- package/src/validation/diagnostics.ts +204 -0
- package/src/validation/docx-comment-proof.ts +720 -0
- package/src/validation/import-diagnostics.ts +128 -0
- package/src/validation/low-priority-word-surfaces.ts +373 -0
- package/dist/canonical-document-BLEbzL2J.d.cts +0 -844
- package/dist/canonical-document-BLEbzL2J.d.ts +0 -844
- package/dist/chunk-2FJS5GZM.js +0 -763
- package/dist/chunk-2FJS5GZM.js.map +0 -1
- package/dist/chunk-2OQBZS3F.js +0 -446
- package/dist/chunk-2OQBZS3F.js.map +0 -1
- package/dist/chunk-2S7W4KFO.js +0 -127
- package/dist/chunk-2S7W4KFO.js.map +0 -1
- package/dist/chunk-2TG72QSW.js +0 -3874
- package/dist/chunk-2TG72QSW.js.map +0 -1
- package/dist/chunk-36QNIZBO.js +0 -532
- package/dist/chunk-36QNIZBO.js.map +0 -1
- package/dist/chunk-4AQOYAW4.js +0 -3069
- package/dist/chunk-4AQOYAW4.js.map +0 -1
- package/dist/chunk-4D5EWJ3P.js +0 -77
- package/dist/chunk-4D5EWJ3P.js.map +0 -1
- package/dist/chunk-5FN54NDH.js +0 -2257
- package/dist/chunk-5FN54NDH.js.map +0 -1
- package/dist/chunk-BOYGQYRQ.js +0 -7306
- package/dist/chunk-BOYGQYRQ.js.map +0 -1
- package/dist/chunk-CN3XMECL.js +0 -212
- package/dist/chunk-CN3XMECL.js.map +0 -1
- package/dist/chunk-EBI3BX6U.js +0 -164
- package/dist/chunk-EBI3BX6U.js.map +0 -1
- package/dist/chunk-EILUG3VB.js +0 -1275
- package/dist/chunk-EILUG3VB.js.map +0 -1
- package/dist/chunk-FUDY333O.js +0 -70
- package/dist/chunk-FUDY333O.js.map +0 -1
- package/dist/chunk-GBVOWFIK.js +0 -1237
- package/dist/chunk-GBVOWFIK.js.map +0 -1
- package/dist/chunk-H4TQ3H3Y.js +0 -262
- package/dist/chunk-H4TQ3H3Y.js.map +0 -1
- package/dist/chunk-JGB3IXZO.js +0 -189
- package/dist/chunk-JGB3IXZO.js.map +0 -1
- package/dist/chunk-KD2QRQPY.js +0 -4342
- package/dist/chunk-KD2QRQPY.js.map +0 -1
- package/dist/chunk-KLMXQVYK.js +0 -369
- package/dist/chunk-KLMXQVYK.js.map +0 -1
- package/dist/chunk-KZUG5KFQ.js +0 -214
- package/dist/chunk-KZUG5KFQ.js.map +0 -1
- package/dist/chunk-QDAQ4CJU.js +0 -345
- package/dist/chunk-QDAQ4CJU.js.map +0 -1
- package/dist/chunk-RMH72RZI.js.map +0 -1
- package/dist/chunk-SWKWQZXM.js +0 -117
- package/dist/chunk-SWKWQZXM.js.map +0 -1
- package/dist/chunk-TJBP2K4T.js.map +0 -1
- package/dist/chunk-TLCEAQDQ.js +0 -542
- package/dist/chunk-TLCEAQDQ.js.map +0 -1
- package/dist/chunk-UZXBISGO.js.map +0 -1
- package/dist/chunk-WGBAKP3Q.js +0 -3220
- package/dist/chunk-WGBAKP3Q.js.map +0 -1
- package/dist/compare/index.cjs +0 -5475
- package/dist/compare/index.cjs.map +0 -1
- package/dist/compare/index.d.cts +0 -114
- package/dist/compare/index.d.ts +0 -114
- package/dist/compare/index.js +0 -731
- package/dist/compare/index.js.map +0 -1
- package/dist/core/commands/formatting-commands.cjs +0 -828
- package/dist/core/commands/formatting-commands.cjs.map +0 -1
- package/dist/core/commands/formatting-commands.d.cts +0 -63
- package/dist/core/commands/formatting-commands.d.ts +0 -63
- package/dist/core/commands/formatting-commands.js +0 -37
- package/dist/core/commands/formatting-commands.js.map +0 -1
- package/dist/core/commands/image-commands.cjs +0 -2023
- package/dist/core/commands/image-commands.cjs.map +0 -1
- package/dist/core/commands/image-commands.d.cts +0 -58
- package/dist/core/commands/image-commands.d.ts +0 -58
- package/dist/core/commands/image-commands.js +0 -18
- package/dist/core/commands/image-commands.js.map +0 -1
- package/dist/core/commands/section-layout-commands.cjs.map +0 -1
- package/dist/core/commands/section-layout-commands.d.cts +0 -62
- package/dist/core/commands/section-layout-commands.d.ts +0 -62
- package/dist/core/commands/section-layout-commands.js +0 -21
- package/dist/core/commands/section-layout-commands.js.map +0 -1
- package/dist/core/commands/style-commands.cjs.map +0 -1
- package/dist/core/commands/style-commands.d.cts +0 -13
- package/dist/core/commands/style-commands.d.ts +0 -13
- package/dist/core/commands/style-commands.js +0 -9
- package/dist/core/commands/style-commands.js.map +0 -1
- package/dist/core/commands/table-structure-commands.cjs +0 -1883
- package/dist/core/commands/table-structure-commands.cjs.map +0 -1
- package/dist/core/commands/table-structure-commands.d.cts +0 -59
- package/dist/core/commands/table-structure-commands.d.ts +0 -59
- package/dist/core/commands/table-structure-commands.js +0 -12
- package/dist/core/commands/table-structure-commands.js.map +0 -1
- package/dist/core/commands/text-commands.cjs +0 -2391
- package/dist/core/commands/text-commands.cjs.map +0 -1
- package/dist/core/commands/text-commands.d.cts +0 -24
- package/dist/core/commands/text-commands.d.ts +0 -24
- package/dist/core/commands/text-commands.js +0 -28
- package/dist/core/commands/text-commands.js.map +0 -1
- package/dist/core/selection/mapping.cjs +0 -200
- package/dist/core/selection/mapping.cjs.map +0 -1
- package/dist/core/selection/mapping.d.cts +0 -2
- package/dist/core/selection/mapping.d.ts +0 -2
- package/dist/core/selection/mapping.js +0 -31
- package/dist/core/selection/mapping.js.map +0 -1
- package/dist/core/state/editor-state.cjs +0 -2278
- package/dist/core/state/editor-state.cjs.map +0 -1
- package/dist/core/state/editor-state.d.cts +0 -2
- package/dist/core/state/editor-state.d.ts +0 -2
- package/dist/core/state/editor-state.js +0 -26
- package/dist/core/state/editor-state.js.map +0 -1
- package/dist/index.cjs +0 -38553
- package/dist/index.cjs.map +0 -1
- package/dist/index.d.cts +0 -15
- package/dist/index.d.ts +0 -15
- package/dist/index.js +0 -7856
- package/dist/index.js.map +0 -1
- package/dist/io/docx-session.cjs +0 -16236
- package/dist/io/docx-session.cjs.map +0 -1
- package/dist/io/docx-session.d.cts +0 -21
- package/dist/io/docx-session.d.ts +0 -21
- package/dist/io/docx-session.js +0 -18
- package/dist/io/docx-session.js.map +0 -1
- package/dist/legal/index.cjs +0 -3900
- package/dist/legal/index.cjs.map +0 -1
- package/dist/legal/index.d.cts +0 -86
- package/dist/legal/index.d.ts +0 -86
- package/dist/legal/index.js +0 -616
- package/dist/legal/index.js.map +0 -1
- package/dist/public-types-7ZL_94cz.d.ts +0 -1573
- package/dist/public-types-CeMaDueh.d.cts +0 -1573
- package/dist/public-types.cjs +0 -19
- package/dist/public-types.cjs.map +0 -1
- package/dist/public-types.d.cts +0 -2
- package/dist/public-types.d.ts +0 -2
- package/dist/public-types.js +0 -1
- package/dist/public-types.js.map +0 -1
- package/dist/runtime/document-runtime.cjs +0 -11140
- package/dist/runtime/document-runtime.cjs.map +0 -1
- package/dist/runtime/document-runtime.d.cts +0 -231
- package/dist/runtime/document-runtime.d.ts +0 -231
- package/dist/runtime/document-runtime.js +0 -21
- package/dist/runtime/document-runtime.js.map +0 -1
- package/dist/structural-helpers-CilgOVhh.d.cts +0 -10
- package/dist/structural-helpers-q0Gd-eBN.d.ts +0 -10
- package/dist/ui-tailwind/editor-surface/search-plugin.cjs +0 -313
- package/dist/ui-tailwind/editor-surface/search-plugin.cjs.map +0 -1
- package/dist/ui-tailwind/editor-surface/search-plugin.d.cts +0 -67
- package/dist/ui-tailwind/editor-surface/search-plugin.d.ts +0 -67
- package/dist/ui-tailwind/editor-surface/search-plugin.js +0 -23
- package/dist/ui-tailwind/editor-surface/search-plugin.js.map +0 -1
- package/dist/ui-tailwind/index.cjs +0 -4833
- package/dist/ui-tailwind/index.cjs.map +0 -1
- package/dist/ui-tailwind/index.d.cts +0 -617
- package/dist/ui-tailwind/index.d.ts +0 -617
- package/dist/ui-tailwind/index.js +0 -575
- package/dist/ui-tailwind/index.js.map +0 -1
|
@@ -0,0 +1,293 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
NumberingCatalog,
|
|
3
|
+
NumberingLevelDefinition,
|
|
4
|
+
NumberingLevelOverrideDefinition,
|
|
5
|
+
NumberingLevelParagraphGeometry,
|
|
6
|
+
ParagraphIndentation,
|
|
7
|
+
ParagraphNode,
|
|
8
|
+
ParagraphSpacing,
|
|
9
|
+
TabStop,
|
|
10
|
+
} from "../model/canonical-document.ts";
|
|
11
|
+
|
|
12
|
+
export const DEFAULT_NUMBERING_START_AT = 1;
|
|
13
|
+
|
|
14
|
+
export interface ResolvedNumberingGeometry {
|
|
15
|
+
markerJustification?: NumberingLevelParagraphGeometry["justification"];
|
|
16
|
+
spacing?: ParagraphSpacing;
|
|
17
|
+
indentation?: ParagraphIndentation;
|
|
18
|
+
tabStops?: TabStop[];
|
|
19
|
+
markerLane?: {
|
|
20
|
+
start: number;
|
|
21
|
+
width: number;
|
|
22
|
+
textStart: number;
|
|
23
|
+
};
|
|
24
|
+
textColumn?: {
|
|
25
|
+
start: number;
|
|
26
|
+
right?: number;
|
|
27
|
+
firstLine?: number;
|
|
28
|
+
hanging?: number;
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
export interface ResolvedNumberingDefinitionSet {
|
|
33
|
+
instance: NumberingCatalog["instances"][string];
|
|
34
|
+
abstractDefinition: NumberingCatalog["abstractDefinitions"][string];
|
|
35
|
+
effectiveLevels: Map<number, NumberingLevelDefinition>;
|
|
36
|
+
effectiveLevel: NumberingLevelDefinition;
|
|
37
|
+
geometry: ResolvedNumberingGeometry;
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
export function resolveNumberingDefinitionSet(
|
|
41
|
+
catalog: NumberingCatalog,
|
|
42
|
+
numbering: ParagraphNode["numbering"] | undefined,
|
|
43
|
+
paragraph?: Pick<ParagraphNode, "spacing" | "indentation" | "tabStops">,
|
|
44
|
+
): ResolvedNumberingDefinitionSet | null {
|
|
45
|
+
if (!numbering) {
|
|
46
|
+
return null;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
const instance = catalog.instances[numbering.numberingInstanceId];
|
|
50
|
+
if (!instance) {
|
|
51
|
+
return null;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
const abstractDefinition = catalog.abstractDefinitions[instance.abstractNumberingId];
|
|
55
|
+
if (!abstractDefinition) {
|
|
56
|
+
return null;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
const effectiveLevels = buildEffectiveLevelDefinitions(abstractDefinition.levels, instance.overrides);
|
|
60
|
+
const effectiveLevel = effectiveLevels.get(numbering.level);
|
|
61
|
+
if (!effectiveLevel) {
|
|
62
|
+
return null;
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
return {
|
|
66
|
+
instance,
|
|
67
|
+
abstractDefinition,
|
|
68
|
+
effectiveLevels,
|
|
69
|
+
effectiveLevel,
|
|
70
|
+
geometry: resolveNumberingGeometry(effectiveLevel.paragraphGeometry, paragraph),
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
function buildEffectiveLevelDefinitions(
|
|
75
|
+
abstractLevels: readonly NumberingLevelDefinition[],
|
|
76
|
+
overrides: NumberingCatalog["instances"][string]["overrides"],
|
|
77
|
+
): Map<number, NumberingLevelDefinition> {
|
|
78
|
+
const effectiveLevels = new Map<number, NumberingLevelDefinition>();
|
|
79
|
+
|
|
80
|
+
for (const level of abstractLevels) {
|
|
81
|
+
effectiveLevels.set(level.level, withDefaultStartAt(level));
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
for (const override of overrides) {
|
|
85
|
+
const base = effectiveLevels.get(override.level);
|
|
86
|
+
const merged = mergeLevelDefinition(base, override.levelDefinition, override.startAt, override.level);
|
|
87
|
+
if (merged) {
|
|
88
|
+
effectiveLevels.set(override.level, merged);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
return effectiveLevels;
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
function mergeLevelDefinition(
|
|
96
|
+
base: NumberingLevelDefinition | undefined,
|
|
97
|
+
override: NumberingLevelOverrideDefinition | undefined,
|
|
98
|
+
startOverride: number | undefined,
|
|
99
|
+
fallbackLevel: number,
|
|
100
|
+
): NumberingLevelDefinition | null {
|
|
101
|
+
if (!base && !override) {
|
|
102
|
+
return null;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
const level = base?.level ?? override?.level ?? fallbackLevel;
|
|
106
|
+
const format = override?.format ?? base?.format ?? "decimal";
|
|
107
|
+
const text = override?.text ?? base?.text ?? `%${level + 1}.`;
|
|
108
|
+
const startAt = startOverride ?? override?.startAt ?? base?.startAt ?? DEFAULT_NUMBERING_START_AT;
|
|
109
|
+
const paragraphGeometry = mergeLevelParagraphGeometry(
|
|
110
|
+
base?.paragraphGeometry,
|
|
111
|
+
override?.paragraphGeometry,
|
|
112
|
+
);
|
|
113
|
+
|
|
114
|
+
return {
|
|
115
|
+
level,
|
|
116
|
+
format,
|
|
117
|
+
text,
|
|
118
|
+
startAt,
|
|
119
|
+
...(override?.paragraphStyleId ?? base?.paragraphStyleId
|
|
120
|
+
? { paragraphStyleId: override?.paragraphStyleId ?? base?.paragraphStyleId }
|
|
121
|
+
: {}),
|
|
122
|
+
...((override?.isLegalNumbering ?? base?.isLegalNumbering) === true
|
|
123
|
+
? { isLegalNumbering: true }
|
|
124
|
+
: {}),
|
|
125
|
+
...(override?.suffix ?? base?.suffix ? { suffix: override?.suffix ?? base?.suffix } : {}),
|
|
126
|
+
...(paragraphGeometry ? { paragraphGeometry } : {}),
|
|
127
|
+
};
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
function withDefaultStartAt(level: NumberingLevelDefinition): NumberingLevelDefinition {
|
|
131
|
+
return {
|
|
132
|
+
...level,
|
|
133
|
+
startAt: level.startAt ?? DEFAULT_NUMBERING_START_AT,
|
|
134
|
+
...(level.paragraphGeometry
|
|
135
|
+
? { paragraphGeometry: cloneLevelParagraphGeometry(level.paragraphGeometry) }
|
|
136
|
+
: {}),
|
|
137
|
+
};
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
function cloneLevelParagraphGeometry(
|
|
141
|
+
geometry: NumberingLevelParagraphGeometry,
|
|
142
|
+
): NumberingLevelParagraphGeometry {
|
|
143
|
+
return {
|
|
144
|
+
...(geometry.justification ? { justification: geometry.justification } : {}),
|
|
145
|
+
...(geometry.spacing ? { spacing: { ...geometry.spacing } } : {}),
|
|
146
|
+
...(geometry.indentation ? { indentation: { ...geometry.indentation } } : {}),
|
|
147
|
+
...(geometry.tabStops && geometry.tabStops.length > 0
|
|
148
|
+
? { tabStops: cloneTabStops(geometry.tabStops) }
|
|
149
|
+
: {}),
|
|
150
|
+
};
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
function mergeLevelParagraphGeometry(
|
|
154
|
+
base: NumberingLevelParagraphGeometry | undefined,
|
|
155
|
+
override: NumberingLevelParagraphGeometry | undefined,
|
|
156
|
+
): NumberingLevelParagraphGeometry | undefined {
|
|
157
|
+
if (!base && !override) {
|
|
158
|
+
return undefined;
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
const spacing = mergeParagraphSpacing(base?.spacing, override?.spacing);
|
|
162
|
+
const indentation = mergeParagraphIndentation(base?.indentation, override?.indentation);
|
|
163
|
+
const tabStops = override?.tabStops?.length
|
|
164
|
+
? cloneTabStops(override.tabStops)
|
|
165
|
+
: base?.tabStops?.length
|
|
166
|
+
? cloneTabStops(base.tabStops)
|
|
167
|
+
: undefined;
|
|
168
|
+
|
|
169
|
+
return {
|
|
170
|
+
...(override?.justification ?? base?.justification
|
|
171
|
+
? { justification: override?.justification ?? base?.justification }
|
|
172
|
+
: {}),
|
|
173
|
+
...(spacing ? { spacing } : {}),
|
|
174
|
+
...(indentation ? { indentation } : {}),
|
|
175
|
+
...(tabStops && tabStops.length > 0 ? { tabStops } : {}),
|
|
176
|
+
};
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
function resolveNumberingGeometry(
|
|
180
|
+
levelGeometry: NumberingLevelParagraphGeometry | undefined,
|
|
181
|
+
paragraph: Pick<ParagraphNode, "spacing" | "indentation" | "tabStops"> | undefined,
|
|
182
|
+
): ResolvedNumberingGeometry {
|
|
183
|
+
const spacing = mergeParagraphSpacing(levelGeometry?.spacing, paragraph?.spacing);
|
|
184
|
+
const indentation = mergeParagraphIndentation(levelGeometry?.indentation, paragraph?.indentation);
|
|
185
|
+
const tabStops = paragraph?.tabStops?.length
|
|
186
|
+
? cloneTabStops(paragraph.tabStops)
|
|
187
|
+
: levelGeometry?.tabStops?.length
|
|
188
|
+
? cloneTabStops(levelGeometry.tabStops)
|
|
189
|
+
: undefined;
|
|
190
|
+
const markerLane = deriveMarkerLane(indentation);
|
|
191
|
+
const textColumn = deriveTextColumn(indentation);
|
|
192
|
+
|
|
193
|
+
return {
|
|
194
|
+
...(levelGeometry?.justification ? { markerJustification: levelGeometry.justification } : {}),
|
|
195
|
+
...(spacing ? { spacing } : {}),
|
|
196
|
+
...(indentation ? { indentation } : {}),
|
|
197
|
+
...(tabStops && tabStops.length > 0 ? { tabStops } : {}),
|
|
198
|
+
...(markerLane ? { markerLane } : {}),
|
|
199
|
+
...(textColumn ? { textColumn } : {}),
|
|
200
|
+
};
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
function mergeParagraphSpacing(
|
|
204
|
+
base: ParagraphSpacing | undefined,
|
|
205
|
+
override: ParagraphSpacing | undefined,
|
|
206
|
+
): ParagraphSpacing | undefined {
|
|
207
|
+
if (!base && !override) {
|
|
208
|
+
return undefined;
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
const spacing: ParagraphSpacing = {
|
|
212
|
+
...(base ?? {}),
|
|
213
|
+
...(override ?? {}),
|
|
214
|
+
};
|
|
215
|
+
|
|
216
|
+
return Object.keys(spacing).length > 0 ? spacing : undefined;
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
function mergeParagraphIndentation(
|
|
220
|
+
base: ParagraphIndentation | undefined,
|
|
221
|
+
override: ParagraphIndentation | undefined,
|
|
222
|
+
): ParagraphIndentation | undefined {
|
|
223
|
+
if (!base && !override) {
|
|
224
|
+
return undefined;
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
const indentation: ParagraphIndentation = {
|
|
228
|
+
...(base ?? {}),
|
|
229
|
+
...(override ?? {}),
|
|
230
|
+
};
|
|
231
|
+
|
|
232
|
+
return Object.keys(indentation).length > 0 ? indentation : undefined;
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
function cloneTabStops(tabStops: readonly TabStop[]): TabStop[] {
|
|
236
|
+
return tabStops.map((tabStop) => ({ ...tabStop }));
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
function deriveMarkerLane(
|
|
240
|
+
indentation: ParagraphIndentation | undefined,
|
|
241
|
+
): ResolvedNumberingGeometry["markerLane"] | undefined {
|
|
242
|
+
if (!indentation) {
|
|
243
|
+
return undefined;
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
const width = resolveHangingWidth(indentation);
|
|
247
|
+
if (width === undefined || width <= 0) {
|
|
248
|
+
return undefined;
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
const textStart = indentation.left ?? 0;
|
|
252
|
+
return {
|
|
253
|
+
start: textStart - width,
|
|
254
|
+
width,
|
|
255
|
+
textStart,
|
|
256
|
+
};
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
function deriveTextColumn(
|
|
260
|
+
indentation: ParagraphIndentation | undefined,
|
|
261
|
+
): ResolvedNumberingGeometry["textColumn"] | undefined {
|
|
262
|
+
if (!indentation) {
|
|
263
|
+
return undefined;
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
const hanging = resolveHangingWidth(indentation);
|
|
267
|
+
const hasGeometry =
|
|
268
|
+
indentation.left !== undefined ||
|
|
269
|
+
indentation.right !== undefined ||
|
|
270
|
+
indentation.firstLine !== undefined ||
|
|
271
|
+
hanging !== undefined;
|
|
272
|
+
|
|
273
|
+
if (!hasGeometry) {
|
|
274
|
+
return undefined;
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
return {
|
|
278
|
+
start: indentation.left ?? 0,
|
|
279
|
+
...(indentation.right !== undefined ? { right: indentation.right } : {}),
|
|
280
|
+
...(indentation.firstLine !== undefined ? { firstLine: indentation.firstLine } : {}),
|
|
281
|
+
...(hanging !== undefined ? { hanging } : {}),
|
|
282
|
+
};
|
|
283
|
+
}
|
|
284
|
+
|
|
285
|
+
function resolveHangingWidth(indentation: ParagraphIndentation): number | undefined {
|
|
286
|
+
if (indentation.hanging !== undefined) {
|
|
287
|
+
return indentation.hanging;
|
|
288
|
+
}
|
|
289
|
+
if (indentation.firstLine !== undefined && indentation.firstLine < 0) {
|
|
290
|
+
return Math.abs(indentation.firstLine);
|
|
291
|
+
}
|
|
292
|
+
return undefined;
|
|
293
|
+
}
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
CommentThreadRecord,
|
|
3
|
+
EditorState,
|
|
4
|
+
EditorWarning,
|
|
5
|
+
} from "../core/state/editor-state.ts";
|
|
6
|
+
import type { TransactionMapping } from "../core/selection/mapping.ts";
|
|
7
|
+
import { remapCommentThreads } from "../review/store/comment-remapping.ts";
|
|
8
|
+
|
|
9
|
+
export interface ApplyReviewRuntimeMappingOptions {
|
|
10
|
+
state: Pick<EditorState, "document" | "warnings" | "runtime">;
|
|
11
|
+
mapping: TransactionMapping;
|
|
12
|
+
nextContent: unknown;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export interface ReviewRuntimeMappingResult {
|
|
16
|
+
comments: Record<string, CommentThreadRecord>;
|
|
17
|
+
warnings: EditorWarning[];
|
|
18
|
+
activeCommentId?: string;
|
|
19
|
+
detachedCommentIds: string[];
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
export function applyReviewRuntimeMapping(
|
|
23
|
+
options: ApplyReviewRuntimeMappingOptions,
|
|
24
|
+
): ReviewRuntimeMappingResult {
|
|
25
|
+
const remapped = remapCommentThreads({
|
|
26
|
+
comments: options.state.document.review.comments,
|
|
27
|
+
mapping: options.mapping,
|
|
28
|
+
nextContent: options.nextContent,
|
|
29
|
+
existingWarnings: options.state.warnings,
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
const activeCommentId =
|
|
33
|
+
options.state.runtime.activeCommentId &&
|
|
34
|
+
remapped.comments[options.state.runtime.activeCommentId]
|
|
35
|
+
? options.state.runtime.activeCommentId
|
|
36
|
+
: undefined;
|
|
37
|
+
|
|
38
|
+
return {
|
|
39
|
+
comments: remapped.comments,
|
|
40
|
+
warnings: remapped.warnings,
|
|
41
|
+
activeCommentId,
|
|
42
|
+
detachedCommentIds: remapped.detachedCommentIds,
|
|
43
|
+
};
|
|
44
|
+
}
|
|
@@ -0,0 +1,107 @@
|
|
|
1
|
+
import type { CanonicalDocumentEnvelope } from "../core/state/editor-state.ts";
|
|
2
|
+
import {
|
|
3
|
+
getReviewCommandIntent,
|
|
4
|
+
isSingleRevisionReviewCommand,
|
|
5
|
+
type ReviewCommand,
|
|
6
|
+
} from "../core/commands/review-commands.ts";
|
|
7
|
+
import type { TransactionMapping } from "../core/selection/mapping.ts";
|
|
8
|
+
import {
|
|
9
|
+
applyRevisionAction,
|
|
10
|
+
type RevisionActionOutcome,
|
|
11
|
+
} from "../review/store/revision-actions.ts";
|
|
12
|
+
import { type RevisionStore } from "../review/store/revision-store.ts";
|
|
13
|
+
|
|
14
|
+
export interface RevisionRuntimeState {
|
|
15
|
+
document: CanonicalDocumentEnvelope;
|
|
16
|
+
store: RevisionStore;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export interface ApplyRevisionRuntimeCommandOptions {
|
|
20
|
+
state: RevisionRuntimeState;
|
|
21
|
+
command: ReviewCommand;
|
|
22
|
+
timestamp: string;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
export interface RevisionRuntimeCommandEffects {
|
|
26
|
+
appliedRevisionIds: string[];
|
|
27
|
+
skippedRevisions: RevisionActionOutcome[];
|
|
28
|
+
detachedRevisionIds: string[];
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export interface RevisionRuntimeCommandResult extends RevisionRuntimeState {
|
|
32
|
+
outcomes: RevisionActionOutcome[];
|
|
33
|
+
mappings: Array<{ revisionId: string; mapping: TransactionMapping; steps: number }>;
|
|
34
|
+
effects: RevisionRuntimeCommandEffects;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export function applyRevisionRuntimeCommand(
|
|
38
|
+
options: ApplyRevisionRuntimeCommandOptions,
|
|
39
|
+
): RevisionRuntimeCommandResult {
|
|
40
|
+
const revisionIds = isSingleRevisionReviewCommand(options.command)
|
|
41
|
+
? [options.command.revisionId]
|
|
42
|
+
: listBatchRevisionIds(options.state.store);
|
|
43
|
+
|
|
44
|
+
const outcomes: RevisionActionOutcome[] = [];
|
|
45
|
+
const mappings: Array<{ revisionId: string; mapping: TransactionMapping; steps: number }> = [];
|
|
46
|
+
const appliedRevisionIds: string[] = [];
|
|
47
|
+
const detachedRevisionIds = new Set<string>();
|
|
48
|
+
|
|
49
|
+
let state = options.state;
|
|
50
|
+
|
|
51
|
+
for (const revisionId of revisionIds) {
|
|
52
|
+
const result = applyRevisionAction({
|
|
53
|
+
document: state.document,
|
|
54
|
+
store: state.store,
|
|
55
|
+
revisionId,
|
|
56
|
+
intent: getReviewCommandIntent(options.command),
|
|
57
|
+
timestamp: options.timestamp,
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
state = {
|
|
61
|
+
document: result.document,
|
|
62
|
+
store: result.store,
|
|
63
|
+
};
|
|
64
|
+
outcomes.push(result.outcome);
|
|
65
|
+
mappings.push({
|
|
66
|
+
revisionId,
|
|
67
|
+
mapping: result.mapping,
|
|
68
|
+
steps: result.mapping.steps.length,
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
if (result.outcome.kind === "applied") {
|
|
72
|
+
appliedRevisionIds.push(revisionId);
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
for (const detachedRevisionId of result.detachedRevisionIds) {
|
|
76
|
+
detachedRevisionIds.add(detachedRevisionId);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
return {
|
|
81
|
+
...state,
|
|
82
|
+
outcomes,
|
|
83
|
+
mappings,
|
|
84
|
+
effects: {
|
|
85
|
+
appliedRevisionIds,
|
|
86
|
+
skippedRevisions: outcomes.filter(
|
|
87
|
+
(outcome): outcome is Extract<RevisionActionOutcome, { kind: "skipped" }> =>
|
|
88
|
+
outcome.kind === "skipped",
|
|
89
|
+
),
|
|
90
|
+
detachedRevisionIds: [...detachedRevisionIds],
|
|
91
|
+
},
|
|
92
|
+
};
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
function listBatchRevisionIds(store: RevisionStore): string[] {
|
|
96
|
+
return Object.values(store.revisions)
|
|
97
|
+
.filter((revision) => revision.status === "active")
|
|
98
|
+
.sort((left, right) => {
|
|
99
|
+
const createdAtDelta = left.createdAt.localeCompare(right.createdAt);
|
|
100
|
+
if (createdAtDelta !== 0) {
|
|
101
|
+
return createdAtDelta;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
return left.revisionId.localeCompare(right.revisionId);
|
|
105
|
+
})
|
|
106
|
+
.map((revision) => revision.revisionId);
|
|
107
|
+
}
|
|
@@ -0,0 +1,192 @@
|
|
|
1
|
+
import type { RuntimeRenderSnapshot, WorkflowScopeSnapshot } from "../api/public-types";
|
|
2
|
+
import {
|
|
3
|
+
createDetachedAnchor,
|
|
4
|
+
createNodeAnchor,
|
|
5
|
+
createRangeAnchor,
|
|
6
|
+
} from "../core/selection/mapping.ts";
|
|
7
|
+
import { canCreateDocxCommentAnchor } from "../core/selection/review-anchors";
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Session capabilities derived from the runtime snapshot.
|
|
11
|
+
*
|
|
12
|
+
* All fields are computed from `RuntimeRenderSnapshot` — no local component
|
|
13
|
+
* state. The UI reads these to decide what controls to enable/disable, what
|
|
14
|
+
* chrome to show, and what mode the editor is in.
|
|
15
|
+
*/
|
|
16
|
+
export interface SessionCapabilities {
|
|
17
|
+
// ── Session phase ──
|
|
18
|
+
/** Current lifecycle phase of the editor session. */
|
|
19
|
+
phase: "loading" | "ready" | "diagnostics" | "error";
|
|
20
|
+
|
|
21
|
+
/** Effective editor mode after accounting for runtime state. */
|
|
22
|
+
mode: "editing" | "review" | "read-only-diagnostics";
|
|
23
|
+
|
|
24
|
+
// ── Document mode ──
|
|
25
|
+
/** Runtime document mode — editing authority, distinct from view/workspace mode. */
|
|
26
|
+
documentMode: "editing" | "suggesting" | "viewing";
|
|
27
|
+
|
|
28
|
+
// ── Command capabilities ──
|
|
29
|
+
canUndo: boolean;
|
|
30
|
+
canRedo: boolean;
|
|
31
|
+
canEdit: boolean;
|
|
32
|
+
canAddComment: boolean;
|
|
33
|
+
canExport: boolean;
|
|
34
|
+
canAcceptChange: boolean;
|
|
35
|
+
canRejectChange: boolean;
|
|
36
|
+
canAcceptAll: boolean;
|
|
37
|
+
canRejectAll: boolean;
|
|
38
|
+
|
|
39
|
+
// ── Review visibility ──
|
|
40
|
+
/** Whether the review rail should be visible by default. */
|
|
41
|
+
reviewRailVisible: boolean;
|
|
42
|
+
|
|
43
|
+
/** Whether tracked changes exist in the document (display toggle is meaningful). */
|
|
44
|
+
trackChangesSupported: boolean;
|
|
45
|
+
|
|
46
|
+
// ── Trust posture ──
|
|
47
|
+
exportBlocked: boolean;
|
|
48
|
+
preserveOnlyCount: number;
|
|
49
|
+
unsupportedFatalCount: number;
|
|
50
|
+
|
|
51
|
+
// ── Protection posture ──
|
|
52
|
+
/** Whether the source package declares document-level protection. */
|
|
53
|
+
hasDocumentProtection: boolean;
|
|
54
|
+
/** Count of permission ranges from the source package. */
|
|
55
|
+
protectedRangeCount: number;
|
|
56
|
+
|
|
57
|
+
// ── Health ──
|
|
58
|
+
/** Total count of health issues (preserve-only + unsupported-fatal + warnings). */
|
|
59
|
+
healthIssueCount: number;
|
|
60
|
+
|
|
61
|
+
// ── Workflow ──
|
|
62
|
+
/** Whether a workflow overlay is currently applied. */
|
|
63
|
+
workflowOverlayPresent: boolean;
|
|
64
|
+
/** Whether the current selection is blocked by workflow scope enforcement. */
|
|
65
|
+
workflowBlocked: boolean;
|
|
66
|
+
|
|
67
|
+
// ── Status ──
|
|
68
|
+
isDirty: boolean;
|
|
69
|
+
isReady: boolean;
|
|
70
|
+
hasFatalError: boolean;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
/**
|
|
74
|
+
* Derive UI capabilities from the runtime snapshot and requested review mode.
|
|
75
|
+
*
|
|
76
|
+
* This is a pure function with no side effects. Call it on every render to
|
|
77
|
+
* get the current capability state.
|
|
78
|
+
*/
|
|
79
|
+
export function deriveCapabilities(
|
|
80
|
+
snapshot: RuntimeRenderSnapshot,
|
|
81
|
+
reviewMode: "editing" | "review",
|
|
82
|
+
workflowScope?: WorkflowScopeSnapshot | null,
|
|
83
|
+
): SessionCapabilities {
|
|
84
|
+
const hasFatalError = Boolean(snapshot.fatalError);
|
|
85
|
+
const isReady = snapshot.isReady;
|
|
86
|
+
const isReadOnly = snapshot.readOnly;
|
|
87
|
+
const exportBlocked = snapshot.compatibility.blockExport;
|
|
88
|
+
const activeStory = snapshot.activeStory ?? { kind: "main" as const };
|
|
89
|
+
const documentMode = snapshot.documentMode ?? "editing";
|
|
90
|
+
|
|
91
|
+
// Phase derivation
|
|
92
|
+
const phase: SessionCapabilities["phase"] = !isReady
|
|
93
|
+
? "loading"
|
|
94
|
+
: hasFatalError
|
|
95
|
+
? "diagnostics"
|
|
96
|
+
: "ready";
|
|
97
|
+
|
|
98
|
+
// Mode derivation: if diagnostics or read-only, override the requested mode
|
|
99
|
+
const mode: SessionCapabilities["mode"] =
|
|
100
|
+
phase === "diagnostics"
|
|
101
|
+
? "read-only-diagnostics"
|
|
102
|
+
: reviewMode;
|
|
103
|
+
|
|
104
|
+
// Command capabilities — document mode "viewing" disables editing
|
|
105
|
+
const canEdit = isReady && !isReadOnly && !hasFatalError && documentMode !== "viewing";
|
|
106
|
+
const canUndo = snapshot.commandState.canUndo && canEdit;
|
|
107
|
+
const canRedo = snapshot.commandState.canRedo && canEdit;
|
|
108
|
+
const canAddComment =
|
|
109
|
+
canEdit &&
|
|
110
|
+
activeStory.kind === "main" &&
|
|
111
|
+
!snapshot.selection.isCollapsed &&
|
|
112
|
+
Boolean(snapshot.surface) &&
|
|
113
|
+
canCreateDocxCommentAnchor(snapshot.surface, toRuntimeAnchor(snapshot.selection.activeRange));
|
|
114
|
+
const canExport = isReady && !exportBlocked && !hasFatalError;
|
|
115
|
+
|
|
116
|
+
// Revision capabilities
|
|
117
|
+
const actionableRevisions = snapshot.trackedChanges.revisions.filter(
|
|
118
|
+
(r) => r.status === "active" && r.actionability === "actionable",
|
|
119
|
+
);
|
|
120
|
+
const canAcceptChange = canEdit && actionableRevisions.some((r) => r.canAccept);
|
|
121
|
+
const canRejectChange = canEdit && actionableRevisions.some((r) => r.canReject);
|
|
122
|
+
const canAcceptAll = canEdit && actionableRevisions.length > 0;
|
|
123
|
+
const canRejectAll = canEdit && actionableRevisions.length > 0;
|
|
124
|
+
|
|
125
|
+
// Trust posture (computed before review visibility since it depends on these)
|
|
126
|
+
const preserveOnlyCount = snapshot.compatibility.featureEntries.filter(
|
|
127
|
+
(e) => e.featureClass === "preserve-only",
|
|
128
|
+
).length;
|
|
129
|
+
const unsupportedFatalCount = snapshot.compatibility.featureEntries.filter(
|
|
130
|
+
(e) => e.featureClass === "unsupported-fatal",
|
|
131
|
+
).length;
|
|
132
|
+
const workflowOverlayPresent = workflowScope?.overlayPresent ?? false;
|
|
133
|
+
const workflowBlocked = (workflowScope?.blockedReasons?.length ?? 0) > 0;
|
|
134
|
+
|
|
135
|
+
// Review visibility
|
|
136
|
+
const trackChangesSupported = snapshot.trackedChanges.totalCount > 0;
|
|
137
|
+
// Rail visible in review/diagnostics mode always.
|
|
138
|
+
// In editing mode, rail stays visible when there are trust/health concerns
|
|
139
|
+
// that the user must be able to reach (per DESIGN.md: health/trust info
|
|
140
|
+
// stays reachable in both modes).
|
|
141
|
+
const hasTrustConcerns = exportBlocked || preserveOnlyCount > 0 || unsupportedFatalCount > 0
|
|
142
|
+
|| snapshot.warnings.length > 0 || hasFatalError;
|
|
143
|
+
const reviewRailVisible = isReady && (workflowOverlayPresent
|
|
144
|
+
? false
|
|
145
|
+
: mode === "review" || mode === "read-only-diagnostics"
|
|
146
|
+
|| (mode === "editing" && hasTrustConcerns));
|
|
147
|
+
|
|
148
|
+
const healthIssueCount = preserveOnlyCount + unsupportedFatalCount + snapshot.warnings.length;
|
|
149
|
+
|
|
150
|
+
const protection = snapshot.protectionSnapshot;
|
|
151
|
+
const hasDocumentProtection = protection?.hasDocumentProtection ?? false;
|
|
152
|
+
const protectedRangeCount = protection?.ranges?.length ?? 0;
|
|
153
|
+
|
|
154
|
+
return {
|
|
155
|
+
phase,
|
|
156
|
+
mode,
|
|
157
|
+
documentMode,
|
|
158
|
+
canUndo,
|
|
159
|
+
canRedo,
|
|
160
|
+
canEdit,
|
|
161
|
+
canAddComment,
|
|
162
|
+
canExport,
|
|
163
|
+
canAcceptChange,
|
|
164
|
+
canRejectChange,
|
|
165
|
+
canAcceptAll,
|
|
166
|
+
canRejectAll,
|
|
167
|
+
reviewRailVisible,
|
|
168
|
+
trackChangesSupported,
|
|
169
|
+
exportBlocked,
|
|
170
|
+
preserveOnlyCount,
|
|
171
|
+
unsupportedFatalCount,
|
|
172
|
+
hasDocumentProtection,
|
|
173
|
+
protectedRangeCount,
|
|
174
|
+
healthIssueCount,
|
|
175
|
+
workflowOverlayPresent,
|
|
176
|
+
workflowBlocked,
|
|
177
|
+
isDirty: snapshot.isDirty,
|
|
178
|
+
isReady,
|
|
179
|
+
hasFatalError,
|
|
180
|
+
};
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
function toRuntimeAnchor(anchor: RuntimeRenderSnapshot["selection"]["activeRange"]) {
|
|
184
|
+
switch (anchor.kind) {
|
|
185
|
+
case "range":
|
|
186
|
+
return createRangeAnchor(anchor.from, anchor.to, anchor.assoc);
|
|
187
|
+
case "node":
|
|
188
|
+
return createNodeAnchor(anchor.at, anchor.assoc);
|
|
189
|
+
case "detached":
|
|
190
|
+
return createDetachedAnchor(anchor.lastKnownRange, anchor.reason);
|
|
191
|
+
}
|
|
192
|
+
}
|