@beyondwork/docx-react-component 1.0.18 → 1.0.20

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 (105) hide show
  1. package/README.md +8 -2
  2. package/package.json +24 -34
  3. package/src/api/README.md +5 -1
  4. package/src/api/public-types.ts +710 -4
  5. package/src/api/session-state.ts +60 -0
  6. package/src/core/commands/formatting-commands.ts +2 -1
  7. package/src/core/commands/image-commands.ts +147 -0
  8. package/src/core/commands/index.ts +19 -3
  9. package/src/core/commands/list-commands.ts +231 -36
  10. package/src/core/commands/paragraph-layout-commands.ts +339 -0
  11. package/src/core/commands/section-layout-commands.ts +680 -0
  12. package/src/core/commands/style-commands.ts +262 -0
  13. package/src/core/search/search-text.ts +357 -0
  14. package/src/core/selection/mapping.ts +41 -0
  15. package/src/core/state/editor-state.ts +4 -1
  16. package/src/index.ts +51 -0
  17. package/src/io/docx-session.ts +623 -56
  18. package/src/io/export/serialize-comments.ts +104 -34
  19. package/src/io/export/serialize-footnotes.ts +198 -1
  20. package/src/io/export/serialize-headers-footers.ts +203 -10
  21. package/src/io/export/serialize-main-document.ts +285 -8
  22. package/src/io/export/serialize-numbering.ts +28 -7
  23. package/src/io/export/split-review-boundaries.ts +181 -19
  24. package/src/io/normalize/normalize-text.ts +144 -32
  25. package/src/io/ooxml/highlight-colors.ts +39 -0
  26. package/src/io/ooxml/numbering-sentinels.ts +44 -0
  27. package/src/io/ooxml/parse-comments.ts +85 -19
  28. package/src/io/ooxml/parse-fields.ts +396 -0
  29. package/src/io/ooxml/parse-footnotes.ts +452 -22
  30. package/src/io/ooxml/parse-headers-footers.ts +657 -29
  31. package/src/io/ooxml/parse-inline-media.ts +30 -0
  32. package/src/io/ooxml/parse-main-document.ts +807 -20
  33. package/src/io/ooxml/parse-numbering.ts +7 -0
  34. package/src/io/ooxml/parse-revisions.ts +317 -38
  35. package/src/io/ooxml/parse-settings.ts +184 -0
  36. package/src/io/ooxml/parse-shapes.ts +25 -0
  37. package/src/io/ooxml/parse-styles.ts +463 -0
  38. package/src/io/ooxml/parse-theme.ts +32 -0
  39. package/src/legal/bookmarks.ts +44 -0
  40. package/src/legal/cross-references.ts +59 -1
  41. package/src/model/canonical-document.ts +250 -4
  42. package/src/model/cds-1.0.0.ts +13 -0
  43. package/src/model/snapshot.ts +87 -2
  44. package/src/review/store/revision-store.ts +6 -0
  45. package/src/review/store/revision-types.ts +1 -0
  46. package/src/runtime/document-layout.ts +332 -0
  47. package/src/runtime/document-navigation.ts +603 -0
  48. package/src/runtime/document-runtime.ts +1754 -78
  49. package/src/runtime/document-search.ts +145 -0
  50. package/src/runtime/numbering-prefix.ts +47 -26
  51. package/src/runtime/page-layout-estimation.ts +212 -0
  52. package/src/runtime/read-only-diagnostics-runtime.ts +9 -0
  53. package/src/runtime/session-capabilities.ts +35 -3
  54. package/src/runtime/story-context.ts +164 -0
  55. package/src/runtime/story-targeting.ts +162 -0
  56. package/src/runtime/surface-projection.ts +324 -36
  57. package/src/runtime/table-schema.ts +89 -7
  58. package/src/runtime/view-state.ts +477 -0
  59. package/src/runtime/workflow-markup.ts +349 -0
  60. package/src/ui/WordReviewEditor.tsx +2469 -1344
  61. package/src/ui/browser-export.ts +52 -0
  62. package/src/ui/editor-command-bag.ts +120 -0
  63. package/src/ui/editor-runtime-boundary.ts +1422 -0
  64. package/src/ui/editor-shell-view.tsx +134 -0
  65. package/src/ui/editor-surface-controller.tsx +51 -0
  66. package/src/ui/headless/preserve-editor-selection.ts +5 -0
  67. package/src/ui/headless/revision-decoration-model.ts +4 -4
  68. package/src/ui/headless/selection-helpers.ts +20 -0
  69. package/src/ui/headless/selection-toolbar-model.ts +22 -0
  70. package/src/ui/headless/use-editor-keyboard.ts +6 -1
  71. package/src/ui/runtime-snapshot-selectors.ts +197 -0
  72. package/src/ui-tailwind/chrome/tw-alert-banner.tsx +18 -2
  73. package/src/ui-tailwind/chrome/tw-image-context-toolbar.tsx +129 -0
  74. package/src/ui-tailwind/chrome/tw-layout-panel.tsx +114 -0
  75. package/src/ui-tailwind/chrome/tw-object-context-toolbar.tsx +34 -0
  76. package/src/ui-tailwind/chrome/tw-page-ruler.tsx +386 -0
  77. package/src/ui-tailwind/chrome/tw-selection-toolbar.tsx +150 -14
  78. package/src/ui-tailwind/chrome/tw-table-context-toolbar.tsx +128 -0
  79. package/src/ui-tailwind/editor-surface/perf-probe.ts +179 -0
  80. package/src/ui-tailwind/editor-surface/pm-command-bridge.ts +46 -7
  81. package/src/ui-tailwind/editor-surface/pm-contextual-ui.ts +31 -0
  82. package/src/ui-tailwind/editor-surface/pm-decorations.ts +35 -0
  83. package/src/ui-tailwind/editor-surface/pm-position-map.ts +3 -3
  84. package/src/ui-tailwind/editor-surface/pm-schema.ts +186 -13
  85. package/src/ui-tailwind/editor-surface/pm-state-from-snapshot.ts +191 -68
  86. package/src/ui-tailwind/editor-surface/search-plugin.ts +19 -68
  87. package/src/ui-tailwind/editor-surface/surface-build-keys.ts +51 -0
  88. package/src/ui-tailwind/editor-surface/tw-inline-token.tsx +11 -0
  89. package/src/ui-tailwind/editor-surface/tw-opaque-block.tsx +7 -1
  90. package/src/ui-tailwind/editor-surface/tw-prosemirror-surface.tsx +528 -85
  91. package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +0 -1
  92. package/src/ui-tailwind/index.ts +2 -1
  93. package/src/ui-tailwind/page-chrome-model.ts +27 -0
  94. package/src/ui-tailwind/review/tw-comment-sidebar.tsx +277 -147
  95. package/src/ui-tailwind/review/tw-health-panel.tsx +31 -2
  96. package/src/ui-tailwind/review/tw-review-rail.tsx +8 -8
  97. package/src/ui-tailwind/review/tw-revision-sidebar.tsx +15 -15
  98. package/src/ui-tailwind/theme/editor-theme.css +127 -0
  99. package/src/ui-tailwind/toolbar/tw-toolbar-icon-button.tsx +4 -0
  100. package/src/ui-tailwind/toolbar/tw-toolbar.tsx +829 -12
  101. package/src/ui-tailwind/tw-review-workspace.tsx +1238 -42
  102. package/src/validation/compatibility-engine.ts +119 -24
  103. package/src/validation/compatibility-report.ts +1 -0
  104. package/src/validation/diagnostics.ts +1 -0
  105. package/src/validation/docx-comment-proof.ts +707 -0
