@beyondwork/docx-react-component 1.0.1 → 1.0.3

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 (172) hide show
  1. package/README.md +44 -104
  2. package/package.json +50 -30
  3. package/src/README.md +85 -0
  4. package/src/api/README.md +22 -0
  5. package/src/api/public-types.ts +525 -0
  6. package/src/compare/diff-engine.ts +530 -0
  7. package/src/compare/export-redlines.ts +162 -0
  8. package/src/compare/snapshot.ts +37 -0
  9. package/src/component-inventory.md +99 -0
  10. package/src/core/README.md +10 -0
  11. package/src/core/commands/README.md +3 -0
  12. package/src/core/commands/formatting-commands.ts +161 -0
  13. package/src/core/commands/image-commands.ts +144 -0
  14. package/src/core/commands/index.ts +1013 -0
  15. package/src/core/commands/list-commands.ts +370 -0
  16. package/src/core/commands/review-commands.ts +108 -0
  17. package/src/core/commands/text-commands.ts +119 -0
  18. package/src/core/schema/README.md +3 -0
  19. package/src/core/schema/text-schema.ts +512 -0
  20. package/src/core/selection/README.md +3 -0
  21. package/src/core/selection/mapping.ts +238 -0
  22. package/src/core/selection/review-anchors.ts +94 -0
  23. package/src/core/state/README.md +3 -0
  24. package/src/core/state/editor-state.ts +580 -0
  25. package/src/core/state/text-transaction.ts +276 -0
  26. package/src/formats/xlsx/io/parse-shared-strings.ts +41 -0
  27. package/src/formats/xlsx/io/parse-sheet.ts +289 -0
  28. package/src/formats/xlsx/io/parse-styles.ts +57 -0
  29. package/src/formats/xlsx/io/parse-workbook.ts +75 -0
  30. package/src/formats/xlsx/io/xlsx-session.ts +306 -0
  31. package/src/formats/xlsx/model/cell.ts +189 -0
  32. package/src/formats/xlsx/model/sheet.ts +244 -0
  33. package/src/formats/xlsx/model/styles.ts +118 -0
  34. package/src/formats/xlsx/model/workbook.ts +449 -0
  35. package/src/index.ts +45 -0
  36. package/src/io/README.md +10 -0
  37. package/src/io/docx-session.ts +1763 -0
  38. package/src/io/export/README.md +3 -0
  39. package/src/io/export/export-session.ts +165 -0
  40. package/src/io/export/minimal-docx.ts +115 -0
  41. package/src/io/export/reattach-preserved-parts.ts +54 -0
  42. package/src/io/export/serialize-comments.ts +876 -0
  43. package/src/io/export/serialize-footnotes.ts +217 -0
  44. package/src/io/export/serialize-headers-footers.ts +200 -0
  45. package/src/io/export/serialize-main-document.ts +982 -0
  46. package/src/io/export/serialize-numbering.ts +97 -0
  47. package/src/io/export/serialize-revisions.ts +389 -0
  48. package/src/io/export/serialize-runtime-revisions.ts +265 -0
  49. package/src/io/export/serialize-tables.ts +147 -0
  50. package/src/io/export/split-review-boundaries.ts +194 -0
  51. package/src/io/normalize/README.md +3 -0
  52. package/src/io/normalize/normalize-text.ts +437 -0
  53. package/src/io/ooxml/README.md +3 -0
  54. package/src/io/ooxml/parse-comments.ts +779 -0
  55. package/src/io/ooxml/parse-complex-content.ts +287 -0
  56. package/src/io/ooxml/parse-fields.ts +438 -0
  57. package/src/io/ooxml/parse-footnotes.ts +403 -0
  58. package/src/io/ooxml/parse-headers-footers.ts +483 -0
  59. package/src/io/ooxml/parse-inline-media.ts +431 -0
  60. package/src/io/ooxml/parse-main-document.ts +1846 -0
  61. package/src/io/ooxml/parse-numbering.ts +425 -0
  62. package/src/io/ooxml/parse-revisions.ts +658 -0
  63. package/src/io/ooxml/parse-shapes.ts +271 -0
  64. package/src/io/ooxml/parse-tables.ts +568 -0
  65. package/src/io/ooxml/parse-theme.ts +314 -0
  66. package/src/io/ooxml/part-manifest.ts +136 -0
  67. package/src/io/ooxml/revision-boundaries.ts +351 -0
  68. package/src/io/opc/README.md +3 -0
  69. package/src/io/opc/corrupt-package.ts +166 -0
  70. package/src/io/opc/docx-package.ts +74 -0
  71. package/src/io/opc/package-reader.ts +325 -0
  72. package/src/io/opc/package-writer.ts +273 -0
  73. package/src/legal/bookmarks.ts +196 -0
  74. package/src/legal/cross-references.ts +356 -0
  75. package/src/legal/defined-terms.ts +203 -0
  76. package/src/model/README.md +3 -0
  77. package/src/model/canonical-document.ts +1911 -0
  78. package/src/model/cds-1.0.0.ts +196 -0
  79. package/src/model/snapshot.ts +393 -0
  80. package/src/preservation/README.md +3 -0
  81. package/src/preservation/markup-compatibility.ts +48 -0
  82. package/src/preservation/opaque-fragment-store.ts +89 -0
  83. package/src/preservation/opaque-region.ts +233 -0
  84. package/src/preservation/package-preservation.ts +120 -0
  85. package/src/preservation/preserved-part-manifest.ts +56 -0
  86. package/src/preservation/relationship-retention.ts +57 -0
  87. package/src/preservation/store.ts +185 -0
  88. package/src/review/README.md +16 -0
  89. package/src/review/store/README.md +3 -0
  90. package/src/review/store/comment-anchors.ts +70 -0
  91. package/src/review/store/comment-remapping.ts +154 -0
  92. package/src/review/store/comment-store.ts +331 -0
  93. package/src/review/store/comment-thread.ts +109 -0
  94. package/src/review/store/revision-actions.ts +394 -0
  95. package/src/review/store/revision-store.ts +303 -0
  96. package/src/review/store/revision-types.ts +168 -0
  97. package/src/review/store/runtime-comment-store.ts +43 -0
  98. package/src/runtime/README.md +3 -0
  99. package/src/runtime/ai-action-policy.ts +764 -0
  100. package/src/runtime/document-runtime.ts +967 -0
  101. package/src/runtime/read-only-diagnostics-runtime.ts +232 -0
  102. package/src/runtime/review-runtime.ts +44 -0
  103. package/src/runtime/revision-runtime.ts +107 -0
  104. package/src/runtime/session-capabilities.ts +138 -0
  105. package/src/runtime/surface-projection.ts +570 -0
  106. package/src/runtime/table-commands.ts +87 -0
  107. package/src/runtime/table-schema.ts +140 -0
  108. package/src/runtime/virtualized-rendering.ts +258 -0
  109. package/src/ui/README.md +30 -0
  110. package/src/ui/WordReviewEditor.tsx +1506 -0
  111. package/src/ui/comments/README.md +3 -0
  112. package/src/ui/compatibility/README.md +3 -0
  113. package/src/ui/editor-surface/README.md +3 -0
  114. package/src/ui/headless/comment-decoration-model.ts +124 -0
  115. package/src/ui/headless/revision-decoration-model.ts +128 -0
  116. package/src/ui/headless/selection-helpers.ts +34 -0
  117. package/src/ui/headless/use-editor-keyboard.ts +98 -0
  118. package/src/ui/review/README.md +3 -0
  119. package/src/ui/shared/revision-filters.ts +31 -0
  120. package/src/ui/status/README.md +3 -0
  121. package/src/ui/theme/README.md +3 -0
  122. package/src/ui/toolbar/README.md +3 -0
  123. package/src/ui-tailwind/chrome/tw-alert-banner.tsx +48 -0
  124. package/src/ui-tailwind/chrome/tw-selection-toolbar.tsx +44 -0
  125. package/src/ui-tailwind/chrome/tw-unsaved-modal.tsx +58 -0
  126. package/src/ui-tailwind/chrome/use-before-unload.ts +20 -0
  127. package/src/ui-tailwind/editor-surface/pm-command-bridge.ts +139 -0
  128. package/src/ui-tailwind/editor-surface/pm-decorations.ts +98 -0
  129. package/src/ui-tailwind/editor-surface/pm-position-map.ts +123 -0
  130. package/src/ui-tailwind/editor-surface/pm-schema.ts +452 -0
  131. package/src/ui-tailwind/editor-surface/pm-state-from-snapshot.ts +327 -0
  132. package/src/ui-tailwind/editor-surface/search-plugin.ts +157 -0
  133. package/src/ui-tailwind/editor-surface/tw-caret.tsx +12 -0
  134. package/src/ui-tailwind/editor-surface/tw-editor-surface.tsx +150 -0
  135. package/src/ui-tailwind/editor-surface/tw-inline-token.tsx +118 -0
  136. package/src/ui-tailwind/editor-surface/tw-opaque-block.tsx +52 -0
  137. package/src/ui-tailwind/editor-surface/tw-paragraph-block.tsx +151 -0
  138. package/src/ui-tailwind/editor-surface/tw-prosemirror-surface.tsx +215 -0
  139. package/src/ui-tailwind/editor-surface/tw-segment-view.tsx +111 -0
  140. package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +122 -0
  141. package/src/ui-tailwind/index.ts +61 -0
  142. package/src/ui-tailwind/review/tw-comment-sidebar.tsx +276 -0
  143. package/src/ui-tailwind/review/tw-health-panel.tsx +120 -0
  144. package/src/ui-tailwind/review/tw-review-rail.tsx +120 -0
  145. package/src/ui-tailwind/review/tw-revision-sidebar.tsx +164 -0
  146. package/src/ui-tailwind/status/tw-status-bar.tsx +58 -0
  147. package/src/ui-tailwind/theme/editor-theme.css +190 -0
  148. package/src/ui-tailwind/toolbar/tw-toolbar-icon-button.tsx +48 -0
  149. package/src/ui-tailwind/toolbar/tw-toolbar.tsx +231 -0
  150. package/src/ui-tailwind/tw-review-workspace.tsx +140 -0
  151. package/src/validation/README.md +3 -0
  152. package/src/validation/compatibility-engine.ts +317 -0
  153. package/src/validation/compatibility-report.ts +160 -0
  154. package/src/validation/diagnostics.ts +203 -0
  155. package/src/validation/import-diagnostics.ts +128 -0
  156. package/src/validation/low-priority-word-surfaces.ts +373 -0
  157. package/dist/chunk-32W6IVQE.js +0 -7725
  158. package/dist/chunk-32W6IVQE.js.map +0 -1
  159. package/dist/index.cjs +0 -23722
  160. package/dist/index.cjs.map +0 -1
  161. package/dist/index.d.cts +0 -7
  162. package/dist/index.d.ts +0 -7
  163. package/dist/index.js +0 -16011
  164. package/dist/index.js.map +0 -1
  165. package/dist/public-types-DqCURAz8.d.cts +0 -1152
  166. package/dist/public-types-DqCURAz8.d.ts +0 -1152
  167. package/dist/tailwind.cjs +0 -8295
  168. package/dist/tailwind.cjs.map +0 -1
  169. package/dist/tailwind.d.cts +0 -323
  170. package/dist/tailwind.d.ts +0 -323
  171. package/dist/tailwind.js +0 -553
  172. package/dist/tailwind.js.map +0 -1
