@beyondwork/docx-react-component 1.0.1 → 1.0.2
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 +44 -104
- package/package.json +76 -46
- package/src/README.md +85 -0
- package/src/api/README.md +22 -0
- package/src/api/public-types.ts +525 -0
- package/src/compare/diff-engine.ts +530 -0
- package/src/compare/export-redlines.ts +162 -0
- package/src/compare/snapshot.ts +37 -0
- package/src/component-inventory.md +99 -0
- package/src/core/README.md +10 -0
- package/src/core/commands/README.md +3 -0
- package/src/core/commands/formatting-commands.ts +161 -0
- package/src/core/commands/image-commands.ts +144 -0
- package/src/core/commands/index.ts +1013 -0
- package/src/core/commands/list-commands.ts +370 -0
- package/src/core/commands/review-commands.ts +108 -0
- package/src/core/commands/text-commands.ts +119 -0
- package/src/core/schema/README.md +3 -0
- package/src/core/schema/text-schema.ts +512 -0
- package/src/core/selection/README.md +3 -0
- package/src/core/selection/mapping.ts +238 -0
- package/src/core/selection/review-anchors.ts +94 -0
- package/src/core/state/README.md +3 -0
- package/src/core/state/editor-state.ts +580 -0
- package/src/core/state/text-transaction.ts +276 -0
- package/src/formats/xlsx/io/parse-shared-strings.ts +41 -0
- package/src/formats/xlsx/io/parse-sheet.ts +289 -0
- package/src/formats/xlsx/io/parse-styles.ts +57 -0
- package/src/formats/xlsx/io/parse-workbook.ts +75 -0
- package/src/formats/xlsx/io/xlsx-session.ts +306 -0
- package/src/formats/xlsx/model/cell.ts +189 -0
- package/src/formats/xlsx/model/sheet.ts +244 -0
- package/src/formats/xlsx/model/styles.ts +118 -0
- package/src/formats/xlsx/model/workbook.ts +449 -0
- package/src/index.ts +45 -0
- package/src/io/README.md +10 -0
- package/src/io/docx-session.ts +1763 -0
- package/src/io/export/README.md +3 -0
- package/src/io/export/export-session.ts +165 -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 +876 -0
- package/src/io/export/serialize-footnotes.ts +217 -0
- package/src/io/export/serialize-headers-footers.ts +200 -0
- package/src/io/export/serialize-main-document.ts +982 -0
- package/src/io/export/serialize-numbering.ts +97 -0
- package/src/io/export/serialize-revisions.ts +389 -0
- package/src/io/export/serialize-runtime-revisions.ts +265 -0
- package/src/io/export/serialize-tables.ts +147 -0
- package/src/io/export/split-review-boundaries.ts +194 -0
- package/src/io/normalize/README.md +3 -0
- package/src/io/normalize/normalize-text.ts +437 -0
- package/src/io/ooxml/README.md +3 -0
- package/src/io/ooxml/parse-comments.ts +779 -0
- package/src/io/ooxml/parse-complex-content.ts +287 -0
- package/src/io/ooxml/parse-fields.ts +438 -0
- package/src/io/ooxml/parse-footnotes.ts +403 -0
- package/src/io/ooxml/parse-headers-footers.ts +483 -0
- package/src/io/ooxml/parse-inline-media.ts +431 -0
- package/src/io/ooxml/parse-main-document.ts +1846 -0
- package/src/io/ooxml/parse-numbering.ts +425 -0
- package/src/io/ooxml/parse-revisions.ts +658 -0
- package/src/io/ooxml/parse-shapes.ts +271 -0
- package/src/io/ooxml/parse-tables.ts +568 -0
- package/src/io/ooxml/parse-theme.ts +314 -0
- package/src/io/ooxml/part-manifest.ts +136 -0
- package/src/io/ooxml/revision-boundaries.ts +351 -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 +320 -0
- package/src/io/opc/package-writer.ts +273 -0
- package/src/legal/bookmarks.ts +196 -0
- package/src/legal/cross-references.ts +356 -0
- package/src/legal/defined-terms.ts +203 -0
- package/src/model/README.md +3 -0
- package/src/model/canonical-document.ts +1911 -0
- package/src/model/cds-1.0.0.ts +196 -0
- package/src/model/snapshot.ts +393 -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 +120 -0
- package/src/preservation/preserved-part-manifest.ts +56 -0
- package/src/preservation/relationship-retention.ts +57 -0
- package/src/preservation/store.ts +185 -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 +331 -0
- package/src/review/store/comment-thread.ts +109 -0
- package/src/review/store/revision-actions.ts +394 -0
- package/src/review/store/revision-store.ts +303 -0
- package/src/review/store/revision-types.ts +168 -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/document-runtime.ts +967 -0
- package/src/runtime/read-only-diagnostics-runtime.ts +232 -0
- package/src/runtime/review-runtime.ts +44 -0
- package/src/runtime/revision-runtime.ts +107 -0
- package/src/runtime/session-capabilities.ts +138 -0
- package/src/runtime/surface-projection.ts +570 -0
- package/src/runtime/table-commands.ts +87 -0
- package/src/runtime/table-schema.ts +140 -0
- package/src/runtime/virtualized-rendering.ts +258 -0
- package/src/ui/README.md +30 -0
- package/src/ui/WordReviewEditor.tsx +1504 -0
- package/src/ui/comments/README.md +3 -0
- package/src/ui/compatibility/README.md +3 -0
- package/src/ui/editor-surface/README.md +3 -0
- package/src/ui/headless/comment-decoration-model.ts +124 -0
- package/src/ui/headless/revision-decoration-model.ts +128 -0
- package/src/ui/headless/selection-helpers.ts +34 -0
- package/src/ui/headless/use-editor-keyboard.ts +98 -0
- package/src/ui/review/README.md +3 -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-tailwind/chrome/tw-alert-banner.tsx +48 -0
- package/src/ui-tailwind/chrome/tw-selection-toolbar.tsx +44 -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/pm-command-bridge.ts +139 -0
- package/src/ui-tailwind/editor-surface/pm-decorations.ts +98 -0
- package/src/ui-tailwind/editor-surface/pm-position-map.ts +123 -0
- package/src/ui-tailwind/editor-surface/pm-schema.ts +452 -0
- package/src/ui-tailwind/editor-surface/pm-state-from-snapshot.ts +327 -0
- package/src/ui-tailwind/editor-surface/search-plugin.ts +157 -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 +118 -0
- package/src/ui-tailwind/editor-surface/tw-opaque-block.tsx +52 -0
- package/src/ui-tailwind/editor-surface/tw-paragraph-block.tsx +151 -0
- package/src/ui-tailwind/editor-surface/tw-prosemirror-surface.tsx +215 -0
- package/src/ui-tailwind/editor-surface/tw-segment-view.tsx +111 -0
- package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +122 -0
- package/src/ui-tailwind/index.ts +61 -0
- package/src/ui-tailwind/review/tw-comment-sidebar.tsx +276 -0
- package/src/ui-tailwind/review/tw-health-panel.tsx +120 -0
- package/src/ui-tailwind/review/tw-review-rail.tsx +120 -0
- package/src/ui-tailwind/review/tw-revision-sidebar.tsx +164 -0
- package/src/ui-tailwind/status/tw-status-bar.tsx +58 -0
- package/src/ui-tailwind/theme/editor-theme.css +190 -0
- package/src/ui-tailwind/toolbar/tw-toolbar-icon-button.tsx +48 -0
- package/src/ui-tailwind/toolbar/tw-toolbar.tsx +231 -0
- package/src/ui-tailwind/tw-review-workspace.tsx +140 -0
- package/src/validation/README.md +3 -0
- package/src/validation/compatibility-engine.ts +317 -0
- package/src/validation/compatibility-report.ts +160 -0
- package/src/validation/diagnostics.ts +203 -0
- package/src/validation/import-diagnostics.ts +128 -0
- package/src/validation/low-priority-word-surfaces.ts +373 -0
- package/dist/chunk-32W6IVQE.js +0 -7725
- package/dist/chunk-32W6IVQE.js.map +0 -1
- package/dist/index.cjs +0 -23722
- package/dist/index.cjs.map +0 -1
- package/dist/index.d.cts +0 -7
- package/dist/index.d.ts +0 -7
- package/dist/index.js +0 -16011
- package/dist/index.js.map +0 -1
- package/dist/public-types-DqCURAz8.d.cts +0 -1152
- package/dist/public-types-DqCURAz8.d.ts +0 -1152
- package/dist/tailwind.cjs +0 -8295
- package/dist/tailwind.cjs.map +0 -1
- package/dist/tailwind.d.cts +0 -323
- package/dist/tailwind.d.ts +0 -323
- package/dist/tailwind.js +0 -553
- package/dist/tailwind.js.map +0 -1
|
@@ -0,0 +1,327 @@
|
|
|
1
|
+
import { Fragment, type Node as PMNode } from "prosemirror-model";
|
|
2
|
+
import { EditorState, type Plugin, TextSelection } from "prosemirror-state";
|
|
3
|
+
|
|
4
|
+
import type {
|
|
5
|
+
EditorSurfaceSnapshot,
|
|
6
|
+
SelectionSnapshot,
|
|
7
|
+
SurfaceBlockSnapshot,
|
|
8
|
+
SurfaceInlineSegment,
|
|
9
|
+
SurfaceTableRowSnapshot,
|
|
10
|
+
SurfaceTableCellSnapshot,
|
|
11
|
+
} from "../../api/public-types";
|
|
12
|
+
import { editorSchema } from "./pm-schema";
|
|
13
|
+
import { buildPositionMap, type PositionMap } from "./pm-position-map";
|
|
14
|
+
|
|
15
|
+
export interface PMStateResult {
|
|
16
|
+
state: EditorState;
|
|
17
|
+
positionMap: PositionMap;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* Create a ProseMirror EditorState from a runtime surface snapshot.
|
|
22
|
+
*
|
|
23
|
+
* The PM document mirrors the snapshot for rendering and input.
|
|
24
|
+
* It is NOT the canonical source — the runtime owns that.
|
|
25
|
+
*/
|
|
26
|
+
export function createPMStateFromSnapshot(
|
|
27
|
+
surface: EditorSurfaceSnapshot,
|
|
28
|
+
selection: SelectionSnapshot,
|
|
29
|
+
plugins: Plugin[],
|
|
30
|
+
): PMStateResult {
|
|
31
|
+
const doc = buildPMDoc(surface);
|
|
32
|
+
const positionMap = buildPositionMap(surface);
|
|
33
|
+
|
|
34
|
+
// Convert runtime selection to PM selection
|
|
35
|
+
const pmAnchor = clamp(
|
|
36
|
+
positionMap.runtimeToPm(selection.anchor),
|
|
37
|
+
1,
|
|
38
|
+
positionMap.pmDocSize - 1,
|
|
39
|
+
);
|
|
40
|
+
const pmHead = clamp(
|
|
41
|
+
positionMap.runtimeToPm(selection.head),
|
|
42
|
+
1,
|
|
43
|
+
positionMap.pmDocSize - 1,
|
|
44
|
+
);
|
|
45
|
+
|
|
46
|
+
let pmSelection: TextSelection;
|
|
47
|
+
try {
|
|
48
|
+
pmSelection = TextSelection.create(doc, pmAnchor, pmHead);
|
|
49
|
+
} catch {
|
|
50
|
+
// If the position is invalid (e.g., inside an atom), fall back to start
|
|
51
|
+
pmSelection = TextSelection.create(doc, 1);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
const state = EditorState.create({
|
|
55
|
+
doc,
|
|
56
|
+
selection: pmSelection,
|
|
57
|
+
plugins,
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
return { state, positionMap };
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
function buildPMDoc(surface: EditorSurfaceSnapshot): PMNode {
|
|
64
|
+
const blocks: PMNode[] = [];
|
|
65
|
+
|
|
66
|
+
for (const block of surface.blocks) {
|
|
67
|
+
if (block.kind === "paragraph") {
|
|
68
|
+
blocks.push(buildParagraph(block));
|
|
69
|
+
} else if (block.kind === "table") {
|
|
70
|
+
blocks.push(buildTable(block));
|
|
71
|
+
} else if (block.kind === "sdt_block") {
|
|
72
|
+
blocks.push(buildSdtBlock(block));
|
|
73
|
+
} else {
|
|
74
|
+
blocks.push(buildOpaqueBlock(block));
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// Ensure at least one block (PM requires non-empty doc)
|
|
79
|
+
if (blocks.length === 0) {
|
|
80
|
+
blocks.push(editorSchema.nodes.paragraph.create());
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
return editorSchema.nodes.doc.create(null, Fragment.from(blocks));
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
function buildParagraph(
|
|
87
|
+
block: Extract<SurfaceBlockSnapshot, { kind: "paragraph" }>,
|
|
88
|
+
): PMNode {
|
|
89
|
+
const content: PMNode[] = [];
|
|
90
|
+
|
|
91
|
+
for (const segment of block.segments) {
|
|
92
|
+
const nodes = buildInlineContent(segment);
|
|
93
|
+
content.push(...nodes);
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
return editorSchema.nodes.paragraph.create(
|
|
97
|
+
{
|
|
98
|
+
styleId: block.styleId ?? null,
|
|
99
|
+
numberingInstanceId: block.numbering?.numberingInstanceId ?? null,
|
|
100
|
+
numberingLevel: block.numbering?.level ?? null,
|
|
101
|
+
},
|
|
102
|
+
content.length > 0 ? Fragment.from(content) : undefined,
|
|
103
|
+
);
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
function buildInlineContent(segment: SurfaceInlineSegment): PMNode[] {
|
|
107
|
+
switch (segment.kind) {
|
|
108
|
+
case "text": {
|
|
109
|
+
if (!segment.text) return [];
|
|
110
|
+
|
|
111
|
+
// Build PM marks from segment marks
|
|
112
|
+
let marks = editorSchema.marks.bold.isInSet([])
|
|
113
|
+
? [] // shouldn't happen, just type safety
|
|
114
|
+
: [];
|
|
115
|
+
|
|
116
|
+
const pmMarks = [];
|
|
117
|
+
if (segment.marks) {
|
|
118
|
+
for (const mark of segment.marks) {
|
|
119
|
+
const pmMark = editorSchema.marks[mark];
|
|
120
|
+
if (pmMark) {
|
|
121
|
+
pmMarks.push(pmMark.create());
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
if (segment.hyperlinkHref) {
|
|
126
|
+
pmMarks.push(editorSchema.marks.link.create({ href: segment.hyperlinkHref }));
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
return [editorSchema.text(segment.text, pmMarks.length > 0 ? pmMarks : undefined)];
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
case "hard_break":
|
|
133
|
+
return [editorSchema.nodes.hard_break.create()];
|
|
134
|
+
|
|
135
|
+
case "tab":
|
|
136
|
+
return [editorSchema.nodes.tab_char.create()];
|
|
137
|
+
|
|
138
|
+
case "image":
|
|
139
|
+
return [
|
|
140
|
+
editorSchema.nodes.image_atom.create({
|
|
141
|
+
mediaId: segment.mediaId,
|
|
142
|
+
altText: segment.altText ?? null,
|
|
143
|
+
state: segment.state,
|
|
144
|
+
display: segment.display ?? "inline",
|
|
145
|
+
detail: segment.detail ?? null,
|
|
146
|
+
}),
|
|
147
|
+
];
|
|
148
|
+
|
|
149
|
+
case "opaque_inline":
|
|
150
|
+
return [buildOpaqueInlineOrComplexAtom(segment)];
|
|
151
|
+
|
|
152
|
+
default:
|
|
153
|
+
return [];
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
function buildTable(
|
|
158
|
+
block: Extract<SurfaceBlockSnapshot, { kind: "table" }>,
|
|
159
|
+
): PMNode {
|
|
160
|
+
const rows: PMNode[] = [];
|
|
161
|
+
for (const row of block.rows) {
|
|
162
|
+
const cells: PMNode[] = [];
|
|
163
|
+
for (const cell of row.cells) {
|
|
164
|
+
const cellContent: PMNode[] = [];
|
|
165
|
+
for (const child of cell.content) {
|
|
166
|
+
if (child.kind === "paragraph") {
|
|
167
|
+
cellContent.push(buildParagraph(child as Extract<SurfaceBlockSnapshot, { kind: "paragraph" }>));
|
|
168
|
+
} else if (child.kind === "table") {
|
|
169
|
+
cellContent.push(buildNestedTablePlaceholder(child as Extract<SurfaceBlockSnapshot, { kind: "table" }>));
|
|
170
|
+
} else if (child.kind === "sdt_block") {
|
|
171
|
+
cellContent.push(buildOpaqueBlock({
|
|
172
|
+
blockId: child.blockId,
|
|
173
|
+
kind: "opaque_block",
|
|
174
|
+
from: child.from,
|
|
175
|
+
to: child.to,
|
|
176
|
+
fragmentId: child.blockId,
|
|
177
|
+
warningId: child.blockId,
|
|
178
|
+
label: child.alias ?? child.tag ?? "Content control",
|
|
179
|
+
detail: "Structured content control remains read-only inside table cells.",
|
|
180
|
+
state: "locked-preserve-only",
|
|
181
|
+
}));
|
|
182
|
+
} else if (child.kind === "opaque_block") {
|
|
183
|
+
cellContent.push(buildOpaqueBlock(child as Extract<SurfaceBlockSnapshot, { kind: "opaque_block" }>));
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
// Ensure at least one paragraph in cell (PM requires non-empty)
|
|
187
|
+
if (cellContent.length === 0) {
|
|
188
|
+
cellContent.push(editorSchema.nodes.paragraph.create());
|
|
189
|
+
}
|
|
190
|
+
cells.push(
|
|
191
|
+
editorSchema.nodes.table_cell.create(
|
|
192
|
+
{
|
|
193
|
+
colspan: cell.colspan,
|
|
194
|
+
rowspan: cell.rowspan,
|
|
195
|
+
gridSpan: cell.gridSpan,
|
|
196
|
+
verticalMerge: cell.verticalMerge,
|
|
197
|
+
},
|
|
198
|
+
Fragment.from(cellContent),
|
|
199
|
+
),
|
|
200
|
+
);
|
|
201
|
+
}
|
|
202
|
+
rows.push(editorSchema.nodes.table_row.create(null, Fragment.from(cells)));
|
|
203
|
+
}
|
|
204
|
+
return editorSchema.nodes.table.create(
|
|
205
|
+
{
|
|
206
|
+
styleId: block.styleId ?? null,
|
|
207
|
+
gridColumns: block.gridColumns,
|
|
208
|
+
},
|
|
209
|
+
Fragment.from(rows),
|
|
210
|
+
);
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
function buildNestedTablePlaceholder(
|
|
214
|
+
block: Extract<SurfaceBlockSnapshot, { kind: "table" }>,
|
|
215
|
+
): PMNode {
|
|
216
|
+
return buildOpaqueBlock({
|
|
217
|
+
blockId: block.blockId,
|
|
218
|
+
kind: "opaque_block",
|
|
219
|
+
from: block.from,
|
|
220
|
+
to: block.to,
|
|
221
|
+
fragmentId: block.blockId,
|
|
222
|
+
warningId: block.blockId,
|
|
223
|
+
label: "Nested table",
|
|
224
|
+
detail: "Nested table remains read-only in the live ProseMirror cell surface.",
|
|
225
|
+
state: "locked-preserve-only",
|
|
226
|
+
});
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
function buildSdtBlock(
|
|
230
|
+
block: Extract<SurfaceBlockSnapshot, { kind: "sdt_block" }>,
|
|
231
|
+
): PMNode {
|
|
232
|
+
const children = block.children.map((child) => {
|
|
233
|
+
if (child.kind === "paragraph") {
|
|
234
|
+
return buildParagraph(child);
|
|
235
|
+
}
|
|
236
|
+
if (child.kind === "table") {
|
|
237
|
+
return buildTable(child);
|
|
238
|
+
}
|
|
239
|
+
if (child.kind === "sdt_block") {
|
|
240
|
+
return buildSdtBlock(child);
|
|
241
|
+
}
|
|
242
|
+
return buildOpaqueBlock(child);
|
|
243
|
+
});
|
|
244
|
+
|
|
245
|
+
if (children.length === 0) {
|
|
246
|
+
children.push(editorSchema.nodes.paragraph.create());
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
return editorSchema.nodes.sdt_block.create(
|
|
250
|
+
{
|
|
251
|
+
sdtType: block.sdtType ?? null,
|
|
252
|
+
alias: block.alias ?? null,
|
|
253
|
+
tag: block.tag ?? null,
|
|
254
|
+
lock: block.lock ?? null,
|
|
255
|
+
},
|
|
256
|
+
Fragment.from(children),
|
|
257
|
+
);
|
|
258
|
+
}
|
|
259
|
+
|
|
260
|
+
/**
|
|
261
|
+
* Map an opaque_inline surface segment to a dedicated complex-rendering PM atom
|
|
262
|
+
* node when the label identifies a known complex content type, or fall back to
|
|
263
|
+
* the generic opaque_inline node.
|
|
264
|
+
*/
|
|
265
|
+
function buildOpaqueInlineOrComplexAtom(
|
|
266
|
+
segment: Extract<import("../../api/public-types").SurfaceInlineSegment, { kind: "opaque_inline" }>,
|
|
267
|
+
): PMNode {
|
|
268
|
+
const label = segment.label;
|
|
269
|
+
const detail = segment.detail;
|
|
270
|
+
|
|
271
|
+
if (label === "Chart") {
|
|
272
|
+
return editorSchema.nodes.chart_atom.create({ detail });
|
|
273
|
+
}
|
|
274
|
+
if (label === "SmartArt") {
|
|
275
|
+
return editorSchema.nodes.smartart_atom.create({ detail });
|
|
276
|
+
}
|
|
277
|
+
if (label === "Shape") {
|
|
278
|
+
// Extract text hint from detail if present
|
|
279
|
+
const textMatch = /Text: "([^"]+)"/.exec(detail);
|
|
280
|
+
const geometryMatch = /Geometry: ([^.]+)\./.exec(detail);
|
|
281
|
+
return editorSchema.nodes.shape_atom.create({
|
|
282
|
+
text: textMatch ? textMatch[1] : null,
|
|
283
|
+
geometry: geometryMatch ? geometryMatch[1] : null,
|
|
284
|
+
detail,
|
|
285
|
+
});
|
|
286
|
+
}
|
|
287
|
+
if (label === "WordArt") {
|
|
288
|
+
const textMatch = /Text: "([^"]+)"/.exec(detail);
|
|
289
|
+
const effectMatch = /Effect: ([^.]+)\./.exec(detail);
|
|
290
|
+
return editorSchema.nodes.wordart_atom.create({
|
|
291
|
+
text: textMatch ? textMatch[1] : "",
|
|
292
|
+
geometry: effectMatch ? effectMatch[1] : null,
|
|
293
|
+
detail,
|
|
294
|
+
});
|
|
295
|
+
}
|
|
296
|
+
if (label === "VML shape") {
|
|
297
|
+
const textMatch = /Text: "([^"]+)"/.exec(detail);
|
|
298
|
+
const typeMatch = /Type: ([^.]+)\./.exec(detail);
|
|
299
|
+
return editorSchema.nodes.vml_atom.create({
|
|
300
|
+
text: textMatch ? textMatch[1] : null,
|
|
301
|
+
shapeType: typeMatch ? typeMatch[1] : null,
|
|
302
|
+
detail,
|
|
303
|
+
});
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
return editorSchema.nodes.opaque_inline.create({
|
|
307
|
+
fragmentId: segment.fragmentId,
|
|
308
|
+
warningId: segment.warningId,
|
|
309
|
+
label,
|
|
310
|
+
detail,
|
|
311
|
+
});
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
function buildOpaqueBlock(
|
|
315
|
+
block: Extract<SurfaceBlockSnapshot, { kind: "opaque_block" }>,
|
|
316
|
+
): PMNode {
|
|
317
|
+
return editorSchema.nodes.opaque_block.create({
|
|
318
|
+
fragmentId: block.fragmentId,
|
|
319
|
+
warningId: block.warningId,
|
|
320
|
+
label: block.label,
|
|
321
|
+
detail: block.detail,
|
|
322
|
+
});
|
|
323
|
+
}
|
|
324
|
+
|
|
325
|
+
function clamp(value: number, min: number, max: number): number {
|
|
326
|
+
return Math.max(min, Math.min(max, value));
|
|
327
|
+
}
|
|
@@ -0,0 +1,157 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ProseMirror search/highlight plugin for the editor surface.
|
|
3
|
+
*
|
|
4
|
+
* Provides in-document search with optional case sensitivity and regex support.
|
|
5
|
+
* Matched ranges are shown as inline decorations with a configurable highlight color.
|
|
6
|
+
*
|
|
7
|
+
* Usage:
|
|
8
|
+
* 1. Include createSearchPlugin() in the editor plugin list.
|
|
9
|
+
* 2. Call performSearch(state, query, options) to compute matches.
|
|
10
|
+
* 3. Dispatch the results into the plugin state via searchPluginKey meta.
|
|
11
|
+
* 4. Call clearSearch(state, dispatch) to remove all highlights.
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
import { Plugin, PluginKey } from "prosemirror-state";
|
|
15
|
+
import type { EditorState, Transaction } from "prosemirror-state";
|
|
16
|
+
import { Decoration, DecorationSet } from "prosemirror-view";
|
|
17
|
+
|
|
18
|
+
// ---------------------------------------------------------------------------
|
|
19
|
+
// Public types
|
|
20
|
+
// ---------------------------------------------------------------------------
|
|
21
|
+
|
|
22
|
+
export interface SearchResult {
|
|
23
|
+
from: number;
|
|
24
|
+
to: number;
|
|
25
|
+
text: string;
|
|
26
|
+
index: number;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export interface SearchOptions {
|
|
30
|
+
caseSensitive?: boolean;
|
|
31
|
+
regex?: boolean;
|
|
32
|
+
highlightColor?: string;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
// ---------------------------------------------------------------------------
|
|
36
|
+
// Plugin state
|
|
37
|
+
// ---------------------------------------------------------------------------
|
|
38
|
+
|
|
39
|
+
interface SearchPluginState {
|
|
40
|
+
results: SearchResult[];
|
|
41
|
+
highlightColor: string | null;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
export const searchPluginKey = new PluginKey<SearchPluginState>("search");
|
|
45
|
+
|
|
46
|
+
// ---------------------------------------------------------------------------
|
|
47
|
+
// Plugin factory
|
|
48
|
+
// ---------------------------------------------------------------------------
|
|
49
|
+
|
|
50
|
+
export function createSearchPlugin(): Plugin<SearchPluginState> {
|
|
51
|
+
return new Plugin<SearchPluginState>({
|
|
52
|
+
key: searchPluginKey,
|
|
53
|
+
|
|
54
|
+
state: {
|
|
55
|
+
init(): SearchPluginState {
|
|
56
|
+
return { results: [], highlightColor: null };
|
|
57
|
+
},
|
|
58
|
+
|
|
59
|
+
apply(tr: Transaction, pluginState: SearchPluginState): SearchPluginState {
|
|
60
|
+
const meta = tr.getMeta(searchPluginKey) as SearchPluginState | undefined;
|
|
61
|
+
if (meta) return meta;
|
|
62
|
+
return pluginState;
|
|
63
|
+
},
|
|
64
|
+
},
|
|
65
|
+
|
|
66
|
+
props: {
|
|
67
|
+
decorations(state: EditorState): DecorationSet {
|
|
68
|
+
const pluginState = searchPluginKey.getState(state);
|
|
69
|
+
if (
|
|
70
|
+
!pluginState ||
|
|
71
|
+
pluginState.results.length === 0 ||
|
|
72
|
+
!pluginState.highlightColor
|
|
73
|
+
) {
|
|
74
|
+
return DecorationSet.empty;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
const decos = pluginState.results.map((r) =>
|
|
78
|
+
Decoration.inline(r.from, r.to, {
|
|
79
|
+
style: `background-color: ${pluginState.highlightColor}; border-radius: 2px;`,
|
|
80
|
+
class: "search-highlight",
|
|
81
|
+
}),
|
|
82
|
+
);
|
|
83
|
+
|
|
84
|
+
return DecorationSet.create(state.doc, decos);
|
|
85
|
+
},
|
|
86
|
+
},
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
// ---------------------------------------------------------------------------
|
|
91
|
+
// Search computation
|
|
92
|
+
// ---------------------------------------------------------------------------
|
|
93
|
+
|
|
94
|
+
/**
|
|
95
|
+
* Scan all text nodes in the document for matches of `query`.
|
|
96
|
+
* Returns an array of match ranges; does not dispatch anything.
|
|
97
|
+
* To apply highlights, set the results on the plugin via:
|
|
98
|
+
* view.dispatch(view.state.tr.setMeta(searchPluginKey, { results, highlightColor }))
|
|
99
|
+
*/
|
|
100
|
+
export function performSearch(
|
|
101
|
+
state: EditorState,
|
|
102
|
+
query: string,
|
|
103
|
+
options: SearchOptions = {},
|
|
104
|
+
): SearchResult[] {
|
|
105
|
+
if (!query) return [];
|
|
106
|
+
|
|
107
|
+
const { caseSensitive = false, regex = false } = options;
|
|
108
|
+
|
|
109
|
+
let pattern: RegExp;
|
|
110
|
+
try {
|
|
111
|
+
if (regex) {
|
|
112
|
+
pattern = new RegExp(query, caseSensitive ? "g" : "gi");
|
|
113
|
+
} else {
|
|
114
|
+
const escaped = query.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
|
|
115
|
+
pattern = new RegExp(escaped, caseSensitive ? "g" : "gi");
|
|
116
|
+
}
|
|
117
|
+
} catch {
|
|
118
|
+
return [];
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
const results: SearchResult[] = [];
|
|
122
|
+
|
|
123
|
+
state.doc.descendants((node, pos) => {
|
|
124
|
+
if (!node.isText || !node.text) return;
|
|
125
|
+
|
|
126
|
+
let match: RegExpExecArray | null;
|
|
127
|
+
pattern.lastIndex = 0;
|
|
128
|
+
while ((match = pattern.exec(node.text)) !== null) {
|
|
129
|
+
results.push({
|
|
130
|
+
from: pos + match.index,
|
|
131
|
+
to: pos + match.index + match[0].length,
|
|
132
|
+
text: match[0],
|
|
133
|
+
index: results.length,
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
});
|
|
137
|
+
|
|
138
|
+
return results;
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
// ---------------------------------------------------------------------------
|
|
142
|
+
// Clear helper (ProseMirror Command signature)
|
|
143
|
+
// ---------------------------------------------------------------------------
|
|
144
|
+
|
|
145
|
+
/**
|
|
146
|
+
* ProseMirror command that clears all search highlights from the plugin state.
|
|
147
|
+
*/
|
|
148
|
+
export function clearSearch(
|
|
149
|
+
state: EditorState,
|
|
150
|
+
dispatch?: (tr: Transaction) => void,
|
|
151
|
+
): boolean {
|
|
152
|
+
if (!dispatch) return true;
|
|
153
|
+
dispatch(
|
|
154
|
+
state.tr.setMeta(searchPluginKey, { results: [], highlightColor: null }),
|
|
155
|
+
);
|
|
156
|
+
return true;
|
|
157
|
+
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import React from "react";
|
|
2
|
+
|
|
3
|
+
import type { SelectionSnapshot } from "../../api/public-types";
|
|
4
|
+
|
|
5
|
+
export function renderTwCaret(selection: SelectionSnapshot, position: number) {
|
|
6
|
+
return selection.isCollapsed && selection.anchor === position ? (
|
|
7
|
+
<span
|
|
8
|
+
aria-hidden="true"
|
|
9
|
+
className="inline-block w-0.5 h-[1.2em] bg-accent rounded-full align-middle animate-wre-blink"
|
|
10
|
+
/>
|
|
11
|
+
) : null;
|
|
12
|
+
}
|
|
@@ -0,0 +1,150 @@
|
|
|
1
|
+
import React, { type FocusEventHandler, useMemo } from "react";
|
|
2
|
+
|
|
3
|
+
import type {
|
|
4
|
+
EditorUser,
|
|
5
|
+
RuntimeRenderSnapshot,
|
|
6
|
+
SelectionSnapshot,
|
|
7
|
+
} from "../../api/public-types";
|
|
8
|
+
import {
|
|
9
|
+
createCommentDecorationModel,
|
|
10
|
+
type MarkupDisplay,
|
|
11
|
+
} from "../../ui/headless/comment-decoration-model";
|
|
12
|
+
import { createRevisionDecorationModel } from "../../ui/headless/revision-decoration-model";
|
|
13
|
+
import { createEditorKeyboardHandler } from "../../ui/headless/use-editor-keyboard";
|
|
14
|
+
import { TwOpaqueBlock } from "./tw-opaque-block";
|
|
15
|
+
import { TwParagraphBlock } from "./tw-paragraph-block";
|
|
16
|
+
|
|
17
|
+
export interface TwEditorSurfaceProps {
|
|
18
|
+
currentUser: EditorUser;
|
|
19
|
+
snapshot: RuntimeRenderSnapshot;
|
|
20
|
+
reviewMode: "editing" | "review";
|
|
21
|
+
markupDisplay: MarkupDisplay;
|
|
22
|
+
activeRevisionId?: string;
|
|
23
|
+
/** When false, revision decorations are suppressed in the document surface. */
|
|
24
|
+
showTrackedChanges?: boolean;
|
|
25
|
+
onFocus: FocusEventHandler<HTMLDivElement>;
|
|
26
|
+
onBlur: FocusEventHandler<HTMLDivElement>;
|
|
27
|
+
onSelectionChange?: (selection: SelectionSnapshot) => void;
|
|
28
|
+
onInsertText?: (text: string) => void;
|
|
29
|
+
onDeleteBackward?: () => void;
|
|
30
|
+
onDeleteForward?: () => void;
|
|
31
|
+
onInsertTab?: () => void;
|
|
32
|
+
onInsertHardBreak?: () => void;
|
|
33
|
+
onSplitParagraph?: () => void;
|
|
34
|
+
onCommentActivated?: (commentId: string) => void;
|
|
35
|
+
onRevisionActivated?: (revisionId: string) => void;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
export function TwEditorSurface(props: TwEditorSurfaceProps) {
|
|
39
|
+
const { currentUser, markupDisplay, onBlur, onFocus, snapshot } = props;
|
|
40
|
+
const surface = snapshot.surface;
|
|
41
|
+
|
|
42
|
+
const canEdit = Boolean(
|
|
43
|
+
surface && snapshot.isReady && !snapshot.readOnly && !snapshot.fatalError,
|
|
44
|
+
);
|
|
45
|
+
|
|
46
|
+
const commentDecorations = useMemo(
|
|
47
|
+
() => createCommentDecorationModel(snapshot.comments),
|
|
48
|
+
[snapshot.comments],
|
|
49
|
+
);
|
|
50
|
+
|
|
51
|
+
const showTrackedChanges = props.showTrackedChanges !== false;
|
|
52
|
+
const revisionDecorations = useMemo(
|
|
53
|
+
() => showTrackedChanges
|
|
54
|
+
? createRevisionDecorationModel(snapshot.trackedChanges, props.activeRevisionId)
|
|
55
|
+
: undefined,
|
|
56
|
+
[snapshot.trackedChanges, props.activeRevisionId, showTrackedChanges],
|
|
57
|
+
);
|
|
58
|
+
|
|
59
|
+
const handleKeyDown = useMemo(() => {
|
|
60
|
+
if (!surface) return undefined;
|
|
61
|
+
return createEditorKeyboardHandler(
|
|
62
|
+
{
|
|
63
|
+
selection: snapshot.selection,
|
|
64
|
+
storySize: surface.storySize,
|
|
65
|
+
canEdit,
|
|
66
|
+
},
|
|
67
|
+
{
|
|
68
|
+
onSelectionChange: props.onSelectionChange,
|
|
69
|
+
onInsertText: props.onInsertText,
|
|
70
|
+
onDeleteBackward: props.onDeleteBackward,
|
|
71
|
+
onDeleteForward: props.onDeleteForward,
|
|
72
|
+
onInsertTab: props.onInsertTab,
|
|
73
|
+
onInsertHardBreak: props.onInsertHardBreak,
|
|
74
|
+
onSplitParagraph: props.onSplitParagraph,
|
|
75
|
+
},
|
|
76
|
+
);
|
|
77
|
+
}, [surface, snapshot.selection, canEdit, props]);
|
|
78
|
+
|
|
79
|
+
const fontClass = markupDisplay === "clean"
|
|
80
|
+
? "font-[family-name:var(--font-legal-sans)]"
|
|
81
|
+
: "font-[family-name:var(--font-legal-serif)]";
|
|
82
|
+
|
|
83
|
+
return (
|
|
84
|
+
<section aria-label="Document canvas" className="min-w-0">
|
|
85
|
+
<div
|
|
86
|
+
aria-label="Document surface"
|
|
87
|
+
aria-multiline="true"
|
|
88
|
+
role="textbox"
|
|
89
|
+
tabIndex={0}
|
|
90
|
+
onFocus={onFocus}
|
|
91
|
+
onBlur={onBlur}
|
|
92
|
+
onKeyDown={handleKeyDown}
|
|
93
|
+
className={`bg-transparent border-none px-12 py-10 outline-offset-1 ${fontClass}`}
|
|
94
|
+
>
|
|
95
|
+
{/* Document header */}
|
|
96
|
+
<div className="flex justify-between gap-4 items-start mb-5">
|
|
97
|
+
<div>
|
|
98
|
+
<h1 className="font-[family-name:var(--font-legal-serif)] text-2xl leading-tight text-primary">
|
|
99
|
+
{snapshot.sourceLabel ?? "Document draft"}
|
|
100
|
+
</h1>
|
|
101
|
+
<p className="text-xs text-secondary mt-2">
|
|
102
|
+
{snapshot.sessionId} · {snapshot.documentStats.storyLength} chars
|
|
103
|
+
</p>
|
|
104
|
+
</div>
|
|
105
|
+
<span className="text-xs text-secondary whitespace-nowrap">
|
|
106
|
+
Reviewer {currentUser.displayName}
|
|
107
|
+
</span>
|
|
108
|
+
</div>
|
|
109
|
+
|
|
110
|
+
{/* Document blocks */}
|
|
111
|
+
<div className="space-y-4 pl-10 relative">
|
|
112
|
+
{surface ? (
|
|
113
|
+
surface.blocks.map((block) =>
|
|
114
|
+
block.kind === "paragraph" ? (
|
|
115
|
+
<TwParagraphBlock
|
|
116
|
+
key={block.blockId}
|
|
117
|
+
block={block}
|
|
118
|
+
selection={snapshot.selection}
|
|
119
|
+
markupDisplay={markupDisplay}
|
|
120
|
+
commentDecorations={commentDecorations}
|
|
121
|
+
revisionDecorations={revisionDecorations}
|
|
122
|
+
onSelectionChange={props.onSelectionChange}
|
|
123
|
+
onCommentActivated={props.onCommentActivated}
|
|
124
|
+
onRevisionActivated={props.onRevisionActivated}
|
|
125
|
+
/>
|
|
126
|
+
) : block.kind === "opaque_block" ? (
|
|
127
|
+
<TwOpaqueBlock
|
|
128
|
+
key={block.blockId}
|
|
129
|
+
block={block}
|
|
130
|
+
selection={snapshot.selection}
|
|
131
|
+
onSelectionChange={props.onSelectionChange}
|
|
132
|
+
/>
|
|
133
|
+
) : null,
|
|
134
|
+
)
|
|
135
|
+
) : (
|
|
136
|
+
<p className="text-sm text-secondary leading-relaxed">
|
|
137
|
+
Loading the review surface. Compatibility cues and edit actions appear here once
|
|
138
|
+
the runtime snapshot is ready.
|
|
139
|
+
</p>
|
|
140
|
+
)}
|
|
141
|
+
{snapshot.fatalError ? (
|
|
142
|
+
<p className="text-sm text-danger">
|
|
143
|
+
Fatal runtime error: {snapshot.fatalError.message}
|
|
144
|
+
</p>
|
|
145
|
+
) : null}
|
|
146
|
+
</div>
|
|
147
|
+
</div>
|
|
148
|
+
</section>
|
|
149
|
+
);
|
|
150
|
+
}
|