@beyondwork/docx-react-component 1.0.29 → 1.0.30

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (381) 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/review-queue-bar.tsx +97 -0
  180. package/src/ui-tailwind/chrome/tw-alert-banner.tsx +64 -0
  181. package/src/ui-tailwind/chrome/tw-context-analytics-summary.tsx +122 -0
  182. package/src/ui-tailwind/chrome/tw-image-context-toolbar.tsx +121 -0
  183. package/src/ui-tailwind/chrome/tw-layout-panel.tsx +114 -0
  184. package/src/ui-tailwind/chrome/tw-object-context-toolbar.tsx +30 -0
  185. package/src/ui-tailwind/chrome/tw-page-ruler.tsx +365 -0
  186. package/src/ui-tailwind/chrome/tw-selection-tool-blocked.tsx +23 -0
  187. package/src/ui-tailwind/chrome/tw-selection-tool-comment.tsx +35 -0
  188. package/src/ui-tailwind/chrome/tw-selection-tool-formatting.tsx +37 -0
  189. package/src/ui-tailwind/chrome/tw-selection-tool-host.tsx +298 -0
  190. package/src/ui-tailwind/chrome/tw-selection-tool-structure.tsx +116 -0
  191. package/src/ui-tailwind/chrome/tw-selection-tool-suggestion.tsx +29 -0
  192. package/src/ui-tailwind/chrome/tw-selection-tool-workflow.tsx +27 -0
  193. package/src/ui-tailwind/chrome/tw-selection-toolbar.tsx +186 -0
  194. package/src/ui-tailwind/chrome/tw-suggestion-card.tsx +139 -0
  195. package/src/ui-tailwind/chrome/tw-table-context-toolbar.tsx +200 -0
  196. package/src/ui-tailwind/chrome/tw-unsaved-modal.tsx +58 -0
  197. package/src/ui-tailwind/chrome/use-before-unload.ts +20 -0
  198. package/src/ui-tailwind/editor-surface/perf-probe.ts +179 -0
  199. package/src/ui-tailwind/editor-surface/pm-command-bridge.ts +189 -0
  200. package/src/ui-tailwind/editor-surface/pm-contextual-ui.ts +31 -0
  201. package/src/ui-tailwind/editor-surface/pm-decorations.ts +411 -0
  202. package/src/ui-tailwind/editor-surface/pm-position-map.ts +123 -0
  203. package/src/ui-tailwind/editor-surface/pm-schema.ts +927 -0
  204. package/src/ui-tailwind/editor-surface/pm-state-from-snapshot.ts +567 -0
  205. package/src/ui-tailwind/editor-surface/search-plugin.ts +168 -0
  206. package/src/ui-tailwind/editor-surface/surface-build-keys.ts +65 -0
  207. package/src/ui-tailwind/editor-surface/tw-caret.tsx +12 -0
  208. package/src/ui-tailwind/editor-surface/tw-editor-surface.tsx +150 -0
  209. package/src/ui-tailwind/editor-surface/tw-inline-token.tsx +129 -0
  210. package/src/ui-tailwind/editor-surface/tw-opaque-block.tsx +58 -0
  211. package/src/ui-tailwind/editor-surface/tw-paragraph-block.tsx +151 -0
  212. package/src/ui-tailwind/editor-surface/tw-prosemirror-surface.tsx +1047 -0
  213. package/src/ui-tailwind/editor-surface/tw-segment-view.tsx +111 -0
  214. package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +503 -0
  215. package/src/ui-tailwind/index.ts +62 -0
  216. package/src/ui-tailwind/page-chrome-model.ts +27 -0
  217. package/src/ui-tailwind/review/tw-comment-sidebar.tsx +406 -0
  218. package/src/ui-tailwind/review/tw-health-panel.tsx +149 -0
  219. package/src/ui-tailwind/review/tw-review-rail.tsx +122 -0
  220. package/src/ui-tailwind/review/tw-revision-sidebar.tsx +164 -0
  221. package/src/ui-tailwind/status/tw-status-bar.tsx +65 -0
  222. package/{dist → src}/ui-tailwind/theme/editor-theme.css +58 -40
  223. package/src/ui-tailwind/toolbar/tw-toolbar-icon-button.tsx +52 -0
  224. package/src/ui-tailwind/toolbar/tw-toolbar.tsx +1133 -0
  225. package/src/ui-tailwind/tw-review-workspace.tsx +1460 -0
  226. package/src/validation/README.md +3 -0
  227. package/src/validation/compatibility-engine.ts +878 -0
  228. package/src/validation/compatibility-report.ts +161 -0
  229. package/src/validation/diagnostics.ts +204 -0
  230. package/src/validation/docx-comment-proof.ts +720 -0
  231. package/src/validation/import-diagnostics.ts +128 -0
  232. package/src/validation/low-priority-word-surfaces.ts +373 -0
  233. package/dist/canonical-document-BLEbzL2J.d.cts +0 -844
  234. package/dist/canonical-document-BLEbzL2J.d.ts +0 -844
  235. package/dist/chunk-2FJS5GZM.js +0 -763
  236. package/dist/chunk-2FJS5GZM.js.map +0 -1
  237. package/dist/chunk-2OQBZS3F.js +0 -446
  238. package/dist/chunk-2OQBZS3F.js.map +0 -1
  239. package/dist/chunk-2S7W4KFO.js +0 -127
  240. package/dist/chunk-2S7W4KFO.js.map +0 -1
  241. package/dist/chunk-2TG72QSW.js +0 -3874
  242. package/dist/chunk-2TG72QSW.js.map +0 -1
  243. package/dist/chunk-36QNIZBO.js +0 -532
  244. package/dist/chunk-36QNIZBO.js.map +0 -1
  245. package/dist/chunk-4AQOYAW4.js +0 -3069
  246. package/dist/chunk-4AQOYAW4.js.map +0 -1
  247. package/dist/chunk-4D5EWJ3P.js +0 -77
  248. package/dist/chunk-4D5EWJ3P.js.map +0 -1
  249. package/dist/chunk-5FN54NDH.js +0 -2257
  250. package/dist/chunk-5FN54NDH.js.map +0 -1
  251. package/dist/chunk-BOYGQYRQ.js +0 -7306
  252. package/dist/chunk-BOYGQYRQ.js.map +0 -1
  253. package/dist/chunk-CN3XMECL.js +0 -212
  254. package/dist/chunk-CN3XMECL.js.map +0 -1
  255. package/dist/chunk-EBI3BX6U.js +0 -164
  256. package/dist/chunk-EBI3BX6U.js.map +0 -1
  257. package/dist/chunk-EILUG3VB.js +0 -1275
  258. package/dist/chunk-EILUG3VB.js.map +0 -1
  259. package/dist/chunk-FUDY333O.js +0 -70
  260. package/dist/chunk-FUDY333O.js.map +0 -1
  261. package/dist/chunk-GBVOWFIK.js +0 -1237
  262. package/dist/chunk-GBVOWFIK.js.map +0 -1
  263. package/dist/chunk-H4TQ3H3Y.js +0 -262
  264. package/dist/chunk-H4TQ3H3Y.js.map +0 -1
  265. package/dist/chunk-JGB3IXZO.js +0 -189
  266. package/dist/chunk-JGB3IXZO.js.map +0 -1
  267. package/dist/chunk-KD2QRQPY.js +0 -4342
  268. package/dist/chunk-KD2QRQPY.js.map +0 -1
  269. package/dist/chunk-KLMXQVYK.js +0 -369
  270. package/dist/chunk-KLMXQVYK.js.map +0 -1
  271. package/dist/chunk-KZUG5KFQ.js +0 -214
  272. package/dist/chunk-KZUG5KFQ.js.map +0 -1
  273. package/dist/chunk-QDAQ4CJU.js +0 -345
  274. package/dist/chunk-QDAQ4CJU.js.map +0 -1
  275. package/dist/chunk-RMH72RZI.js.map +0 -1
  276. package/dist/chunk-SWKWQZXM.js +0 -117
  277. package/dist/chunk-SWKWQZXM.js.map +0 -1
  278. package/dist/chunk-TJBP2K4T.js.map +0 -1
  279. package/dist/chunk-TLCEAQDQ.js +0 -542
  280. package/dist/chunk-TLCEAQDQ.js.map +0 -1
  281. package/dist/chunk-UZXBISGO.js.map +0 -1
  282. package/dist/chunk-WGBAKP3Q.js +0 -3220
  283. package/dist/chunk-WGBAKP3Q.js.map +0 -1
  284. package/dist/compare/index.cjs +0 -5475
  285. package/dist/compare/index.cjs.map +0 -1
  286. package/dist/compare/index.d.cts +0 -114
  287. package/dist/compare/index.d.ts +0 -114
  288. package/dist/compare/index.js +0 -731
  289. package/dist/compare/index.js.map +0 -1
  290. package/dist/core/commands/formatting-commands.cjs +0 -828
  291. package/dist/core/commands/formatting-commands.cjs.map +0 -1
  292. package/dist/core/commands/formatting-commands.d.cts +0 -63
  293. package/dist/core/commands/formatting-commands.d.ts +0 -63
  294. package/dist/core/commands/formatting-commands.js +0 -37
  295. package/dist/core/commands/formatting-commands.js.map +0 -1
  296. package/dist/core/commands/image-commands.cjs +0 -2023
  297. package/dist/core/commands/image-commands.cjs.map +0 -1
  298. package/dist/core/commands/image-commands.d.cts +0 -58
  299. package/dist/core/commands/image-commands.d.ts +0 -58
  300. package/dist/core/commands/image-commands.js +0 -18
  301. package/dist/core/commands/image-commands.js.map +0 -1
  302. package/dist/core/commands/section-layout-commands.cjs.map +0 -1
  303. package/dist/core/commands/section-layout-commands.d.cts +0 -62
  304. package/dist/core/commands/section-layout-commands.d.ts +0 -62
  305. package/dist/core/commands/section-layout-commands.js +0 -21
  306. package/dist/core/commands/section-layout-commands.js.map +0 -1
  307. package/dist/core/commands/style-commands.cjs.map +0 -1
  308. package/dist/core/commands/style-commands.d.cts +0 -13
  309. package/dist/core/commands/style-commands.d.ts +0 -13
  310. package/dist/core/commands/style-commands.js +0 -9
  311. package/dist/core/commands/style-commands.js.map +0 -1
  312. package/dist/core/commands/table-structure-commands.cjs +0 -1883
  313. package/dist/core/commands/table-structure-commands.cjs.map +0 -1
  314. package/dist/core/commands/table-structure-commands.d.cts +0 -59
  315. package/dist/core/commands/table-structure-commands.d.ts +0 -59
  316. package/dist/core/commands/table-structure-commands.js +0 -12
  317. package/dist/core/commands/table-structure-commands.js.map +0 -1
  318. package/dist/core/commands/text-commands.cjs +0 -2391
  319. package/dist/core/commands/text-commands.cjs.map +0 -1
  320. package/dist/core/commands/text-commands.d.cts +0 -24
  321. package/dist/core/commands/text-commands.d.ts +0 -24
  322. package/dist/core/commands/text-commands.js +0 -28
  323. package/dist/core/commands/text-commands.js.map +0 -1
  324. package/dist/core/selection/mapping.cjs +0 -200
  325. package/dist/core/selection/mapping.cjs.map +0 -1
  326. package/dist/core/selection/mapping.d.cts +0 -2
  327. package/dist/core/selection/mapping.d.ts +0 -2
  328. package/dist/core/selection/mapping.js +0 -31
  329. package/dist/core/selection/mapping.js.map +0 -1
  330. package/dist/core/state/editor-state.cjs +0 -2278
  331. package/dist/core/state/editor-state.cjs.map +0 -1
  332. package/dist/core/state/editor-state.d.cts +0 -2
  333. package/dist/core/state/editor-state.d.ts +0 -2
  334. package/dist/core/state/editor-state.js +0 -26
  335. package/dist/core/state/editor-state.js.map +0 -1
  336. package/dist/index.cjs +0 -38553
  337. package/dist/index.cjs.map +0 -1
  338. package/dist/index.d.cts +0 -15
  339. package/dist/index.d.ts +0 -15
  340. package/dist/index.js +0 -7856
  341. package/dist/index.js.map +0 -1
  342. package/dist/io/docx-session.cjs +0 -16236
  343. package/dist/io/docx-session.cjs.map +0 -1
  344. package/dist/io/docx-session.d.cts +0 -21
  345. package/dist/io/docx-session.d.ts +0 -21
  346. package/dist/io/docx-session.js +0 -18
  347. package/dist/io/docx-session.js.map +0 -1
  348. package/dist/legal/index.cjs +0 -3900
  349. package/dist/legal/index.cjs.map +0 -1
  350. package/dist/legal/index.d.cts +0 -86
  351. package/dist/legal/index.d.ts +0 -86
  352. package/dist/legal/index.js +0 -616
  353. package/dist/legal/index.js.map +0 -1
  354. package/dist/public-types-7ZL_94cz.d.ts +0 -1573
  355. package/dist/public-types-CeMaDueh.d.cts +0 -1573
  356. package/dist/public-types.cjs +0 -19
  357. package/dist/public-types.cjs.map +0 -1
  358. package/dist/public-types.d.cts +0 -2
  359. package/dist/public-types.d.ts +0 -2
  360. package/dist/public-types.js +0 -1
  361. package/dist/public-types.js.map +0 -1
  362. package/dist/runtime/document-runtime.cjs +0 -11140
  363. package/dist/runtime/document-runtime.cjs.map +0 -1
  364. package/dist/runtime/document-runtime.d.cts +0 -231
  365. package/dist/runtime/document-runtime.d.ts +0 -231
  366. package/dist/runtime/document-runtime.js +0 -21
  367. package/dist/runtime/document-runtime.js.map +0 -1
  368. package/dist/structural-helpers-CilgOVhh.d.cts +0 -10
  369. package/dist/structural-helpers-q0Gd-eBN.d.ts +0 -10
  370. package/dist/ui-tailwind/editor-surface/search-plugin.cjs +0 -313
  371. package/dist/ui-tailwind/editor-surface/search-plugin.cjs.map +0 -1
  372. package/dist/ui-tailwind/editor-surface/search-plugin.d.cts +0 -67
  373. package/dist/ui-tailwind/editor-surface/search-plugin.d.ts +0 -67
  374. package/dist/ui-tailwind/editor-surface/search-plugin.js +0 -23
  375. package/dist/ui-tailwind/editor-surface/search-plugin.js.map +0 -1
  376. package/dist/ui-tailwind/index.cjs +0 -4833
  377. package/dist/ui-tailwind/index.cjs.map +0 -1
  378. package/dist/ui-tailwind/index.d.cts +0 -617
  379. package/dist/ui-tailwind/index.d.ts +0 -617
  380. package/dist/ui-tailwind/index.js +0 -575
  381. package/dist/ui-tailwind/index.js.map +0 -1