@@ -0,0 +1,373 @@
1
+ /**
2
+ * Low-Priority Word Surfaces Validation
3
+ *
4
+ * This module defines the explicit bounded stance for Word surfaces that are
5
+ * intentionally de-prioritized in the current legal review editor scope.
6
+ * These include OLE objects, embedded objects, macros, and other complex
7
+ * Word features that are outside the core document editing workflow.
8
+ */
9
+
10
+ /**
11
+ * Classification of low-priority Word surfaces based on legal review importance
12
+ * and technical complexity.
13
+ */
14
+ export type LowPriorityWordSurface =
15
+ | 'ole-objects'
16
+ | 'embedded-objects'
17
+ | 'macros'
18
+ | 'advanced-word-automation'
19
+ | 'custom-ribbons'
20
+ | 'word-addins'
21
+ | 'external-data-connections'
22
+ | 'mail-merge-fields'
23
+ | 'legacy-form-fields'
24
+ | 'activex-controls'
25
+ | 'vba-scripts'
26
+ | 'external-links';
27
+
28
+ /**
29
+ * Support stance for each low-priority surface
30
+ */
31
+ export type SupportStance =
32
+ | 'preserve-only' // Keep intact during round-trip, but no editing
33
+ | 'blocked' // Explicitly blocked for security/complexity reasons
34
+ | 'unsupported-fatal' // Causes import to fail
35
+ | 'placeholder-only'; // Render placeholder, preserve original
36
+
37
+ /**
38
+ * Detailed policy for each low-priority Word surface
39
+ */
40
+ export interface LowPrioritySurfacePolicy {
41
+ surface: LowPriorityWordSurface;
42
+ stance: SupportStance;
43
+ rationale: string;
44
+ fixtureEvidence?: string[];
45
+ userVisibleBehavior: {
46
+ onImport: 'preserve' | 'block' | 'fail' | 'placeholder';
47
+ onEdit: 'locked' | 'blocked' | 'unavailable';
48
+ onExport: 'preserve' | 'omit' | 'block';
49
+ warningLevel: 'none' | 'info' | 'warning' | 'error';
50
+ };
51
+ wordCompatibility: {
52
+ roundTripSafe: boolean;
53
+ wordReopenBehavior: 'normal' | 'repair-prompt' | 'blocked';
54
+ };
55
+ }
56
+
57
+ /**
58
+ * Complete policy matrix for all low-priority Word surfaces
59
+ */
60
+ export const LOW_PRIORITY_WORD_SURFACE_POLICIES: LowPrioritySurfacePolicy[] = [
61
+ {
62
+ surface: 'ole-objects',
63
+ stance: 'preserve-only',
64
+ rationale: 'OLE objects are complex binary content that would require format-specific handling. Preservation maintains compatibility without security risks.',
65
+ fixtureEvidence: ['F20'],
66
+ userVisibleBehavior: {
67
+ onImport: 'preserve',
68
+ onEdit: 'locked',
69
+ onExport: 'preserve',
70
+ warningLevel: 'info'
71
+ },
72
+ wordCompatibility: {
73
+ roundTripSafe: true,
74
+ wordReopenBehavior: 'normal'
75
+ }
76
+ },
77
+ {
78
+ surface: 'embedded-objects',
79
+ stance: 'preserve-only',
80
+ rationale: 'Embedded objects (Excel charts, PowerPoint slides) are preserved to maintain document integrity but not editable within the text editor scope.',
81
+ fixtureEvidence: ['F20'],
82
+ userVisibleBehavior: {
83
+ onImport: 'preserve',
84
+ onEdit: 'locked',
85
+ onExport: 'preserve',
86
+ warningLevel: 'info'
87
+ },
88
+ wordCompatibility: {
89
+ roundTripSafe: true,
90
+ wordReopenBehavior: 'normal'
91
+ }
92
+ },
93
+ {
94
+ surface: 'macros',
95
+ stance: 'blocked',
96
+ rationale: 'Macros pose security risks and are explicitly blocked. Legal document review workflows should not depend on executable code.',
97
+ fixtureEvidence: [],
98
+ userVisibleBehavior: {
99
+ onImport: 'block',
100
+ onEdit: 'blocked',
101
+ onExport: 'omit',
102
+ warningLevel: 'warning'
103
+ },
104
+ wordCompatibility: {
105
+ roundTripSafe: false,
106
+ wordReopenBehavior: 'normal'
107
+ }
108
+ },
109
+ {
110
+ surface: 'vba-scripts',
111
+ stance: 'blocked',
112
+ rationale: 'VBA scripts are executable code that poses security risks and is outside the scope of document review workflows.',
113
+ fixtureEvidence: [],
114
+ userVisibleBehavior: {
115
+ onImport: 'block',
116
+ onEdit: 'blocked',
117
+ onExport: 'omit',
118
+ warningLevel: 'warning'
119
+ },
120
+ wordCompatibility: {
121
+ roundTripSafe: false,
122
+ wordReopenBehavior: 'normal'
123
+ }
124
+ },
125
+ {
126
+ surface: 'advanced-word-automation',
127
+ stance: 'unsupported-fatal',
128
+ rationale: 'Advanced automation features are complex and outside the legal review scope. Documents depending on them should use Word directly.',
129
+ fixtureEvidence: [],
130
+ userVisibleBehavior: {
131
+ onImport: 'fail',
132
+ onEdit: 'unavailable',
133
+ onExport: 'block',
134
+ warningLevel: 'error'
135
+ },
136
+ wordCompatibility: {
137
+ roundTripSafe: false,
138
+ wordReopenBehavior: 'blocked'
139
+ }
140
+ },
141
+ {
142
+ surface: 'custom-ribbons',
143
+ stance: 'preserve-only',
144
+ rationale: 'Custom ribbon definitions are UI customizations that can be preserved but are not relevant to document content editing.',
145
+ fixtureEvidence: ['F20'],
146
+ userVisibleBehavior: {
147
+ onImport: 'preserve',
148
+ onEdit: 'unavailable',
149
+ onExport: 'preserve',
150
+ warningLevel: 'none'
151
+ },
152
+ wordCompatibility: {
153
+ roundTripSafe: true,
154
+ wordReopenBehavior: 'normal'
155
+ }
156
+ },
157
+ {
158
+ surface: 'word-addins',
159
+ stance: 'preserve-only',
160
+ rationale: 'Add-in definitions are application-level customizations that can be preserved but are not editable within the document scope.',
161
+ fixtureEvidence: ['F20'],
162
+ userVisibleBehavior: {
163
+ onImport: 'preserve',
164
+ onEdit: 'unavailable',
165
+ onExport: 'preserve',
166
+ warningLevel: 'none'
167
+ },
168
+ wordCompatibility: {
169
+ roundTripSafe: true,
170
+ wordReopenBehavior: 'normal'
171
+ }
172
+ },
173
+ {
174
+ surface: 'external-data-connections',
175
+ stance: 'blocked',
176
+ rationale: 'External data connections pose security risks and can create dependencies outside the document. Blocked for legal review workflows.',
177
+ fixtureEvidence: [],
178
+ userVisibleBehavior: {
179
+ onImport: 'block',
180
+ onEdit: 'blocked',
181
+ onExport: 'omit',
182
+ warningLevel: 'warning'
183
+ },
184
+ wordCompatibility: {
185
+ roundTripSafe: false,
186
+ wordReopenBehavior: 'normal'
187
+ }
188
+ },
189
+ {
190
+ surface: 'mail-merge-fields',
191
+ stance: 'preserve-only',
192
+ rationale: 'Mail merge fields can be preserved as static content but merge operations are outside the document editing scope.',
193
+ fixtureEvidence: ['F17'],
194
+ userVisibleBehavior: {
195
+ onImport: 'preserve',
196
+ onEdit: 'locked',
197
+ onExport: 'preserve',
198
+ warningLevel: 'info'
199
+ },
200
+ wordCompatibility: {
201
+ roundTripSafe: true,
202
+ wordReopenBehavior: 'normal'
203
+ }
204
+ },
205
+ {
206
+ surface: 'legacy-form-fields',
207
+ stance: 'preserve-only',
208
+ rationale: 'Legacy form fields are preserved for compatibility but modern content controls (SDTs) are preferred for structured input.',
209
+ fixtureEvidence: ['F18'],
210
+ userVisibleBehavior: {
211
+ onImport: 'preserve',
212
+ onEdit: 'locked',
213
+ onExport: 'preserve',
214
+ warningLevel: 'info'
215
+ },
216
+ wordCompatibility: {
217
+ roundTripSafe: true,
218
+ wordReopenBehavior: 'normal'
219
+ }
220
+ },
221
+ {
222
+ surface: 'activex-controls',
223
+ stance: 'blocked',
224
+ rationale: 'ActiveX controls are deprecated and pose security risks. Not appropriate for legal document workflows.',
225
+ fixtureEvidence: [],
226
+ userVisibleBehavior: {
227
+ onImport: 'block',
228
+ onEdit: 'blocked',
229
+ onExport: 'omit',
230
+ warningLevel: 'error'
231
+ },
232
+ wordCompatibility: {
233
+ roundTripSafe: false,
234
+ wordReopenBehavior: 'normal'
235
+ }
236
+ },
237
+ {
238
+ surface: 'external-links',
239
+ stance: 'preserve-only',
240
+ rationale: 'External links to files or resources outside the document are preserved but not resolved within the editor.',
241
+ fixtureEvidence: ['F20'],
242
+ userVisibleBehavior: {
243
+ onImport: 'preserve',
244
+ onEdit: 'locked',
245
+ onExport: 'preserve',
246
+ warningLevel: 'info'
247
+ },
248
+ wordCompatibility: {
249
+ roundTripSafe: true,
250
+ wordReopenBehavior: 'normal'
251
+ }
252
+ }
253
+ ];
254
+
255
+ /**
256
+ * Get the policy for a specific low-priority surface
257
+ */
258
+ export function getLowPrioritySurfacePolicy(surface: LowPriorityWordSurface): LowPrioritySurfacePolicy {
259
+ const policy = LOW_PRIORITY_WORD_SURFACE_POLICIES.find(p => p.surface === surface);
260
+ if (!policy) {
261
+ throw new Error(`No policy defined for surface: ${surface}`);
262
+ }
263
+ return policy;
264
+ }
265
+
266
+ /**
267
+ * Check if a surface is supported for editing (as opposed to preserve-only)
268
+ */
269
+ export function isSurfaceEditingSupported(surface: LowPriorityWordSurface): boolean {
270
+ const policy = getLowPrioritySurfacePolicy(surface);
271
+ return policy.userVisibleBehavior.onEdit !== 'locked' &&
272
+ policy.userVisibleBehavior.onEdit !== 'blocked' &&
273
+ policy.userVisibleBehavior.onEdit !== 'unavailable';
274
+ }
275
+
276
+ /**
277
+ * Check if a surface should block document import
278
+ */
279
+ export function shouldBlockImport(surface: LowPriorityWordSurface): boolean {
280
+ const policy = getLowPrioritySurfacePolicy(surface);
281
+ return policy.userVisibleBehavior.onImport === 'block' ||
282
+ policy.userVisibleBehavior.onImport === 'fail';
283
+ }
284
+
285
+ /**
286
+ * Get warning level for a detected surface
287
+ */
288
+ export function getWarningLevel(surface: LowPriorityWordSurface): 'none' | 'info' | 'warning' | 'error' {
289
+ const policy = getLowPrioritySurfacePolicy(surface);
290
+ return policy.userVisibleBehavior.warningLevel;
291
+ }
292
+
293
+ /**
294
+ * Generate compatibility report entry for a low-priority surface
295
+ */
296
+ export interface LowPrioritySurfaceCompatibilityEntry {
297
+ surface: LowPriorityWordSurface;
298
+ stance: SupportStance;
299
+ detected: boolean;
300
+ userImpact: 'none' | 'locked' | 'blocked' | 'preserved';
301
+ wordCompatible: boolean;
302
+ recommendation: string;
303
+ }
304
+
305
+ /**
306
+ * Generate compatibility entries for detected low-priority surfaces
307
+ */
308
+ export function generateCompatibilityEntries(
309
+ detectedSurfaces: LowPriorityWordSurface[]
310
+ ): LowPrioritySurfaceCompatibilityEntry[] {
311
+ return detectedSurfaces.map(surface => {
312
+ const policy = getLowPrioritySurfacePolicy(surface);
313
+
314
+ let userImpact: 'none' | 'locked' | 'blocked' | 'preserved';
315
+ let recommendation: string;
316
+
317
+ switch (policy.stance) {
318
+ case 'preserve-only':
319
+ userImpact = 'locked';
320
+ recommendation = 'Content is preserved but cannot be edited within this editor. Use Microsoft Word for full editing capabilities.';
321
+ break;
322
+ case 'blocked':
323
+ userImpact = 'blocked';
324
+ recommendation = 'This content type is blocked for security reasons. Remove or replace before importing.';
325
+ break;
326
+ case 'unsupported-fatal':
327
+ userImpact = 'blocked';
328
+ recommendation = 'This document contains features that cannot be safely handled. Please use Microsoft Word directly.';
329
+ break;
330
+ case 'placeholder-only':
331
+ userImpact = 'preserved';
332
+ recommendation = 'Content is displayed as a placeholder. Original formatting preserved for export.';
333
+ break;
334
+ }
335
+
336
+ return {
337
+ surface,
338
+ stance: policy.stance,
339
+ detected: true,
340
+ userImpact,
341
+ wordCompatible: policy.wordCompatibility.roundTripSafe,
342
+ recommendation
343
+ };
344
+ });
345
+ }
346
+
347
+ /**
348
+ * Summary statistics for low-priority surface handling
349
+ */
350
+ export interface LowPrioritySurfaceSummary {
351
+ totalSurfaces: number;
352
+ preserveOnlyCount: number;
353
+ blockedCount: number;
354
+ unsupportedFatalCount: number;
355
+ wordCompatibleCount: number;
356
+ surfacesWithWarnings: number;
357
+ }
358
+
359
+ /**
360
+ * Generate summary statistics for the low-priority surface policy matrix
361
+ */
362
+ export function generatePolicySummary(): LowPrioritySurfaceSummary {
363
+ const policies = LOW_PRIORITY_WORD_SURFACE_POLICIES;
364
+
365
+ return {
366
+ totalSurfaces: policies.length,
367
+ preserveOnlyCount: policies.filter(p => p.stance === 'preserve-only').length,
368
+ blockedCount: policies.filter(p => p.stance === 'blocked').length,
369
+ unsupportedFatalCount: policies.filter(p => p.stance === 'unsupported-fatal').length,
370
+ wordCompatibleCount: policies.filter(p => p.wordCompatibility.roundTripSafe).length,
371
+ surfacesWithWarnings: policies.filter(p => p.userVisibleBehavior.warningLevel !== 'none').length
372
+ };
373
+ }