@beyondwork/docx-react-component 1.0.29 → 1.0.31

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 (383) hide show
  1. package/package.json +65 -96
  2. package/src/README.md +85 -0
  3. package/src/api/README.md +26 -0
  4. package/src/api/public-types.ts +1952 -0
  5. package/src/api/session-state.ts +62 -0
  6. package/src/compare/diff-engine.ts +623 -0
  7. package/src/compare/export-redlines.ts +280 -0
  8. package/src/compare/index.ts +25 -0
  9. package/src/compare/snapshot.ts +97 -0
  10. package/src/component-inventory.md +99 -0
  11. package/src/core/README.md +10 -0
  12. package/src/core/commands/README.md +3 -0
  13. package/{dist/chunk-TJBP2K4T.js → src/core/commands/formatting-commands.ts} +536 -196
  14. package/src/core/commands/image-commands.ts +373 -0
  15. package/src/core/commands/index.ts +1879 -0
  16. package/src/core/commands/list-commands.ts +565 -0
  17. package/src/core/commands/paragraph-layout-commands.ts +339 -0
  18. package/src/core/commands/review-commands.ts +108 -0
  19. package/{dist/core/commands/section-layout-commands.cjs → src/core/commands/section-layout-commands.ts} +340 -137
  20. package/src/core/commands/structural-helpers.ts +309 -0
  21. package/{dist/core/commands/style-commands.cjs → src/core/commands/style-commands.ts} +113 -65
  22. package/src/core/commands/table-structure-commands.ts +854 -0
  23. package/{dist/chunk-UZXBISGO.js → src/core/commands/text-commands.ts} +142 -86
  24. package/src/core/schema/README.md +3 -0
  25. package/src/core/schema/text-schema.ts +516 -0
  26. package/src/core/search/search-text.ts +357 -0
  27. package/src/core/selection/README.md +3 -0
  28. package/src/core/selection/mapping.ts +289 -0
  29. package/src/core/selection/review-anchors.ts +183 -0
  30. package/src/core/state/README.md +3 -0
  31. package/src/core/state/editor-state.ts +892 -0
  32. package/src/core/state/text-transaction.ts +869 -0
  33. package/src/formats/xlsx/io/parse-shared-strings.ts +41 -0
  34. package/src/formats/xlsx/io/parse-sheet.ts +459 -0
  35. package/src/formats/xlsx/io/parse-styles.ts +59 -0
  36. package/src/formats/xlsx/io/parse-workbook.ts +75 -0
  37. package/src/formats/xlsx/io/serialize-shared-strings.ts +72 -0
  38. package/src/formats/xlsx/io/serialize-sheet.ts +333 -0
  39. package/src/formats/xlsx/io/serialize-styles.ts +98 -0
  40. package/src/formats/xlsx/io/serialize-workbook.ts +429 -0
  41. package/src/formats/xlsx/io/xlsx-session.ts +314 -0
  42. package/src/formats/xlsx/model/cell.ts +189 -0
  43. package/src/formats/xlsx/model/sheet.ts +326 -0
  44. package/src/formats/xlsx/model/styles.ts +118 -0
  45. package/src/formats/xlsx/model/workbook.ts +453 -0
  46. package/src/formats/xlsx/runtime/cell-commands.ts +567 -0
  47. package/src/formats/xlsx/runtime/sheet-commands.ts +206 -0
  48. package/src/formats/xlsx/runtime/workbook-runtime.ts +177 -0
  49. package/src/formats/xlsx/runtime/workbook-transaction.ts +822 -0
  50. package/src/index.ts +142 -0
  51. package/src/io/README.md +10 -0
  52. package/src/io/docx-session.ts +3175 -0
  53. package/src/io/export/README.md +3 -0
  54. package/src/io/export/export-session.ts +220 -0
  55. package/src/io/export/minimal-docx.ts +115 -0
  56. package/src/io/export/reattach-preserved-parts.ts +54 -0
  57. package/src/io/export/serialize-comments.ts +947 -0
  58. package/src/io/export/serialize-footnotes.ts +394 -0
  59. package/src/io/export/serialize-headers-footers.ts +368 -0
  60. package/src/io/export/serialize-main-document.ts +1342 -0
  61. package/src/io/export/serialize-numbering.ts +218 -0
  62. package/src/io/export/serialize-revisions.ts +389 -0
  63. package/src/io/export/serialize-runtime-revisions.ts +463 -0
  64. package/src/io/export/serialize-tables.ts +174 -0
  65. package/src/io/export/split-review-boundaries.ts +356 -0
  66. package/src/io/export/split-story-blocks-for-runtime-revisions.ts +252 -0
  67. package/src/io/export/table-properties-xml.ts +318 -0
  68. package/src/io/normalize/README.md +3 -0
  69. package/src/io/normalize/normalize-text.ts +670 -0
  70. package/src/io/ooxml/README.md +3 -0
  71. package/src/io/ooxml/highlight-colors.ts +39 -0
  72. package/src/io/ooxml/numbering-sentinels.ts +44 -0
  73. package/src/io/ooxml/parse-comments.ts +852 -0
  74. package/src/io/ooxml/parse-complex-content.ts +287 -0
  75. package/src/io/ooxml/parse-fields.ts +834 -0
  76. package/src/io/ooxml/parse-footnotes.ts +952 -0
  77. package/src/io/ooxml/parse-headers-footers.ts +1212 -0
  78. package/src/io/ooxml/parse-inline-media.ts +461 -0
  79. package/src/io/ooxml/parse-main-document.ts +2947 -0
  80. package/src/io/ooxml/parse-numbering.ts +747 -0
  81. package/src/io/ooxml/parse-revisions.ts +1045 -0
  82. package/src/io/ooxml/parse-settings.ts +184 -0
  83. package/src/io/ooxml/parse-shapes.ts +296 -0
  84. package/src/io/ooxml/parse-styles.ts +639 -0
  85. package/src/io/ooxml/parse-tables.ts +627 -0
  86. package/src/io/ooxml/parse-theme.ts +346 -0
  87. package/src/io/ooxml/part-manifest.ts +136 -0
  88. package/src/io/ooxml/revision-boundaries.ts +475 -0
  89. package/src/io/ooxml/workflow-payload.ts +544 -0
  90. package/src/io/opc/README.md +3 -0
  91. package/src/io/opc/corrupt-package.ts +166 -0
  92. package/src/io/opc/docx-package.ts +74 -0
  93. package/src/io/opc/package-reader.ts +325 -0
  94. package/src/io/opc/package-writer.ts +273 -0
  95. package/src/io/source-package-provenance.ts +241 -0
  96. package/{dist/chunk-RMH72RZI.js → src/legal/bookmarks.ts} +130 -44
  97. package/src/legal/cross-references.ts +414 -0
  98. package/src/legal/defined-terms.ts +203 -0
  99. package/src/legal/index.ts +32 -0
  100. package/src/legal/signature-blocks.ts +259 -0
  101. package/src/model/README.md +3 -0
  102. package/src/model/canonical-document.ts +2722 -0
  103. package/src/model/cds-1.0.0.ts +212 -0
  104. package/src/model/snapshot.ts +760 -0
  105. package/src/preservation/README.md +3 -0
  106. package/src/preservation/markup-compatibility.ts +48 -0
  107. package/src/preservation/opaque-fragment-store.ts +89 -0
  108. package/src/preservation/opaque-region.ts +233 -0
  109. package/src/preservation/package-preservation.ts +113 -0
  110. package/src/preservation/preserved-part-manifest.ts +56 -0
  111. package/src/preservation/relationship-retention.ts +57 -0
  112. package/src/preservation/store.ts +255 -0
  113. package/src/review/README.md +16 -0
  114. package/src/review/store/README.md +3 -0
  115. package/src/review/store/comment-anchors.ts +70 -0
  116. package/src/review/store/comment-remapping.ts +154 -0
  117. package/src/review/store/comment-store.ts +349 -0
  118. package/src/review/store/comment-thread.ts +109 -0
  119. package/src/review/store/revision-actions.ts +423 -0
  120. package/src/review/store/revision-store.ts +323 -0
  121. package/src/review/store/revision-types.ts +182 -0
  122. package/src/review/store/runtime-comment-store.ts +43 -0
  123. package/src/runtime/README.md +3 -0
  124. package/src/runtime/ai-action-policy.ts +764 -0
  125. package/src/runtime/context-analytics.ts +824 -0
  126. package/src/runtime/document-layout.ts +332 -0
  127. package/src/runtime/document-locations.ts +521 -0
  128. package/src/runtime/document-navigation.ts +616 -0
  129. package/src/runtime/document-outline.ts +440 -0
  130. package/src/runtime/document-runtime.ts +4055 -0
  131. package/src/runtime/document-search.ts +145 -0
  132. package/src/runtime/event-refresh-hints.ts +137 -0
  133. package/src/runtime/numbering-prefix.ts +244 -0
  134. package/src/runtime/page-layout-estimation.ts +305 -0
  135. package/src/runtime/read-only-diagnostics-runtime.ts +241 -0
  136. package/src/runtime/resolved-numbering-geometry.ts +293 -0
  137. package/src/runtime/review-runtime.ts +44 -0
  138. package/src/runtime/revision-runtime.ts +107 -0
  139. package/src/runtime/session-capabilities.ts +192 -0
  140. package/src/runtime/story-context.ts +164 -0
  141. package/src/runtime/story-targeting.ts +162 -0
  142. package/src/runtime/suggestions-snapshot.ts +137 -0
  143. package/src/runtime/surface-projection.ts +1553 -0
  144. package/src/runtime/table-commands.ts +173 -0
  145. package/src/runtime/table-schema.ts +309 -0
  146. package/src/runtime/table-style-resolver.ts +409 -0
  147. package/src/runtime/view-state.ts +493 -0
  148. package/src/runtime/virtualized-rendering.ts +258 -0
  149. package/src/runtime/workflow-markup.ts +393 -0
  150. package/src/ui/README.md +30 -0
  151. package/src/ui/WordReviewEditor.tsx +5268 -0
  152. package/src/ui/browser-export.ts +52 -0
  153. package/src/ui/comments/README.md +3 -0
  154. package/src/ui/compatibility/README.md +3 -0
  155. package/src/ui/editor-command-bag.ts +127 -0
  156. package/src/ui/editor-runtime-boundary.ts +1558 -0
  157. package/src/ui/editor-shell-view.tsx +144 -0
  158. package/src/ui/editor-surface/README.md +3 -0
  159. package/src/ui/editor-surface-controller.tsx +66 -0
  160. package/src/ui/headless/comment-decoration-model.ts +124 -0
  161. package/src/ui/headless/preserve-editor-selection.ts +5 -0
  162. package/src/ui/headless/revision-decoration-model.ts +128 -0
  163. package/src/ui/headless/selection-helpers.ts +54 -0
  164. package/src/ui/headless/selection-tool-context.ts +19 -0
  165. package/src/ui/headless/selection-tool-resolver.ts +752 -0
  166. package/src/ui/headless/selection-tool-types.ts +129 -0
  167. package/src/ui/headless/selection-toolbar-model.ts +11 -0
  168. package/src/ui/headless/use-editor-keyboard.ts +103 -0
  169. package/src/ui/review/README.md +3 -0
  170. package/src/ui/runtime-shortcut-dispatch.ts +365 -0
  171. package/src/ui/runtime-snapshot-selectors.ts +197 -0
  172. package/src/ui/shared/revision-filters.ts +31 -0
  173. package/src/ui/status/README.md +3 -0
  174. package/src/ui/theme/README.md +3 -0
  175. package/src/ui/toolbar/README.md +3 -0
  176. package/src/ui/workflow-surface-blocked-rails.ts +94 -0
  177. package/src/ui-tailwind/chrome/chrome-preset-model.ts +107 -0
  178. package/src/ui-tailwind/chrome/chrome-preset-toolbar.tsx +15 -0
  179. package/src/ui-tailwind/chrome/responsive-chrome.ts +46 -0
  180. package/src/ui-tailwind/chrome/review-queue-bar.tsx +97 -0
  181. package/src/ui-tailwind/chrome/tw-alert-banner.tsx +64 -0
  182. package/src/ui-tailwind/chrome/tw-context-analytics-summary.tsx +122 -0
  183. package/src/ui-tailwind/chrome/tw-image-context-toolbar.tsx +121 -0
  184. package/src/ui-tailwind/chrome/tw-layout-panel.tsx +114 -0
  185. package/src/ui-tailwind/chrome/tw-object-context-toolbar.tsx +30 -0
  186. package/src/ui-tailwind/chrome/tw-page-ruler.tsx +365 -0
  187. package/src/ui-tailwind/chrome/tw-selection-tool-blocked.tsx +23 -0
  188. package/src/ui-tailwind/chrome/tw-selection-tool-comment.tsx +35 -0
  189. package/src/ui-tailwind/chrome/tw-selection-tool-formatting.tsx +37 -0
  190. package/src/ui-tailwind/chrome/tw-selection-tool-host.tsx +303 -0
  191. package/src/ui-tailwind/chrome/tw-selection-tool-structure.tsx +116 -0
  192. package/src/ui-tailwind/chrome/tw-selection-tool-suggestion.tsx +29 -0
  193. package/src/ui-tailwind/chrome/tw-selection-tool-workflow.tsx +27 -0
  194. package/src/ui-tailwind/chrome/tw-selection-toolbar.tsx +186 -0
  195. package/src/ui-tailwind/chrome/tw-suggestion-card.tsx +139 -0
  196. package/src/ui-tailwind/chrome/tw-table-context-toolbar.tsx +250 -0
  197. package/src/ui-tailwind/chrome/tw-unsaved-modal.tsx +58 -0
  198. package/src/ui-tailwind/chrome/use-before-unload.ts +20 -0
  199. package/src/ui-tailwind/editor-surface/perf-probe.ts +179 -0
  200. package/src/ui-tailwind/editor-surface/pm-command-bridge.ts +189 -0
  201. package/src/ui-tailwind/editor-surface/pm-contextual-ui.ts +31 -0
  202. package/src/ui-tailwind/editor-surface/pm-decorations.ts +411 -0
  203. package/src/ui-tailwind/editor-surface/pm-position-map.ts +123 -0
  204. package/src/ui-tailwind/editor-surface/pm-schema.ts +927 -0
  205. package/src/ui-tailwind/editor-surface/pm-state-from-snapshot.ts +567 -0
  206. package/src/ui-tailwind/editor-surface/search-plugin.ts +168 -0
  207. package/src/ui-tailwind/editor-surface/surface-build-keys.ts +63 -0
  208. package/src/ui-tailwind/editor-surface/tw-caret.tsx +12 -0
  209. package/src/ui-tailwind/editor-surface/tw-editor-surface.tsx +150 -0
  210. package/src/ui-tailwind/editor-surface/tw-inline-token.tsx +129 -0
  211. package/src/ui-tailwind/editor-surface/tw-opaque-block.tsx +58 -0
  212. package/src/ui-tailwind/editor-surface/tw-paragraph-block.tsx +151 -0
  213. package/src/ui-tailwind/editor-surface/tw-prosemirror-surface.tsx +1047 -0
  214. package/src/ui-tailwind/editor-surface/tw-segment-view.tsx +111 -0
  215. package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +503 -0
  216. package/src/ui-tailwind/index.ts +62 -0
  217. package/src/ui-tailwind/page-chrome-model.ts +27 -0
  218. package/src/ui-tailwind/review/tw-comment-sidebar.tsx +406 -0
  219. package/src/ui-tailwind/review/tw-health-panel.tsx +149 -0
  220. package/src/ui-tailwind/review/tw-review-rail.tsx +130 -0
  221. package/src/ui-tailwind/review/tw-revision-sidebar.tsx +164 -0
  222. package/src/ui-tailwind/status/tw-status-bar.tsx +65 -0
  223. package/{dist → src}/ui-tailwind/theme/editor-theme.css +58 -40
  224. package/src/ui-tailwind/toolbar/toolbar-layout.ts +47 -0
  225. package/src/ui-tailwind/toolbar/tw-toolbar-icon-button.tsx +52 -0
  226. package/src/ui-tailwind/toolbar/tw-toolbar.tsx +1478 -0
  227. package/src/ui-tailwind/tw-review-workspace.tsx +1587 -0
  228. package/src/validation/README.md +3 -0
  229. package/src/validation/compatibility-engine.ts +878 -0
  230. package/src/validation/compatibility-report.ts +161 -0
  231. package/src/validation/diagnostics.ts +204 -0
  232. package/src/validation/docx-comment-proof.ts +720 -0
  233. package/src/validation/import-diagnostics.ts +128 -0
  234. package/src/validation/low-priority-word-surfaces.ts +373 -0
  235. package/dist/canonical-document-BLEbzL2J.d.cts +0 -844
  236. package/dist/canonical-document-BLEbzL2J.d.ts +0 -844
  237. package/dist/chunk-2FJS5GZM.js +0 -763
  238. package/dist/chunk-2FJS5GZM.js.map +0 -1
  239. package/dist/chunk-2OQBZS3F.js +0 -446
  240. package/dist/chunk-2OQBZS3F.js.map +0 -1
  241. package/dist/chunk-2S7W4KFO.js +0 -127
  242. package/dist/chunk-2S7W4KFO.js.map +0 -1
  243. package/dist/chunk-2TG72QSW.js +0 -3874
  244. package/dist/chunk-2TG72QSW.js.map +0 -1
  245. package/dist/chunk-36QNIZBO.js +0 -532
  246. package/dist/chunk-36QNIZBO.js.map +0 -1
  247. package/dist/chunk-4AQOYAW4.js +0 -3069
  248. package/dist/chunk-4AQOYAW4.js.map +0 -1
  249. package/dist/chunk-4D5EWJ3P.js +0 -77
  250. package/dist/chunk-4D5EWJ3P.js.map +0 -1
  251. package/dist/chunk-5FN54NDH.js +0 -2257
  252. package/dist/chunk-5FN54NDH.js.map +0 -1
  253. package/dist/chunk-BOYGQYRQ.js +0 -7306
  254. package/dist/chunk-BOYGQYRQ.js.map +0 -1
  255. package/dist/chunk-CN3XMECL.js +0 -212
  256. package/dist/chunk-CN3XMECL.js.map +0 -1
  257. package/dist/chunk-EBI3BX6U.js +0 -164
  258. package/dist/chunk-EBI3BX6U.js.map +0 -1
  259. package/dist/chunk-EILUG3VB.js +0 -1275
  260. package/dist/chunk-EILUG3VB.js.map +0 -1
  261. package/dist/chunk-FUDY333O.js +0 -70
  262. package/dist/chunk-FUDY333O.js.map +0 -1
  263. package/dist/chunk-GBVOWFIK.js +0 -1237
  264. package/dist/chunk-GBVOWFIK.js.map +0 -1
  265. package/dist/chunk-H4TQ3H3Y.js +0 -262
  266. package/dist/chunk-H4TQ3H3Y.js.map +0 -1
  267. package/dist/chunk-JGB3IXZO.js +0 -189
  268. package/dist/chunk-JGB3IXZO.js.map +0 -1
  269. package/dist/chunk-KD2QRQPY.js +0 -4342
  270. package/dist/chunk-KD2QRQPY.js.map +0 -1
  271. package/dist/chunk-KLMXQVYK.js +0 -369
  272. package/dist/chunk-KLMXQVYK.js.map +0 -1
  273. package/dist/chunk-KZUG5KFQ.js +0 -214
  274. package/dist/chunk-KZUG5KFQ.js.map +0 -1
  275. package/dist/chunk-QDAQ4CJU.js +0 -345
  276. package/dist/chunk-QDAQ4CJU.js.map +0 -1
  277. package/dist/chunk-RMH72RZI.js.map +0 -1
  278. package/dist/chunk-SWKWQZXM.js +0 -117
  279. package/dist/chunk-SWKWQZXM.js.map +0 -1
  280. package/dist/chunk-TJBP2K4T.js.map +0 -1
  281. package/dist/chunk-TLCEAQDQ.js +0 -542
  282. package/dist/chunk-TLCEAQDQ.js.map +0 -1
  283. package/dist/chunk-UZXBISGO.js.map +0 -1
  284. package/dist/chunk-WGBAKP3Q.js +0 -3220
  285. package/dist/chunk-WGBAKP3Q.js.map +0 -1
  286. package/dist/compare/index.cjs +0 -5475
  287. package/dist/compare/index.cjs.map +0 -1
  288. package/dist/compare/index.d.cts +0 -114
  289. package/dist/compare/index.d.ts +0 -114
  290. package/dist/compare/index.js +0 -731
  291. package/dist/compare/index.js.map +0 -1
  292. package/dist/core/commands/formatting-commands.cjs +0 -828
  293. package/dist/core/commands/formatting-commands.cjs.map +0 -1
  294. package/dist/core/commands/formatting-commands.d.cts +0 -63
  295. package/dist/core/commands/formatting-commands.d.ts +0 -63
  296. package/dist/core/commands/formatting-commands.js +0 -37
  297. package/dist/core/commands/formatting-commands.js.map +0 -1
  298. package/dist/core/commands/image-commands.cjs +0 -2023
  299. package/dist/core/commands/image-commands.cjs.map +0 -1
  300. package/dist/core/commands/image-commands.d.cts +0 -58
  301. package/dist/core/commands/image-commands.d.ts +0 -58
  302. package/dist/core/commands/image-commands.js +0 -18
  303. package/dist/core/commands/image-commands.js.map +0 -1
  304. package/dist/core/commands/section-layout-commands.cjs.map +0 -1
  305. package/dist/core/commands/section-layout-commands.d.cts +0 -62
  306. package/dist/core/commands/section-layout-commands.d.ts +0 -62
  307. package/dist/core/commands/section-layout-commands.js +0 -21
  308. package/dist/core/commands/section-layout-commands.js.map +0 -1
  309. package/dist/core/commands/style-commands.cjs.map +0 -1
  310. package/dist/core/commands/style-commands.d.cts +0 -13
  311. package/dist/core/commands/style-commands.d.ts +0 -13
  312. package/dist/core/commands/style-commands.js +0 -9
  313. package/dist/core/commands/style-commands.js.map +0 -1
  314. package/dist/core/commands/table-structure-commands.cjs +0 -1883
  315. package/dist/core/commands/table-structure-commands.cjs.map +0 -1
  316. package/dist/core/commands/table-structure-commands.d.cts +0 -59
  317. package/dist/core/commands/table-structure-commands.d.ts +0 -59
  318. package/dist/core/commands/table-structure-commands.js +0 -12
  319. package/dist/core/commands/table-structure-commands.js.map +0 -1
  320. package/dist/core/commands/text-commands.cjs +0 -2391
  321. package/dist/core/commands/text-commands.cjs.map +0 -1
  322. package/dist/core/commands/text-commands.d.cts +0 -24
  323. package/dist/core/commands/text-commands.d.ts +0 -24
  324. package/dist/core/commands/text-commands.js +0 -28
  325. package/dist/core/commands/text-commands.js.map +0 -1
  326. package/dist/core/selection/mapping.cjs +0 -200
  327. package/dist/core/selection/mapping.cjs.map +0 -1
  328. package/dist/core/selection/mapping.d.cts +0 -2
  329. package/dist/core/selection/mapping.d.ts +0 -2
  330. package/dist/core/selection/mapping.js +0 -31
  331. package/dist/core/selection/mapping.js.map +0 -1
  332. package/dist/core/state/editor-state.cjs +0 -2278
  333. package/dist/core/state/editor-state.cjs.map +0 -1
  334. package/dist/core/state/editor-state.d.cts +0 -2
  335. package/dist/core/state/editor-state.d.ts +0 -2
  336. package/dist/core/state/editor-state.js +0 -26
  337. package/dist/core/state/editor-state.js.map +0 -1
  338. package/dist/index.cjs +0 -38553
  339. package/dist/index.cjs.map +0 -1
  340. package/dist/index.d.cts +0 -15
  341. package/dist/index.d.ts +0 -15
  342. package/dist/index.js +0 -7856
  343. package/dist/index.js.map +0 -1
  344. package/dist/io/docx-session.cjs +0 -16236
  345. package/dist/io/docx-session.cjs.map +0 -1
  346. package/dist/io/docx-session.d.cts +0 -21
  347. package/dist/io/docx-session.d.ts +0 -21
  348. package/dist/io/docx-session.js +0 -18
  349. package/dist/io/docx-session.js.map +0 -1
  350. package/dist/legal/index.cjs +0 -3900
  351. package/dist/legal/index.cjs.map +0 -1
  352. package/dist/legal/index.d.cts +0 -86
  353. package/dist/legal/index.d.ts +0 -86
  354. package/dist/legal/index.js +0 -616
  355. package/dist/legal/index.js.map +0 -1
  356. package/dist/public-types-7ZL_94cz.d.ts +0 -1573
  357. package/dist/public-types-CeMaDueh.d.cts +0 -1573
  358. package/dist/public-types.cjs +0 -19
  359. package/dist/public-types.cjs.map +0 -1
  360. package/dist/public-types.d.cts +0 -2
  361. package/dist/public-types.d.ts +0 -2
  362. package/dist/public-types.js +0 -1
  363. package/dist/public-types.js.map +0 -1
  364. package/dist/runtime/document-runtime.cjs +0 -11140
  365. package/dist/runtime/document-runtime.cjs.map +0 -1
  366. package/dist/runtime/document-runtime.d.cts +0 -231
  367. package/dist/runtime/document-runtime.d.ts +0 -231
  368. package/dist/runtime/document-runtime.js +0 -21
  369. package/dist/runtime/document-runtime.js.map +0 -1
  370. package/dist/structural-helpers-CilgOVhh.d.cts +0 -10
  371. package/dist/structural-helpers-q0Gd-eBN.d.ts +0 -10
  372. package/dist/ui-tailwind/editor-surface/search-plugin.cjs +0 -313
  373. package/dist/ui-tailwind/editor-surface/search-plugin.cjs.map +0 -1
  374. package/dist/ui-tailwind/editor-surface/search-plugin.d.cts +0 -67
  375. package/dist/ui-tailwind/editor-surface/search-plugin.d.ts +0 -67
  376. package/dist/ui-tailwind/editor-surface/search-plugin.js +0 -23
  377. package/dist/ui-tailwind/editor-surface/search-plugin.js.map +0 -1
  378. package/dist/ui-tailwind/index.cjs +0 -4833
  379. package/dist/ui-tailwind/index.cjs.map +0 -1
  380. package/dist/ui-tailwind/index.d.cts +0 -617
  381. package/dist/ui-tailwind/index.d.ts +0 -617
  382. package/dist/ui-tailwind/index.js +0 -575
  383. package/dist/ui-tailwind/index.js.map +0 -1
