@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,747 @@
1
+ import type {
2
+ NumberingCatalog,
3
+ NumberingLevelDefinition,
4
+ NumberingLevelParagraphGeometry,
5
+ NumberingLevelOverride,
6
+ NumberingLevelOverrideDefinition,
7
+ ParagraphSpacing,
8
+ ParagraphIndentation,
9
+ TabStop,
10
+ } from "../../model/canonical-document.ts";
11
+
12
+ export interface ParsedParagraphNumberingReference {
13
+ paragraphIndex: number;
14
+ numberingInstanceId: string;
15
+ level: number;
16
+ }
17
+
18
+ interface XmlElementNode {
19
+ type: "element";
20
+ name: string;
21
+ attributes: Record<string, string>;
22
+ children: XmlNode[];
23
+ }
24
+
25
+ interface XmlTextNode {
26
+ type: "text";
27
+ text: string;
28
+ }
29
+
30
+ type XmlNode = XmlElementNode | XmlTextNode;
31
+
32
+ export function parseNumberingXml(xml: string): NumberingCatalog {
33
+ const root = parseXml(xml);
34
+ const numberingElement = findChildElement(root, "numbering");
35
+ const abstractDefinitions: NumberingCatalog["abstractDefinitions"] = {};
36
+ const instances: NumberingCatalog["instances"] = {};
37
+
38
+ for (const child of numberingElement.children) {
39
+ if (child.type !== "element") {
40
+ continue;
41
+ }
42
+
43
+ switch (localName(child.name)) {
44
+ case "abstractNum": {
45
+ const rawId = child.attributes["w:abstractNumId"] ?? child.attributes.abstractNumId;
46
+ if (!rawId) {
47
+ continue;
48
+ }
49
+
50
+ const abstractNumberingId = toCanonicalAbstractNumberingId(rawId);
51
+ abstractDefinitions[abstractNumberingId] = {
52
+ abstractNumberingId,
53
+ levels: readLevels(child),
54
+ };
55
+ break;
56
+ }
57
+ case "num": {
58
+ const rawId = child.attributes["w:numId"] ?? child.attributes.numId;
59
+ const abstractReference = findChildElementOptional(child, "abstractNumId");
60
+ const rawAbstractId =
61
+ abstractReference?.attributes["w:val"] ?? abstractReference?.attributes.val;
62
+
63
+ if (!rawId || !rawAbstractId) {
64
+ continue;
65
+ }
66
+
67
+ const numberingInstanceId = toCanonicalNumberingInstanceId(rawId);
68
+ instances[numberingInstanceId] = {
69
+ numberingInstanceId,
70
+ abstractNumberingId: toCanonicalAbstractNumberingId(rawAbstractId),
71
+ overrides: readOverrides(child),
72
+ };
73
+ break;
74
+ }
75
+ }
76
+ }
77
+
78
+ return {
79
+ abstractDefinitions,
80
+ instances,
81
+ };
82
+ }
83
+
84
+ export function parseParagraphNumberingReferences(
85
+ documentXml: string,
86
+ ): ParsedParagraphNumberingReference[] {
87
+ const root = parseXml(documentXml);
88
+ const documentElement = findChildElement(root, "document");
89
+ const bodyElement = findChildElement(documentElement, "body");
90
+ const references: ParsedParagraphNumberingReference[] = [];
91
+ let paragraphIndex = 0;
92
+
93
+ for (const child of bodyElement.children) {
94
+ if (child.type !== "element" || localName(child.name) !== "p") {
95
+ continue;
96
+ }
97
+
98
+ const paragraphProperties = findChildElementOptional(child, "pPr");
99
+ const numberingProperties = paragraphProperties
100
+ ? findChildElementOptional(paragraphProperties, "numPr")
101
+ : undefined;
102
+
103
+ if (numberingProperties) {
104
+ const levelNode = findChildElementOptional(numberingProperties, "ilvl");
105
+ const instanceNode = findChildElementOptional(numberingProperties, "numId");
106
+ const rawLevel = levelNode?.attributes["w:val"] ?? levelNode?.attributes.val;
107
+ const rawInstanceId = instanceNode?.attributes["w:val"] ?? instanceNode?.attributes.val;
108
+
109
+ if (rawLevel !== undefined && rawInstanceId) {
110
+ const level = parseInteger(rawLevel);
111
+ if (level !== undefined) {
112
+ references.push({
113
+ paragraphIndex,
114
+ numberingInstanceId: toCanonicalNumberingInstanceId(rawInstanceId),
115
+ level,
116
+ });
117
+ }
118
+ }
119
+ }
120
+
121
+ paragraphIndex += 1;
122
+ }
123
+
124
+ return references;
125
+ }
126
+
127
+ export function toCanonicalAbstractNumberingId(value: string): string {
128
+ return value.startsWith("abstract-num:") ? value : `abstract-num:${value}`;
129
+ }
130
+
131
+ export function toCanonicalNumberingInstanceId(value: string): string {
132
+ return value.startsWith("num:") ? value : `num:${value}`;
133
+ }
134
+
135
+ function readLevels(abstractNode: XmlElementNode): NumberingLevelDefinition[] {
136
+ const levels: NumberingLevelDefinition[] = [];
137
+
138
+ for (const child of abstractNode.children) {
139
+ if (child.type !== "element" || localName(child.name) !== "lvl") {
140
+ continue;
141
+ }
142
+
143
+ const rawLevel = child.attributes["w:ilvl"] ?? child.attributes.ilvl;
144
+ const level = rawLevel === undefined ? undefined : parseInteger(rawLevel);
145
+ if (level === undefined) {
146
+ continue;
147
+ }
148
+ const definition = readLevelDefinition(child, level);
149
+ if (!definition) {
150
+ continue;
151
+ }
152
+
153
+ levels.push(definition);
154
+ }
155
+
156
+ return levels.sort((left, right) => left.level - right.level);
157
+ }
158
+
159
+ function readOverrides(numNode: XmlElementNode): NumberingLevelOverride[] {
160
+ const overrides: NumberingLevelOverride[] = [];
161
+
162
+ for (const child of numNode.children) {
163
+ if (child.type !== "element" || localName(child.name) !== "lvlOverride") {
164
+ continue;
165
+ }
166
+
167
+ const rawLevel = child.attributes["w:ilvl"] ?? child.attributes.ilvl;
168
+ const level = rawLevel === undefined ? undefined : parseInteger(rawLevel);
169
+ if (level === undefined) {
170
+ continue;
171
+ }
172
+
173
+ const startOverrideNode = findChildElementOptional(child, "startOverride");
174
+ const rawStart = startOverrideNode?.attributes["w:val"] ?? startOverrideNode?.attributes.val;
175
+ const startAt = rawStart === undefined ? undefined : parseInteger(rawStart);
176
+ const levelDefinitionNode = findChildElementOptional(child, "lvl");
177
+ const levelDefinition = levelDefinitionNode
178
+ ? readLevelOverrideDefinition(levelDefinitionNode, level)
179
+ : undefined;
180
+
181
+ overrides.push({
182
+ level,
183
+ ...(startAt !== undefined ? { startAt } : {}),
184
+ ...(levelDefinition ? { levelDefinition } : {}),
185
+ });
186
+ }
187
+
188
+ return overrides.sort((left, right) => left.level - right.level);
189
+ }
190
+
191
+ function readLevelDefinition(
192
+ levelNode: XmlElementNode,
193
+ fallbackLevel?: number,
194
+ ): NumberingLevelDefinition | undefined {
195
+ const rawLevel = levelNode.attributes["w:ilvl"] ?? levelNode.attributes.ilvl;
196
+ const level = rawLevel === undefined ? fallbackLevel : parseInteger(rawLevel);
197
+ if (level === undefined) {
198
+ return undefined;
199
+ }
200
+
201
+ const startNode = findChildElementOptional(levelNode, "start");
202
+ const formatNode = findChildElementOptional(levelNode, "numFmt");
203
+ const textNode = findChildElementOptional(levelNode, "lvlText");
204
+ const paragraphStyleNode = findChildElementOptional(levelNode, "pStyle");
205
+ const rawStart = startNode?.attributes["w:val"] ?? startNode?.attributes.val;
206
+ const startAt = rawStart === undefined ? undefined : parseInteger(rawStart);
207
+ const format = formatNode?.attributes["w:val"] ?? formatNode?.attributes.val ?? "decimal";
208
+ const text = textNode?.attributes["w:val"] ?? textNode?.attributes.val ?? `%${level + 1}.`;
209
+ const paragraphStyleId =
210
+ paragraphStyleNode?.attributes["w:val"] ?? paragraphStyleNode?.attributes.val;
211
+ const isLegalNode = findChildElementOptional(levelNode, "isLgl");
212
+ const isLegalNumbering = readIsLegalNumberingValue(isLegalNode);
213
+ const suffixNode = findChildElementOptional(levelNode, "suff");
214
+ const suffixVal = suffixNode?.attributes["w:val"] ?? suffixNode?.attributes.val;
215
+ const suffix =
216
+ suffixVal === "space" || suffixVal === "nothing"
217
+ ? suffixVal
218
+ : suffixVal === "tab"
219
+ ? "tab"
220
+ : undefined;
221
+ const paragraphGeometry = readLevelParagraphGeometry(levelNode);
222
+
223
+ return {
224
+ level,
225
+ format,
226
+ text,
227
+ ...(startAt !== undefined ? { startAt } : {}),
228
+ ...(paragraphStyleId ? { paragraphStyleId } : {}),
229
+ ...(isLegalNumbering ? { isLegalNumbering: true } : {}),
230
+ ...(suffix ? { suffix } : {}),
231
+ ...(paragraphGeometry ? { paragraphGeometry } : {}),
232
+ };
233
+ }
234
+
235
+ function readLevelOverrideDefinition(
236
+ levelNode: XmlElementNode,
237
+ fallbackLevel?: number,
238
+ ): NumberingLevelOverrideDefinition | undefined {
239
+ const rawLevel = levelNode.attributes["w:ilvl"] ?? levelNode.attributes.ilvl;
240
+ const level = rawLevel === undefined ? fallbackLevel : parseInteger(rawLevel);
241
+ if (level === undefined) {
242
+ return undefined;
243
+ }
244
+
245
+ const startNode = findChildElementOptional(levelNode, "start");
246
+ const formatNode = findChildElementOptional(levelNode, "numFmt");
247
+ const textNode = findChildElementOptional(levelNode, "lvlText");
248
+ const paragraphStyleNode = findChildElementOptional(levelNode, "pStyle");
249
+ const rawStart = startNode?.attributes["w:val"] ?? startNode?.attributes.val;
250
+ const startAt = rawStart === undefined ? undefined : parseInteger(rawStart);
251
+ const format = formatNode?.attributes["w:val"] ?? formatNode?.attributes.val;
252
+ const text = textNode?.attributes["w:val"] ?? textNode?.attributes.val;
253
+ const paragraphStyleId =
254
+ paragraphStyleNode?.attributes["w:val"] ?? paragraphStyleNode?.attributes.val;
255
+ const isLegalNode = findChildElementOptional(levelNode, "isLgl");
256
+ const isLegalNumbering = readIsLegalNumberingValue(isLegalNode);
257
+ const suffixNode = findChildElementOptional(levelNode, "suff");
258
+ const suffixVal = suffixNode?.attributes["w:val"] ?? suffixNode?.attributes.val;
259
+ const suffix =
260
+ suffixVal === "space" || suffixVal === "nothing"
261
+ ? suffixVal
262
+ : suffixVal === "tab"
263
+ ? "tab"
264
+ : undefined;
265
+ const paragraphGeometry = readLevelParagraphGeometry(levelNode);
266
+
267
+ const hasExplicitFields =
268
+ startAt !== undefined ||
269
+ format !== undefined ||
270
+ text !== undefined ||
271
+ paragraphStyleId !== undefined ||
272
+ isLegalNumbering !== undefined ||
273
+ suffix !== undefined ||
274
+ paragraphGeometry !== undefined;
275
+
276
+ if (!hasExplicitFields) {
277
+ return undefined;
278
+ }
279
+
280
+ return {
281
+ level,
282
+ ...(startAt !== undefined ? { startAt } : {}),
283
+ ...(format !== undefined ? { format } : {}),
284
+ ...(text !== undefined ? { text } : {}),
285
+ ...(paragraphStyleId ? { paragraphStyleId } : {}),
286
+ ...(isLegalNumbering !== undefined ? { isLegalNumbering } : {}),
287
+ ...(suffix ? { suffix } : {}),
288
+ ...(paragraphGeometry ? { paragraphGeometry } : {}),
289
+ };
290
+ }
291
+
292
+ function readLevelParagraphGeometry(
293
+ levelNode: XmlElementNode,
294
+ ): NumberingLevelDefinition["paragraphGeometry"] | undefined {
295
+ const paragraphPropertiesNode = findChildElementOptional(levelNode, "pPr");
296
+ const justification = readLevelJustification(levelNode);
297
+ const spacing = paragraphPropertiesNode ? readParagraphSpacing(paragraphPropertiesNode) : undefined;
298
+ const indentation = paragraphPropertiesNode
299
+ ? readParagraphIndentation(paragraphPropertiesNode)
300
+ : undefined;
301
+ const tabStops = paragraphPropertiesNode ? readParagraphTabStops(paragraphPropertiesNode) : undefined;
302
+
303
+ if (!justification && !spacing && !indentation && (!tabStops || tabStops.length === 0)) {
304
+ return undefined;
305
+ }
306
+
307
+ return {
308
+ ...(justification ? { justification } : {}),
309
+ ...(spacing ? { spacing } : {}),
310
+ ...(indentation ? { indentation } : {}),
311
+ ...(tabStops && tabStops.length > 0 ? { tabStops } : {}),
312
+ };
313
+ }
314
+
315
+ function readLevelJustification(
316
+ levelNode: XmlElementNode,
317
+ ): NumberingLevelParagraphGeometry["justification"] {
318
+ const justificationNode = findChildElementOptional(levelNode, "lvlJc");
319
+ if (!justificationNode) {
320
+ return undefined;
321
+ }
322
+
323
+ const rawValue = (
324
+ justificationNode.attributes["w:val"] ??
325
+ justificationNode.attributes.val ??
326
+ ""
327
+ ).toLowerCase();
328
+
329
+ if (rawValue === "start") {
330
+ return "left";
331
+ }
332
+ if (rawValue === "end") {
333
+ return "right";
334
+ }
335
+ if (
336
+ rawValue === "left" ||
337
+ rawValue === "center" ||
338
+ rawValue === "right" ||
339
+ rawValue === "both" ||
340
+ rawValue === "distribute"
341
+ ) {
342
+ return rawValue;
343
+ }
344
+
345
+ return undefined;
346
+ }
347
+
348
+ function readParagraphIndentation(node: XmlElementNode): ParagraphIndentation | undefined {
349
+ const indNode = findChildElementOptional(node, "ind");
350
+ if (!indNode) {
351
+ return undefined;
352
+ }
353
+
354
+ const indentation: ParagraphIndentation = {};
355
+ const left =
356
+ indNode.attributes["w:start"] ??
357
+ indNode.attributes.start ??
358
+ indNode.attributes["w:left"] ??
359
+ indNode.attributes.left;
360
+ const right =
361
+ indNode.attributes["w:end"] ??
362
+ indNode.attributes.end ??
363
+ indNode.attributes["w:right"] ??
364
+ indNode.attributes.right;
365
+ const firstLine = indNode.attributes["w:firstLine"] ?? indNode.attributes.firstLine;
366
+ const hanging = indNode.attributes["w:hanging"] ?? indNode.attributes.hanging;
367
+
368
+ if (left !== undefined) {
369
+ const value = Number.parseInt(left, 10);
370
+ if (Number.isFinite(value)) indentation.left = value;
371
+ }
372
+ if (right !== undefined) {
373
+ const value = Number.parseInt(right, 10);
374
+ if (Number.isFinite(value)) indentation.right = value;
375
+ }
376
+ if (firstLine !== undefined) {
377
+ const value = Number.parseInt(firstLine, 10);
378
+ if (Number.isFinite(value)) indentation.firstLine = value;
379
+ }
380
+ if (hanging !== undefined) {
381
+ const value = Number.parseInt(hanging, 10);
382
+ if (Number.isFinite(value)) indentation.hanging = value;
383
+ }
384
+
385
+ if (
386
+ indentation.left === undefined &&
387
+ indentation.right === undefined &&
388
+ indentation.firstLine === undefined &&
389
+ indentation.hanging === undefined
390
+ ) {
391
+ return undefined;
392
+ }
393
+
394
+ return indentation;
395
+ }
396
+
397
+ function readParagraphSpacing(node: XmlElementNode): ParagraphSpacing | undefined {
398
+ const spacingNode = findChildElementOptional(node, "spacing");
399
+ if (!spacingNode) {
400
+ return undefined;
401
+ }
402
+
403
+ const spacing: ParagraphSpacing = {};
404
+ const before = spacingNode.attributes["w:before"] ?? spacingNode.attributes.before;
405
+ const after = spacingNode.attributes["w:after"] ?? spacingNode.attributes.after;
406
+ const line = spacingNode.attributes["w:line"] ?? spacingNode.attributes.line;
407
+ const lineRule = spacingNode.attributes["w:lineRule"] ?? spacingNode.attributes.lineRule;
408
+
409
+ if (before !== undefined) {
410
+ const value = Number.parseInt(before, 10);
411
+ if (Number.isFinite(value)) spacing.before = value;
412
+ }
413
+ if (after !== undefined) {
414
+ const value = Number.parseInt(after, 10);
415
+ if (Number.isFinite(value)) spacing.after = value;
416
+ }
417
+ if (line !== undefined) {
418
+ const value = Number.parseInt(line, 10);
419
+ if (Number.isFinite(value)) spacing.line = value;
420
+ }
421
+ if (lineRule !== undefined) {
422
+ const normalized = lineRule.toLowerCase();
423
+ if (normalized === "auto" || normalized === "exact") {
424
+ spacing.lineRule = normalized;
425
+ } else if (normalized === "atleast") {
426
+ spacing.lineRule = "atLeast";
427
+ }
428
+ }
429
+
430
+ if (
431
+ spacing.before === undefined &&
432
+ spacing.after === undefined &&
433
+ spacing.line === undefined &&
434
+ spacing.lineRule === undefined
435
+ ) {
436
+ return undefined;
437
+ }
438
+
439
+ return spacing;
440
+ }
441
+
442
+ function readParagraphTabStops(node: XmlElementNode): TabStop[] | undefined {
443
+ const tabsNode = findChildElementOptional(node, "tabs");
444
+ if (!tabsNode) {
445
+ return undefined;
446
+ }
447
+
448
+ const tabStops: TabStop[] = [];
449
+ for (const child of tabsNode.children) {
450
+ if (child.type !== "element" || localName(child.name) !== "tab") {
451
+ continue;
452
+ }
453
+
454
+ const pos = child.attributes["w:pos"] ?? child.attributes.pos;
455
+ const val = (child.attributes["w:val"] ?? child.attributes.val ?? "left").toLowerCase();
456
+ const leader = (child.attributes["w:leader"] ?? child.attributes.leader ?? "none").toLowerCase();
457
+ if (pos === undefined) {
458
+ continue;
459
+ }
460
+
461
+ const position = Number.parseInt(pos, 10);
462
+ if (!Number.isFinite(position)) {
463
+ continue;
464
+ }
465
+
466
+ const align = (["left", "center", "right", "decimal", "num", "bar", "clear"] as const).includes(
467
+ val as TabStop["align"],
468
+ )
469
+ ? (val as TabStop["align"])
470
+ : "left";
471
+ const leaderValue =
472
+ leader === "none" ||
473
+ leader === "dot" ||
474
+ leader === "hyphen" ||
475
+ leader === "underscore" ||
476
+ leader === "heavy"
477
+ ? (leader as Exclude<TabStop["leader"], "middleDot">)
478
+ : leader === "middledot"
479
+ ? "middleDot"
480
+ : undefined;
481
+
482
+ tabStops.push({
483
+ position,
484
+ align,
485
+ ...(leaderValue && leaderValue !== "none" ? { leader: leaderValue } : {}),
486
+ });
487
+ }
488
+
489
+ return tabStops.length > 0 ? tabStops : undefined;
490
+ }
491
+
492
+ function readIsLegalNumberingValue(
493
+ isLegalNode: XmlElementNode | undefined,
494
+ ): boolean | undefined {
495
+ if (!isLegalNode) {
496
+ return undefined;
497
+ }
498
+
499
+ const rawValue = (
500
+ isLegalNode.attributes["w:val"] ??
501
+ isLegalNode.attributes.val ??
502
+ "1"
503
+ ).toLowerCase();
504
+
505
+ if (rawValue === "0" || rawValue === "false" || rawValue === "off") {
506
+ return false;
507
+ }
508
+
509
+ return true;
510
+ }
511
+
512
+ function findChildElement(node: XmlElementNode, childLocalName: string): XmlElementNode {
513
+ const child = findChildElementOptional(node, childLocalName);
514
+ if (!child) {
515
+ throw new Error(`Expected <${childLocalName}> element in numbering XML.`);
516
+ }
517
+
518
+ return child;
519
+ }
520
+
521
+ function findChildElementOptional(
522
+ node: XmlElementNode,
523
+ childLocalName: string,
524
+ ): XmlElementNode | undefined {
525
+ return node.children.find(
526
+ (entry): entry is XmlElementNode =>
527
+ entry.type === "element" && localName(entry.name) === childLocalName,
528
+ );
529
+ }
530
+
531
+ function localName(name: string): string {
532
+ const separatorIndex = name.indexOf(":");
533
+ return separatorIndex >= 0 ? name.slice(separatorIndex + 1) : name;
534
+ }
535
+
536
+ function parseInteger(value: string): number | undefined {
537
+ if (!/^-?\d+$/.test(value)) {
538
+ return undefined;
539
+ }
540
+
541
+ return Number.parseInt(value, 10);
542
+ }
543
+
544
+ function parseXml(xml: string): XmlElementNode {
545
+ const root: XmlElementNode = {
546
+ type: "element",
547
+ name: "__root__",
548
+ attributes: {},
549
+ children: [],
550
+ };
551
+ const stack: XmlElementNode[] = [root];
552
+ let cursor = 0;
553
+
554
+ while (cursor < xml.length) {
555
+ if (xml.startsWith("<!--", cursor)) {
556
+ const end = xml.indexOf("-->", cursor);
557
+ cursor = end >= 0 ? end + 3 : xml.length;
558
+ continue;
559
+ }
560
+
561
+ if (xml.startsWith("<?", cursor)) {
562
+ const end = xml.indexOf("?>", cursor);
563
+ cursor = end >= 0 ? end + 2 : xml.length;
564
+ continue;
565
+ }
566
+
567
+ if (xml.startsWith("<![CDATA[", cursor)) {
568
+ const end = xml.indexOf("]]>", cursor);
569
+ const textEnd = end >= 0 ? end : xml.length;
570
+ stack[stack.length - 1]?.children.push({
571
+ type: "text",
572
+ text: xml.slice(cursor + 9, textEnd),
573
+ });
574
+ cursor = end >= 0 ? end + 3 : xml.length;
575
+ continue;
576
+ }
577
+
578
+ if (xml[cursor] !== "<") {
579
+ const nextTag = xml.indexOf("<", cursor);
580
+ const end = nextTag >= 0 ? nextTag : xml.length;
581
+ const text = decodeXmlEntities(xml.slice(cursor, end));
582
+ if (text.length > 0) {
583
+ stack[stack.length - 1]?.children.push({
584
+ type: "text",
585
+ text,
586
+ });
587
+ }
588
+ cursor = end;
589
+ continue;
590
+ }
591
+
592
+ if (xml[cursor + 1] === "/") {
593
+ const end = xml.indexOf(">", cursor);
594
+ if (end < 0) {
595
+ throw new Error("Malformed XML: missing closing >.");
596
+ }
597
+
598
+ const name = xml.slice(cursor + 2, end).trim();
599
+ const current = stack.pop();
600
+ if (!current || localName(current.name) !== localName(name)) {
601
+ throw new Error(`Malformed XML: unexpected closing tag </${name}>.`);
602
+ }
603
+
604
+ cursor = end + 1;
605
+ continue;
606
+ }
607
+
608
+ const tagEnd = findTagEnd(xml, cursor);
609
+ const tagBody = xml.slice(cursor + 1, tagEnd);
610
+ const selfClosing = /\/\s*$/.test(tagBody);
611
+ const { name, attributes } = parseTag(tagBody.replace(/\/\s*$/, "").trim());
612
+ const element: XmlElementNode = {
613
+ type: "element",
614
+ name,
615
+ attributes,
616
+ children: [],
617
+ };
618
+ stack[stack.length - 1]?.children.push(element);
619
+
620
+ if (!selfClosing) {
621
+ stack.push(element);
622
+ }
623
+
624
+ cursor = tagEnd + 1;
625
+ }
626
+
627
+ if (stack.length !== 1) {
628
+ throw new Error("Malformed XML: unclosed element.");
629
+ }
630
+
631
+ return root;
632
+ }
633
+
634
+ function parseTag(tagBody: string): { name: string; attributes: Record<string, string> } {
635
+ let cursor = 0;
636
+ while (cursor < tagBody.length && /\s/.test(tagBody[cursor] ?? "")) {
637
+ cursor += 1;
638
+ }
639
+
640
+ const nameStart = cursor;
641
+ while (cursor < tagBody.length && !/\s/.test(tagBody[cursor] ?? "")) {
642
+ cursor += 1;
643
+ }
644
+
645
+ const name = tagBody.slice(nameStart, cursor);
646
+ const attributes: Record<string, string> = {};
647
+
648
+ while (cursor < tagBody.length) {
649
+ while (cursor < tagBody.length && /\s/.test(tagBody[cursor] ?? "")) {
650
+ cursor += 1;
651
+ }
652
+ if (cursor >= tagBody.length) {
653
+ break;
654
+ }
655
+
656
+ const keyStart = cursor;
657
+ while (cursor < tagBody.length && !/[\s=]/.test(tagBody[cursor] ?? "")) {
658
+ cursor += 1;
659
+ }
660
+ const key = tagBody.slice(keyStart, cursor);
661
+
662
+ while (cursor < tagBody.length && /\s/.test(tagBody[cursor] ?? "")) {
663
+ cursor += 1;
664
+ }
665
+
666
+ if (tagBody[cursor] !== "=") {
667
+ attributes[key] = "";
668
+ continue;
669
+ }
670
+ cursor += 1;
671
+
672
+ while (cursor < tagBody.length && /\s/.test(tagBody[cursor] ?? "")) {
673
+ cursor += 1;
674
+ }
675
+
676
+ const quote = tagBody[cursor];
677
+ if (quote !== `"` && quote !== `'`) {
678
+ throw new Error(`Malformed XML attribute ${key}.`);
679
+ }
680
+ cursor += 1;
681
+
682
+ const valueStart = cursor;
683
+ while (cursor < tagBody.length && tagBody[cursor] !== quote) {
684
+ cursor += 1;
685
+ }
686
+ const rawValue = tagBody.slice(valueStart, cursor);
687
+ attributes[key] = decodeXmlEntities(rawValue);
688
+ cursor += 1;
689
+ }
690
+
691
+ return { name, attributes };
692
+ }
693
+
694
+ function findTagEnd(xml: string, start: number): number {
695
+ let cursor = start + 1;
696
+ let quote: string | null = null;
697
+
698
+ while (cursor < xml.length) {
699
+ const current = xml[cursor];
700
+ if (quote) {
701
+ if (current === quote) {
702
+ quote = null;
703
+ }
704
+ cursor += 1;
705
+ continue;
706
+ }
707
+
708
+ if (current === `"` || current === `'`) {
709
+ quote = current;
710
+ cursor += 1;
711
+ continue;
712
+ }
713
+
714
+ if (current === ">") {
715
+ return cursor;
716
+ }
717
+
718
+ cursor += 1;
719
+ }
720
+
721
+ throw new Error("Malformed XML: missing >.");
722
+ }
723
+
724
+ function decodeXmlEntities(value: string): string {
725
+ return value.replace(/&(#x[0-9a-fA-F]+|#\d+|amp|lt|gt|quot|apos);/g, (match, entity) => {
726
+ switch (entity) {
727
+ case "amp":
728
+ return "&";
729
+ case "lt":
730
+ return "<";
731
+ case "gt":
732
+ return ">";
733
+ case "quot":
734
+ return `"`;
735
+ case "apos":
736
+ return "'";
737
+ default:
738
+ if (entity.startsWith("#x")) {
739
+ return String.fromCodePoint(Number.parseInt(entity.slice(2), 16));
740
+ }
741
+ if (entity.startsWith("#")) {
742
+ return String.fromCodePoint(Number.parseInt(entity.slice(1), 10));
743
+ }
744
+ return match;
745
+ }
746
+ });
747
+ }