@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,947 @@
1
+ import type { CommentEntry, CommentThread } from "../../review/store/comment-store.ts";
2
+ import type { RevisionParagraphBoundary } from "../ooxml/revision-boundaries.ts";
3
+ import type { ImportedCommentDefinition } from "../ooxml/parse-comments.ts";
4
+
5
+ interface XmlElementNode {
6
+ type: "element";
7
+ name: string;
8
+ attributes: Record<string, string>;
9
+ children: XmlNode[];
10
+ start: number;
11
+ end: number;
12
+ }
13
+
14
+ interface XmlTextNode {
15
+ type: "text";
16
+ text: string;
17
+ start: number;
18
+ end: number;
19
+ }
20
+
21
+ type XmlNode = XmlElementNode | XmlTextNode;
22
+
23
+ interface ParagraphBoundaryMap {
24
+ paragraphIndex: number;
25
+ start: number;
26
+ end: number;
27
+ boundaries: Map<number, number>;
28
+ }
29
+
30
+ interface SerializableCommentEntry {
31
+ thread: CommentThread;
32
+ entry: CommentEntry;
33
+ exportCommentId: string;
34
+ paraId: string;
35
+ durableId?: string;
36
+ isRoot: boolean;
37
+ }
38
+
39
+ export interface SerializedCommentsResult {
40
+ commentsXml: string;
41
+ serializedCommentIds: string[];
42
+ commentsExtendedXml?: string;
43
+ commentsIdsXml?: string;
44
+ peopleXml?: string;
45
+ }
46
+
47
+ export interface SerializeMergedCommentsXmlOptions {
48
+ preservedDefinitions?: readonly ImportedCommentDefinition[];
49
+ sourceRootTag?: string;
50
+ sourceExtendedRootTag?: string;
51
+ sourceIdsRootTag?: string;
52
+ sourcePeopleRootTag?: string;
53
+ peopleAuthors?: readonly string[];
54
+ exportCommentIds?: ReadonlyMap<string, string>;
55
+ }
56
+
57
+ export interface SerializeCommentAnchorsOptions {
58
+ exportCommentIds?: ReadonlyMap<string, string>;
59
+ }
60
+
61
+ export interface SerializedCommentDocumentResult {
62
+ documentXml: string;
63
+ serializedCommentIds: string[];
64
+ skippedCommentIds: string[];
65
+ }
66
+
67
+ const DEFAULT_COMMENTS_ROOT_TAG =
68
+ `<w:comments xmlns:w="http://schemas.openxmlformats.org/wordprocessingml/2006/main" xmlns:w14="http://schemas.microsoft.com/office/word/2010/wordml" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="w14">`;
69
+ const DEFAULT_COMMENTS_EXTENDED_ROOT_TAG =
70
+ `<w15:commentsEx xmlns:w15="http://schemas.microsoft.com/office/word/2012/wordml">`;
71
+ const DEFAULT_COMMENTS_IDS_ROOT_TAG =
72
+ `<w16cid:commentsIds xmlns:w16cid="http://schemas.microsoft.com/office/word/2016/wordml/cid">`;
73
+ const DEFAULT_PEOPLE_ROOT_TAG =
74
+ `<w15:people xmlns:w15="http://schemas.microsoft.com/office/word/2012/wordml">`;
75
+
76
+ export function serializeCommentsXml(
77
+ threads: readonly CommentThread[],
78
+ options: {
79
+ exportCommentIds?: ReadonlyMap<string, string>;
80
+ } = {},
81
+ ): SerializedCommentsResult {
82
+ return serializeMergedCommentsXml(threads, {
83
+ exportCommentIds: options.exportCommentIds,
84
+ });
85
+ }
86
+
87
+ export function serializeMergedCommentsXml(
88
+ threads: readonly CommentThread[],
89
+ options: SerializeMergedCommentsXmlOptions = {},
90
+ ): SerializedCommentsResult {
91
+ const serializableThreads = threads.filter((thread) => thread.entries.length > 0);
92
+ const preservedDefinitions = (options.preservedDefinitions ?? [])
93
+ .slice()
94
+ .sort((left, right) => left.order - right.order);
95
+ const runtimeThreadById = new Map(
96
+ serializableThreads.map((thread) => [thread.commentId, thread]),
97
+ );
98
+ const exportCommentIds =
99
+ options.exportCommentIds ??
100
+ createCommentExportIdMap(serializableThreads, preservedDefinitions);
101
+ const serializableEntries = createSerializableEntries(
102
+ serializableThreads,
103
+ preservedDefinitions,
104
+ exportCommentIds,
105
+ );
106
+ const serializedEntryByOoxmlId = new Map(
107
+ serializableEntries.map((entry) => [entry.exportCommentId, entry]),
108
+ );
109
+ const emittedThreadIds = new Set<string>();
110
+ const mergedComments: string[] = [];
111
+
112
+ for (const definition of preservedDefinitions) {
113
+ const ownedThread = runtimeThreadById.get(definition.commentId);
114
+ if (ownedThread) {
115
+ if (emittedThreadIds.has(ownedThread.commentId)) {
116
+ continue;
117
+ }
118
+ mergedComments.push(
119
+ ...serializeThreadEntries(
120
+ serializableEntries.filter((entry) => entry.thread.commentId === ownedThread.commentId),
121
+ ),
122
+ );
123
+ emittedThreadIds.add(ownedThread.commentId);
124
+ continue;
125
+ }
126
+
127
+ if (serializedEntryByOoxmlId.has(definition.commentId)) {
128
+ continue;
129
+ }
130
+
131
+ mergedComments.push(definition.rawXml);
132
+ }
133
+
134
+ for (const thread of serializableThreads) {
135
+ if (emittedThreadIds.has(thread.commentId)) {
136
+ continue;
137
+ }
138
+
139
+ mergedComments.push(
140
+ ...serializeThreadEntries(
141
+ serializableEntries.filter((entry) => entry.thread.commentId === thread.commentId),
142
+ ),
143
+ );
144
+ emittedThreadIds.add(thread.commentId);
145
+ }
146
+
147
+ const mergedExtendedEntries = [
148
+ ...serializePreservedCommentExtensions(
149
+ preservedDefinitions.filter((definition) => !serializedEntryByOoxmlId.has(definition.commentId)),
150
+ ),
151
+ ...serializableEntries
152
+ .map((entry) => serializeCommentExtension(entry))
153
+ .filter((xml): xml is string => Boolean(xml)),
154
+ ];
155
+ const mergedDurableIds = [
156
+ ...serializePreservedCommentIds(
157
+ preservedDefinitions.filter((definition) => !serializedEntryByOoxmlId.has(definition.commentId)),
158
+ ),
159
+ ...serializableEntries
160
+ .map((entry) => serializeCommentDurableId(entry))
161
+ .filter((xml): xml is string => Boolean(xml)),
162
+ ];
163
+ const peopleAuthors = new Set([
164
+ ...(options.peopleAuthors ?? []),
165
+ ...preservedDefinitions
166
+ .map((definition) => definition.authorId)
167
+ .filter((authorId): authorId is string => typeof authorId === "string" && authorId.length > 0),
168
+ ...serializableEntries
169
+ .map((entry) => entry.entry.authorId)
170
+ .filter((authorId): authorId is string => typeof authorId === "string" && authorId.length > 0),
171
+ ]);
172
+
173
+ return {
174
+ commentsXml: [
175
+ `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>`,
176
+ normalizeCommentsRootTag(options.sourceRootTag ?? DEFAULT_COMMENTS_ROOT_TAG),
177
+ ...mergedComments,
178
+ `</w:comments>`,
179
+ ].join("\n"),
180
+ serializedCommentIds: serializableThreads.map((thread) => thread.commentId),
181
+ commentsExtendedXml:
182
+ mergedExtendedEntries.length > 0
183
+ ? [
184
+ `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>`,
185
+ options.sourceExtendedRootTag ?? DEFAULT_COMMENTS_EXTENDED_ROOT_TAG,
186
+ ...mergedExtendedEntries,
187
+ `</w15:commentsEx>`,
188
+ ].join("\n")
189
+ : undefined,
190
+ commentsIdsXml:
191
+ mergedDurableIds.length > 0
192
+ ? [
193
+ `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>`,
194
+ options.sourceIdsRootTag ?? DEFAULT_COMMENTS_IDS_ROOT_TAG,
195
+ ...mergedDurableIds,
196
+ `</w16cid:commentsIds>`,
197
+ ].join("\n")
198
+ : undefined,
199
+ peopleXml:
200
+ peopleAuthors.size > 0
201
+ ? [
202
+ `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>`,
203
+ options.sourcePeopleRootTag ?? DEFAULT_PEOPLE_ROOT_TAG,
204
+ ...[...peopleAuthors]
205
+ .sort((left, right) => left.localeCompare(right))
206
+ .map(
207
+ (authorId) =>
208
+ `<w15:person w15:author="${escapeAttribute(authorId)}" />`,
209
+ ),
210
+ `</w15:people>`,
211
+ ].join("\n")
212
+ : undefined,
213
+ };
214
+ }
215
+
216
+ function normalizeCommentsRootTag(rootTag: string): string {
217
+ let normalized = rootTag;
218
+ if (!/\bxmlns:w14=/u.test(normalized)) {
219
+ normalized = normalized.replace(
220
+ />$/u,
221
+ ` xmlns:w14="http://schemas.microsoft.com/office/word/2010/wordml">`,
222
+ );
223
+ }
224
+ if (!/\bxmlns:mc=/u.test(normalized)) {
225
+ normalized = normalized.replace(
226
+ />$/u,
227
+ ` xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006">`,
228
+ );
229
+ }
230
+ if (!/\bmc:Ignorable=/u.test(normalized)) {
231
+ normalized = normalized.replace(/>$/u, ` mc:Ignorable="w14">`);
232
+ } else if (!/\bmc:Ignorable="[^"]*\bw14\b/u.test(normalized)) {
233
+ normalized = normalized.replace(
234
+ /\bmc:Ignorable="([^"]*)"/u,
235
+ (_match, value: string) =>
236
+ `mc:Ignorable="${`${value} w14`.trim().replace(/\s+/gu, " ")}"`,
237
+ );
238
+ }
239
+ return normalized;
240
+ }
241
+
242
+ export function serializeCommentAnchorsIntoDocumentXml(
243
+ documentXml: string,
244
+ threads: readonly CommentThread[],
245
+ paragraphs: readonly Pick<
246
+ RevisionParagraphBoundary,
247
+ "paragraphIndex" | "start" | "end" | "boundaries"
248
+ >[] = mapParagraphBoundaries(documentXml),
249
+ options: SerializeCommentAnchorsOptions = {},
250
+ ): SerializedCommentDocumentResult {
251
+ const insertions = new Map<number, string[]>();
252
+ const serializedCommentIds: string[] = [];
253
+ const skippedCommentIds: string[] = [];
254
+ const exportCommentIds =
255
+ options.exportCommentIds ?? createCommentExportIdMap(threads);
256
+
257
+ for (const thread of threads) {
258
+ const { anchor } = thread;
259
+ if (anchor.kind !== "range") {
260
+ skippedCommentIds.push(thread.commentId);
261
+ continue;
262
+ }
263
+
264
+ const paragraph = paragraphs.find(
265
+ (candidate) =>
266
+ anchor.range.from >= candidate.start &&
267
+ anchor.range.to <= candidate.end,
268
+ );
269
+
270
+ if (!paragraph) {
271
+ skippedCommentIds.push(thread.commentId);
272
+ continue;
273
+ }
274
+
275
+ const startIndex = paragraph.boundaries.get(anchor.range.from);
276
+ const endIndex = paragraph.boundaries.get(anchor.range.to);
277
+
278
+ if (startIndex === undefined || endIndex === undefined) {
279
+ skippedCommentIds.push(thread.commentId);
280
+ continue;
281
+ }
282
+
283
+ const exportCommentId = exportCommentIds.get(thread.commentId) ?? thread.commentId;
284
+ pushInsertion(
285
+ insertions,
286
+ startIndex,
287
+ `<w:commentRangeStart w:id="${escapeAttribute(exportCommentId)}"/>`,
288
+ );
289
+ pushInsertion(
290
+ insertions,
291
+ endIndex,
292
+ `<w:commentRangeEnd w:id="${escapeAttribute(exportCommentId)}"/>`,
293
+ );
294
+ pushInsertion(
295
+ insertions,
296
+ endIndex,
297
+ `<w:r><w:commentReference w:id="${escapeAttribute(exportCommentId)}"/></w:r>`,
298
+ );
299
+ serializedCommentIds.push(thread.commentId);
300
+ }
301
+
302
+ const sortedInsertions = [...insertions.entries()].sort((left, right) => left[0] - right[0]);
303
+ let cursor = 0;
304
+ let output = "";
305
+
306
+ for (const [index, snippets] of sortedInsertions) {
307
+ output += documentXml.slice(cursor, index);
308
+ output += snippets.join("");
309
+ cursor = index;
310
+ }
311
+
312
+ output += documentXml.slice(cursor);
313
+
314
+ return {
315
+ documentXml: output,
316
+ serializedCommentIds,
317
+ skippedCommentIds,
318
+ };
319
+ }
320
+
321
+ export function createCommentExportIdMap(
322
+ threads: readonly CommentThread[],
323
+ preservedDefinitions: readonly ImportedCommentDefinition[] = [],
324
+ ): ReadonlyMap<string, string> {
325
+ const exportIds = new Map<string, string>();
326
+ const reservedNumericIds = new Set<number>();
327
+
328
+ for (const definition of preservedDefinitions) {
329
+ const numericId = parseOoxmlNumericId(definition.commentId);
330
+ if (numericId !== undefined) {
331
+ reservedNumericIds.add(numericId);
332
+ }
333
+ }
334
+
335
+ for (const thread of threads) {
336
+ const rootEntry = getRootEntry(thread);
337
+ const preferredValue =
338
+ thread.metadata?.rootOoxmlCommentId ??
339
+ rootEntry?.metadata?.ooxmlCommentId ??
340
+ thread.commentId;
341
+ const numericId = parseOoxmlNumericId(preferredValue);
342
+ if (numericId !== undefined && !reservedNumericIds.has(numericId)) {
343
+ exportIds.set(thread.commentId, String(numericId));
344
+ reservedNumericIds.add(numericId);
345
+ }
346
+ }
347
+
348
+ let nextGeneratedId =
349
+ reservedNumericIds.size > 0 ? Math.max(...reservedNumericIds) + 1 : 1;
350
+
351
+ for (const thread of threads) {
352
+ if (exportIds.has(thread.commentId)) {
353
+ continue;
354
+ }
355
+
356
+ while (reservedNumericIds.has(nextGeneratedId)) {
357
+ nextGeneratedId += 1;
358
+ }
359
+
360
+ exportIds.set(thread.commentId, String(nextGeneratedId));
361
+ reservedNumericIds.add(nextGeneratedId);
362
+ nextGeneratedId += 1;
363
+ }
364
+
365
+ return exportIds;
366
+ }
367
+
368
+ function createSerializableEntries(
369
+ threads: readonly CommentThread[],
370
+ preservedDefinitions: readonly ImportedCommentDefinition[],
371
+ exportCommentIds: ReadonlyMap<string, string>,
372
+ ): SerializableCommentEntry[] {
373
+ const entries: SerializableCommentEntry[] = [];
374
+ const reservedNumericIds = new Set<number>();
375
+
376
+ for (const definition of preservedDefinitions) {
377
+ const numericId = parseOoxmlNumericId(definition.commentId);
378
+ if (numericId !== undefined) {
379
+ reservedNumericIds.add(numericId);
380
+ }
381
+ }
382
+
383
+ for (const thread of threads) {
384
+ const rootEntry = getRootEntry(thread);
385
+ if (!rootEntry) {
386
+ continue;
387
+ }
388
+
389
+ const rootExportCommentId = exportCommentIds.get(thread.commentId) ?? thread.commentId;
390
+ const rootNumericId = parseOoxmlNumericId(rootExportCommentId);
391
+ if (rootNumericId !== undefined) {
392
+ reservedNumericIds.add(rootNumericId);
393
+ }
394
+
395
+ const rootParaId =
396
+ rootEntry.metadata?.paraId ??
397
+ thread.metadata?.rootParaId ??
398
+ generateParaId(thread.commentId, 0);
399
+ const rootDurableId =
400
+ rootEntry.metadata?.durableId ??
401
+ createDurableId(rootEntry, rootExportCommentId, 0);
402
+
403
+ entries.push({
404
+ thread,
405
+ entry: rootEntry,
406
+ exportCommentId: rootExportCommentId,
407
+ paraId: rootParaId,
408
+ durableId: rootDurableId,
409
+ isRoot: true,
410
+ });
411
+
412
+ for (let index = 1; index < thread.entries.length; index += 1) {
413
+ const entry = thread.entries[index];
414
+ if (!entry) {
415
+ continue;
416
+ }
417
+
418
+ const preferredCommentId = entry.metadata?.ooxmlCommentId;
419
+ let exportCommentId: string | undefined;
420
+ const preferredNumericId = preferredCommentId
421
+ ? parseOoxmlNumericId(preferredCommentId)
422
+ : undefined;
423
+ if (
424
+ preferredNumericId !== undefined &&
425
+ !reservedNumericIds.has(preferredNumericId)
426
+ ) {
427
+ exportCommentId = String(preferredNumericId);
428
+ reservedNumericIds.add(preferredNumericId);
429
+ } else {
430
+ let nextGeneratedId =
431
+ reservedNumericIds.size > 0 ? Math.max(...reservedNumericIds) + 1 : 1;
432
+ while (reservedNumericIds.has(nextGeneratedId)) {
433
+ nextGeneratedId += 1;
434
+ }
435
+ exportCommentId = String(nextGeneratedId);
436
+ reservedNumericIds.add(nextGeneratedId);
437
+ }
438
+
439
+ entries.push({
440
+ thread,
441
+ entry,
442
+ exportCommentId,
443
+ paraId: entry.metadata?.paraId ?? generateParaId(thread.commentId, index),
444
+ durableId:
445
+ entry.metadata?.durableId ??
446
+ createDurableId(entry, exportCommentId, index),
447
+ isRoot: false,
448
+ });
449
+ }
450
+ }
451
+
452
+ return entries;
453
+ }
454
+
455
+ function serializeThreadEntries(entries: readonly SerializableCommentEntry[]): string[] {
456
+ return entries.map((entry) => serializeCommentEntry(entry));
457
+ }
458
+
459
+ function serializeCommentEntry(entry: SerializableCommentEntry): string {
460
+ const author = escapeAttribute(entry.entry.authorId);
461
+ const createdAt = escapeAttribute(entry.entry.createdAt);
462
+ const initials = entry.entry.metadata?.initials;
463
+ const paragraphXml = serializeCommentParagraphs(entry.entry.body, entry.paraId);
464
+
465
+ return `<w:comment w:id="${escapeAttribute(entry.exportCommentId)}"${initials ? ` w:initials="${escapeAttribute(initials)}"` : ""} w:author="${author}" w:date="${createdAt}">${paragraphXml}</w:comment>`;
466
+ }
467
+
468
+ function serializeCommentParagraphs(body: string, paraId: string): string {
469
+ const paragraphs = body.length > 0 ? body.split("\n") : [""];
470
+ const textId = deriveTextIdFromParaId(paraId);
471
+ return paragraphs
472
+ .map((paragraph, index) => {
473
+ const attributes =
474
+ index === 0
475
+ ? ` w14:paraId="${escapeAttribute(paraId)}" w14:textId="${escapeAttribute(textId)}"`
476
+ : "";
477
+ return `<w:p${attributes}><w:r>${serializeText(paragraph)}</w:r></w:p>`;
478
+ })
479
+ .join("");
480
+ }
481
+
482
+ function serializePreservedCommentExtensions(
483
+ definitions: readonly ImportedCommentDefinition[],
484
+ ): string[] {
485
+ return definitions
486
+ .map((definition) =>
487
+ definition.paraId
488
+ ? serializePreservedCommentExtension(definition)
489
+ : undefined,
490
+ )
491
+ .filter((xml): xml is string => Boolean(xml));
492
+ }
493
+
494
+ function serializePreservedCommentExtension(
495
+ definition: ImportedCommentDefinition,
496
+ ): string {
497
+ const doneValue = definition.isDone ? "true" : "false";
498
+ return `<w15:commentEx w15:paraId="${escapeAttribute(definition.paraId!)}"${definition.parentParaId ? ` w15:paraIdParent="${escapeAttribute(definition.parentParaId)}"` : ""} w15:done="${doneValue}" />`;
499
+ }
500
+
501
+ function serializeCommentExtension(
502
+ entry: SerializableCommentEntry,
503
+ ): string | undefined {
504
+ return `<w15:commentEx w15:paraId="${escapeAttribute(entry.paraId)}"${entry.isRoot ? "" : ` w15:paraIdParent="${escapeAttribute(findRootParaId(entry.thread, entry.paraId))}"`} w15:done="${entry.isRoot && entry.thread.status === "resolved" ? "true" : "false"}" />`;
505
+ }
506
+
507
+ function serializePreservedCommentIds(
508
+ definitions: readonly ImportedCommentDefinition[],
509
+ ): string[] {
510
+ return definitions
511
+ .map((definition) =>
512
+ definition.paraId && definition.durableId
513
+ ? `<w16cid:commentId w16cid:paraId="${escapeAttribute(definition.paraId)}" w16cid:durableId="${escapeAttribute(definition.durableId)}" />`
514
+ : undefined,
515
+ )
516
+ .filter((xml): xml is string => Boolean(xml));
517
+ }
518
+
519
+ function serializeCommentDurableId(
520
+ entry: SerializableCommentEntry,
521
+ ): string | undefined {
522
+ if (!entry.durableId) {
523
+ return undefined;
524
+ }
525
+
526
+ return `<w16cid:commentId w16cid:paraId="${escapeAttribute(entry.paraId)}" w16cid:durableId="${escapeAttribute(entry.durableId)}" />`;
527
+ }
528
+
529
+ function getRootEntry(thread: CommentThread): CommentEntry | undefined {
530
+ return thread.entries[0];
531
+ }
532
+
533
+ function findRootParaId(thread: CommentThread, fallback: string): string {
534
+ return (
535
+ thread.entries[0]?.metadata?.paraId ??
536
+ thread.metadata?.rootParaId ??
537
+ fallback
538
+ );
539
+ }
540
+
541
+ function generateParaId(seed: string, index: number): string {
542
+ const normalized = `${seed}:${index}`;
543
+ let hash = 0;
544
+
545
+ for (let cursor = 0; cursor < normalized.length; cursor += 1) {
546
+ hash = (hash * 31 + normalized.charCodeAt(cursor)) >>> 0;
547
+ }
548
+
549
+ return hash.toString(16).toUpperCase().padStart(8, "0").slice(-8);
550
+ }
551
+
552
+ function createDurableId(
553
+ entry: CommentEntry,
554
+ exportCommentId: string,
555
+ index: number,
556
+ ): string {
557
+ const seed = `${entry.entryId}:${exportCommentId}:${index}`;
558
+ let hash = 0n;
559
+
560
+ for (let cursor = 0; cursor < seed.length; cursor += 1) {
561
+ hash = (hash * 131n + BigInt(seed.charCodeAt(cursor))) & 0xffffffffffffffffn;
562
+ }
563
+
564
+ return hash.toString(16).toUpperCase().padStart(16, "0").slice(-16);
565
+ }
566
+
567
+ function deriveTextIdFromParaId(paraId: string): string {
568
+ return paraId.slice(-8).padStart(8, "0");
569
+ }
570
+
571
+ function parseOoxmlNumericId(value: string): number | undefined {
572
+ if (!/^-?\d+$/u.test(value)) {
573
+ return undefined;
574
+ }
575
+
576
+ const numericId = Number.parseInt(value, 10);
577
+ return Number.isFinite(numericId) ? numericId : undefined;
578
+ }
579
+
580
+ export function mapParagraphBoundaries(documentXml: string): ParagraphBoundaryMap[] {
581
+ const root = parseXml(documentXml);
582
+ const documentElement = findChildElement(root, "document");
583
+ const bodyElement = findChildElement(documentElement, "body");
584
+ const paragraphs: ParagraphBoundaryMap[] = [];
585
+ walkBlockNodesForParagraphBoundaries(
586
+ bodyElement.children,
587
+ documentXml,
588
+ paragraphs,
589
+ 0,
590
+ -1,
591
+ true,
592
+ );
593
+
594
+ return paragraphs;
595
+ }
596
+
597
+ function walkBlockNodesForParagraphBoundaries(
598
+ nodes: readonly XmlNode[],
599
+ documentXml: string,
600
+ paragraphs: ParagraphBoundaryMap[],
601
+ globalCursor: number,
602
+ paragraphIndex: number,
603
+ useSurfaceParagraphSeparators: boolean,
604
+ ): {
605
+ globalCursor: number;
606
+ paragraphIndex: number;
607
+ } {
608
+ let nextCursor = globalCursor;
609
+ let nextParagraphIndex = paragraphIndex;
610
+ let elementIndex = -1;
611
+
612
+ for (const node of nodes) {
613
+ if (node.type !== "element") {
614
+ continue;
615
+ }
616
+ elementIndex += 1;
617
+
618
+ const name = localName(node.name);
619
+ if (name === "p") {
620
+ if (useSurfaceParagraphSeparators && elementIndex > 0) {
621
+ nextCursor += 1;
622
+ }
623
+ nextParagraphIndex += 1;
624
+ const boundaries = new Map<number, number>();
625
+ boundaries.set(nextCursor, node.start + openingTagLength(documentXml, node.start));
626
+
627
+ walkParagraphForBoundaries(
628
+ node,
629
+ documentXml,
630
+ boundaries,
631
+ () => nextCursor,
632
+ (next) => {
633
+ nextCursor = next;
634
+ },
635
+ );
636
+
637
+ if (!boundaries.has(nextCursor)) {
638
+ boundaries.set(nextCursor, node.end - 4);
639
+ }
640
+ paragraphs.push({
641
+ paragraphIndex: nextParagraphIndex,
642
+ start: Math.min(...boundaries.keys()),
643
+ end: Math.max(...boundaries.keys()),
644
+ boundaries,
645
+ });
646
+ continue;
647
+ }
648
+
649
+ if (name === "tbl") {
650
+ for (const child of node.children) {
651
+ if (child.type !== "element" || localName(child.name) !== "tr") {
652
+ continue;
653
+ }
654
+ for (const rowChild of child.children) {
655
+ if (rowChild.type !== "element" || localName(rowChild.name) !== "tc") {
656
+ continue;
657
+ }
658
+ const result = walkBlockNodesForParagraphBoundaries(
659
+ rowChild.children,
660
+ documentXml,
661
+ paragraphs,
662
+ nextCursor,
663
+ nextParagraphIndex,
664
+ false,
665
+ );
666
+ nextCursor = result.globalCursor;
667
+ nextParagraphIndex = result.paragraphIndex;
668
+ }
669
+ }
670
+ continue;
671
+ }
672
+
673
+ if (name === "sdt") {
674
+ const sdtContent = findChildElement(node, "sdtContent");
675
+ const result = walkBlockNodesForParagraphBoundaries(
676
+ sdtContent.children,
677
+ documentXml,
678
+ paragraphs,
679
+ nextCursor,
680
+ nextParagraphIndex,
681
+ false,
682
+ );
683
+ nextCursor = result.globalCursor;
684
+ nextParagraphIndex = result.paragraphIndex;
685
+ continue;
686
+ }
687
+
688
+ if (name === "customXml") {
689
+ nextCursor += 1;
690
+ continue;
691
+ }
692
+
693
+ nextCursor += 0;
694
+ }
695
+
696
+ return {
697
+ globalCursor: nextCursor,
698
+ paragraphIndex: nextParagraphIndex,
699
+ };
700
+ }
701
+
702
+ function walkParagraphForBoundaries(
703
+ paragraph: XmlElementNode,
704
+ sourceXml: string,
705
+ boundaries: Map<number, number>,
706
+ getCursor: () => number,
707
+ setCursor: (next: number) => void,
708
+ ): void {
709
+ for (const child of paragraph.children) {
710
+ walkInlineNodeForBoundaries(child, sourceXml, boundaries, getCursor, setCursor);
711
+ }
712
+ }
713
+
714
+ function walkInlineNodeForBoundaries(
715
+ node: XmlNode,
716
+ sourceXml: string,
717
+ boundaries: Map<number, number>,
718
+ getCursor: () => number,
719
+ setCursor: (next: number) => void,
720
+ ): void {
721
+ if (node.type !== "element") {
722
+ return;
723
+ }
724
+
725
+ switch (localName(node.name)) {
726
+ case "r": {
727
+ if (!boundaries.has(getCursor())) {
728
+ boundaries.set(getCursor(), node.start);
729
+ }
730
+
731
+ for (const child of node.children) {
732
+ walkInlineNodeForBoundaries(child, sourceXml, boundaries, getCursor, setCursor);
733
+ }
734
+
735
+ boundaries.set(getCursor(), node.end);
736
+ return;
737
+ }
738
+ case "t": {
739
+ const text = node.children
740
+ .filter((child): child is XmlTextNode => child.type === "text")
741
+ .map((child) => child.text)
742
+ .join("");
743
+ setCursor(getCursor() + Array.from(text).length);
744
+ return;
745
+ }
746
+ case "tab":
747
+ case "br": {
748
+ const startCursor = getCursor();
749
+ boundaries.set(startCursor, node.start);
750
+ const nextCursor = startCursor + 1;
751
+ boundaries.set(nextCursor, node.end);
752
+ setCursor(nextCursor);
753
+ return;
754
+ }
755
+ default:
756
+ for (const child of node.children) {
757
+ walkInlineNodeForBoundaries(child, sourceXml, boundaries, getCursor, setCursor);
758
+ }
759
+ }
760
+ }
761
+
762
+ function pushInsertion(
763
+ insertions: Map<number, string[]>,
764
+ index: number,
765
+ xml: string,
766
+ ): void {
767
+ const bucket = insertions.get(index);
768
+ if (bucket) {
769
+ bucket.push(xml);
770
+ return;
771
+ }
772
+
773
+ insertions.set(index, [xml]);
774
+ }
775
+
776
+ function serializeText(text: string): string {
777
+ const preserve = requiresPreservedSpace(text) ? ` xml:space="preserve"` : "";
778
+ return `<w:t${preserve}>${escapeXml(text)}</w:t>`;
779
+ }
780
+
781
+ function requiresPreservedSpace(text: string): boolean {
782
+ return /^\s/u.test(text) || /\s$/u.test(text) || /\s{2,}/u.test(text);
783
+ }
784
+
785
+ function escapeXml(value: string): string {
786
+ return value
787
+ .replace(/&/g, "&amp;")
788
+ .replace(/</g, "&lt;")
789
+ .replace(/>/g, "&gt;");
790
+ }
791
+
792
+ function escapeAttribute(value: string): string {
793
+ return escapeXml(value).replace(/"/g, "&quot;");
794
+ }
795
+
796
+ function openingTagLength(xml: string, start: number): number {
797
+ const end = xml.indexOf(">", start);
798
+ if (end < 0) {
799
+ throw new Error("Malformed XML: missing tag close.");
800
+ }
801
+
802
+ return end - start + 1;
803
+ }
804
+
805
+ function parseXml(xml: string): XmlElementNode {
806
+ const root: XmlElementNode = {
807
+ type: "element",
808
+ name: "#document",
809
+ attributes: {},
810
+ children: [],
811
+ start: 0,
812
+ end: xml.length,
813
+ };
814
+ const stack: XmlElementNode[] = [root];
815
+ const tokenPattern =
816
+ /<!--[\s\S]*?-->|<\?[\s\S]*?\?>|<!DOCTYPE[\s\S]*?>|<!\[CDATA\[[\s\S]*?\]\]>|<[^>]+>|[^<]+/gu;
817
+
818
+ for (const match of xml.matchAll(tokenPattern)) {
819
+ const token = match[0] ?? "";
820
+ const start = match.index ?? 0;
821
+ const end = start + token.length;
822
+
823
+ if (token.startsWith("<?") || token.startsWith("<!DOCTYPE") || token.startsWith("<!--")) {
824
+ continue;
825
+ }
826
+
827
+ if (token.startsWith("<![CDATA[")) {
828
+ const text = token.slice(9, -3);
829
+ stack[stack.length - 1]?.children.push({
830
+ type: "text",
831
+ text,
832
+ start,
833
+ end,
834
+ });
835
+ continue;
836
+ }
837
+
838
+ if (token.startsWith("</")) {
839
+ const node = stack.pop();
840
+ if (!node) {
841
+ throw new Error("Malformed XML: unexpected closing tag.");
842
+ }
843
+ node.end = end;
844
+ continue;
845
+ }
846
+
847
+ if (token.startsWith("<")) {
848
+ const selfClosing = /\/>$/.test(token);
849
+ const tagBody = token.slice(1, token.length - (selfClosing ? 2 : 1)).trim();
850
+ const { name, attributes } = parseTag(tagBody);
851
+ const node: XmlElementNode = {
852
+ type: "element",
853
+ name,
854
+ attributes,
855
+ children: [],
856
+ start,
857
+ end,
858
+ };
859
+ stack[stack.length - 1]?.children.push(node);
860
+ if (!selfClosing) {
861
+ stack.push(node);
862
+ }
863
+ continue;
864
+ }
865
+
866
+ const text = decodeXmlText(token);
867
+ if (text.length > 0) {
868
+ stack[stack.length - 1]?.children.push({
869
+ type: "text",
870
+ text,
871
+ start,
872
+ end,
873
+ });
874
+ }
875
+ }
876
+
877
+ if (stack.length !== 1) {
878
+ throw new Error("Malformed XML: unclosed tag.");
879
+ }
880
+
881
+ return root;
882
+ }
883
+
884
+ function parseTag(tagBody: string): { name: string; attributes: Record<string, string> } {
885
+ const whitespaceIndex = tagBody.search(/\s/u);
886
+ const name = whitespaceIndex === -1 ? tagBody : tagBody.slice(0, whitespaceIndex);
887
+ const rawAttributes = whitespaceIndex === -1 ? "" : tagBody.slice(whitespaceIndex + 1);
888
+ const attributes: Record<string, string> = {};
889
+ const pattern = /([A-Za-z_][A-Za-z0-9:._-]*)\s*=\s*("([^"]*)"|'([^']*)')/gu;
890
+
891
+ for (const match of rawAttributes.matchAll(pattern)) {
892
+ const key = match[1];
893
+ const value = match[3] ?? match[4] ?? "";
894
+ if (key) {
895
+ attributes[key] = decodeXmlText(value);
896
+ }
897
+ }
898
+
899
+ return { name, attributes };
900
+ }
901
+
902
+ function findChildElement(node: XmlElementNode, name: string): XmlElementNode {
903
+ const match = node.children.find(
904
+ (child): child is XmlElementNode =>
905
+ child.type === "element" && localName(child.name) === name,
906
+ );
907
+
908
+ if (!match) {
909
+ throw new Error(`Expected XML element ${name}.`);
910
+ }
911
+
912
+ return match;
913
+ }
914
+
915
+ function localName(name: string): string {
916
+ const index = name.indexOf(":");
917
+ return index === -1 ? name : name.slice(index + 1);
918
+ }
919
+
920
+ function decodeXmlText(text: string): string {
921
+ return text.replace(
922
+ /&(?:#x([0-9A-Fa-f]+)|#([0-9]+)|([A-Za-z]+));/gu,
923
+ (_, hex, dec, named) => {
924
+ if (hex) {
925
+ return String.fromCodePoint(Number.parseInt(hex, 16));
926
+ }
927
+ if (dec) {
928
+ return String.fromCodePoint(Number.parseInt(dec, 10));
929
+ }
930
+
931
+ switch (named) {
932
+ case "amp":
933
+ return "&";
934
+ case "lt":
935
+ return "<";
936
+ case "gt":
937
+ return ">";
938
+ case "quot":
939
+ return "\"";
940
+ case "apos":
941
+ return "'";
942
+ default:
943
+ return `&${named};`;
944
+ }
945
+ },
946
+ );
947
+ }