@@ -0,0 +1,463 @@
1
+ import type { RevisionRecord } from "../../review/store/revision-types.ts";
2
+ import {
3
+ mapRevisionBoundaries,
4
+ type RevisionParagraphBoundary,
5
+ } from "../ooxml/revision-boundaries.ts";
6
+
7
+ interface XmlReplacement {
8
+ start: number;
9
+ end: number;
10
+ replacement: string;
11
+ }
12
+
13
+ export interface SerializedRuntimeRevisionsResult {
14
+ documentXml: string;
15
+ serializedRevisionIds: string[];
16
+ skippedRevisionIds: string[];
17
+ }
18
+
19
+ const STORY_WRAPPER_PREFIX =
20
+ `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>` +
21
+ `<w:document xmlns:w="http://schemas.openxmlformats.org/wordprocessingml/2006/main"><w:body>`;
22
+ const STORY_WRAPPER_SUFFIX = `</w:body></w:document>`;
23
+
24
+ export function serializeRuntimeRevisionsIntoDocumentXml(
25
+ documentXml: string,
26
+ revisions: readonly RevisionRecord[],
27
+ boundaries: readonly RevisionParagraphBoundary[] = mapRevisionBoundaries(documentXml),
28
+ ): SerializedRuntimeRevisionsResult {
29
+ const replacements: XmlReplacement[] = [];
30
+ const serializedRevisionIds: string[] = [];
31
+ const skippedRevisionIds: string[] = [];
32
+ const paragraphMarkers = new Map<
33
+ number,
34
+ { boundary: RevisionParagraphBoundary; markers: string[]; revisionIds: string[] }
35
+ >();
36
+
37
+ for (const revision of revisions) {
38
+ if (revision.status !== "active" || revision.anchor.kind !== "range") {
39
+ continue;
40
+ }
41
+
42
+ if (revision.kind === "property-change") {
43
+ const propertyChangeReplacement = createPropertyChangeReplacement(documentXml, boundaries, revision);
44
+ if (!propertyChangeReplacement) {
45
+ skippedRevisionIds.push(revision.revisionId);
46
+ continue;
47
+ }
48
+ replacements.push(propertyChangeReplacement);
49
+ serializedRevisionIds.push(revision.revisionId);
50
+ continue;
51
+ }
52
+
53
+ if (revision.kind !== "insertion" && revision.kind !== "deletion") {
54
+ continue;
55
+ }
56
+
57
+ const form = revision.metadata.importedRevisionForm;
58
+ if (form === "paragraph-insertion" || form === "paragraph-deletion") {
59
+ const paragraphBoundary = findParagraphBoundaryForAnchor(boundaries, revision);
60
+ if (!paragraphBoundary) {
61
+ skippedRevisionIds.push(revision.revisionId);
62
+ continue;
63
+ }
64
+
65
+ const entry = paragraphMarkers.get(paragraphBoundary.paragraphIndex) ?? {
66
+ boundary: paragraphBoundary,
67
+ markers: [],
68
+ revisionIds: [],
69
+ };
70
+ entry.markers.push(createParagraphRevisionMarker(revision));
71
+ entry.revisionIds.push(revision.revisionId);
72
+ paragraphMarkers.set(paragraphBoundary.paragraphIndex, entry);
73
+ serializedRevisionIds.push(revision.revisionId);
74
+ continue;
75
+ }
76
+
77
+ const rangeReplacement = createRangeRevisionReplacement(documentXml, boundaries, revision);
78
+ if (!rangeReplacement) {
79
+ skippedRevisionIds.push(revision.revisionId);
80
+ continue;
81
+ }
82
+
83
+ replacements.push(rangeReplacement);
84
+ serializedRevisionIds.push(revision.revisionId);
85
+ }
86
+
87
+ for (const { boundary, markers, revisionIds } of paragraphMarkers.values()) {
88
+ const paragraphInsertion = createParagraphRevisionInsertion(
89
+ documentXml,
90
+ boundary,
91
+ markers,
92
+ );
93
+ if (!paragraphInsertion) {
94
+ skippedRevisionIds.push(...revisionIds);
95
+ continue;
96
+ }
97
+
98
+ replacements.push(paragraphInsertion);
99
+ }
100
+
101
+ return {
102
+ documentXml: applyReplacements(documentXml, replacements),
103
+ serializedRevisionIds,
104
+ skippedRevisionIds,
105
+ };
106
+ }
107
+
108
+ function createPropertyChangeReplacement(
109
+ documentXml: string,
110
+ boundaries: readonly RevisionParagraphBoundary[],
111
+ revision: RevisionRecord,
112
+ ): XmlReplacement | undefined {
113
+ const propertyChange = revision.metadata.propertyChangeData;
114
+ if (!propertyChange) {
115
+ return undefined;
116
+ }
117
+ if (propertyChange.xmlTag === "pPrChange") {
118
+ return createParagraphPropertyChangeReplacement(documentXml, boundaries, revision, propertyChange.beforeXml);
119
+ }
120
+ if (propertyChange.xmlTag === "rPrChange") {
121
+ return createRunPropertyChangeReplacement(documentXml, boundaries, revision, propertyChange.beforeXml);
122
+ }
123
+ return undefined;
124
+ }
125
+
126
+ function createParagraphPropertyChangeReplacement(
127
+ documentXml: string,
128
+ boundaries: readonly RevisionParagraphBoundary[],
129
+ revision: RevisionRecord,
130
+ beforeXml: string,
131
+ ): XmlReplacement | undefined {
132
+ if (revision.anchor.kind !== "range") {
133
+ return undefined;
134
+ }
135
+ const paragraphBoundary = findParagraphBoundaryForRange(
136
+ boundaries,
137
+ revision.anchor.range.from,
138
+ revision.anchor.range.to,
139
+ );
140
+ if (!paragraphBoundary) {
141
+ return undefined;
142
+ }
143
+ const insertionIndex = findClosingTagInsertionIndex(
144
+ documentXml,
145
+ paragraphBoundary.paragraphPropertiesStart,
146
+ paragraphBoundary.paragraphPropertiesEnd,
147
+ "w:pPr",
148
+ );
149
+ if (insertionIndex === undefined) {
150
+ return undefined;
151
+ }
152
+ return {
153
+ start: insertionIndex,
154
+ end: insertionIndex,
155
+ replacement: `<w:pPrChange${serializeRevisionAttributes(revision)}>${beforeXml}</w:pPrChange>`,
156
+ };
157
+ }
158
+
159
+ function createRunPropertyChangeReplacement(
160
+ documentXml: string,
161
+ boundaries: readonly RevisionParagraphBoundary[],
162
+ revision: RevisionRecord,
163
+ beforeXml: string,
164
+ ): XmlReplacement | undefined {
165
+ if (revision.anchor.kind !== "range") {
166
+ return undefined;
167
+ }
168
+ const paragraphBoundary = findParagraphBoundaryForRange(
169
+ boundaries,
170
+ revision.anchor.range.from,
171
+ revision.anchor.range.to,
172
+ );
173
+ if (!paragraphBoundary) {
174
+ return undefined;
175
+ }
176
+ const startIndex = paragraphBoundary.boundaries.get(revision.anchor.range.from);
177
+ const endIndex = paragraphBoundary.boundaries.get(revision.anchor.range.to);
178
+ if (startIndex === undefined || endIndex === undefined || endIndex < startIndex) {
179
+ return undefined;
180
+ }
181
+ const runXml = documentXml.slice(startIndex, endIndex);
182
+ if (!/^<w:r[\s>][\s\S]*<\/w:r>$/u.test(runXml)) {
183
+ return undefined;
184
+ }
185
+ const changeXml = `<w:rPrChange${serializeRevisionAttributes(revision)}>${beforeXml}</w:rPrChange>`;
186
+ let replacement: string;
187
+ if (/<w:rPr[\s>][\s\S]*<\/w:rPr>/u.test(runXml)) {
188
+ replacement = runXml.replace(/<\/w:rPr>/u, `${changeXml}</w:rPr>`);
189
+ } else {
190
+ replacement = runXml.replace(/^(<w:r[\s>][^>]*>)/u, `$1<w:rPr>${changeXml}</w:rPr>`);
191
+ }
192
+ if (replacement === runXml) {
193
+ return undefined;
194
+ }
195
+ return {
196
+ start: startIndex,
197
+ end: endIndex,
198
+ replacement,
199
+ };
200
+ }
201
+
202
+ export function serializeRuntimeRevisionsIntoStoryXml(
203
+ storyXml: string,
204
+ revisions: readonly RevisionRecord[],
205
+ ): SerializedRuntimeRevisionsResult {
206
+ if (revisions.length === 0) {
207
+ return {
208
+ documentXml: storyXml,
209
+ serializedRevisionIds: [],
210
+ skippedRevisionIds: [],
211
+ };
212
+ }
213
+
214
+ const xmlDeclMatch = storyXml.match(/^<\?xml[\s\S]*?\?>\s*/u);
215
+ const xmlDecl = xmlDeclMatch?.[0] ?? "";
216
+ const withoutDecl = xmlDeclMatch ? storyXml.slice(xmlDecl.length) : storyXml;
217
+ const rootMatch = withoutDecl.match(/^<([A-Za-z0-9:._-]+)([^>]*)>([\s\S]*)<\/\1>\s*$/u);
218
+ if (!rootMatch) {
219
+ return serializeRuntimeRevisionsIntoDocumentXml(storyXml, revisions);
220
+ }
221
+
222
+ const [, tagName, attrs = "", innerXml = ""] = rootMatch;
223
+ const wrapped = `${STORY_WRAPPER_PREFIX}${innerXml}${STORY_WRAPPER_SUFFIX}`;
224
+ const serialized = serializeRuntimeRevisionsIntoDocumentXml(wrapped, revisions);
225
+ return {
226
+ documentXml:
227
+ `${xmlDecl}<${tagName}${attrs}>` +
228
+ serialized.documentXml.slice(
229
+ STORY_WRAPPER_PREFIX.length,
230
+ serialized.documentXml.length - STORY_WRAPPER_SUFFIX.length,
231
+ ) +
232
+ `</${tagName}>`,
233
+ serializedRevisionIds: serialized.serializedRevisionIds,
234
+ skippedRevisionIds: serialized.skippedRevisionIds,
235
+ };
236
+ }
237
+
238
+ function createRangeRevisionReplacement(
239
+ documentXml: string,
240
+ boundaries: readonly RevisionParagraphBoundary[],
241
+ revision: RevisionRecord,
242
+ ): XmlReplacement | undefined {
243
+ const { anchor } = revision;
244
+ if (anchor.kind !== "range") {
245
+ return undefined;
246
+ }
247
+ const paragraphBoundary = findParagraphBoundaryForRange(boundaries, anchor.range.from, anchor.range.to);
248
+ if (!paragraphBoundary) {
249
+ return undefined;
250
+ }
251
+
252
+ const startIndex = paragraphBoundary.boundaries.get(anchor.range.from);
253
+ const endIndex = paragraphBoundary.boundaries.get(anchor.range.to);
254
+ if (startIndex === undefined || endIndex === undefined || endIndex < startIndex) {
255
+ return undefined;
256
+ }
257
+
258
+ const xml = documentXml.slice(startIndex, endIndex);
259
+ const { leadingMarkers, coreXml, trailingMarkers } = peelZeroWidthEdgeMarkers(xml);
260
+ if (coreXml.length === 0) {
261
+ return undefined;
262
+ }
263
+ const attributes = serializeRevisionAttributes(revision);
264
+ return {
265
+ start: startIndex,
266
+ end: endIndex,
267
+ replacement:
268
+ revision.kind === "insertion"
269
+ ? `${leadingMarkers}<w:ins${attributes}>${coreXml}</w:ins>${trailingMarkers}`
270
+ : `${leadingMarkers}<w:del${attributes}>${convertRunsToDeletedContent(coreXml)}</w:del>${trailingMarkers}`,
271
+ };
272
+ }
273
+
274
+ function createParagraphRevisionMarker(revision: RevisionRecord): string {
275
+ const markerName = revision.kind === "insertion" ? "w:ins" : "w:del";
276
+ return `<${markerName}${serializeRevisionAttributes(revision)}/>`;
277
+ }
278
+
279
+ function createParagraphRevisionInsertion(
280
+ documentXml: string,
281
+ paragraphBoundary: RevisionParagraphBoundary,
282
+ markers: readonly string[],
283
+ ): XmlReplacement | undefined {
284
+ const paragraphXml = documentXml.slice(
285
+ paragraphBoundary.paragraphStart,
286
+ paragraphBoundary.paragraphEnd,
287
+ );
288
+ const markerXml = markers.join("");
289
+ const paragraphRunPropertiesInsertionIndex = findClosingTagInsertionIndex(
290
+ documentXml,
291
+ paragraphBoundary.paragraphRunPropertiesStart,
292
+ paragraphBoundary.paragraphRunPropertiesEnd,
293
+ "w:rPr",
294
+ );
295
+ if (paragraphRunPropertiesInsertionIndex !== undefined) {
296
+ return {
297
+ start: paragraphRunPropertiesInsertionIndex,
298
+ end: paragraphRunPropertiesInsertionIndex,
299
+ replacement: markerXml,
300
+ };
301
+ }
302
+
303
+ const paragraphPropertiesInsertionIndex = findClosingTagInsertionIndex(
304
+ documentXml,
305
+ paragraphBoundary.paragraphPropertiesStart,
306
+ paragraphBoundary.paragraphPropertiesEnd,
307
+ "w:pPr",
308
+ );
309
+ if (paragraphPropertiesInsertionIndex !== undefined) {
310
+ return {
311
+ start: paragraphPropertiesInsertionIndex,
312
+ end: paragraphPropertiesInsertionIndex,
313
+ replacement: `<w:rPr>${markerXml}</w:rPr>`,
314
+ };
315
+ }
316
+
317
+ if (!/<w:p[\s>]/u.test(paragraphXml)) {
318
+ return undefined;
319
+ }
320
+
321
+ return {
322
+ start: paragraphBoundary.paragraphStartTagEnd,
323
+ end: paragraphBoundary.paragraphStartTagEnd,
324
+ replacement: `<w:pPr><w:rPr>${markerXml}</w:rPr></w:pPr>`,
325
+ };
326
+ }
327
+
328
+ function findParagraphBoundaryForRange(
329
+ boundaries: readonly RevisionParagraphBoundary[],
330
+ from: number,
331
+ to: number,
332
+ ): RevisionParagraphBoundary | undefined {
333
+ return boundaries.find(
334
+ (boundary) => from >= boundary.start && to <= boundary.end,
335
+ );
336
+ }
337
+
338
+ function findParagraphBoundaryForAnchor(
339
+ boundaries: readonly RevisionParagraphBoundary[],
340
+ revision: RevisionRecord,
341
+ ): RevisionParagraphBoundary | undefined {
342
+ const anchor = revision.anchor.kind === "range" ? revision.anchor.range.from : undefined;
343
+ if (anchor === undefined) {
344
+ return undefined;
345
+ }
346
+
347
+ return boundaries.find(
348
+ (boundary) =>
349
+ boundary.end === anchor ||
350
+ (anchor >= boundary.start && anchor <= boundary.end),
351
+ );
352
+ }
353
+
354
+ function serializeRevisionAttributes(revision: RevisionRecord): string {
355
+ const attributes = {
356
+ "w:id": revision.metadata.ooxmlRevisionId ?? sanitizeRevisionId(revision.revisionId),
357
+ "w:author": revision.authorId,
358
+ "w:date": revision.createdAt,
359
+ };
360
+
361
+ return Object.entries(attributes)
362
+ .filter(([, value]) => value && value.length > 0)
363
+ .map(([name, value]) => ` ${name}="${escapeAttribute(value)}"`)
364
+ .join("");
365
+ }
366
+
367
+ function peelZeroWidthEdgeMarkers(xml: string): {
368
+ leadingMarkers: string;
369
+ coreXml: string;
370
+ trailingMarkers: string;
371
+ } {
372
+ const markerPattern =
373
+ /^(?:\s|<(?:\w+:)?(?:bookmarkStart|bookmarkEnd|permStart|permEnd|commentRangeStart|commentRangeEnd|proofErr|lastRenderedPageBreak)\b[^>]*\/>)+$/u;
374
+ let coreXml = xml;
375
+ let leadingMarkers = "";
376
+ let trailingMarkers = "";
377
+
378
+ while (true) {
379
+ const match = coreXml.match(/^\s*(<(?:\w+:)?(?:bookmarkStart|bookmarkEnd|permStart|permEnd|commentRangeStart|commentRangeEnd|proofErr|lastRenderedPageBreak)\b[^>]*\/>)/u);
380
+ if (!match) {
381
+ break;
382
+ }
383
+ leadingMarkers += match[0];
384
+ coreXml = coreXml.slice(match[0].length);
385
+ }
386
+
387
+ while (true) {
388
+ const match = coreXml.match(/(<(?:\w+:)?(?:bookmarkStart|bookmarkEnd|permStart|permEnd|commentRangeStart|commentRangeEnd|proofErr|lastRenderedPageBreak)\b[^>]*\/>)\s*$/u);
389
+ if (!match) {
390
+ break;
391
+ }
392
+ trailingMarkers = match[0] + trailingMarkers;
393
+ coreXml = coreXml.slice(0, coreXml.length - match[0].length);
394
+ }
395
+
396
+ if (markerPattern.test(coreXml)) {
397
+ return {
398
+ leadingMarkers: xml,
399
+ coreXml: "",
400
+ trailingMarkers: "",
401
+ };
402
+ }
403
+
404
+ return {
405
+ leadingMarkers,
406
+ coreXml,
407
+ trailingMarkers,
408
+ };
409
+ }
410
+
411
+ function findClosingTagInsertionIndex(
412
+ documentXml: string,
413
+ start: number | undefined,
414
+ end: number | undefined,
415
+ tagName: string,
416
+ ): number | undefined {
417
+ if (start === undefined || end === undefined) {
418
+ return undefined;
419
+ }
420
+
421
+ const closingTag = `</${tagName}>`;
422
+ const closingIndex = documentXml.lastIndexOf(closingTag, end);
423
+ if (closingIndex < start) {
424
+ return undefined;
425
+ }
426
+
427
+ return closingIndex;
428
+ }
429
+
430
+ function sanitizeRevisionId(revisionId: string): string {
431
+ const numericTail = /(\d+)$/.exec(revisionId)?.[1];
432
+ return numericTail ?? revisionId.replace(/[^A-Za-z0-9._-]/g, "-");
433
+ }
434
+
435
+ function convertRunsToDeletedContent(xml: string): string {
436
+ return xml
437
+ .replace(/<(\/?)w:t\b/g, "<$1w:delText")
438
+ .replace(/<(\/?)w:instrText\b/g, "<$1w:delInstrText");
439
+ }
440
+
441
+ function applyReplacements(documentXml: string, replacements: readonly XmlReplacement[]): string {
442
+ const sorted = replacements
443
+ .slice()
444
+ .sort((left, right) => right.start - left.start || right.end - left.end);
445
+ let output = documentXml;
446
+
447
+ for (const replacement of sorted) {
448
+ output =
449
+ output.slice(0, replacement.start) +
450
+ replacement.replacement +
451
+ output.slice(replacement.end);
452
+ }
453
+
454
+ return output;
455
+ }
456
+
457
+ function escapeAttribute(value: string): string {
458
+ return value
459
+ .replace(/&/g, "&amp;")
460
+ .replace(/</g, "&lt;")
461
+ .replace(/>/g, "&gt;")
462
+ .replace(/"/g, "&quot;");
463
+ }
@@ -0,0 +1,174 @@
1
+ import type {
2
+ ParsedBorderSpec,
3
+ ParsedCellMargins,
4
+ ParsedCellShading,
5
+ ParsedTable,
6
+ ParsedTableBorders,
7
+ ParsedTableCell,
8
+ ParsedTableCellBorders,
9
+ ParsedTableLook,
10
+ ParsedTableRow,
11
+ ParsedTableWidth,
12
+ } from "../ooxml/parse-tables.ts";
13
+
14
+ export function serializeTable(table: ParsedTable): string {
15
+ const propertiesXml = table.propertiesXml ?? buildTablePropertiesXml(table);
16
+ const gridXml =
17
+ table.gridColumns.length > 0
18
+ ? `<w:tblGrid>${table.gridColumns
19
+ .map((width) => `<w:gridCol w:w="${width}"/>`)
20
+ .join("")}</w:tblGrid>`
21
+ : "";
22
+ const rowsXml = table.rows.map(serializeRow).join("");
23
+
24
+ return `<w:tbl>${propertiesXml}${gridXml}${rowsXml}</w:tbl>`;
25
+ }
26
+
27
+ function serializeRow(row: ParsedTableRow): string {
28
+ const propertiesXml = row.propertiesXml ?? buildRowPropertiesXml(row);
29
+ return `<w:tr>${propertiesXml}${row.cells.map(serializeCell).join("")}</w:tr>`;
30
+ }
31
+
32
+ function serializeCell(cell: ParsedTableCell): string {
33
+ const propertiesXml = ensureCellProperties(cell);
34
+ const blocksXml = cell.blocksXml.length > 0 ? cell.blocksXml.join("") : "<w:p/>";
35
+ return `<w:tc>${propertiesXml}${blocksXml}</w:tc>`;
36
+ }
37
+
38
+ function buildTablePropertiesXml(table: ParsedTable): string {
39
+ const children: string[] = [];
40
+ if (table.styleId) {
41
+ children.push(`<w:tblStyle w:val="${table.styleId}"/>`);
42
+ }
43
+ if (table.width) {
44
+ children.push(serializeWidth("tblW", table.width));
45
+ }
46
+ if (table.alignment) {
47
+ children.push(`<w:jc w:val="${table.alignment}"/>`);
48
+ }
49
+ if (table.borders) {
50
+ const bordersXml = serializeTableBorders(table.borders);
51
+ if (bordersXml) children.push(`<w:tblBorders>${bordersXml}</w:tblBorders>`);
52
+ }
53
+ if (table.cellMargins) {
54
+ const marginsXml = serializeTableCellMargins(table.cellMargins);
55
+ if (marginsXml) children.push(`<w:tblCellMar>${marginsXml}</w:tblCellMar>`);
56
+ }
57
+ if (table.tblLook) {
58
+ const tblLookXml = serializeTableLook(table.tblLook);
59
+ if (tblLookXml) children.push(tblLookXml);
60
+ }
61
+ return children.length > 0 ? `<w:tblPr>${children.join("")}</w:tblPr>` : "";
62
+ }
63
+
64
+ function buildRowPropertiesXml(row: ParsedTableRow): string {
65
+ const children: string[] = [];
66
+ if (row.height !== undefined) {
67
+ const hRuleAttr = row.heightRule ? ` w:hRule="${row.heightRule}"` : "";
68
+ children.push(`<w:trHeight w:val="${row.height}"${hRuleAttr}/>`);
69
+ }
70
+ if (row.isHeader) {
71
+ children.push(`<w:tblHeader/>`);
72
+ }
73
+ return children.length > 0 ? `<w:trPr>${children.join("")}</w:trPr>` : "";
74
+ }
75
+
76
+ function ensureCellProperties(cell: ParsedTableCell): string {
77
+ if (cell.propertiesXml) {
78
+ return cell.propertiesXml;
79
+ }
80
+
81
+ const children: string[] = [];
82
+ if (cell.width) {
83
+ children.push(serializeWidth("tcW", cell.width));
84
+ }
85
+ if (cell.gridSpan && cell.gridSpan > 1) {
86
+ children.push(`<w:gridSpan w:val="${cell.gridSpan}"/>`);
87
+ }
88
+ if (cell.verticalMerge) {
89
+ children.push(
90
+ cell.verticalMerge === "restart"
91
+ ? `<w:vMerge w:val="restart"/>`
92
+ : `<w:vMerge/>`,
93
+ );
94
+ }
95
+ if (cell.borders) {
96
+ const bordersXml = serializeCellBorders(cell.borders);
97
+ if (bordersXml) children.push(`<w:tcBorders>${bordersXml}</w:tcBorders>`);
98
+ }
99
+ if (cell.shading) {
100
+ children.push(serializeCellShading(cell.shading));
101
+ }
102
+ if (cell.verticalAlign) {
103
+ children.push(`<w:vAlign w:val="${cell.verticalAlign}"/>`);
104
+ }
105
+
106
+ return children.length > 0 ? `<w:tcPr>${children.join("")}</w:tcPr>` : "";
107
+ }
108
+
109
+ function serializeWidth(element: string, width: ParsedTableWidth): string {
110
+ return `<w:${element} w:w="${width.value}" w:type="${width.type}"/>`;
111
+ }
112
+
113
+ function serializeBorderSpec(element: string, spec: ParsedBorderSpec): string {
114
+ const attrs: string[] = [];
115
+ if (spec.value) attrs.push(`w:val="${spec.value}"`);
116
+ if (spec.size !== undefined) attrs.push(`w:sz="${spec.size}"`);
117
+ if (spec.space !== undefined) attrs.push(`w:space="${spec.space}"`);
118
+ if (spec.color) attrs.push(`w:color="${spec.color}"`);
119
+ const attrsStr = attrs.length > 0 ? ` ${attrs.join(" ")}` : "";
120
+ return `<w:${element}${attrsStr}/>`;
121
+ }
122
+
123
+ function serializeTableBorders(borders: ParsedTableBorders): string {
124
+ const sides = ["top", "left", "bottom", "right", "insideH", "insideV"] as const;
125
+ return sides
126
+ .filter((side) => borders[side] !== undefined)
127
+ .map((side) => serializeBorderSpec(side, borders[side]!))
128
+ .join("");
129
+ }
130
+
131
+ function serializeCellBorders(borders: ParsedTableCellBorders): string {
132
+ const sides = ["top", "left", "bottom", "right", "insideH", "insideV"] as const;
133
+ return sides
134
+ .filter((side) => borders[side] !== undefined)
135
+ .map((side) => serializeBorderSpec(side, borders[side]!))
136
+ .join("");
137
+ }
138
+
139
+ function serializeTableLook(tblLook: ParsedTableLook): string {
140
+ const attrs: string[] = [];
141
+ if (tblLook.val) attrs.push(`w:val="${tblLook.val}"`);
142
+ for (const [key, attr] of [
143
+ ["firstRow", "w:firstRow"],
144
+ ["lastRow", "w:lastRow"],
145
+ ["firstColumn", "w:firstColumn"],
146
+ ["lastColumn", "w:lastColumn"],
147
+ ["noHBand", "w:noHBand"],
148
+ ["noVBand", "w:noVBand"],
149
+ ] as const) {
150
+ const value = tblLook[key];
151
+ if (value !== undefined) {
152
+ attrs.push(`${attr}="${value ? "1" : "0"}"`);
153
+ }
154
+ }
155
+ return attrs.length > 0 ? `<w:tblLook ${attrs.join(" ")}/>` : "";
156
+ }
157
+
158
+ function serializeCellShading(shading: ParsedCellShading): string {
159
+ const attrs: string[] = [];
160
+ if (shading.val) attrs.push(`w:val="${shading.val}"`);
161
+ if (shading.color) attrs.push(`w:color="${shading.color}"`);
162
+ if (shading.fill) attrs.push(`w:fill="${shading.fill}"`);
163
+ const attrsStr = attrs.length > 0 ? ` ${attrs.join(" ")}` : "";
164
+ return `<w:shd${attrsStr}/>`;
165
+ }
166
+
167
+ function serializeTableCellMargins(margins: ParsedCellMargins): string {
168
+ const parts: string[] = [];
169
+ if (margins.top !== undefined) parts.push(`<w:top w:w="${margins.top}" w:type="dxa"/>`);
170
+ if (margins.left !== undefined) parts.push(`<w:left w:w="${margins.left}" w:type="dxa"/>`);
171
+ if (margins.bottom !== undefined) parts.push(`<w:bottom w:w="${margins.bottom}" w:type="dxa"/>`);
172
+ if (margins.right !== undefined) parts.push(`<w:right w:w="${margins.right}" w:type="dxa"/>`);
173
+ return parts.join("");
174
+ }