@beyondwork/docx-react-component 1.0.28 → 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.
Files changed (92) hide show
  1. package/package.json +26 -37
  2. package/src/api/public-types.ts +531 -0
  3. package/src/api/session-state.ts +2 -0
  4. package/src/core/commands/index.ts +201 -79
  5. package/src/core/commands/table-structure-commands.ts +138 -5
  6. package/src/core/state/text-transaction.ts +370 -3
  7. package/src/index.ts +41 -0
  8. package/src/io/docx-session.ts +318 -25
  9. package/src/io/export/serialize-footnotes.ts +41 -46
  10. package/src/io/export/serialize-headers-footers.ts +36 -40
  11. package/src/io/export/serialize-main-document.ts +55 -89
  12. package/src/io/export/serialize-numbering.ts +104 -4
  13. package/src/io/export/serialize-runtime-revisions.ts +196 -2
  14. package/src/io/export/split-story-blocks-for-runtime-revisions.ts +252 -0
  15. package/src/io/export/table-properties-xml.ts +318 -0
  16. package/src/io/normalize/normalize-text.ts +34 -3
  17. package/src/io/ooxml/parse-comments.ts +6 -0
  18. package/src/io/ooxml/parse-footnotes.ts +69 -13
  19. package/src/io/ooxml/parse-headers-footers.ts +54 -11
  20. package/src/io/ooxml/parse-main-document.ts +112 -42
  21. package/src/io/ooxml/parse-numbering.ts +341 -26
  22. package/src/io/ooxml/parse-revisions.ts +118 -4
  23. package/src/io/ooxml/parse-styles.ts +176 -0
  24. package/src/io/ooxml/parse-tables.ts +34 -25
  25. package/src/io/ooxml/revision-boundaries.ts +127 -3
  26. package/src/io/ooxml/workflow-payload.ts +544 -0
  27. package/src/model/canonical-document.ts +91 -1
  28. package/src/model/snapshot.ts +112 -1
  29. package/src/preservation/store.ts +73 -3
  30. package/src/review/store/comment-store.ts +19 -1
  31. package/src/review/store/revision-actions.ts +29 -0
  32. package/src/review/store/revision-store.ts +12 -1
  33. package/src/review/store/revision-types.ts +11 -0
  34. package/src/runtime/context-analytics.ts +824 -0
  35. package/src/runtime/document-locations.ts +521 -0
  36. package/src/runtime/document-navigation.ts +14 -1
  37. package/src/runtime/document-outline.ts +440 -0
  38. package/src/runtime/document-runtime.ts +941 -45
  39. package/src/runtime/event-refresh-hints.ts +137 -0
  40. package/src/runtime/numbering-prefix.ts +67 -39
  41. package/src/runtime/page-layout-estimation.ts +100 -7
  42. package/src/runtime/resolved-numbering-geometry.ts +293 -0
  43. package/src/runtime/session-capabilities.ts +2 -2
  44. package/src/runtime/suggestions-snapshot.ts +137 -0
  45. package/src/runtime/surface-projection.ts +223 -27
  46. package/src/runtime/table-style-resolver.ts +409 -0
  47. package/src/runtime/view-state.ts +17 -1
  48. package/src/runtime/workflow-markup.ts +54 -14
  49. package/src/ui/WordReviewEditor.tsx +1269 -87
  50. package/src/ui/editor-command-bag.ts +7 -0
  51. package/src/ui/editor-runtime-boundary.ts +111 -10
  52. package/src/ui/editor-shell-view.tsx +17 -15
  53. package/src/ui/editor-surface-controller.tsx +5 -0
  54. package/src/ui/headless/selection-tool-context.ts +19 -0
  55. package/src/ui/headless/selection-tool-resolver.ts +752 -0
  56. package/src/ui/headless/selection-tool-types.ts +129 -0
  57. package/src/ui/headless/selection-toolbar-model.ts +10 -33
  58. package/src/ui/runtime-shortcut-dispatch.ts +365 -0
  59. package/src/ui-tailwind/chrome/chrome-preset-model.ts +107 -0
  60. package/src/ui-tailwind/chrome/chrome-preset-toolbar.tsx +15 -0
  61. package/src/ui-tailwind/chrome/review-queue-bar.tsx +97 -0
  62. package/src/ui-tailwind/chrome/tw-context-analytics-summary.tsx +122 -0
  63. package/src/ui-tailwind/chrome/tw-image-context-toolbar.tsx +1 -9
  64. package/src/ui-tailwind/chrome/tw-object-context-toolbar.tsx +1 -5
  65. package/src/ui-tailwind/chrome/tw-page-ruler.tsx +8 -29
  66. package/src/ui-tailwind/chrome/tw-selection-tool-blocked.tsx +23 -0
  67. package/src/ui-tailwind/chrome/tw-selection-tool-comment.tsx +35 -0
  68. package/src/ui-tailwind/chrome/tw-selection-tool-formatting.tsx +37 -0
  69. package/src/ui-tailwind/chrome/tw-selection-tool-host.tsx +298 -0
  70. package/src/ui-tailwind/chrome/tw-selection-tool-structure.tsx +116 -0
  71. package/src/ui-tailwind/chrome/tw-selection-tool-suggestion.tsx +29 -0
  72. package/src/ui-tailwind/chrome/tw-selection-tool-workflow.tsx +27 -0
  73. package/src/ui-tailwind/chrome/tw-selection-toolbar.tsx +3 -3
  74. package/src/ui-tailwind/chrome/tw-suggestion-card.tsx +3 -3
  75. package/src/ui-tailwind/chrome/tw-table-context-toolbar.tsx +86 -14
  76. package/src/ui-tailwind/editor-surface/pm-command-bridge.ts +57 -52
  77. package/src/ui-tailwind/editor-surface/pm-decorations.ts +36 -52
  78. package/src/ui-tailwind/editor-surface/pm-schema.ts +56 -5
  79. package/src/ui-tailwind/editor-surface/pm-state-from-snapshot.ts +87 -24
  80. package/src/ui-tailwind/editor-surface/surface-build-keys.ts +4 -0
  81. package/src/ui-tailwind/editor-surface/tw-prosemirror-surface.tsx +135 -32
  82. package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +74 -7
  83. package/src/ui-tailwind/review/tw-comment-sidebar.tsx +17 -17
  84. package/src/ui-tailwind/review/tw-review-rail.tsx +19 -17
  85. package/src/ui-tailwind/review/tw-revision-sidebar.tsx +10 -10
  86. package/src/ui-tailwind/status/tw-status-bar.tsx +10 -6
  87. package/src/ui-tailwind/theme/editor-theme.css +58 -40
  88. package/src/ui-tailwind/toolbar/tw-toolbar-icon-button.tsx +4 -4
  89. package/src/ui-tailwind/toolbar/tw-toolbar.tsx +250 -181
  90. package/src/ui-tailwind/tw-review-workspace.tsx +323 -280
  91. package/src/validation/compatibility-engine.ts +246 -2
  92. package/src/validation/docx-comment-proof.ts +24 -11
