@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.
- package/README.md +8 -2
- package/package.json +24 -34
- package/src/api/README.md +5 -1
- package/src/api/public-types.ts +710 -4
- package/src/api/session-state.ts +60 -0
- package/src/core/commands/formatting-commands.ts +2 -1
- package/src/core/commands/image-commands.ts +147 -0
- package/src/core/commands/index.ts +19 -3
- package/src/core/commands/list-commands.ts +231 -36
- package/src/core/commands/paragraph-layout-commands.ts +339 -0
- package/src/core/commands/section-layout-commands.ts +680 -0
- package/src/core/commands/style-commands.ts +262 -0
- package/src/core/search/search-text.ts +357 -0
- package/src/core/selection/mapping.ts +41 -0
- package/src/core/state/editor-state.ts +4 -1
- package/src/index.ts +51 -0
- package/src/io/docx-session.ts +623 -56
- package/src/io/export/serialize-comments.ts +104 -34
- package/src/io/export/serialize-footnotes.ts +198 -1
- package/src/io/export/serialize-headers-footers.ts +203 -10
- package/src/io/export/serialize-main-document.ts +285 -8
- package/src/io/export/serialize-numbering.ts +28 -7
- package/src/io/export/split-review-boundaries.ts +181 -19
- package/src/io/normalize/normalize-text.ts +144 -32
- 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 +85 -19
- package/src/io/ooxml/parse-fields.ts +396 -0
- package/src/io/ooxml/parse-footnotes.ts +452 -22
- package/src/io/ooxml/parse-headers-footers.ts +657 -29
- package/src/io/ooxml/parse-inline-media.ts +30 -0
- package/src/io/ooxml/parse-main-document.ts +807 -20
- package/src/io/ooxml/parse-numbering.ts +7 -0
- package/src/io/ooxml/parse-revisions.ts +317 -38
- package/src/io/ooxml/parse-settings.ts +184 -0
- package/src/io/ooxml/parse-shapes.ts +25 -0
- package/src/io/ooxml/parse-styles.ts +463 -0
- package/src/io/ooxml/parse-theme.ts +32 -0
- package/src/legal/bookmarks.ts +44 -0
- package/src/legal/cross-references.ts +59 -1
- package/src/model/canonical-document.ts +250 -4
- package/src/model/cds-1.0.0.ts +13 -0
- package/src/model/snapshot.ts +87 -2
- package/src/review/store/revision-store.ts +6 -0
- package/src/review/store/revision-types.ts +1 -0
- package/src/runtime/document-layout.ts +332 -0
- package/src/runtime/document-navigation.ts +603 -0
- package/src/runtime/document-runtime.ts +1754 -78
- package/src/runtime/document-search.ts +145 -0
- package/src/runtime/numbering-prefix.ts +47 -26
- package/src/runtime/page-layout-estimation.ts +212 -0
- package/src/runtime/read-only-diagnostics-runtime.ts +9 -0
- package/src/runtime/session-capabilities.ts +35 -3
- package/src/runtime/story-context.ts +164 -0
- package/src/runtime/story-targeting.ts +162 -0
- package/src/runtime/surface-projection.ts +324 -36
- package/src/runtime/table-schema.ts +89 -7
- package/src/runtime/view-state.ts +477 -0
- package/src/runtime/workflow-markup.ts +349 -0
- package/src/ui/WordReviewEditor.tsx +2469 -1344
- package/src/ui/browser-export.ts +52 -0
- package/src/ui/editor-command-bag.ts +120 -0
- package/src/ui/editor-runtime-boundary.ts +1422 -0
- package/src/ui/editor-shell-view.tsx +134 -0
- package/src/ui/editor-surface-controller.tsx +51 -0
- package/src/ui/headless/preserve-editor-selection.ts +5 -0
- package/src/ui/headless/revision-decoration-model.ts +4 -4
- package/src/ui/headless/selection-helpers.ts +20 -0
- package/src/ui/headless/selection-toolbar-model.ts +22 -0
- package/src/ui/headless/use-editor-keyboard.ts +6 -1
- package/src/ui/runtime-snapshot-selectors.ts +197 -0
- package/src/ui-tailwind/chrome/tw-alert-banner.tsx +18 -2
- package/src/ui-tailwind/chrome/tw-image-context-toolbar.tsx +129 -0
- package/src/ui-tailwind/chrome/tw-layout-panel.tsx +114 -0
- package/src/ui-tailwind/chrome/tw-object-context-toolbar.tsx +34 -0
- package/src/ui-tailwind/chrome/tw-page-ruler.tsx +386 -0
- package/src/ui-tailwind/chrome/tw-selection-toolbar.tsx +150 -14
- package/src/ui-tailwind/chrome/tw-table-context-toolbar.tsx +128 -0
- package/src/ui-tailwind/editor-surface/perf-probe.ts +179 -0
- package/src/ui-tailwind/editor-surface/pm-command-bridge.ts +46 -7
- package/src/ui-tailwind/editor-surface/pm-contextual-ui.ts +31 -0
- package/src/ui-tailwind/editor-surface/pm-decorations.ts +35 -0
- package/src/ui-tailwind/editor-surface/pm-position-map.ts +3 -3
- package/src/ui-tailwind/editor-surface/pm-schema.ts +186 -13
- package/src/ui-tailwind/editor-surface/pm-state-from-snapshot.ts +191 -68
- package/src/ui-tailwind/editor-surface/search-plugin.ts +19 -68
- package/src/ui-tailwind/editor-surface/surface-build-keys.ts +51 -0
- package/src/ui-tailwind/editor-surface/tw-inline-token.tsx +11 -0
- package/src/ui-tailwind/editor-surface/tw-opaque-block.tsx +7 -1
- package/src/ui-tailwind/editor-surface/tw-prosemirror-surface.tsx +528 -85
- package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +0 -1
- package/src/ui-tailwind/index.ts +2 -1
- package/src/ui-tailwind/page-chrome-model.ts +27 -0
- package/src/ui-tailwind/review/tw-comment-sidebar.tsx +277 -147
- package/src/ui-tailwind/review/tw-health-panel.tsx +31 -2
- package/src/ui-tailwind/review/tw-review-rail.tsx +8 -8
- package/src/ui-tailwind/review/tw-revision-sidebar.tsx +15 -15
- package/src/ui-tailwind/theme/editor-theme.css +127 -0
- package/src/ui-tailwind/toolbar/tw-toolbar-icon-button.tsx +4 -0
- package/src/ui-tailwind/toolbar/tw-toolbar.tsx +829 -12
- package/src/ui-tailwind/tw-review-workspace.tsx +1238 -42
- package/src/validation/compatibility-engine.ts +119 -24
- package/src/validation/compatibility-report.ts +1 -0
- package/src/validation/diagnostics.ts +1 -0
- 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
|
-
|
|
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
|
-
|
|
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: "
|
|
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: "
|
|
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
|
+
}
|