@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,258 @@
1
+ import type {
2
+ EditorSurfaceSnapshot,
3
+ SurfaceBlockSnapshot,
4
+ } from "../api/public-types";
5
+
6
+ export interface VirtualizedRenderingMetrics {
7
+ blockId: string;
8
+ index: number;
9
+ top: number;
10
+ height: number;
11
+ bottom: number;
12
+ }
13
+
14
+ export interface VirtualizedRenderingSession {
15
+ blocks: SurfaceBlockSnapshot[];
16
+ metrics: VirtualizedRenderingMetrics[];
17
+ totalHeight: number;
18
+ averageBlockHeight: number;
19
+ estimatedPageCount: number;
20
+ }
21
+
22
+ export interface CreateVirtualizedRenderingSessionOptions {
23
+ averagePageHeight?: number;
24
+ paragraphBaseHeight?: number;
25
+ paragraphLineHeight?: number;
26
+ segmentWrapWidth?: number;
27
+ tableRowHeight?: number;
28
+ opaqueBlockHeight?: number;
29
+ sdtPaddingHeight?: number;
30
+ }
31
+
32
+ export interface VirtualizedViewport {
33
+ scrollTop: number;
34
+ viewportHeight: number;
35
+ overscanHeight?: number;
36
+ }
37
+
38
+ export interface VirtualizedWindow {
39
+ startIndex: number;
40
+ endIndex: number;
41
+ totalHeight: number;
42
+ offsetTop: number;
43
+ offsetBottom: number;
44
+ visibleBlocks: SurfaceBlockSnapshot[];
45
+ visibleMetrics: VirtualizedRenderingMetrics[];
46
+ }
47
+
48
+ const DEFAULT_PAGE_HEIGHT = 1120;
49
+ const DEFAULT_PARAGRAPH_BASE_HEIGHT = 28;
50
+ const DEFAULT_PARAGRAPH_LINE_HEIGHT = 18;
51
+ const DEFAULT_SEGMENT_WRAP_WIDTH = 96;
52
+ const DEFAULT_TABLE_ROW_HEIGHT = 44;
53
+ const DEFAULT_OPAQUE_BLOCK_HEIGHT = 72;
54
+ const DEFAULT_SDT_PADDING_HEIGHT = 24;
55
+
56
+ export function createVirtualizedRenderingSession(
57
+ surface: EditorSurfaceSnapshot,
58
+ previous: VirtualizedRenderingSession | undefined = undefined,
59
+ options: CreateVirtualizedRenderingSessionOptions = {},
60
+ ): VirtualizedRenderingSession {
61
+ const blocks = surface.blocks;
62
+ const metrics = previous?.metrics ?? [];
63
+
64
+ if (metrics.length > blocks.length) {
65
+ metrics.length = blocks.length;
66
+ }
67
+
68
+ let cursor = 0;
69
+ for (let index = 0; index < blocks.length; index += 1) {
70
+ const block = blocks[index]!;
71
+ const height = estimateBlockHeight(block, options);
72
+ const top = cursor;
73
+ const bottom = top + height;
74
+ const metric = metrics[index];
75
+ if (metric) {
76
+ metric.blockId = block.blockId;
77
+ metric.index = index;
78
+ metric.top = top;
79
+ metric.height = height;
80
+ metric.bottom = bottom;
81
+ } else {
82
+ metrics.push({
83
+ blockId: block.blockId,
84
+ index,
85
+ top,
86
+ height,
87
+ bottom,
88
+ });
89
+ }
90
+ cursor = bottom;
91
+ }
92
+
93
+ const totalHeight = cursor;
94
+ const averageBlockHeight = blocks.length > 0 ? totalHeight / blocks.length : 0;
95
+ const estimatedPageCount =
96
+ totalHeight > 0
97
+ ? Math.max(
98
+ 1,
99
+ Math.ceil(totalHeight / (options.averagePageHeight ?? DEFAULT_PAGE_HEIGHT)),
100
+ )
101
+ : 0;
102
+
103
+ if (previous) {
104
+ previous.blocks = blocks;
105
+ previous.metrics = metrics;
106
+ previous.totalHeight = totalHeight;
107
+ previous.averageBlockHeight = averageBlockHeight;
108
+ previous.estimatedPageCount = estimatedPageCount;
109
+ return previous;
110
+ }
111
+
112
+ return {
113
+ blocks,
114
+ metrics,
115
+ totalHeight,
116
+ averageBlockHeight,
117
+ estimatedPageCount,
118
+ };
119
+ }
120
+
121
+ export function computeVirtualizedWindow(
122
+ session: VirtualizedRenderingSession,
123
+ viewport: VirtualizedViewport,
124
+ ): VirtualizedWindow {
125
+ if (session.metrics.length === 0) {
126
+ return {
127
+ startIndex: 0,
128
+ endIndex: 0,
129
+ totalHeight: 0,
130
+ offsetTop: 0,
131
+ offsetBottom: 0,
132
+ visibleBlocks: [],
133
+ visibleMetrics: [],
134
+ };
135
+ }
136
+
137
+ const overscanHeight = Math.max(0, viewport.overscanHeight ?? viewport.viewportHeight * 0.5);
138
+ const windowTop = Math.max(0, viewport.scrollTop - overscanHeight);
139
+ const windowBottom = Math.min(
140
+ session.totalHeight,
141
+ viewport.scrollTop + viewport.viewportHeight + overscanHeight,
142
+ );
143
+ const startIndex = findFirstVisibleIndex(session.metrics, windowTop);
144
+ const endIndexExclusive = findLastVisibleIndex(session.metrics, windowBottom) + 1;
145
+ const safeEndIndex = Math.max(startIndex + 1, endIndexExclusive);
146
+ const visibleMetrics = session.metrics.slice(startIndex, safeEndIndex);
147
+ const visibleBlocks = session.blocks.slice(startIndex, safeEndIndex);
148
+ const lastMetric = visibleMetrics[visibleMetrics.length - 1]!;
149
+
150
+ return {
151
+ startIndex,
152
+ endIndex: safeEndIndex,
153
+ totalHeight: session.totalHeight,
154
+ offsetTop: visibleMetrics[0]?.top ?? 0,
155
+ offsetBottom: Math.max(0, session.totalHeight - lastMetric.bottom),
156
+ visibleBlocks,
157
+ visibleMetrics,
158
+ };
159
+ }
160
+
161
+ function findFirstVisibleIndex(
162
+ metrics: VirtualizedRenderingMetrics[],
163
+ top: number,
164
+ ): number {
165
+ let low = 0;
166
+ let high = metrics.length - 1;
167
+ let result = 0;
168
+
169
+ while (low <= high) {
170
+ const middle = (low + high) >> 1;
171
+ const metric = metrics[middle]!;
172
+ if (metric.bottom >= top) {
173
+ result = middle;
174
+ high = middle - 1;
175
+ } else {
176
+ low = middle + 1;
177
+ }
178
+ }
179
+
180
+ return result;
181
+ }
182
+
183
+ function findLastVisibleIndex(
184
+ metrics: VirtualizedRenderingMetrics[],
185
+ bottom: number,
186
+ ): number {
187
+ let low = 0;
188
+ let high = metrics.length - 1;
189
+ let result = metrics.length - 1;
190
+
191
+ while (low <= high) {
192
+ const middle = (low + high) >> 1;
193
+ const metric = metrics[middle]!;
194
+ if (metric.top <= bottom) {
195
+ result = middle;
196
+ low = middle + 1;
197
+ } else {
198
+ high = middle - 1;
199
+ }
200
+ }
201
+
202
+ return result;
203
+ }
204
+
205
+ function estimateBlockHeight(
206
+ block: SurfaceBlockSnapshot,
207
+ options: CreateVirtualizedRenderingSessionOptions,
208
+ ): number {
209
+ switch (block.kind) {
210
+ case "paragraph":
211
+ return estimateParagraphHeight(block, options);
212
+ case "table":
213
+ return Math.max(
214
+ options.tableRowHeight ?? DEFAULT_TABLE_ROW_HEIGHT,
215
+ block.rows.length * (options.tableRowHeight ?? DEFAULT_TABLE_ROW_HEIGHT),
216
+ );
217
+ case "opaque_block":
218
+ return options.opaqueBlockHeight ?? DEFAULT_OPAQUE_BLOCK_HEIGHT;
219
+ case "sdt_block":
220
+ return (
221
+ (options.sdtPaddingHeight ?? DEFAULT_SDT_PADDING_HEIGHT) +
222
+ block.children.reduce(
223
+ (total, child) => total + estimateBlockHeight(child, options),
224
+ 0,
225
+ )
226
+ );
227
+ default:
228
+ return options.paragraphBaseHeight ?? DEFAULT_PARAGRAPH_BASE_HEIGHT;
229
+ }
230
+ }
231
+
232
+ function estimateParagraphHeight(
233
+ block: Extract<SurfaceBlockSnapshot, { kind: "paragraph" }>,
234
+ options: CreateVirtualizedRenderingSessionOptions,
235
+ ): number {
236
+ const wrapWidth = options.segmentWrapWidth ?? DEFAULT_SEGMENT_WRAP_WIDTH;
237
+ let consumedCharacters = 0;
238
+
239
+ for (const segment of block.segments) {
240
+ switch (segment.kind) {
241
+ case "text":
242
+ consumedCharacters += segment.text.length;
243
+ break;
244
+ case "image":
245
+ consumedCharacters += wrapWidth;
246
+ break;
247
+ default:
248
+ consumedCharacters += 4;
249
+ break;
250
+ }
251
+ }
252
+
253
+ const estimatedLines = Math.max(1, Math.ceil(consumedCharacters / wrapWidth));
254
+ return (
255
+ (options.paragraphBaseHeight ?? DEFAULT_PARAGRAPH_BASE_HEIGHT) +
256
+ estimatedLines * (options.paragraphLineHeight ?? DEFAULT_PARAGRAPH_LINE_HEIGHT)
257
+ );
258
+ }
@@ -0,0 +1,393 @@
1
+ import type {
2
+ CommentSidebarSnapshot,
3
+ EditorAnchorProjection,
4
+ EditorStoryTarget,
5
+ FieldSnapshot,
6
+ ProtectionSnapshot,
7
+ RuntimeRenderSnapshot,
8
+ SurfaceBlockSnapshot,
9
+ SurfaceInlineSegment,
10
+ TrackedChangesSnapshot,
11
+ WorkflowCandidateRange,
12
+ WorkflowCandidateRangeOptions,
13
+ WorkflowFieldMarkup,
14
+ WorkflowHighlightMarkup,
15
+ WorkflowMetadataMarkup,
16
+ WorkflowMetadataSnapshot,
17
+ WorkflowMarkupItem,
18
+ WorkflowMarkupSnapshot,
19
+ WorkflowOpaqueFragmentMarkup,
20
+ WorkflowProtectedRangeMarkup,
21
+ WorkflowRevisionMarkup,
22
+ WorkflowCommentMarkup,
23
+ } from "../api/public-types";
24
+ import { MAIN_STORY_TARGET } from "../core/selection/mapping.ts";
25
+ import { searchSurfaceBlocks } from "../core/search/search-text.ts";
26
+ import { describeOpaqueFragment, isBlockedImportFeatureKey } from "../preservation/store.ts";
27
+ import type { CanonicalDocumentEnvelope } from "../core/state/editor-state.ts";
28
+
29
+ export function collectWorkflowMarkupSnapshot(input: {
30
+ renderSnapshot: RuntimeRenderSnapshot;
31
+ fieldSnapshot: FieldSnapshot;
32
+ protectionSnapshot: ProtectionSnapshot;
33
+ preservation: CanonicalDocumentEnvelope["preservation"];
34
+ workflowMetadataSnapshot?: WorkflowMetadataSnapshot;
35
+ }): WorkflowMarkupSnapshot {
36
+ const highlights: WorkflowHighlightMarkup[] = [];
37
+ const metadata = collectWorkflowMetadataMarkup(input.workflowMetadataSnapshot);
38
+ const fields: WorkflowFieldMarkup[] = [];
39
+ const opaqueFragments: WorkflowOpaqueFragmentMarkup[] = [];
40
+ const surface = input.renderSnapshot.surface;
41
+
42
+ if (surface) {
43
+ collectSurfaceMarkup(
44
+ surface.blocks,
45
+ MAIN_STORY_TARGET,
46
+ input.preservation,
47
+ highlights,
48
+ opaqueFragments,
49
+ );
50
+ for (const story of surface.secondaryStories) {
51
+ collectSurfaceMarkup(
52
+ story.blocks,
53
+ story.target,
54
+ input.preservation,
55
+ highlights,
56
+ opaqueFragments,
57
+ );
58
+ }
59
+
60
+ fields.push(...collectFieldMarkup(surface, input.fieldSnapshot));
61
+ }
62
+ opaqueFragments.push(...collectOpaqueFragmentMarkup(input.preservation, opaqueFragments));
63
+
64
+ const comments = input.renderSnapshot.comments.threads.map((thread): WorkflowCommentMarkup => ({
65
+ markupId: `comment:${thread.commentId}`,
66
+ kind: "comment",
67
+ commentId: thread.commentId,
68
+ anchor: thread.anchor,
69
+ label: thread.anchorLabel,
70
+ excerpt: thread.excerpt,
71
+ status: thread.status,
72
+ warningCount: thread.warningCount,
73
+ entryCount: thread.entryCount,
74
+ createdAt: thread.createdAt,
75
+ createdBy: thread.createdBy,
76
+ }));
77
+
78
+ const revisions = input.renderSnapshot.trackedChanges.revisions.map(
79
+ (revision): WorkflowRevisionMarkup => ({
80
+ markupId: `revision:${revision.revisionId}`,
81
+ kind: "revision",
82
+ revisionId: revision.revisionId,
83
+ revisionKind: revision.kind,
84
+ anchor: revision.anchor,
85
+ label: revision.label,
86
+ excerpt: revision.excerpt,
87
+ status: revision.status,
88
+ actionability: revision.actionability,
89
+ authorId: revision.authorId,
90
+ createdAt: revision.createdAt,
91
+ preserveOnlyReason: revision.preserveOnlyReason,
92
+ }),
93
+ );
94
+
95
+ const protectedRanges = input.protectionSnapshot.ranges
96
+ .filter(
97
+ (range): range is typeof range & { start: number; end: number } =>
98
+ typeof range.start === "number" && typeof range.end === "number",
99
+ )
100
+ .map(
101
+ (range): WorkflowProtectedRangeMarkup => ({
102
+ markupId: `protected-range:${range.rangeId}`,
103
+ kind: "protected_range",
104
+ rangeId: range.rangeId,
105
+ anchor: createRangeAnchor(range.start, range.end),
106
+ storyTarget: MAIN_STORY_TARGET,
107
+ label: `Protected range ${range.rangeId}`,
108
+ excerpt: range.enforcementReason,
109
+ enforced: range.enforced,
110
+ enforcementReason: range.enforcementReason,
111
+ editorGroup: range.editorGroup,
112
+ editor: range.editor,
113
+ }),
114
+ );
115
+
116
+ const items: WorkflowMarkupItem[] = [
117
+ ...highlights,
118
+ ...metadata,
119
+ ...comments,
120
+ ...revisions,
121
+ ...fields,
122
+ ...protectedRanges,
123
+ ...opaqueFragments,
124
+ ];
125
+
126
+ return {
127
+ totalCount: items.length,
128
+ items,
129
+ highlights,
130
+ metadata,
131
+ comments,
132
+ revisions,
133
+ fields,
134
+ protectedRanges,
135
+ opaqueFragments,
136
+ };
137
+ }
138
+
139
+ function collectWorkflowMetadataMarkup(
140
+ snapshot?: WorkflowMetadataSnapshot,
141
+ ): WorkflowMetadataMarkup[] {
142
+ if (!snapshot) {
143
+ return [];
144
+ }
145
+
146
+ const definitionsById = new Map(
147
+ snapshot.definitions.map((definition) => [definition.metadataId, definition] as const),
148
+ );
149
+
150
+ return snapshot.entries.flatMap((entry) => {
151
+ const definition = definitionsById.get(entry.metadataId);
152
+ if (!definition) {
153
+ return [];
154
+ }
155
+
156
+ return [{
157
+ markupId: `metadata:${entry.entryId}`,
158
+ kind: "metadata",
159
+ entryId: entry.entryId,
160
+ metadataId: entry.metadataId,
161
+ anchor: entry.anchor,
162
+ storyTarget: entry.storyTarget,
163
+ label: definition.label,
164
+ excerpt: definition.kind,
165
+ color: definition.color,
166
+ icon: definition.icon,
167
+ persistence: definition.persistence,
168
+ value: entry.value,
169
+ scopeId: entry.scopeId,
170
+ workItemId: entry.workItemId,
171
+ } satisfies WorkflowMetadataMarkup];
172
+ });
173
+ }
174
+
175
+ export function deriveWorkflowCandidateRangesFromMarkup(
176
+ snapshot: WorkflowMarkupSnapshot,
177
+ options: WorkflowCandidateRangeOptions = {},
178
+ ): WorkflowCandidateRange[] {
179
+ const allowedKinds = options.kinds ? new Set(options.kinds) : null;
180
+ const includeDetached = options.includeDetached === true;
181
+
182
+ return snapshot.items
183
+ .filter((item) => (allowedKinds ? allowedKinds.has(item.kind) : true))
184
+ .filter((item) => includeDetached || item.anchor.kind !== "detached")
185
+ .map(
186
+ (item): WorkflowCandidateRange => ({
187
+ candidateId: item.markupId,
188
+ storyTarget: item.storyTarget,
189
+ anchor: item.anchor,
190
+ label: item.label,
191
+ source: options.source ?? "host",
192
+ }),
193
+ );
194
+ }
195
+
196
+ function collectSurfaceMarkup(
197
+ blocks: readonly SurfaceBlockSnapshot[],
198
+ storyTarget: EditorStoryTarget,
199
+ preservation: CanonicalDocumentEnvelope["preservation"],
200
+ highlights: WorkflowHighlightMarkup[],
201
+ opaqueFragments: WorkflowOpaqueFragmentMarkup[],
202
+ ): void {
203
+ for (const block of blocks) {
204
+ if (block.kind === "paragraph") {
205
+ for (const segment of block.segments) {
206
+ collectSegmentMarkup(segment, storyTarget, preservation, highlights, opaqueFragments);
207
+ }
208
+ continue;
209
+ }
210
+
211
+ if (block.kind === "table") {
212
+ for (const row of block.rows) {
213
+ for (const cell of row.cells) {
214
+ collectSurfaceMarkup(cell.content, storyTarget, preservation, highlights, opaqueFragments);
215
+ }
216
+ }
217
+ continue;
218
+ }
219
+
220
+ if (block.kind === "sdt_block") {
221
+ collectSurfaceMarkup(block.children, storyTarget, preservation, highlights, opaqueFragments);
222
+ continue;
223
+ }
224
+
225
+ const fragment = preservation.opaqueFragments[block.fragmentId];
226
+ const descriptor = fragment ? describeOpaqueFragment(fragment) : null;
227
+ opaqueFragments.push({
228
+ markupId: `opaque:${block.fragmentId}`,
229
+ kind: "opaque_fragment",
230
+ fragmentId: block.fragmentId,
231
+ warningId: block.warningId,
232
+ anchor: createRangeAnchor(block.from, block.to),
233
+ storyTarget,
234
+ label: block.label,
235
+ excerpt: block.detail,
236
+ detail: block.detail,
237
+ blockedReasonCode:
238
+ block.blockedReasonCode ??
239
+ (fragment && descriptor && isBlockedImportFeatureKey(descriptor.featureKey)
240
+ ? "workflow_blocked_import"
241
+ : "workflow_preserve_only"),
242
+ });
243
+ }
244
+ }
245
+
246
+ function collectSegmentMarkup(
247
+ segment: SurfaceInlineSegment,
248
+ storyTarget: EditorStoryTarget,
249
+ preservation: CanonicalDocumentEnvelope["preservation"],
250
+ highlights: WorkflowHighlightMarkup[],
251
+ opaqueFragments: WorkflowOpaqueFragmentMarkup[],
252
+ ): void {
253
+ if (segment.kind === "text" && segment.markAttrs?.backgroundColor) {
254
+ highlights.push({
255
+ markupId: `highlight:${storyTargetKey(storyTarget)}:${segment.from}:${segment.to}:${segment.markAttrs.backgroundColor}`,
256
+ kind: "highlight",
257
+ anchor: createRangeAnchor(segment.from, segment.to),
258
+ storyTarget,
259
+ label: `Highlight ${segment.markAttrs.backgroundColor}`,
260
+ excerpt: segment.text,
261
+ color: segment.markAttrs.backgroundColor,
262
+ text: segment.text,
263
+ });
264
+ return;
265
+ }
266
+
267
+ if (segment.kind === "opaque_inline") {
268
+ const fragment = preservation.opaqueFragments[segment.fragmentId];
269
+ const descriptor = fragment ? describeOpaqueFragment(fragment) : null;
270
+ opaqueFragments.push({
271
+ markupId: `opaque:${segment.fragmentId}`,
272
+ kind: "opaque_fragment",
273
+ fragmentId: segment.fragmentId,
274
+ warningId: segment.warningId,
275
+ anchor: createRangeAnchor(segment.from, segment.to),
276
+ storyTarget,
277
+ label: segment.label,
278
+ excerpt: segment.detail,
279
+ detail: segment.detail,
280
+ blockedReasonCode:
281
+ segment.blockedReasonCode ??
282
+ (fragment && descriptor && isBlockedImportFeatureKey(descriptor.featureKey)
283
+ ? "workflow_blocked_import"
284
+ : "workflow_preserve_only"),
285
+ });
286
+ }
287
+ }
288
+
289
+ function collectFieldMarkup(
290
+ surface: RuntimeRenderSnapshot["surface"],
291
+ fieldSnapshot: FieldSnapshot,
292
+ ): WorkflowFieldMarkup[] {
293
+ if (!surface) {
294
+ return [];
295
+ }
296
+
297
+ const stories = [
298
+ { blocks: surface.blocks, storyTarget: MAIN_STORY_TARGET },
299
+ ...surface.secondaryStories.map((story) => ({
300
+ blocks: story.blocks,
301
+ storyTarget: story.target,
302
+ })),
303
+ ];
304
+
305
+ return fieldSnapshot.fields.flatMap((field) => {
306
+ const displayText = field.displayText.trim();
307
+ if (!displayText) {
308
+ return [];
309
+ }
310
+
311
+ for (const story of stories) {
312
+ const matches = searchSurfaceBlocks(story.blocks, displayText, { limit: 2 });
313
+ if (matches.length === 1) {
314
+ const match = matches[0]!;
315
+ return [
316
+ {
317
+ markupId: `field:${field.index}`,
318
+ kind: "field",
319
+ anchor: createRangeAnchor(match.from, match.to),
320
+ storyTarget: story.storyTarget,
321
+ label: displayText,
322
+ excerpt: field.instruction,
323
+ fieldIndex: field.index,
324
+ fieldFamily: field.fieldFamily,
325
+ fieldTarget: field.fieldTarget,
326
+ refreshStatus: field.refreshStatus,
327
+ displayText: field.displayText,
328
+ } satisfies WorkflowFieldMarkup,
329
+ ];
330
+ }
331
+ }
332
+
333
+ return [];
334
+ });
335
+ }
336
+
337
+ function collectOpaqueFragmentMarkup(
338
+ preservation: CanonicalDocumentEnvelope["preservation"],
339
+ existing: WorkflowOpaqueFragmentMarkup[],
340
+ ): WorkflowOpaqueFragmentMarkup[] {
341
+ const seen = new Set(existing.map((item) => item.fragmentId));
342
+
343
+ return Object.values(preservation.opaqueFragments)
344
+ .filter(
345
+ (fragment) =>
346
+ !seen.has(fragment.fragmentId)
347
+ && fragment.packagePartName === "/word/document.xml",
348
+ )
349
+ .map((fragment) => {
350
+ const descriptor = describeOpaqueFragment(fragment);
351
+ const blockedReasonCode = isBlockedImportFeatureKey(descriptor.featureKey)
352
+ ? "workflow_blocked_import"
353
+ : "workflow_preserve_only";
354
+
355
+ return {
356
+ markupId: `opaque:${fragment.fragmentId}`,
357
+ kind: "opaque_fragment",
358
+ fragmentId: fragment.fragmentId,
359
+ warningId: fragment.warningId,
360
+ anchor: createRangeAnchor(fragment.lastKnownRange.from, fragment.lastKnownRange.to),
361
+ storyTarget: MAIN_STORY_TARGET,
362
+ label: descriptor.label,
363
+ excerpt: descriptor.detail,
364
+ detail: descriptor.detail,
365
+ blockedReasonCode,
366
+ } satisfies WorkflowOpaqueFragmentMarkup;
367
+ });
368
+ }
369
+
370
+ function createRangeAnchor(from: number, to: number): EditorAnchorProjection {
371
+ return {
372
+ kind: "range",
373
+ from,
374
+ to,
375
+ assoc: {
376
+ start: -1,
377
+ end: 1,
378
+ },
379
+ };
380
+ }
381
+
382
+ function storyTargetKey(storyTarget: EditorStoryTarget): string {
383
+ switch (storyTarget.kind) {
384
+ case "main":
385
+ return "main";
386
+ case "header":
387
+ case "footer":
388
+ return `${storyTarget.kind}:${storyTarget.relationshipId}:${storyTarget.variant}:${storyTarget.sectionIndex ?? "none"}`;
389
+ case "footnote":
390
+ case "endnote":
391
+ return `${storyTarget.kind}:${storyTarget.noteId}`;
392
+ }
393
+ }
@@ -0,0 +1,30 @@
1
+ # UI
2
+
3
+ This directory contains the public entry point and shared headless logic for the editor UI.
4
+
5
+ ## Entry Point
6
+
7
+ `WordReviewEditor.tsx` is the public component. It bridges the `DocumentRuntime` to the Tailwind rendering layer in `src/ui-tailwind/`.
8
+
9
+ ## Headless Logic
10
+
11
+ `headless/` contains pure, framework-free utilities shared across implementations:
12
+
13
+ - `use-editor-keyboard.ts` — Keyboard event handler factory
14
+ - `comment-decoration-model.ts` — Comment anchor → highlight class computation
15
+ - `revision-decoration-model.ts` — Revision anchor → highlight class computation
16
+ - `selection-helpers.ts` — Selection snapshot utilities
17
+
18
+ ## Session Capabilities
19
+
20
+ `src/runtime/session-capabilities.ts` derives all UI control states from `RuntimeRenderSnapshot`. The `deriveCapabilities(snapshot, reviewMode)` function is called on every render in `WordReviewEditor.tsx` and the result is passed to the workspace components.
21
+
22
+ ## Shared Utilities
23
+
24
+ `shared/revision-filters.ts` — Shared revision filtering and empty state helpers.
25
+
26
+ ## Rendering
27
+
28
+ All visual rendering is in `src/ui-tailwind/`. See `docs/reference/word-review-editor-frontend-architecture.md` for the canonical architecture.
29
+
30
+ Legacy inline-CSSProperties components have been removed.