@@ -0,0 +1,521 @@
1
+ import type {
2
+ CommentSidebarSnapshot,
3
+ DocumentChunkSnapshot,
4
+ DocumentLocationSnapshot,
5
+ DocumentNavigationSnapshot,
6
+ DocumentSectionSnapshot,
7
+ DocumentTextToken,
8
+ EditorAnchorProjection,
9
+ EditorStoryTarget,
10
+ EditorSurfaceSnapshot,
11
+ RestorePointSnapshot,
12
+ ReviewWorkSnapshot,
13
+ RuntimeRenderSnapshot,
14
+ SelectionSnapshot,
15
+ StoryTextStreamSnapshot,
16
+ TrackedChangesSnapshot,
17
+ WorkflowMarkupSnapshot,
18
+ } from "../api/public-types";
19
+ import type { CanonicalDocumentEnvelope } from "../core/state/editor-state.ts";
20
+ import { MAIN_STORY_TARGET } from "../core/selection/mapping.ts";
21
+ import {
22
+ createDocumentSectionSnapshots,
23
+ findBookmarkNameForOffset,
24
+ resolveHeadingPath,
25
+ } from "./document-outline.ts";
26
+ import { storyTargetKey } from "./story-targeting.ts";
27
+
28
+ function getAnchorOffset(anchor: EditorAnchorProjection): number | undefined {
29
+ switch (anchor.kind) {
30
+ case "range":
31
+ return anchor.from;
32
+ case "node":
33
+ return anchor.at;
34
+ case "detached":
35
+ return anchor.lastKnownRange.from;
36
+ }
37
+ }
38
+
39
+ function createLocationId(anchor: EditorAnchorProjection, storyTarget?: EditorStoryTarget): string {
40
+ switch (anchor.kind) {
41
+ case "range":
42
+ return `location:${storyTargetKey(storyTarget ?? MAIN_STORY_TARGET)}:range:${anchor.from}:${anchor.to}`;
43
+ case "node":
44
+ return `location:${storyTargetKey(storyTarget ?? MAIN_STORY_TARGET)}:node:${anchor.at}`;
45
+ case "detached":
46
+ return `location:${storyTargetKey(storyTarget ?? MAIN_STORY_TARGET)}:detached:${anchor.lastKnownRange.from}:${anchor.lastKnownRange.to}:${anchor.reason}`;
47
+ }
48
+ }
49
+
50
+ function createPublicRangeAnchor(from: number, to: number): EditorAnchorProjection {
51
+ return {
52
+ kind: "range",
53
+ from,
54
+ to,
55
+ assoc: { start: -1, end: 1 },
56
+ };
57
+ }
58
+
59
+ function resolveOffsetMetadata(
60
+ navigation: DocumentNavigationSnapshot,
61
+ offset: number,
62
+ ): { pageIndex?: number; sectionIndex?: number } {
63
+ const page = navigation.pages.find(
64
+ (entry) => offset >= entry.startOffset && offset < entry.endOffset,
65
+ );
66
+ return {
67
+ ...(typeof page?.pageIndex === "number" ? { pageIndex: page.pageIndex } : {}),
68
+ ...(typeof page?.sectionIndex === "number"
69
+ ? { sectionIndex: page.sectionIndex }
70
+ : {}),
71
+ };
72
+ }
73
+
74
+ export function createDocumentLocationSnapshot(input: {
75
+ document: CanonicalDocumentEnvelope;
76
+ navigation: DocumentNavigationSnapshot;
77
+ sections?: DocumentSectionSnapshot[];
78
+ anchor: EditorAnchorProjection;
79
+ storyTarget?: EditorStoryTarget;
80
+ source: DocumentLocationSnapshot["source"];
81
+ }): DocumentLocationSnapshot {
82
+ const storyTarget = input.storyTarget ?? MAIN_STORY_TARGET;
83
+ const offset = getAnchorOffset(input.anchor);
84
+
85
+ if (storyTarget.kind !== "main") {
86
+ return {
87
+ locationId: createLocationId(input.anchor, storyTarget),
88
+ anchor: input.anchor,
89
+ storyTarget,
90
+ ...("sectionIndex" in storyTarget && typeof storyTarget.sectionIndex === "number"
91
+ ? { sectionIndex: storyTarget.sectionIndex }
92
+ : {}),
93
+ source: input.source,
94
+ };
95
+ }
96
+
97
+ const page = input.navigation.pages.find(
98
+ (entry) => offset !== undefined && offset >= entry.startOffset && offset < entry.endOffset,
99
+ );
100
+ const section =
101
+ input.sections?.find((entry) => entry.sectionIndex === page?.sectionIndex) ??
102
+ createDocumentSectionSnapshots(input.document, input.navigation).find(
103
+ (entry) => entry.sectionIndex === page?.sectionIndex,
104
+ );
105
+ const headingContext = resolveHeadingPath(input.navigation.headings, offset);
106
+ const bookmarkName = findBookmarkNameForOffset(input.document, offset);
107
+
108
+ return {
109
+ locationId: createLocationId(input.anchor, storyTarget),
110
+ anchor: input.anchor,
111
+ storyTarget,
112
+ ...(section ? { sectionIndex: section.sectionIndex } : {}),
113
+ ...(typeof page?.pageIndex === "number" ? { pageIndex: page.pageIndex } : {}),
114
+ ...(headingContext.headingId ? { headingId: headingContext.headingId } : {}),
115
+ ...(headingContext.headingPath.length > 0
116
+ ? { headingPath: headingContext.headingPath }
117
+ : {}),
118
+ ...(bookmarkName ? { bookmarkName } : {}),
119
+ source: input.source,
120
+ };
121
+ }
122
+
123
+ export function createSelectionFromAnchor(
124
+ anchor: EditorAnchorProjection,
125
+ storyTarget?: EditorStoryTarget,
126
+ ): SelectionSnapshot {
127
+ switch (anchor.kind) {
128
+ case "range":
129
+ return {
130
+ anchor: anchor.from,
131
+ head: anchor.to,
132
+ isCollapsed: anchor.from === anchor.to,
133
+ activeRange: anchor,
134
+ ...(storyTarget ? { storyTarget } : {}),
135
+ };
136
+ case "node":
137
+ return {
138
+ anchor: anchor.at,
139
+ head: anchor.at,
140
+ isCollapsed: true,
141
+ activeRange: anchor,
142
+ ...(storyTarget ? { storyTarget } : {}),
143
+ };
144
+ case "detached":
145
+ return {
146
+ anchor: anchor.lastKnownRange.from,
147
+ head: anchor.lastKnownRange.to,
148
+ isCollapsed: anchor.lastKnownRange.from === anchor.lastKnownRange.to,
149
+ activeRange: anchor,
150
+ ...(storyTarget ? { storyTarget } : {}),
151
+ };
152
+ }
153
+ }
154
+
155
+ function pushParagraphTokens(
156
+ tokens: DocumentTextToken[],
157
+ block: Extract<EditorSurfaceSnapshot["blocks"][number], { kind: "paragraph" }>,
158
+ storyTarget: EditorStoryTarget,
159
+ navigation: DocumentNavigationSnapshot,
160
+ ): void {
161
+ const location = resolveOffsetMetadata(navigation, block.from);
162
+
163
+ tokens.push({
164
+ tokenId: `${block.blockId}:boundary`,
165
+ storyTarget,
166
+ blockId: block.blockId,
167
+ blockKind: block.kind,
168
+ kind: "paragraph_boundary",
169
+ from: block.from,
170
+ to: block.from,
171
+ ...(typeof location.pageIndex === "number" ? { pageIndex: location.pageIndex } : {}),
172
+ ...(typeof location.sectionIndex === "number"
173
+ ? { sectionIndex: location.sectionIndex }
174
+ : {}),
175
+ });
176
+
177
+ for (const segment of block.segments) {
178
+ tokens.push({
179
+ tokenId: `${block.blockId}:${segment.segmentId}`,
180
+ storyTarget,
181
+ blockId: block.blockId,
182
+ blockKind: block.kind,
183
+ segmentId: segment.segmentId,
184
+ kind: segment.kind,
185
+ from: segment.from,
186
+ to: segment.to,
187
+ ...("text" in segment && segment.text ? { text: segment.text } : {}),
188
+ ...("marks" in segment && segment.marks ? { marks: segment.marks } : {}),
189
+ ...(typeof location.pageIndex === "number" ? { pageIndex: location.pageIndex } : {}),
190
+ ...(typeof location.sectionIndex === "number"
191
+ ? { sectionIndex: location.sectionIndex }
192
+ : {}),
193
+ });
194
+ }
195
+ }
196
+
197
+ function flattenParagraphText(
198
+ block: Extract<EditorSurfaceSnapshot["blocks"][number], { kind: "paragraph" }>,
199
+ ): string {
200
+ return block.segments
201
+ .map((segment) => {
202
+ switch (segment.kind) {
203
+ case "text":
204
+ return segment.text;
205
+ case "tab":
206
+ return "\t";
207
+ case "hard_break":
208
+ return "\n";
209
+ case "image":
210
+ return segment.altText ?? "[image]";
211
+ case "note_ref":
212
+ case "field_ref":
213
+ return segment.label;
214
+ case "opaque_inline":
215
+ return segment.label;
216
+ }
217
+ })
218
+ .join("");
219
+ }
220
+
221
+ function getChunkHeadingPath(
222
+ navigation: DocumentNavigationSnapshot,
223
+ offset: number,
224
+ ): string[] {
225
+ return resolveHeadingPath(navigation.headings, offset).headingPath;
226
+ }
227
+
228
+ function getChunkCautionFlags(
229
+ block: EditorSurfaceSnapshot["blocks"][number],
230
+ ): DocumentChunkSnapshot["cautionFlags"] {
231
+ switch (block.kind) {
232
+ case "table":
233
+ return ["contains_table"];
234
+ case "opaque_block":
235
+ return ["contains_preserve_only"];
236
+ case "paragraph":
237
+ return block.segments.some((segment) => segment.kind === "opaque_inline")
238
+ ? ["contains_preserve_only"]
239
+ : [];
240
+ case "sdt_block":
241
+ return [];
242
+ }
243
+ }
244
+
245
+ export function createDocumentTextStreamSnapshots(input: {
246
+ surface: EditorSurfaceSnapshot;
247
+ navigation: DocumentNavigationSnapshot;
248
+ }): StoryTextStreamSnapshot[] {
249
+ const streams: StoryTextStreamSnapshot[] = [];
250
+ const stories = [
251
+ {
252
+ target: MAIN_STORY_TARGET,
253
+ label: "Main document",
254
+ storySize: input.surface.storySize,
255
+ blocks: input.surface.blocks,
256
+ },
257
+ ...input.surface.secondaryStories,
258
+ ];
259
+
260
+ for (const story of stories) {
261
+ const tokens: DocumentTextToken[] = [];
262
+ for (const block of story.blocks) {
263
+ if (block.kind === "paragraph") {
264
+ pushParagraphTokens(tokens, block, story.target, input.navigation);
265
+ continue;
266
+ }
267
+ if (block.kind === "table") {
268
+ tokens.push({
269
+ tokenId: `${block.blockId}:table`,
270
+ storyTarget: story.target,
271
+ blockId: block.blockId,
272
+ blockKind: block.kind,
273
+ kind: "table_boundary",
274
+ from: block.from,
275
+ to: block.to,
276
+ });
277
+ continue;
278
+ }
279
+ if (block.kind === "sdt_block") {
280
+ for (const child of block.children) {
281
+ if (child.kind === "paragraph") {
282
+ pushParagraphTokens(tokens, child, story.target, input.navigation);
283
+ }
284
+ }
285
+ }
286
+ }
287
+ streams.push({
288
+ target: story.target,
289
+ label: story.label,
290
+ storySize: story.storySize,
291
+ tokens,
292
+ });
293
+ }
294
+
295
+ return streams;
296
+ }
297
+
298
+ export function createDocumentChunks(input: {
299
+ document: CanonicalDocumentEnvelope;
300
+ renderSnapshot: RuntimeRenderSnapshot;
301
+ navigation: DocumentNavigationSnapshot;
302
+ }): DocumentChunkSnapshot[] {
303
+ const surface = input.renderSnapshot.surface;
304
+ if (!surface) {
305
+ return [];
306
+ }
307
+
308
+ const chunks: DocumentChunkSnapshot[] = [];
309
+ for (const block of surface.blocks) {
310
+ const publicAnchor = createPublicRangeAnchor(block.from, block.to);
311
+ const location = createDocumentLocationSnapshot({
312
+ document: input.document,
313
+ navigation: input.navigation,
314
+ anchor: publicAnchor,
315
+ storyTarget: MAIN_STORY_TARGET,
316
+ source: { kind: "navigation" },
317
+ });
318
+ const headingPath = getChunkHeadingPath(input.navigation, block.from);
319
+
320
+ if (block.kind === "paragraph") {
321
+ chunks.push({
322
+ chunkId: `chunk:${block.blockId}`,
323
+ kind: "retrieval",
324
+ label: headingPath.at(-1) ?? "Paragraph",
325
+ text: flattenParagraphText(block),
326
+ storyTarget: MAIN_STORY_TARGET,
327
+ anchor: publicAnchor,
328
+ ...(typeof location.sectionIndex === "number"
329
+ ? { sectionIndex: location.sectionIndex }
330
+ : {}),
331
+ ...(typeof location.pageIndex === "number"
332
+ ? { pageRange: { start: location.pageIndex, end: location.pageIndex } }
333
+ : {}),
334
+ headingPath,
335
+ blockRefs: [{ blockId: block.blockId, kind: block.kind }],
336
+ source: { type: "surface" },
337
+ cautionFlags: getChunkCautionFlags(block),
338
+ });
339
+ continue;
340
+ }
341
+
342
+ if (block.kind === "table") {
343
+ chunks.push({
344
+ chunkId: `chunk:${block.blockId}`,
345
+ kind: "retrieval",
346
+ label: headingPath.at(-1) ?? "Table",
347
+ text: "[table]",
348
+ storyTarget: MAIN_STORY_TARGET,
349
+ anchor: publicAnchor,
350
+ ...(typeof location.sectionIndex === "number"
351
+ ? { sectionIndex: location.sectionIndex }
352
+ : {}),
353
+ ...(typeof location.pageIndex === "number"
354
+ ? { pageRange: { start: location.pageIndex, end: location.pageIndex } }
355
+ : {}),
356
+ headingPath,
357
+ blockRefs: [{ blockId: block.blockId, kind: block.kind }],
358
+ source: { type: "surface" },
359
+ cautionFlags: getChunkCautionFlags(block),
360
+ });
361
+ continue;
362
+ }
363
+
364
+ if (block.kind === "opaque_block") {
365
+ chunks.push({
366
+ chunkId: `chunk:${block.blockId}`,
367
+ kind: "retrieval",
368
+ label: block.label,
369
+ text: block.detail,
370
+ storyTarget: MAIN_STORY_TARGET,
371
+ anchor: publicAnchor,
372
+ ...(typeof location.sectionIndex === "number"
373
+ ? { sectionIndex: location.sectionIndex }
374
+ : {}),
375
+ ...(typeof location.pageIndex === "number"
376
+ ? { pageRange: { start: location.pageIndex, end: location.pageIndex } }
377
+ : {}),
378
+ headingPath,
379
+ blockRefs: [{ blockId: block.blockId, kind: block.kind }],
380
+ source: { type: "surface" },
381
+ cautionFlags: getChunkCautionFlags(block),
382
+ });
383
+ }
384
+ }
385
+
386
+ return chunks;
387
+ }
388
+
389
+ export function createWorkflowChunks(input: {
390
+ document: CanonicalDocumentEnvelope;
391
+ navigation: DocumentNavigationSnapshot;
392
+ workflowMarkup: WorkflowMarkupSnapshot;
393
+ }): DocumentChunkSnapshot[] {
394
+ return input.workflowMarkup.items.map((item) => {
395
+ const location = createDocumentLocationSnapshot({
396
+ document: input.document,
397
+ navigation: input.navigation,
398
+ anchor: item.anchor,
399
+ storyTarget: item.storyTarget,
400
+ source: { kind: "workflow", markupId: item.markupId },
401
+ });
402
+ return {
403
+ chunkId: `workflow:${item.markupId}`,
404
+ kind: "workflow",
405
+ label: item.label,
406
+ text: item.excerpt ?? item.label,
407
+ ...(item.storyTarget ? { storyTarget: item.storyTarget } : {}),
408
+ anchor: item.anchor,
409
+ ...(typeof location.sectionIndex === "number"
410
+ ? { sectionIndex: location.sectionIndex }
411
+ : {}),
412
+ ...(typeof location.pageIndex === "number"
413
+ ? { pageRange: { start: location.pageIndex, end: location.pageIndex } }
414
+ : {}),
415
+ headingPath: location.headingPath ?? [],
416
+ blockRefs: [],
417
+ source: { type: "workflow", markupId: item.markupId, candidateId: item.markupId },
418
+ cautionFlags:
419
+ item.anchor.kind === "detached"
420
+ ? ["contains_detached_anchor"]
421
+ : item.kind === "protected_range"
422
+ ? ["contains_protected_range"]
423
+ : item.kind === "opaque_fragment"
424
+ ? ["contains_preserve_only"]
425
+ : [],
426
+ };
427
+ });
428
+ }
429
+
430
+ export function createReviewWorkSnapshot(input: {
431
+ comments: CommentSidebarSnapshot;
432
+ trackedChanges: TrackedChangesSnapshot;
433
+ workflowMarkup: WorkflowMarkupSnapshot;
434
+ document: CanonicalDocumentEnvelope;
435
+ navigation: DocumentNavigationSnapshot;
436
+ }): ReviewWorkSnapshot {
437
+ const items = input.workflowMarkup.items
438
+ .filter((item) =>
439
+ item.kind === "comment" ||
440
+ item.kind === "revision" ||
441
+ item.kind === "protected_range" ||
442
+ item.kind === "opaque_fragment",
443
+ )
444
+ .map((item) => {
445
+ const location = createDocumentLocationSnapshot({
446
+ document: input.document,
447
+ navigation: input.navigation,
448
+ anchor: item.anchor,
449
+ storyTarget: item.storyTarget,
450
+ source: { kind: "workflow", markupId: item.markupId },
451
+ });
452
+ return {
453
+ workId: item.markupId,
454
+ kind: item.kind,
455
+ label: item.label,
456
+ excerpt: item.excerpt ?? item.label,
457
+ anchor: item.anchor,
458
+ ...(item.storyTarget ? { storyTarget: item.storyTarget } : {}),
459
+ ...(typeof location.sectionIndex === "number"
460
+ ? { sectionIndex: location.sectionIndex }
461
+ : {}),
462
+ ...(typeof location.pageIndex === "number"
463
+ ? { pageIndex: location.pageIndex }
464
+ : {}),
465
+ };
466
+ });
467
+
468
+ return {
469
+ openCommentCount: input.comments.openCommentIds.length,
470
+ actionableRevisionCount: input.trackedChanges.actionableChangeIds.length,
471
+ items,
472
+ };
473
+ }
474
+
475
+ export function createRestorePoint(input: {
476
+ location: DocumentLocationSnapshot;
477
+ revisionToken: string;
478
+ createdAt: string;
479
+ checkpointType: RestorePointSnapshot["checkpointType"];
480
+ }): RestorePointSnapshot {
481
+ return {
482
+ restorePointId: `restore:${input.revisionToken}:${input.location.locationId}`,
483
+ location: input.location,
484
+ revisionToken: input.revisionToken,
485
+ createdAt: input.createdAt,
486
+ checkpointType: input.checkpointType,
487
+ };
488
+ }
489
+
490
+ export function createCurrentLocation(input: {
491
+ document: CanonicalDocumentEnvelope;
492
+ renderSnapshot: RuntimeRenderSnapshot;
493
+ navigation: DocumentNavigationSnapshot;
494
+ sections?: DocumentSectionSnapshot[];
495
+ }): DocumentLocationSnapshot | null {
496
+ const selection = input.renderSnapshot.selection;
497
+ return createDocumentLocationSnapshot({
498
+ document: input.document,
499
+ navigation: input.navigation,
500
+ sections: input.sections,
501
+ anchor: selection.activeRange,
502
+ storyTarget: selection.storyTarget ?? input.renderSnapshot.activeStory,
503
+ source: { kind: "selection" },
504
+ });
505
+ }
506
+
507
+ export function createLocationFromSelection(input: {
508
+ document: CanonicalDocumentEnvelope;
509
+ navigation: DocumentNavigationSnapshot;
510
+ sections?: DocumentSectionSnapshot[];
511
+ selection: SelectionSnapshot;
512
+ }): DocumentLocationSnapshot | null {
513
+ return createDocumentLocationSnapshot({
514
+ document: input.document,
515
+ navigation: input.navigation,
516
+ sections: input.sections,
517
+ anchor: input.selection.activeRange,
518
+ storyTarget: input.selection.storyTarget ?? MAIN_STORY_TARGET,
519
+ source: { kind: "selection" },
520
+ });
521
+ }
@@ -599,5 +599,18 @@ function extractParagraphText(
599
599
  break;
600
600
  }
601
601
  }
602
- return text;
602
+ if (!block.numberingPrefix) {
603
+ return text;
604
+ }
605
+ const trimmedText = text.trimStart();
606
+ if (
607
+ trimmedText === block.numberingPrefix ||
608
+ trimmedText.startsWith(`${block.numberingPrefix} `) ||
609
+ trimmedText.startsWith(`${block.numberingPrefix}\t`) ||
610
+ (block.numberingSuffix === "nothing" && trimmedText.startsWith(block.numberingPrefix))
611
+ ) {
612
+ return text;
613
+ }
614
+ const separator = block.numberingSuffix === "nothing" ? "" : " ";
615
+ return `${block.numberingPrefix}${separator}${text}`;
603
616
  }