@@ -0,0 +1,878 @@
1
+ import { createRangeAnchor } from "../core/selection/mapping.ts";
2
+ import type {
3
+ CanonicalDocumentEnvelope,
4
+ CompatibilityFeatureEntry,
5
+ CompatibilityReport,
6
+ EditorError,
7
+ EditorWarning,
8
+ } from "../core/state/editor-state.ts";
9
+ import type {
10
+ BlockNode,
11
+ DocumentRootNode,
12
+ InlineNode,
13
+ ParagraphNode,
14
+ } from "../model/canonical-document.ts";
15
+ import {
16
+ describeOpaqueFragment,
17
+ describeStructuredWrapperBlock,
18
+ listOpaqueFragments,
19
+ listPreservedPackageParts,
20
+ } from "../preservation/store.ts";
21
+
22
+ export interface BuildCompatibilityReportInput {
23
+ document: CanonicalDocumentEnvelope;
24
+ warnings?: readonly EditorWarning[];
25
+ fatalError?: EditorError;
26
+ generatedAt: string;
27
+ }
28
+
29
+ export function buildCompatibilityReport(
30
+ input: BuildCompatibilityReportInput,
31
+ ): CompatibilityReport {
32
+ const content = normalizeDocumentRoot(input.document.content);
33
+ const contentFeatures = collectContentFeatures(content);
34
+ if (hasSupportedRuntimeComments(input.document.review.comments)) {
35
+ contentFeatures.push(
36
+ supportedEntry(
37
+ "comments-single-paragraph",
38
+ "Single-paragraph review comments stay mappable through runtime selections.",
39
+ ),
40
+ );
41
+ }
42
+ const featureEntries: CompatibilityFeatureEntry[] = [
43
+ ...contentFeatures,
44
+ ...collectStructuredWrapperFeatures(content),
45
+ ...collectReviewFeatures(input.document),
46
+ ...collectPreservationFeatures(input.document),
47
+ ];
48
+ const warnings = dedupeWarnings([
49
+ ...(input.warnings ?? []),
50
+ ...collectDiagnosticWarnings(input.document),
51
+ ]);
52
+ const errors = dedupeErrors([
53
+ ...collectDiagnosticErrors(input.document),
54
+ ...(input.fatalError ? [input.fatalError] : []),
55
+ ]);
56
+
57
+ return {
58
+ reportVersion: "compatibility-report/1",
59
+ generatedAt: input.generatedAt,
60
+ blockExport:
61
+ featureEntries.some((entry) => entry.featureClass === "unsupported-fatal") ||
62
+ errors.some((error) => error.isFatal),
63
+ featureEntries,
64
+ warnings,
65
+ errors,
66
+ };
67
+ }
68
+
69
+ function hasSupportedRuntimeComments(
70
+ comments: CanonicalDocumentEnvelope["review"]["comments"],
71
+ ): boolean {
72
+ return Object.values(comments).some((comment) => comment.anchor.kind !== "detached");
73
+ }
74
+
75
+ function collectReviewFeatures(
76
+ document: CanonicalDocumentEnvelope,
77
+ ): CompatibilityFeatureEntry[] {
78
+ const entries: CompatibilityFeatureEntry[] = [];
79
+ const comments = Object.values(document.review.comments ?? {});
80
+ const revisions = Object.values(document.review.revisions ?? {});
81
+
82
+ const multiParagraphComment = comments.find(
83
+ (comment) => comment.metadata?.detachedReason === "multi-paragraph" && comment.anchor.kind === "detached",
84
+ );
85
+ if (multiParagraphComment?.anchor.kind === "detached") {
86
+ entries.push({
87
+ featureEntryId: "feature:comments-multi-paragraph-ranges",
88
+ featureKey: "comments-multi-paragraph-ranges",
89
+ featureClass: "preserve-only",
90
+ message:
91
+ "Cross-paragraph comment anchors remain visible but detached to preserve package truth and export safety.",
92
+ affectedAnchor: createRangeAnchor(
93
+ multiParagraphComment.anchor.lastKnownRange.from,
94
+ multiParagraphComment.anchor.lastKnownRange.to,
95
+ ),
96
+ details: {
97
+ commentId: multiParagraphComment.commentId,
98
+ detachedReason: multiParagraphComment.metadata?.detachedReason,
99
+ actionabilityNote: multiParagraphComment.metadata?.actionabilityNote,
100
+ },
101
+ });
102
+ }
103
+
104
+ const moveRevision = revisions.find((revision) => revision.kind === "move");
105
+ if (moveRevision) {
106
+ entries.push({
107
+ featureEntryId: "feature:tracked-moves",
108
+ featureKey: "tracked-moves",
109
+ featureClass: "preserve-only",
110
+ message:
111
+ "Tracked move revisions are preserved for round-trip and review visibility but remain non-actionable in the current runtime.",
112
+ affectedAnchor: toRevisionAffectedAnchor(moveRevision),
113
+ details: {
114
+ revisionId: moveRevision.changeId,
115
+ moveDirection: moveRevision.metadata?.moveData?.direction,
116
+ linkedRevisionId: moveRevision.metadata?.moveData?.linkedRevisionId,
117
+ },
118
+ });
119
+ }
120
+
121
+ const formattingRevisions = revisions.filter(
122
+ (revision) =>
123
+ revision.kind === "formatting" ||
124
+ (revision.kind === "property-change" && revision.metadata?.propertyChangeData?.xmlTag === "rPrChange"),
125
+ );
126
+ if (formattingRevisions.length > 0) {
127
+ const featureClass = formattingRevisions.some(
128
+ (revision) => typeof revision.metadata?.preserveOnlyReason === "string" && revision.metadata.preserveOnlyReason.length > 0,
129
+ )
130
+ ? "preserve-only"
131
+ : "supported-roundtrip";
132
+ entries.push({
133
+ featureEntryId: "feature:formatting-revisions",
134
+ featureKey: "formatting-revisions",
135
+ featureClass,
136
+ message:
137
+ featureClass === "preserve-only"
138
+ ? "Formatting revisions remain visible for review, but at least one formatting-change slice is still preserve-only."
139
+ : "Formatting-change revisions remain structured and round-trip through the runtime review model.",
140
+ affectedAnchor: toRevisionAffectedAnchor(formattingRevisions[0]!),
141
+ details: {
142
+ revisionCount: formattingRevisions.length,
143
+ },
144
+ });
145
+ }
146
+
147
+ const structuralRevisions = revisions.filter(
148
+ (revision) =>
149
+ revision.kind === "property-change" &&
150
+ revision.metadata?.propertyChangeData?.xmlTag !== "rPrChange",
151
+ );
152
+ if (structuralRevisions.length > 0) {
153
+ const featureClass = structuralRevisions.some(
154
+ (revision) => typeof revision.metadata?.preserveOnlyReason === "string" && revision.metadata.preserveOnlyReason.length > 0,
155
+ )
156
+ ? "preserve-only"
157
+ : "supported-roundtrip";
158
+ entries.push({
159
+ featureEntryId: "feature:structural-revisions",
160
+ featureKey: "structural-revisions",
161
+ featureClass,
162
+ message:
163
+ featureClass === "preserve-only"
164
+ ? "Structural property revisions are preserved for round-trip, but at least one slice remains non-actionable."
165
+ : "Structured paragraph, table, or section property revisions remain mapped in the runtime review model.",
166
+ affectedAnchor: toRevisionAffectedAnchor(structuralRevisions[0]!),
167
+ details: {
168
+ revisionCount: structuralRevisions.length,
169
+ revisionKinds: structuralRevisions.map(
170
+ (revision) => revision.metadata?.propertyChangeData?.xmlTag ?? revision.kind,
171
+ ),
172
+ },
173
+ });
174
+ }
175
+
176
+ return entries;
177
+ }
178
+
179
+ function normalizeDocumentRoot(content: unknown): DocumentRootNode {
180
+ if (content && typeof content === "object" && (content as { type?: string }).type === "doc") {
181
+ return content as DocumentRootNode;
182
+ }
183
+
184
+ if (Array.isArray(content)) {
185
+ return {
186
+ type: "doc",
187
+ children: content.filter(
188
+ (value): value is DocumentRootNode["children"][number] =>
189
+ Boolean(value) &&
190
+ typeof value === "object" &&
191
+ ((value as { type?: string }).type === "paragraph" ||
192
+ (value as { type?: string }).type === "opaque_block"),
193
+ ),
194
+ };
195
+ }
196
+
197
+ return {
198
+ type: "doc",
199
+ children: [{ type: "paragraph", children: [] }],
200
+ };
201
+ }
202
+
203
+ function collectContentFeatures(
204
+ content: DocumentRootNode,
205
+ ): CompatibilityFeatureEntry[] {
206
+ const flags = {
207
+ paragraphs: false,
208
+ runs: false,
209
+ whitespace: false,
210
+ headings: false,
211
+ lists: false,
212
+ hyperlinks: false,
213
+ images: false,
214
+ tables: false,
215
+ mergedCells: false,
216
+ sections: false,
217
+ contentControls: false,
218
+ };
219
+
220
+ for (let index = 0; index < content.children.length; index += 1) {
221
+ measureBlock(content.children[index], flags);
222
+ }
223
+
224
+ const entries: CompatibilityFeatureEntry[] = [];
225
+ if (flags.paragraphs) {
226
+ entries.push(supportedEntry("paragraphs", "Paragraph structure is editable and round-trippable."));
227
+ }
228
+ if (flags.runs) {
229
+ entries.push(supportedEntry("runs", "Runs and inline text are editable through runtime commands."));
230
+ }
231
+ if (flags.whitespace) {
232
+ entries.push(supportedEntry("whitespace", "Whitespace-sensitive text units stay explicit in the runtime model."));
233
+ }
234
+ if (flags.headings) {
235
+ entries.push(supportedEntry("headings", "Heading styles remain attached to paragraph structure."));
236
+ }
237
+ if (flags.lists) {
238
+ entries.push(supportedEntry("lists", "Numbering metadata stays attached to paragraph blocks."));
239
+ }
240
+ if (flags.hyperlinks) {
241
+ entries.push(supportedEntry("hyperlinks", "Hyperlink relationships are preserved and re-serialized."));
242
+ }
243
+ if (flags.images) {
244
+ entries.push(supportedEntry("inline-images", "Inline image placements stay attached to preserved media parts."));
245
+ }
246
+ if (flags.tables) {
247
+ entries.push(
248
+ supportedEntry(
249
+ "tables",
250
+ "Structured tables keep bounded topology and common visual properties through runtime editing and export.",
251
+ ),
252
+ );
253
+ }
254
+ if (flags.mergedCells) {
255
+ entries.push(
256
+ supportedEntry(
257
+ "merged-cells",
258
+ "Merged-cell topology survives through the supported table runtime and export path.",
259
+ ),
260
+ );
261
+ }
262
+ if (flags.sections) {
263
+ entries.push(
264
+ supportedEntry(
265
+ "sections",
266
+ "Section boundaries and page-layout properties remain structured and export-safe.",
267
+ ),
268
+ );
269
+ }
270
+ if (flags.contentControls) {
271
+ entries.push(
272
+ supportedEntry(
273
+ "content-controls",
274
+ "Common content controls render through structured SDT state and round-trip without flattening to opaque placeholders.",
275
+ ),
276
+ );
277
+ }
278
+ return entries;
279
+ }
280
+
281
+ function measureBlock(
282
+ block: BlockNode,
283
+ flags: {
284
+ paragraphs: boolean;
285
+ runs: boolean;
286
+ whitespace: boolean;
287
+ headings: boolean;
288
+ lists: boolean;
289
+ hyperlinks: boolean;
290
+ images: boolean;
291
+ tables: boolean;
292
+ mergedCells: boolean;
293
+ sections: boolean;
294
+ contentControls: boolean;
295
+ },
296
+ ): number {
297
+ switch (block.type) {
298
+ case "paragraph":
299
+ flags.paragraphs = true;
300
+ if (block.styleId?.toLowerCase().startsWith("heading")) {
301
+ flags.headings = true;
302
+ }
303
+ if (block.numbering) {
304
+ flags.lists = true;
305
+ }
306
+ return measureParagraph(block, flags);
307
+ case "table":
308
+ flags.tables = true;
309
+ if (
310
+ block.rows.some((row) =>
311
+ row.cells.some((cell) => (cell.gridSpan ?? 1) > 1 || cell.verticalMerge != null),
312
+ )
313
+ ) {
314
+ flags.mergedCells = true;
315
+ }
316
+ return block.rows.reduce(
317
+ (size, row) =>
318
+ size +
319
+ row.cells.reduce(
320
+ (cellSize, cell) =>
321
+ cellSize + cell.children.reduce((childSize, child) => childSize + measureBlock(child, flags), 0),
322
+ 0,
323
+ ),
324
+ 0,
325
+ );
326
+ case "sdt":
327
+ if (describeStructuredWrapperBlock(block)?.featureKey === "content-controls") {
328
+ return 1;
329
+ }
330
+ flags.contentControls = true;
331
+ return block.children.reduce((size, child) => size + measureBlock(child, flags), 0);
332
+ case "custom_xml":
333
+ return 1;
334
+ case "alt_chunk":
335
+ return 1;
336
+ case "section_break":
337
+ flags.sections = true;
338
+ return 1;
339
+ default:
340
+ return 0;
341
+ }
342
+ }
343
+
344
+ function collectStructuredWrapperFeatures(
345
+ content: DocumentRootNode,
346
+ ): CompatibilityFeatureEntry[] {
347
+ const entries: CompatibilityFeatureEntry[] = [];
348
+ let structuredWrapperIndex = 0;
349
+
350
+ function visitBlock(block: BlockNode): void {
351
+ const descriptor = describeStructuredWrapperBlock(block);
352
+ if (descriptor) {
353
+ entries.push({
354
+ featureEntryId: `feature:structured-wrapper:${structuredWrapperIndex}`,
355
+ featureKey: descriptor.featureKey,
356
+ featureClass: "preserve-only",
357
+ message: descriptor.label,
358
+ details: {
359
+ detail: descriptor.detail,
360
+ surface: "structured-wrapper",
361
+ },
362
+ });
363
+ structuredWrapperIndex += 1;
364
+ return;
365
+ }
366
+
367
+ if (block.type === "table") {
368
+ for (const row of block.rows) {
369
+ for (const cell of row.cells) {
370
+ for (const child of cell.children) {
371
+ visitBlock(child);
372
+ }
373
+ }
374
+ }
375
+ return;
376
+ }
377
+
378
+ if (block.type === "sdt") {
379
+ for (const child of block.children) {
380
+ visitBlock(child);
381
+ }
382
+ }
383
+ }
384
+
385
+ for (const block of content.children) {
386
+ visitBlock(block);
387
+ }
388
+
389
+ return entries;
390
+ }
391
+
392
+ function measureParagraph(
393
+ paragraph: ParagraphNode,
394
+ flags: {
395
+ paragraphs: boolean;
396
+ runs: boolean;
397
+ whitespace: boolean;
398
+ hyperlinks: boolean;
399
+ images: boolean;
400
+ tables: boolean;
401
+ sections: boolean;
402
+ contentControls: boolean;
403
+ },
404
+ ): number {
405
+ let size = 0;
406
+ const children = Array.isArray(paragraph.children) ? paragraph.children : [];
407
+
408
+ for (const child of children) {
409
+ size += measureInlineNode(child, flags);
410
+ }
411
+
412
+ return size;
413
+ }
414
+
415
+ function measureInlineNode(
416
+ node: InlineNode,
417
+ flags: {
418
+ paragraphs: boolean;
419
+ runs: boolean;
420
+ whitespace: boolean;
421
+ hyperlinks: boolean;
422
+ images: boolean;
423
+ tables: boolean;
424
+ sections: boolean;
425
+ contentControls: boolean;
426
+ },
427
+ ): number {
428
+ switch (node.type) {
429
+ case "text":
430
+ flags.runs = flags.runs || node.text.length > 0;
431
+ flags.whitespace =
432
+ flags.whitespace ||
433
+ /^\s/u.test(node.text) ||
434
+ /\s$/u.test(node.text) ||
435
+ node.text.includes(" ");
436
+ return Array.from(node.text).length;
437
+ case "tab":
438
+ case "hard_break":
439
+ flags.runs = true;
440
+ flags.whitespace = true;
441
+ return 1;
442
+ case "hyperlink":
443
+ flags.runs = true;
444
+ flags.hyperlinks = true;
445
+ return node.children.reduce((size, child) => size + measureInlineNode(child, flags), 0);
446
+ case "image":
447
+ flags.images = true;
448
+ flags.runs = true;
449
+ return 1;
450
+ case "opaque_inline":
451
+ flags.runs = true;
452
+ return 1;
453
+ case "column_break":
454
+ case "symbol":
455
+ case "field":
456
+ case "bookmark_start":
457
+ case "bookmark_end":
458
+ case "footnote_ref":
459
+ case "chart_preview":
460
+ case "smartart_preview":
461
+ case "shape":
462
+ case "wordart":
463
+ case "vml_shape":
464
+ default:
465
+ flags.runs = true;
466
+ return 1;
467
+ }
468
+ }
469
+
470
+ function collectPreservationFeatures(
471
+ document: CanonicalDocumentEnvelope,
472
+ ): CompatibilityFeatureEntry[] {
473
+ const entries: CompatibilityFeatureEntry[] = [];
474
+ const structuredSubPartPaths = collectStructuredSubPartPaths(document);
475
+
476
+ entries.push(...collectSubPartFeatures(document));
477
+
478
+ for (const fragment of listOpaqueFragments(document.preservation as never)) {
479
+ const descriptor = describeOpaqueFragment(fragment);
480
+ entries.push({
481
+ featureEntryId: `feature:${fragment.fragmentId}`,
482
+ featureKey: descriptor.featureKey,
483
+ featureClass: "preserve-only",
484
+ message: descriptor.label,
485
+ affectedAnchor: createRangeAnchor(
486
+ fragment.lastKnownRange.from,
487
+ fragment.lastKnownRange.to,
488
+ ),
489
+ details: {
490
+ fragmentId: fragment.fragmentId,
491
+ warningId: fragment.warningId,
492
+ detail: descriptor.detail,
493
+ },
494
+ });
495
+ }
496
+
497
+ for (const packagePart of listPreservedPackageParts(document.preservation as never)) {
498
+ if (structuredSubPartPaths.has(packagePart.packagePartName)) {
499
+ continue;
500
+ }
501
+
502
+ entries.push({
503
+ featureEntryId: `feature:package:${packagePart.packagePartName}`,
504
+ featureKey: "unknown-package-parts",
505
+ featureClass: "preserve-only",
506
+ message: `Preserved package part ${packagePart.packagePartName}.`,
507
+ details: {
508
+ packagePartName: packagePart.packagePartName,
509
+ contentType: packagePart.contentType,
510
+ relationshipIds: packagePart.relationshipIds,
511
+ },
512
+ });
513
+ }
514
+
515
+ return entries;
516
+ }
517
+
518
+ function collectStructuredSubPartPaths(
519
+ document: CanonicalDocumentEnvelope,
520
+ ): ReadonlySet<string> {
521
+ const subParts = document.subParts;
522
+ if (!subParts) {
523
+ return new Set<string>();
524
+ }
525
+
526
+ const paths = new Set<string>();
527
+
528
+ for (const header of subParts.headers ?? []) {
529
+ paths.add(header.partPath);
530
+ }
531
+
532
+ for (const footer of subParts.footers ?? []) {
533
+ paths.add(footer.partPath);
534
+ }
535
+
536
+ if (subParts.footnoteCollection) {
537
+ paths.add("/word/footnotes.xml");
538
+ paths.add("/word/endnotes.xml");
539
+ }
540
+
541
+ if (subParts.theme) {
542
+ paths.add("/word/theme/theme1.xml");
543
+ }
544
+
545
+ return paths;
546
+ }
547
+
548
+ function collectSubPartFeatures(
549
+ document: CanonicalDocumentEnvelope,
550
+ ): CompatibilityFeatureEntry[] {
551
+ const subParts = document.subParts;
552
+ if (!subParts) {
553
+ return [];
554
+ }
555
+
556
+ const entries: CompatibilityFeatureEntry[] = collectLossySubPartFeatures(subParts);
557
+ const hasHeaderFooterContent = (subParts.headers?.length ?? 0) > 0 || (subParts.footers?.length ?? 0) > 0;
558
+ const noteCount =
559
+ Object.keys(subParts.footnoteCollection?.footnotes ?? {}).length +
560
+ Object.keys(subParts.footnoteCollection?.endnotes ?? {}).length;
561
+ const headerFooterBookmarkCount =
562
+ [...(subParts.headers ?? []), ...(subParts.footers ?? [])].reduce(
563
+ (count, subPart) => count + countBookmarksInBlocks(subPart.blocks),
564
+ 0,
565
+ );
566
+ const noteBookmarkCount =
567
+ [
568
+ ...Object.values(subParts.footnoteCollection?.footnotes ?? {}),
569
+ ...Object.values(subParts.footnoteCollection?.endnotes ?? {}),
570
+ ].reduce((count, note) => count + countBookmarksInBlocks(note.blocks), 0);
571
+
572
+ if (hasHeaderFooterContent && !entries.some((entry) => entry.featureKey === "headers-footers-lossy")) {
573
+ entries.push({
574
+ featureEntryId: "feature:subparts:headers-footers",
575
+ featureKey: "headers-footers",
576
+ featureClass: "supported-roundtrip",
577
+ message: "Headers and footers resolve as structured secondary stories with export-safe ownership.",
578
+ details: {
579
+ headerCount: subParts.headers?.length ?? 0,
580
+ footerCount: subParts.footers?.length ?? 0,
581
+ bookmarkCount: headerFooterBookmarkCount,
582
+ },
583
+ });
584
+ }
585
+
586
+ if (noteCount > 0 && !entries.some((entry) => entry.featureKey === "notes-lossy")) {
587
+ entries.push({
588
+ featureEntryId: "feature:subparts:notes",
589
+ featureKey: "notes",
590
+ featureClass: "supported-roundtrip",
591
+ message: "Footnotes and endnotes resolve as structured secondary stories with export-safe ownership.",
592
+ details: {
593
+ footnoteCount: Object.keys(subParts.footnoteCollection?.footnotes ?? {}).length,
594
+ endnoteCount: Object.keys(subParts.footnoteCollection?.endnotes ?? {}).length,
595
+ bookmarkCount: noteBookmarkCount,
596
+ },
597
+ });
598
+ }
599
+
600
+ if (subParts.theme) {
601
+ entries.push({
602
+ featureEntryId: "feature:subparts:theme",
603
+ featureKey: "unknown-package-parts",
604
+ featureClass: "preserve-only",
605
+ message: "Theme metadata is preserved through structured sub-part ownership.",
606
+ details: {
607
+ surface: "theme-subpart",
608
+ themeName: subParts.theme.name,
609
+ },
610
+ });
611
+ }
612
+
613
+ return entries;
614
+ }
615
+
616
+ function collectLossySubPartFeatures(
617
+ subParts: NonNullable<CanonicalDocumentEnvelope["subParts"]>,
618
+ ): CompatibilityFeatureEntry[] {
619
+ const entries: CompatibilityFeatureEntry[] = [];
620
+
621
+ const headerFooterLossy = [
622
+ ...(subParts.headers ?? []).flatMap((header) =>
623
+ collectLossyBlocks(header.blocks, `header:${header.partPath}`),
624
+ ),
625
+ ...(subParts.footers ?? []).flatMap((footer) =>
626
+ collectLossyBlocks(footer.blocks, `footer:${footer.partPath}`),
627
+ ),
628
+ ];
629
+ if (headerFooterLossy.length > 0) {
630
+ entries.push({
631
+ featureEntryId: "feature:subparts:headers-footers-lossy",
632
+ featureKey: "headers-footers-lossy",
633
+ featureClass: "unsupported-fatal",
634
+ message: "Headers or footers contain content the current sub-part serializer cannot re-emit safely.",
635
+ details: {
636
+ issues: headerFooterLossy,
637
+ },
638
+ });
639
+ }
640
+
641
+ const noteLossy = [
642
+ ...Object.values(subParts.footnoteCollection?.footnotes ?? {}).flatMap((note) =>
643
+ collectLossyBlocks(note.blocks, `footnote:${note.noteId}`),
644
+ ),
645
+ ...Object.values(subParts.footnoteCollection?.endnotes ?? {}).flatMap((note) =>
646
+ collectLossyBlocks(note.blocks, `endnote:${note.noteId}`),
647
+ ),
648
+ ];
649
+ if (noteLossy.length > 0) {
650
+ entries.push({
651
+ featureEntryId: "feature:subparts:notes-lossy",
652
+ featureKey: "notes-lossy",
653
+ featureClass: "unsupported-fatal",
654
+ message: "Footnotes or endnotes contain content the current sub-part serializer cannot re-emit safely.",
655
+ details: {
656
+ issues: noteLossy,
657
+ },
658
+ });
659
+ }
660
+
661
+ return entries;
662
+ }
663
+
664
+ function countBookmarksInBlocks(blocks: readonly BlockNode[]): number {
665
+ return blocks.reduce((count, block) => count + countBookmarksInBlock(block), 0);
666
+ }
667
+
668
+ function countBookmarksInBlock(block: BlockNode): number {
669
+ switch (block.type) {
670
+ case "paragraph":
671
+ return (block.children ?? []).reduce((count, child) => count + countBookmarksInInlineNode(child), 0);
672
+ case "table":
673
+ return block.rows.reduce(
674
+ (count, row) =>
675
+ count +
676
+ row.cells.reduce(
677
+ (cellCount, cell) =>
678
+ cellCount + cell.children.reduce((childCount, child) => childCount + countBookmarksInBlock(child), 0),
679
+ 0,
680
+ ),
681
+ 0,
682
+ );
683
+ case "sdt":
684
+ case "custom_xml":
685
+ return block.children.reduce((count, child) => count + countBookmarksInBlock(child), 0);
686
+ default:
687
+ return 0;
688
+ }
689
+ }
690
+
691
+ function countBookmarksInInlineNode(node: InlineNode): number {
692
+ switch (node.type) {
693
+ case "bookmark_start":
694
+ return 1;
695
+ case "hyperlink":
696
+ case "field":
697
+ return (node.children ?? []).reduce((count, child) => count + countBookmarksInInlineNode(child), 0);
698
+ default:
699
+ return 0;
700
+ }
701
+ }
702
+
703
+ function collectLossyBlocks(
704
+ blocks: readonly BlockNode[],
705
+ surface: string,
706
+ ): string[] {
707
+ const issues: string[] = [];
708
+
709
+ for (const block of blocks) {
710
+ if (block.type === "table") {
711
+ for (const row of block.rows) {
712
+ for (const cell of row.cells) {
713
+ issues.push(...collectLossyBlocks(cell.children, `${surface}:table-cell`));
714
+ }
715
+ }
716
+ continue;
717
+ }
718
+
719
+ if (block.type !== "paragraph") {
720
+ issues.push(`${surface}:${block.type}`);
721
+ continue;
722
+ }
723
+
724
+ if (
725
+ block.numbering !== undefined ||
726
+ block.keepNext !== undefined ||
727
+ block.keepLines !== undefined ||
728
+ block.outlineLevel !== undefined ||
729
+ block.pageBreakBefore !== undefined ||
730
+ block.widowControl !== undefined ||
731
+ block.borders !== undefined ||
732
+ block.shading !== undefined ||
733
+ block.bidi !== undefined ||
734
+ block.suppressLineNumbers !== undefined ||
735
+ block.cnfStyle !== undefined
736
+ ) {
737
+ issues.push(`${surface}:paragraph-properties`);
738
+ }
739
+
740
+ for (const child of block.children) {
741
+ issues.push(...collectLossyInlineContent(child, surface));
742
+ }
743
+ }
744
+
745
+ return [...new Set(issues)];
746
+ }
747
+
748
+ function collectLossyInlineContent(
749
+ node: InlineNode,
750
+ surface: string,
751
+ ): string[] {
752
+ switch (node.type) {
753
+ case "text": {
754
+ const allowSecondaryStoryColorMarks =
755
+ surface.startsWith("header:") || surface.startsWith("footer:");
756
+ const unsupportedMarks = (node.marks ?? [])
757
+ .filter(
758
+ (mark) =>
759
+ mark.type !== "bold" &&
760
+ mark.type !== "italic" &&
761
+ mark.type !== "underline" &&
762
+ mark.type !== "strikethrough" &&
763
+ mark.type !== "doubleStrikethrough" &&
764
+ mark.type !== "fontFamily" &&
765
+ mark.type !== "fontSize" &&
766
+ mark.type !== "textColor" &&
767
+ (!allowSecondaryStoryColorMarks || mark.type !== "backgroundColor") &&
768
+ (!allowSecondaryStoryColorMarks || mark.type !== "highlight") &&
769
+ mark.type !== "smallCaps" &&
770
+ mark.type !== "allCaps",
771
+ )
772
+ .map((mark) => `${surface}:mark:${mark.type}`);
773
+ return unsupportedMarks;
774
+ }
775
+ case "tab":
776
+ case "hard_break":
777
+ case "footnote_ref":
778
+ case "field":
779
+ case "bookmark_start":
780
+ case "bookmark_end":
781
+ case "shape":
782
+ case "wordart":
783
+ case "vml_shape":
784
+ case "chart_preview":
785
+ case "smartart_preview":
786
+ return [];
787
+ default:
788
+ return [`${surface}:${node.type}`];
789
+ }
790
+ }
791
+
792
+ function collectDiagnosticWarnings(
793
+ document: CanonicalDocumentEnvelope,
794
+ ): EditorWarning[] {
795
+ const diagnostics = Array.isArray(document.diagnostics?.warnings)
796
+ ? document.diagnostics.warnings
797
+ : [];
798
+
799
+ return diagnostics.map((warning) => ({
800
+ warningId: warning.warningId,
801
+ code:
802
+ warning.source === "validation" || warning.source === "export"
803
+ ? "export_roundtrip_risk"
804
+ : warning.source === "preservation"
805
+ ? "unsupported_ooxml_preserved"
806
+ : "import_normalized",
807
+ severity: "warning",
808
+ message: warning.message,
809
+ source: warning.source,
810
+ }));
811
+ }
812
+
813
+ function collectDiagnosticErrors(
814
+ document: CanonicalDocumentEnvelope,
815
+ ): EditorError[] {
816
+ const diagnostics = Array.isArray(document.diagnostics?.errors)
817
+ ? document.diagnostics.errors
818
+ : [];
819
+
820
+ return diagnostics.map((error) => ({
821
+ errorId: error.diagnosticId,
822
+ code:
823
+ error.code === "load_failed"
824
+ ? "import_failed"
825
+ : error.code,
826
+ message: error.message,
827
+ isFatal: error.isFatal,
828
+ source: error.source,
829
+ }));
830
+ }
831
+
832
+ function supportedEntry(
833
+ featureKey: CompatibilityFeatureEntry["featureKey"],
834
+ message: string,
835
+ ): CompatibilityFeatureEntry {
836
+ return {
837
+ featureEntryId: `feature:${featureKey}`,
838
+ featureKey,
839
+ featureClass: "supported-roundtrip",
840
+ message,
841
+ };
842
+ }
843
+
844
+ function toRevisionAffectedAnchor(
845
+ revision: CanonicalDocumentEnvelope["review"]["revisions"][string],
846
+ ) {
847
+ switch (revision.anchor.kind) {
848
+ case "range":
849
+ return createRangeAnchor(revision.anchor.range.from, revision.anchor.range.to);
850
+ case "node":
851
+ return createRangeAnchor(revision.anchor.at, revision.anchor.at + 1);
852
+ case "detached":
853
+ return createRangeAnchor(
854
+ revision.anchor.lastKnownRange.from,
855
+ revision.anchor.lastKnownRange.to,
856
+ );
857
+ }
858
+ }
859
+
860
+ function dedupeWarnings(warnings: EditorWarning[]): EditorWarning[] {
861
+ const byId = new Map<string, EditorWarning>();
862
+
863
+ for (const warning of warnings) {
864
+ byId.set(warning.warningId, warning);
865
+ }
866
+
867
+ return [...byId.values()];
868
+ }
869
+
870
+ function dedupeErrors(errors: EditorError[]): EditorError[] {
871
+ const byId = new Map<string, EditorError>();
872
+
873
+ for (const error of errors) {
874
+ byId.set(error.errorId, error);
875
+ }
876
+
877
+ return [...byId.values()];
878
+ }