@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,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
+ }