@@ -11,6 +11,25 @@
11
11
 
12
12
  import type { NodeSpec } from "prosemirror-model";
13
13
 
14
+ /** Characters that must never appear in CSS values derived from OOXML tokens. */
15
+ const CSS_INJECTION_RE = /[;{}()[\]\\@!]/;
16
+
17
+ /** Validate a CSS color token — allows #hex (3/6/8 digits) and named colors only. */
18
+ function safeCssColor(raw: string | null | undefined): string | null {
19
+ if (!raw) return null;
20
+ if (/^#[0-9A-Fa-f]{3,8}$/.test(raw)) return raw;
21
+ if (/^[a-zA-Z]+$/.test(raw) && raw !== "expression") return raw;
22
+ return null;
23
+ }
24
+
25
+ /** Sanitize a composite CSS border shorthand (e.g. "1px solid #abc"). Rejects injection attempts. */
26
+ function safeCssBorder(raw: string | null | undefined): string | null {
27
+ if (!raw) return null;
28
+ if (CSS_INJECTION_RE.test(raw)) return null;
29
+ if (raw.length > 100) return null;
30
+ return raw;
31
+ }
32
+
14
33
  type TableCellAttrs = {
15
34
  colspan?: number | null;
16
35
  rowspan?: number | null;
@@ -18,6 +37,11 @@ type TableCellAttrs = {
18
37
  gridSpan?: number | null;
19
38
  verticalMerge?: "restart" | "continue" | null;
20
39
  backgroundColor?: string | null;
40
+ verticalAlign?: "top" | "center" | "bottom" | null;
41
+ borderTop?: string | null;
42
+ borderRight?: string | null;
43
+ borderBottom?: string | null;
44
+ borderLeft?: string | null;
21
45
  };
22
46
 
23
47
  function resolveRenderedColspan(attrs: {
@@ -55,6 +79,7 @@ function getCellAttrs(dom: HTMLElement): TableCellAttrs {
55
79
  const gridSpan = gridSpanAttr ? Number.parseInt(gridSpanAttr, 10) : colspan;
56
80
  const backgroundColor =
57
81
  dom.getAttribute("data-cell-background") ?? dom.style.backgroundColor ?? null;
82
+ const verticalAlign = dom.getAttribute("data-vertical-align") as "top" | "center" | "bottom" | null;
58
83
 
59
84
  return {
60
85
  colspan,
@@ -66,6 +91,11 @@ function getCellAttrs(dom: HTMLElement): TableCellAttrs {
66
91
  ? verticalMergeAttr
67
92
  : null,
68
93
  backgroundColor,
94
+ verticalAlign: verticalAlign === "top" || verticalAlign === "center" || verticalAlign === "bottom" ? verticalAlign : null,
95
+ borderTop: dom.getAttribute("data-border-top"),
96
+ borderRight: dom.getAttribute("data-border-right"),
97
+ borderBottom: dom.getAttribute("data-border-bottom"),
98
+ borderLeft: dom.getAttribute("data-border-left"),
69
99
  };
70
100
  }
71
101
 
@@ -91,8 +121,28 @@ function setCellDomAttrs(nodeAttrs: TableCellAttrs, className: string): Record<s
91
121
  }
92
122
  if (nodeAttrs.backgroundColor) {
93
123
  attrs["data-cell-background"] = nodeAttrs.backgroundColor;
94
- attrs.style = `background-color: ${nodeAttrs.backgroundColor}`;
95
124
  }
125
+ if (nodeAttrs.verticalAlign && nodeAttrs.verticalAlign !== "top") {
126
+ attrs["data-vertical-align"] = nodeAttrs.verticalAlign;
127
+ }
128
+ if (nodeAttrs.borderTop) attrs["data-border-top"] = nodeAttrs.borderTop;
129
+ if (nodeAttrs.borderRight) attrs["data-border-right"] = nodeAttrs.borderRight;
130
+ if (nodeAttrs.borderBottom) attrs["data-border-bottom"] = nodeAttrs.borderBottom;
131
+ if (nodeAttrs.borderLeft) attrs["data-border-left"] = nodeAttrs.borderLeft;
132
+
133
+ const styles: string[] = [];
134
+ const bgColor = safeCssColor(nodeAttrs.backgroundColor);
135
+ if (bgColor) styles.push(`background-color: ${bgColor}`);
136
+ if (nodeAttrs.verticalAlign) styles.push(`vertical-align: ${nodeAttrs.verticalAlign === "center" ? "middle" : nodeAttrs.verticalAlign}`);
137
+ const bTop = safeCssBorder(nodeAttrs.borderTop);
138
+ if (bTop) styles.push(`border-top: ${bTop}`);
139
+ const bRight = safeCssBorder(nodeAttrs.borderRight);
140
+ if (bRight) styles.push(`border-right: ${bRight}`);
141
+ const bBottom = safeCssBorder(nodeAttrs.borderBottom);
142
+ if (bBottom) styles.push(`border-bottom: ${bBottom}`);
143
+ const bLeft = safeCssBorder(nodeAttrs.borderLeft);
144
+ if (bLeft) styles.push(`border-left: ${bLeft}`);
145
+ if (styles.length > 0) attrs.style = styles.join("; ");
96
146
 
97
147
  return attrs;
98
148
  }
@@ -126,6 +176,11 @@ const tableCellSpecAttrs = {
126
176
  rowspan: { default: 1, validate: "number" },
127
177
  colwidth: { default: null, validate: validateColwidth },
128
178
  backgroundColor: { default: null },
179
+ verticalAlign: { default: null },
180
+ borderTop: { default: null },
181
+ borderRight: { default: null },
182
+ borderBottom: { default: null },
183
+ borderLeft: { default: null },
129
184
  } as const;
130
185
 
131
186
  export const tableNodeSpec: NodeSpec = {
@@ -137,10 +192,29 @@ export const tableNodeSpec: NodeSpec = {
137
192
  styleId: { default: null },
138
193
  propertiesXml: { default: null },
139
194
  gridColumns: { default: [] },
195
+ alignment: { default: null },
196
+ tblLookFirstRow: { default: false },
197
+ tblLookLastRow: { default: false },
198
+ tblLookFirstColumn: { default: false },
199
+ tblLookLastColumn: { default: false },
200
+ tblLookNoHBand: { default: false },
201
+ tblLookNoVBand: { default: false },
140
202
  },
141
203
  parseDOM: [{ tag: "table" }],
142
- toDOM() {
143
- return ["table", { class: "border-collapse w-full my-2 text-sm" }, ["tbody", 0]];
204
+ toDOM(node) {
205
+ const style = node.attrs.alignment === "center"
206
+ ? "margin-left: auto; margin-right: auto"
207
+ : node.attrs.alignment === "right"
208
+ ? "margin-left: auto"
209
+ : undefined;
210
+ return [
211
+ "table",
212
+ {
213
+ class: "border-collapse w-full my-2 text-sm",
214
+ ...(style ? { style } : {}),
215
+ },
216
+ ["tbody", 0],
217
+ ];
144
218
  },
145
219
  };
146
220
 
@@ -149,15 +223,23 @@ export const tableRowNodeSpec: NodeSpec = {
149
223
  tableRole: "row",
150
224
  attrs: {
151
225
  propertiesXml: { default: null },
226
+ height: { default: null },
227
+ heightRule: { default: null },
228
+ isHeader: { default: false },
152
229
  },
153
230
  parseDOM: [{ tag: "tr" }],
154
- toDOM() {
155
- return ["tr", 0];
231
+ toDOM(node) {
232
+ const style = node.attrs.heightRule === "exact" && node.attrs.height
233
+ ? `height: ${Math.round((node.attrs.height as number) / 20)}pt`
234
+ : node.attrs.heightRule === "atLeast" && node.attrs.height
235
+ ? `min-height: ${Math.round((node.attrs.height as number) / 20)}pt`
236
+ : undefined;
237
+ return ["tr", { ...(style ? { style } : {}) }, 0];
156
238
  },
157
239
  };
158
240
 
159
241
  export const tableCellNodeSpec: NodeSpec = {
160
- content: "paragraph+",
242
+ content: "block+",
161
243
  tableRole: "cell",
162
244
  isolating: true,
163
245
  attrs: tableCellSpecAttrs,
@@ -182,7 +264,7 @@ export const tableCellNodeSpec: NodeSpec = {
182
264
  };
183
265
 
184
266
  export const tableHeaderCellNodeSpec: NodeSpec = {
185
- content: "paragraph+",
267
+ content: "block+",
186
268
  tableRole: "header_cell",
187
269
  isolating: true,
188
270
  attrs: tableCellSpecAttrs,
@@ -0,0 +1,477 @@
1
+ /**
2
+ * Runtime-owned view state — non-canonical interaction model.
3
+ *
4
+ * This module owns the editor's interaction and measurement state that is
5
+ * separate from the canonical document model. View state is never serialized
6
+ * into persisted snapshots or exported DOCX; it exists only while the editor
7
+ * session is live.
8
+ *
9
+ * Design boundaries:
10
+ * - Canonical document state lives in EditorState / CanonicalDocumentEnvelope.
11
+ * - View state is runtime-owned, non-canonical, non-exported.
12
+ * - Builds on Wave 35's activeStory and PageLayoutSnapshot model.
13
+ * - Runtime viewMode models editor posture ("editing" | "review" | "view").
14
+ * - Workspace mode and zoom are runtime-owned so hosts and the mounted shell
15
+ * share one view-state contract.
16
+ */
17
+
18
+ import type {
19
+ ActiveListContext,
20
+ ActiveNoteContext,
21
+ CaretAffinity,
22
+ DocumentMode,
23
+ EditorStoryTarget,
24
+ EditorSurfaceSnapshot,
25
+ EditorViewStateSnapshot,
26
+ LayoutMeasurement,
27
+ PageLayoutSnapshot,
28
+ PageRegionHitTest,
29
+ SelectionSnapshot,
30
+ SurfaceBlockSnapshot,
31
+ SurfaceInlineSegment,
32
+ ViewMode,
33
+ WorkspaceMode,
34
+ ZoomLevel,
35
+ } from "../api/public-types";
36
+ import type { NumberingCatalog } from "../model/canonical-document.ts";
37
+
38
+ export interface ViewState {
39
+ viewMode: ViewMode;
40
+ documentMode: DocumentMode;
41
+ workspaceMode: WorkspaceMode;
42
+ zoomLevel: ZoomLevel;
43
+ isFocused: boolean;
44
+ caretAffinity: CaretAffinity;
45
+ activePageRegion: PageRegionHitTest | null;
46
+ activeObjectFrame: LayoutMeasurement["objectFrame"] | null;
47
+ }
48
+
49
+ const MIN_ZOOM_PERCENT = 50;
50
+ const MAX_ZOOM_PERCENT = 200;
51
+
52
+ const DEFAULT_VIEW_STATE: ViewState = {
53
+ viewMode: "editing",
54
+ documentMode: "editing",
55
+ workspaceMode: "canvas",
56
+ zoomLevel: 100,
57
+ isFocused: false,
58
+ caretAffinity: "none",
59
+ activePageRegion: null,
60
+ activeObjectFrame: null,
61
+ };
62
+
63
+ export function createViewState(initial?: Partial<ViewState>): ViewState {
64
+ return { ...DEFAULT_VIEW_STATE, ...initial };
65
+ }
66
+
67
+ export function setViewMode(state: ViewState, mode: ViewMode): ViewState {
68
+ if (state.viewMode === mode) return state;
69
+ return { ...state, viewMode: mode };
70
+ }
71
+
72
+ export function setDocumentMode(state: ViewState, mode: DocumentMode): ViewState {
73
+ if (state.documentMode === mode) return state;
74
+ return { ...state, documentMode: mode };
75
+ }
76
+
77
+ export function setWorkspaceMode(
78
+ state: ViewState,
79
+ workspaceMode: WorkspaceMode,
80
+ ): ViewState {
81
+ if (state.workspaceMode === workspaceMode) {
82
+ return state;
83
+ }
84
+ return { ...state, workspaceMode };
85
+ }
86
+
87
+ export function setZoomLevel(state: ViewState, zoomLevel: ZoomLevel): ViewState {
88
+ const normalizedZoom = normalizeZoomLevel(zoomLevel, state.zoomLevel);
89
+ if (state.zoomLevel === normalizedZoom) {
90
+ return state;
91
+ }
92
+ return { ...state, zoomLevel: normalizedZoom };
93
+ }
94
+
95
+ export function setFocused(state: ViewState, focused: boolean): ViewState {
96
+ if (state.isFocused === focused) return state;
97
+ return { ...state, isFocused: focused };
98
+ }
99
+
100
+ export function setCaretAffinity(state: ViewState, affinity: CaretAffinity): ViewState {
101
+ if (state.caretAffinity === affinity) return state;
102
+ return { ...state, caretAffinity: affinity };
103
+ }
104
+
105
+ export function setActivePageRegion(state: ViewState, region: PageRegionHitTest | null): ViewState {
106
+ return { ...state, activePageRegion: region };
107
+ }
108
+
109
+ export function setActiveObjectFrame(
110
+ state: ViewState,
111
+ frame: LayoutMeasurement["objectFrame"] | null,
112
+ ): ViewState {
113
+ return { ...state, activeObjectFrame: frame };
114
+ }
115
+
116
+ /**
117
+ * Derive list context from the surface block at the current selection head.
118
+ */
119
+ export function deriveActiveListContext(
120
+ surface: EditorSurfaceSnapshot | undefined,
121
+ selectionHead: number,
122
+ numberingCatalog?: NumberingCatalog,
123
+ ): ActiveListContext | null {
124
+ if (!surface) return null;
125
+ const block = findBlockAtPosition(surface.blocks, selectionHead);
126
+ if (!block || block.kind !== "paragraph" || !block.numbering) return null;
127
+
128
+ const isOrdered = resolveListOrdering(block.numbering, numberingCatalog);
129
+ return {
130
+ numberingInstanceId: block.numbering.numberingInstanceId,
131
+ level: block.numbering.level,
132
+ isOrdered,
133
+ ...(block.numberingPrefix ? { markerText: block.numberingPrefix } : {}),
134
+ };
135
+ }
136
+
137
+ /**
138
+ * Derive note context when the active story targets a footnote or endnote.
139
+ */
140
+ export function deriveActiveNoteContext(
141
+ activeStory: EditorStoryTarget,
142
+ mainSurface: EditorSurfaceSnapshot | undefined,
143
+ ): ActiveNoteContext | null {
144
+ if (activeStory.kind !== "footnote" && activeStory.kind !== "endnote") {
145
+ return null;
146
+ }
147
+
148
+ const referencePosition = findNoteReferencePosition(mainSurface, activeStory);
149
+ return {
150
+ noteKind: activeStory.kind,
151
+ noteId: activeStory.noteId,
152
+ referencePosition,
153
+ };
154
+ }
155
+
156
+ /**
157
+ * Derive layout measurement from the surface at the current selection.
158
+ */
159
+ export function deriveLayoutMeasurement(
160
+ surface: EditorSurfaceSnapshot | undefined,
161
+ selectionOrPosition: SelectionSnapshot | number,
162
+ viewState: ViewState,
163
+ ): LayoutMeasurement {
164
+ const selectionHead =
165
+ typeof selectionOrPosition === "number"
166
+ ? selectionOrPosition
167
+ : selectionOrPosition.activeRange.kind === "node"
168
+ ? selectionOrPosition.activeRange.at
169
+ : selectionOrPosition.head;
170
+ const block = surface ? findBlockAtPosition(surface.blocks, selectionHead) : null;
171
+ const tabStops = block?.kind === "paragraph" && block.tabStops ? block.tabStops : [];
172
+ const listMarkerLane = deriveListMarkerLane(block);
173
+
174
+ return {
175
+ pageRegions: viewState.activePageRegion ? [viewState.activePageRegion] : [],
176
+ caretAffinity: viewState.caretAffinity,
177
+ tabStops,
178
+ listMarkerLane: listMarkerLane ?? undefined,
179
+ objectFrame: viewState.activeObjectFrame ?? undefined,
180
+ };
181
+ }
182
+
183
+ /**
184
+ * Build the full EditorViewStateSnapshot from runtime + surface state.
185
+ */
186
+ export function createEditorViewStateSnapshot(
187
+ viewState: ViewState,
188
+ activeStory: EditorStoryTarget,
189
+ selection: SelectionSnapshot,
190
+ surface: EditorSurfaceSnapshot | undefined,
191
+ mainSurface: EditorSurfaceSnapshot | undefined,
192
+ pageLayout: PageLayoutSnapshot | null | undefined,
193
+ numberingCatalog?: NumberingCatalog,
194
+ ): EditorViewStateSnapshot {
195
+ const selectionPosition =
196
+ selection.activeRange.kind === "node" ? selection.activeRange.at : selection.head;
197
+ const derivedViewState = deriveInteractionViewState(
198
+ viewState,
199
+ activeStory,
200
+ selection,
201
+ surface,
202
+ pageLayout,
203
+ );
204
+ const activeListContext = deriveActiveListContext(surface, selectionPosition, numberingCatalog);
205
+ const activeNoteContext = deriveActiveNoteContext(activeStory, mainSurface);
206
+ const measurement = deriveLayoutMeasurement(surface, selection, derivedViewState);
207
+
208
+ return {
209
+ viewMode: derivedViewState.viewMode,
210
+ documentMode: derivedViewState.documentMode,
211
+ workspaceMode: derivedViewState.workspaceMode,
212
+ zoomLevel: derivedViewState.zoomLevel,
213
+ activeStory,
214
+ selection,
215
+ caretAffinity: derivedViewState.caretAffinity,
216
+ activeListContext,
217
+ activeNoteContext,
218
+ activePageRegion: derivedViewState.activePageRegion,
219
+ activeObjectFrame: derivedViewState.activeObjectFrame,
220
+ measurement,
221
+ isFocused: derivedViewState.isFocused,
222
+ };
223
+ }
224
+
225
+ // ---------------------------------------------------------------------------
226
+ // Internal helpers
227
+ // ---------------------------------------------------------------------------
228
+
229
+ function findBlockAtPosition(
230
+ blocks: readonly SurfaceBlockSnapshot[],
231
+ position: number,
232
+ ): SurfaceBlockSnapshot | null {
233
+ for (const block of blocks) {
234
+ if (position >= block.from && position <= block.to) {
235
+ if (block.kind === "sdt_block") {
236
+ const inner = findBlockAtPosition(block.children, position);
237
+ if (inner) return inner;
238
+ }
239
+ if (block.kind === "table") {
240
+ for (const row of block.rows) {
241
+ for (const cell of row.cells) {
242
+ const inner = findBlockAtPosition(cell.content, position);
243
+ if (inner) return inner;
244
+ }
245
+ }
246
+ }
247
+ return block;
248
+ }
249
+ }
250
+ return null;
251
+ }
252
+
253
+ function resolveListOrdering(
254
+ numbering: NonNullable<Extract<SurfaceBlockSnapshot, { kind: "paragraph" }>["numbering"]>,
255
+ numberingCatalog?: NumberingCatalog,
256
+ ): boolean {
257
+ if (!numberingCatalog) {
258
+ return true;
259
+ }
260
+
261
+ const instance = numberingCatalog.instances[numbering.numberingInstanceId];
262
+ const definition = instance
263
+ ? numberingCatalog.abstractDefinitions[instance.abstractNumberingId]
264
+ : undefined;
265
+ const levelDefinition = definition?.levels.find((level) => level.level === numbering.level);
266
+
267
+ if (!levelDefinition) {
268
+ return true;
269
+ }
270
+
271
+ return levelDefinition.format !== "bullet";
272
+ }
273
+
274
+ function deriveInteractionViewState(
275
+ viewState: ViewState,
276
+ activeStory: EditorStoryTarget,
277
+ selection: SelectionSnapshot,
278
+ surface: EditorSurfaceSnapshot | undefined,
279
+ pageLayout: PageLayoutSnapshot | null | undefined,
280
+ ): ViewState {
281
+ const activePageRegion = deriveActivePageRegion(activeStory, pageLayout) ?? viewState.activePageRegion;
282
+ const activeObjectFrame = deriveActiveObjectFrame(surface, selection);
283
+ const caretAffinity = deriveCaretAffinity(surface, selection);
284
+
285
+ return {
286
+ ...viewState,
287
+ caretAffinity,
288
+ activePageRegion,
289
+ activeObjectFrame,
290
+ };
291
+ }
292
+
293
+ function deriveActivePageRegion(
294
+ activeStory: EditorStoryTarget,
295
+ pageLayout: PageLayoutSnapshot | null | undefined,
296
+ ): PageRegionHitTest | null {
297
+ if (!pageLayout) {
298
+ return null;
299
+ }
300
+ const sectionIndex = pageLayout?.sectionIndex ?? 0;
301
+ switch (activeStory.kind) {
302
+ case "header":
303
+ return { region: "header", sectionIndex, columnIndex: 0 };
304
+ case "footer":
305
+ return { region: "footer", sectionIndex, columnIndex: 0 };
306
+ default:
307
+ return { region: "body", sectionIndex, columnIndex: 0 };
308
+ }
309
+ }
310
+
311
+ function deriveActiveObjectFrame(
312
+ surface: EditorSurfaceSnapshot | undefined,
313
+ selection: SelectionSnapshot,
314
+ ): LayoutMeasurement["objectFrame"] | null {
315
+ if (!surface) {
316
+ return null;
317
+ }
318
+
319
+ const position =
320
+ selection.activeRange.kind === "node" ? selection.activeRange.at : selection.head;
321
+ const block = findBlockAtPosition(surface.blocks, position);
322
+ if (!block || block.kind !== "paragraph") {
323
+ return null;
324
+ }
325
+
326
+ const segment = findObjectSegmentAtPosition(block.segments, position);
327
+ if (!segment) {
328
+ return null;
329
+ }
330
+
331
+ if (segment.kind === "image") {
332
+ return {
333
+ kind: "image",
334
+ anchorPos: segment.from,
335
+ display: segment.display === "floating" ? "floating" : "inline",
336
+ };
337
+ }
338
+
339
+ const objectKind = inferOpaqueObjectKind(segment);
340
+ if (!objectKind) {
341
+ return null;
342
+ }
343
+
344
+ return {
345
+ kind: objectKind,
346
+ anchorPos: segment.from,
347
+ display: "inline",
348
+ };
349
+ }
350
+
351
+ function deriveCaretAffinity(
352
+ surface: EditorSurfaceSnapshot | undefined,
353
+ selection: SelectionSnapshot,
354
+ ): CaretAffinity {
355
+ if (selection.activeRange.kind === "node") {
356
+ return selection.activeRange.assoc < 0 ? "backward" : "forward";
357
+ }
358
+ if (!selection.isCollapsed || !surface) {
359
+ return "none";
360
+ }
361
+ const block = findBlockAtPosition(surface.blocks, selection.head);
362
+ if (!block || block.kind !== "paragraph") {
363
+ return "none";
364
+ }
365
+ for (const segment of block.segments) {
366
+ if (!isObjectLikeSegment(segment)) {
367
+ continue;
368
+ }
369
+ if (selection.head === segment.from) {
370
+ return "forward";
371
+ }
372
+ if (selection.head === segment.to) {
373
+ return "backward";
374
+ }
375
+ }
376
+ return "none";
377
+ }
378
+
379
+ export function findNoteReferencePosition(
380
+ mainSurface: EditorSurfaceSnapshot | undefined,
381
+ target: EditorStoryTarget & { kind: "footnote" | "endnote" },
382
+ ): number {
383
+ if (!mainSurface) return 0;
384
+ for (const block of mainSurface.blocks) {
385
+ if (block.kind === "paragraph") {
386
+ for (const segment of block.segments) {
387
+ if (
388
+ segment.kind === "note_ref" &&
389
+ segment.noteKind === target.kind &&
390
+ segment.noteId === target.noteId
391
+ ) {
392
+ return segment.from;
393
+ }
394
+ // Fallback: opaque_inline references from older surface projections
395
+ if (
396
+ segment.kind === "opaque_inline" &&
397
+ segment.label.toLowerCase().includes(target.kind) &&
398
+ segment.detail.includes(target.noteId)
399
+ ) {
400
+ return segment.from;
401
+ }
402
+ }
403
+ }
404
+ }
405
+ return 0;
406
+ }
407
+
408
+ function normalizeZoomLevel(
409
+ zoomLevel: ZoomLevel,
410
+ currentZoom: ZoomLevel,
411
+ ): ZoomLevel {
412
+ if (zoomLevel === "pageWidth" || zoomLevel === "onePage") {
413
+ return zoomLevel;
414
+ }
415
+ if (!Number.isFinite(zoomLevel) || zoomLevel <= 0) {
416
+ return currentZoom;
417
+ }
418
+ return Math.max(
419
+ MIN_ZOOM_PERCENT,
420
+ Math.min(MAX_ZOOM_PERCENT, Math.round(zoomLevel)),
421
+ );
422
+ }
423
+
424
+ function findObjectSegmentAtPosition(
425
+ segments: readonly SurfaceInlineSegment[],
426
+ position: number,
427
+ ): Extract<SurfaceInlineSegment, { kind: "image" | "opaque_inline" }> | null {
428
+ for (const segment of segments) {
429
+ if (!isObjectLikeSegment(segment)) {
430
+ continue;
431
+ }
432
+ if (position >= segment.from && position <= segment.to) {
433
+ return segment;
434
+ }
435
+ }
436
+ return null;
437
+ }
438
+
439
+ function isObjectLikeSegment(
440
+ segment: SurfaceInlineSegment,
441
+ ): segment is Extract<SurfaceInlineSegment, { kind: "image" | "opaque_inline" }> {
442
+ if (segment.kind === "image") {
443
+ return true;
444
+ }
445
+ if (segment.kind !== "opaque_inline") {
446
+ return false;
447
+ }
448
+ return inferOpaqueObjectKind(segment) !== null;
449
+ }
450
+
451
+ function inferOpaqueObjectKind(
452
+ segment: Extract<SurfaceInlineSegment, { kind: "opaque_inline" }>,
453
+ ): "textbox" | "shape" | null {
454
+ if (segment.label === "Text box") {
455
+ return "textbox";
456
+ }
457
+ if (segment.label === "Drawing shape") {
458
+ return "shape";
459
+ }
460
+ if (segment.label === "Legacy VML drawing") {
461
+ return segment.detail.includes("#_x0000_t202")
462
+ ? "textbox"
463
+ : "shape";
464
+ }
465
+ return null;
466
+ }
467
+
468
+ function deriveListMarkerLane(
469
+ block: SurfaceBlockSnapshot | null,
470
+ ): { indent: number; markerWidth: number } | null {
471
+ if (!block || block.kind !== "paragraph" || !block.numbering) return null;
472
+ const indent = block.indentation?.hanging ?? block.indentation?.left ?? 360;
473
+ return {
474
+ indent,
475
+ markerWidth: Math.min(indent, 360),
476
+ };
477
+ }