@beyondwork/docx-react-component 1.0.29 → 1.0.30

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (381) hide show
  1. package/package.json +65 -96
  2. package/src/README.md +85 -0
  3. package/src/api/README.md +26 -0
  4. package/src/api/public-types.ts +1952 -0
  5. package/src/api/session-state.ts +62 -0
  6. package/src/compare/diff-engine.ts +623 -0
  7. package/src/compare/export-redlines.ts +280 -0
  8. package/src/compare/index.ts +25 -0
  9. package/src/compare/snapshot.ts +97 -0
  10. package/src/component-inventory.md +99 -0
  11. package/src/core/README.md +10 -0
  12. package/src/core/commands/README.md +3 -0
  13. package/{dist/chunk-TJBP2K4T.js → src/core/commands/formatting-commands.ts} +536 -196
  14. package/src/core/commands/image-commands.ts +373 -0
  15. package/src/core/commands/index.ts +1879 -0
  16. package/src/core/commands/list-commands.ts +565 -0
  17. package/src/core/commands/paragraph-layout-commands.ts +339 -0
  18. package/src/core/commands/review-commands.ts +108 -0
  19. package/{dist/core/commands/section-layout-commands.cjs → src/core/commands/section-layout-commands.ts} +340 -137
  20. package/src/core/commands/structural-helpers.ts +309 -0
  21. package/{dist/core/commands/style-commands.cjs → src/core/commands/style-commands.ts} +113 -65
  22. package/src/core/commands/table-structure-commands.ts +854 -0
  23. package/{dist/chunk-UZXBISGO.js → src/core/commands/text-commands.ts} +142 -86
  24. package/src/core/schema/README.md +3 -0
  25. package/src/core/schema/text-schema.ts +516 -0
  26. package/src/core/search/search-text.ts +357 -0
  27. package/src/core/selection/README.md +3 -0
  28. package/src/core/selection/mapping.ts +289 -0
  29. package/src/core/selection/review-anchors.ts +183 -0
  30. package/src/core/state/README.md +3 -0
  31. package/src/core/state/editor-state.ts +892 -0
  32. package/src/core/state/text-transaction.ts +869 -0
  33. package/src/formats/xlsx/io/parse-shared-strings.ts +41 -0
  34. package/src/formats/xlsx/io/parse-sheet.ts +459 -0
  35. package/src/formats/xlsx/io/parse-styles.ts +59 -0
  36. package/src/formats/xlsx/io/parse-workbook.ts +75 -0
  37. package/src/formats/xlsx/io/serialize-shared-strings.ts +72 -0
  38. package/src/formats/xlsx/io/serialize-sheet.ts +333 -0
  39. package/src/formats/xlsx/io/serialize-styles.ts +98 -0
  40. package/src/formats/xlsx/io/serialize-workbook.ts +429 -0
  41. package/src/formats/xlsx/io/xlsx-session.ts +314 -0
  42. package/src/formats/xlsx/model/cell.ts +189 -0
  43. package/src/formats/xlsx/model/sheet.ts +326 -0
  44. package/src/formats/xlsx/model/styles.ts +118 -0
  45. package/src/formats/xlsx/model/workbook.ts +453 -0
  46. package/src/formats/xlsx/runtime/cell-commands.ts +567 -0
  47. package/src/formats/xlsx/runtime/sheet-commands.ts +206 -0
  48. package/src/formats/xlsx/runtime/workbook-runtime.ts +177 -0
  49. package/src/formats/xlsx/runtime/workbook-transaction.ts +822 -0
  50. package/src/index.ts +142 -0
  51. package/src/io/README.md +10 -0
  52. package/src/io/docx-session.ts +3175 -0
  53. package/src/io/export/README.md +3 -0
  54. package/src/io/export/export-session.ts +220 -0
  55. package/src/io/export/minimal-docx.ts +115 -0
  56. package/src/io/export/reattach-preserved-parts.ts +54 -0
  57. package/src/io/export/serialize-comments.ts +947 -0
  58. package/src/io/export/serialize-footnotes.ts +394 -0
  59. package/src/io/export/serialize-headers-footers.ts +368 -0
  60. package/src/io/export/serialize-main-document.ts +1342 -0
  61. package/src/io/export/serialize-numbering.ts +218 -0
  62. package/src/io/export/serialize-revisions.ts +389 -0
  63. package/src/io/export/serialize-runtime-revisions.ts +463 -0
  64. package/src/io/export/serialize-tables.ts +174 -0
  65. package/src/io/export/split-review-boundaries.ts +356 -0
  66. package/src/io/export/split-story-blocks-for-runtime-revisions.ts +252 -0
  67. package/src/io/export/table-properties-xml.ts +318 -0
  68. package/src/io/normalize/README.md +3 -0
  69. package/src/io/normalize/normalize-text.ts +670 -0
  70. package/src/io/ooxml/README.md +3 -0
  71. package/src/io/ooxml/highlight-colors.ts +39 -0
  72. package/src/io/ooxml/numbering-sentinels.ts +44 -0
  73. package/src/io/ooxml/parse-comments.ts +852 -0
  74. package/src/io/ooxml/parse-complex-content.ts +287 -0
  75. package/src/io/ooxml/parse-fields.ts +834 -0
  76. package/src/io/ooxml/parse-footnotes.ts +952 -0
  77. package/src/io/ooxml/parse-headers-footers.ts +1212 -0
  78. package/src/io/ooxml/parse-inline-media.ts +461 -0
  79. package/src/io/ooxml/parse-main-document.ts +2947 -0
  80. package/src/io/ooxml/parse-numbering.ts +747 -0
  81. package/src/io/ooxml/parse-revisions.ts +1045 -0
  82. package/src/io/ooxml/parse-settings.ts +184 -0
  83. package/src/io/ooxml/parse-shapes.ts +296 -0
  84. package/src/io/ooxml/parse-styles.ts +639 -0
  85. package/src/io/ooxml/parse-tables.ts +627 -0
  86. package/src/io/ooxml/parse-theme.ts +346 -0
  87. package/src/io/ooxml/part-manifest.ts +136 -0
  88. package/src/io/ooxml/revision-boundaries.ts +475 -0
  89. package/src/io/ooxml/workflow-payload.ts +544 -0
  90. package/src/io/opc/README.md +3 -0
  91. package/src/io/opc/corrupt-package.ts +166 -0
  92. package/src/io/opc/docx-package.ts +74 -0
  93. package/src/io/opc/package-reader.ts +325 -0
  94. package/src/io/opc/package-writer.ts +273 -0
  95. package/src/io/source-package-provenance.ts +241 -0
  96. package/{dist/chunk-RMH72RZI.js → src/legal/bookmarks.ts} +130 -44
  97. package/src/legal/cross-references.ts +414 -0
  98. package/src/legal/defined-terms.ts +203 -0
  99. package/src/legal/index.ts +32 -0
  100. package/src/legal/signature-blocks.ts +259 -0
  101. package/src/model/README.md +3 -0
  102. package/src/model/canonical-document.ts +2722 -0
  103. package/src/model/cds-1.0.0.ts +212 -0
  104. package/src/model/snapshot.ts +760 -0
  105. package/src/preservation/README.md +3 -0
  106. package/src/preservation/markup-compatibility.ts +48 -0
  107. package/src/preservation/opaque-fragment-store.ts +89 -0
  108. package/src/preservation/opaque-region.ts +233 -0
  109. package/src/preservation/package-preservation.ts +113 -0
  110. package/src/preservation/preserved-part-manifest.ts +56 -0
  111. package/src/preservation/relationship-retention.ts +57 -0
  112. package/src/preservation/store.ts +255 -0
  113. package/src/review/README.md +16 -0
  114. package/src/review/store/README.md +3 -0
  115. package/src/review/store/comment-anchors.ts +70 -0
  116. package/src/review/store/comment-remapping.ts +154 -0
  117. package/src/review/store/comment-store.ts +349 -0
  118. package/src/review/store/comment-thread.ts +109 -0
  119. package/src/review/store/revision-actions.ts +423 -0
  120. package/src/review/store/revision-store.ts +323 -0
  121. package/src/review/store/revision-types.ts +182 -0
  122. package/src/review/store/runtime-comment-store.ts +43 -0
  123. package/src/runtime/README.md +3 -0
  124. package/src/runtime/ai-action-policy.ts +764 -0
  125. package/src/runtime/context-analytics.ts +824 -0
  126. package/src/runtime/document-layout.ts +332 -0
  127. package/src/runtime/document-locations.ts +521 -0
  128. package/src/runtime/document-navigation.ts +616 -0
  129. package/src/runtime/document-outline.ts +440 -0
  130. package/src/runtime/document-runtime.ts +4055 -0
  131. package/src/runtime/document-search.ts +145 -0
  132. package/src/runtime/event-refresh-hints.ts +137 -0
  133. package/src/runtime/numbering-prefix.ts +244 -0
  134. package/src/runtime/page-layout-estimation.ts +305 -0
  135. package/src/runtime/read-only-diagnostics-runtime.ts +241 -0
  136. package/src/runtime/resolved-numbering-geometry.ts +293 -0
  137. package/src/runtime/review-runtime.ts +44 -0
  138. package/src/runtime/revision-runtime.ts +107 -0
  139. package/src/runtime/session-capabilities.ts +192 -0
  140. package/src/runtime/story-context.ts +164 -0
  141. package/src/runtime/story-targeting.ts +162 -0
  142. package/src/runtime/suggestions-snapshot.ts +137 -0
  143. package/src/runtime/surface-projection.ts +1553 -0
  144. package/src/runtime/table-commands.ts +173 -0
  145. package/src/runtime/table-schema.ts +309 -0
  146. package/src/runtime/table-style-resolver.ts +409 -0
  147. package/src/runtime/view-state.ts +493 -0
  148. package/src/runtime/virtualized-rendering.ts +258 -0
  149. package/src/runtime/workflow-markup.ts +393 -0
  150. package/src/ui/README.md +30 -0
  151. package/src/ui/WordReviewEditor.tsx +5268 -0
  152. package/src/ui/browser-export.ts +52 -0
  153. package/src/ui/comments/README.md +3 -0
  154. package/src/ui/compatibility/README.md +3 -0
  155. package/src/ui/editor-command-bag.ts +127 -0
  156. package/src/ui/editor-runtime-boundary.ts +1558 -0
  157. package/src/ui/editor-shell-view.tsx +144 -0
  158. package/src/ui/editor-surface/README.md +3 -0
  159. package/src/ui/editor-surface-controller.tsx +66 -0
  160. package/src/ui/headless/comment-decoration-model.ts +124 -0
  161. package/src/ui/headless/preserve-editor-selection.ts +5 -0
  162. package/src/ui/headless/revision-decoration-model.ts +128 -0
  163. package/src/ui/headless/selection-helpers.ts +54 -0
  164. package/src/ui/headless/selection-tool-context.ts +19 -0
  165. package/src/ui/headless/selection-tool-resolver.ts +752 -0
  166. package/src/ui/headless/selection-tool-types.ts +129 -0
  167. package/src/ui/headless/selection-toolbar-model.ts +11 -0
  168. package/src/ui/headless/use-editor-keyboard.ts +103 -0
  169. package/src/ui/review/README.md +3 -0
  170. package/src/ui/runtime-shortcut-dispatch.ts +365 -0
  171. package/src/ui/runtime-snapshot-selectors.ts +197 -0
  172. package/src/ui/shared/revision-filters.ts +31 -0
  173. package/src/ui/status/README.md +3 -0
  174. package/src/ui/theme/README.md +3 -0
  175. package/src/ui/toolbar/README.md +3 -0
  176. package/src/ui/workflow-surface-blocked-rails.ts +94 -0
  177. package/src/ui-tailwind/chrome/chrome-preset-model.ts +107 -0
  178. package/src/ui-tailwind/chrome/chrome-preset-toolbar.tsx +15 -0
  179. package/src/ui-tailwind/chrome/review-queue-bar.tsx +97 -0
  180. package/src/ui-tailwind/chrome/tw-alert-banner.tsx +64 -0
  181. package/src/ui-tailwind/chrome/tw-context-analytics-summary.tsx +122 -0
  182. package/src/ui-tailwind/chrome/tw-image-context-toolbar.tsx +121 -0
  183. package/src/ui-tailwind/chrome/tw-layout-panel.tsx +114 -0
  184. package/src/ui-tailwind/chrome/tw-object-context-toolbar.tsx +30 -0
  185. package/src/ui-tailwind/chrome/tw-page-ruler.tsx +365 -0
  186. package/src/ui-tailwind/chrome/tw-selection-tool-blocked.tsx +23 -0
  187. package/src/ui-tailwind/chrome/tw-selection-tool-comment.tsx +35 -0
  188. package/src/ui-tailwind/chrome/tw-selection-tool-formatting.tsx +37 -0
  189. package/src/ui-tailwind/chrome/tw-selection-tool-host.tsx +298 -0
  190. package/src/ui-tailwind/chrome/tw-selection-tool-structure.tsx +116 -0
  191. package/src/ui-tailwind/chrome/tw-selection-tool-suggestion.tsx +29 -0
  192. package/src/ui-tailwind/chrome/tw-selection-tool-workflow.tsx +27 -0
  193. package/src/ui-tailwind/chrome/tw-selection-toolbar.tsx +186 -0
  194. package/src/ui-tailwind/chrome/tw-suggestion-card.tsx +139 -0
  195. package/src/ui-tailwind/chrome/tw-table-context-toolbar.tsx +200 -0
  196. package/src/ui-tailwind/chrome/tw-unsaved-modal.tsx +58 -0
  197. package/src/ui-tailwind/chrome/use-before-unload.ts +20 -0
  198. package/src/ui-tailwind/editor-surface/perf-probe.ts +179 -0
  199. package/src/ui-tailwind/editor-surface/pm-command-bridge.ts +189 -0
  200. package/src/ui-tailwind/editor-surface/pm-contextual-ui.ts +31 -0
  201. package/src/ui-tailwind/editor-surface/pm-decorations.ts +411 -0
  202. package/src/ui-tailwind/editor-surface/pm-position-map.ts +123 -0
  203. package/src/ui-tailwind/editor-surface/pm-schema.ts +927 -0
  204. package/src/ui-tailwind/editor-surface/pm-state-from-snapshot.ts +567 -0
  205. package/src/ui-tailwind/editor-surface/search-plugin.ts +168 -0
  206. package/src/ui-tailwind/editor-surface/surface-build-keys.ts +65 -0
  207. package/src/ui-tailwind/editor-surface/tw-caret.tsx +12 -0
  208. package/src/ui-tailwind/editor-surface/tw-editor-surface.tsx +150 -0
  209. package/src/ui-tailwind/editor-surface/tw-inline-token.tsx +129 -0
  210. package/src/ui-tailwind/editor-surface/tw-opaque-block.tsx +58 -0
  211. package/src/ui-tailwind/editor-surface/tw-paragraph-block.tsx +151 -0
  212. package/src/ui-tailwind/editor-surface/tw-prosemirror-surface.tsx +1047 -0
  213. package/src/ui-tailwind/editor-surface/tw-segment-view.tsx +111 -0
  214. package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +503 -0
  215. package/src/ui-tailwind/index.ts +62 -0
  216. package/src/ui-tailwind/page-chrome-model.ts +27 -0
  217. package/src/ui-tailwind/review/tw-comment-sidebar.tsx +406 -0
  218. package/src/ui-tailwind/review/tw-health-panel.tsx +149 -0
  219. package/src/ui-tailwind/review/tw-review-rail.tsx +122 -0
  220. package/src/ui-tailwind/review/tw-revision-sidebar.tsx +164 -0
  221. package/src/ui-tailwind/status/tw-status-bar.tsx +65 -0
  222. package/{dist → src}/ui-tailwind/theme/editor-theme.css +58 -40
  223. package/src/ui-tailwind/toolbar/tw-toolbar-icon-button.tsx +52 -0
  224. package/src/ui-tailwind/toolbar/tw-toolbar.tsx +1133 -0
  225. package/src/ui-tailwind/tw-review-workspace.tsx +1460 -0
  226. package/src/validation/README.md +3 -0
  227. package/src/validation/compatibility-engine.ts +878 -0
  228. package/src/validation/compatibility-report.ts +161 -0
  229. package/src/validation/diagnostics.ts +204 -0
  230. package/src/validation/docx-comment-proof.ts +720 -0
  231. package/src/validation/import-diagnostics.ts +128 -0
  232. package/src/validation/low-priority-word-surfaces.ts +373 -0
  233. package/dist/canonical-document-BLEbzL2J.d.cts +0 -844
  234. package/dist/canonical-document-BLEbzL2J.d.ts +0 -844
  235. package/dist/chunk-2FJS5GZM.js +0 -763
  236. package/dist/chunk-2FJS5GZM.js.map +0 -1
  237. package/dist/chunk-2OQBZS3F.js +0 -446
  238. package/dist/chunk-2OQBZS3F.js.map +0 -1
  239. package/dist/chunk-2S7W4KFO.js +0 -127
  240. package/dist/chunk-2S7W4KFO.js.map +0 -1
  241. package/dist/chunk-2TG72QSW.js +0 -3874
  242. package/dist/chunk-2TG72QSW.js.map +0 -1
  243. package/dist/chunk-36QNIZBO.js +0 -532
  244. package/dist/chunk-36QNIZBO.js.map +0 -1
  245. package/dist/chunk-4AQOYAW4.js +0 -3069
  246. package/dist/chunk-4AQOYAW4.js.map +0 -1
  247. package/dist/chunk-4D5EWJ3P.js +0 -77
  248. package/dist/chunk-4D5EWJ3P.js.map +0 -1
  249. package/dist/chunk-5FN54NDH.js +0 -2257
  250. package/dist/chunk-5FN54NDH.js.map +0 -1
  251. package/dist/chunk-BOYGQYRQ.js +0 -7306
  252. package/dist/chunk-BOYGQYRQ.js.map +0 -1
  253. package/dist/chunk-CN3XMECL.js +0 -212
  254. package/dist/chunk-CN3XMECL.js.map +0 -1
  255. package/dist/chunk-EBI3BX6U.js +0 -164
  256. package/dist/chunk-EBI3BX6U.js.map +0 -1
  257. package/dist/chunk-EILUG3VB.js +0 -1275
  258. package/dist/chunk-EILUG3VB.js.map +0 -1
  259. package/dist/chunk-FUDY333O.js +0 -70
  260. package/dist/chunk-FUDY333O.js.map +0 -1
  261. package/dist/chunk-GBVOWFIK.js +0 -1237
  262. package/dist/chunk-GBVOWFIK.js.map +0 -1
  263. package/dist/chunk-H4TQ3H3Y.js +0 -262
  264. package/dist/chunk-H4TQ3H3Y.js.map +0 -1
  265. package/dist/chunk-JGB3IXZO.js +0 -189
  266. package/dist/chunk-JGB3IXZO.js.map +0 -1
  267. package/dist/chunk-KD2QRQPY.js +0 -4342
  268. package/dist/chunk-KD2QRQPY.js.map +0 -1
  269. package/dist/chunk-KLMXQVYK.js +0 -369
  270. package/dist/chunk-KLMXQVYK.js.map +0 -1
  271. package/dist/chunk-KZUG5KFQ.js +0 -214
  272. package/dist/chunk-KZUG5KFQ.js.map +0 -1
  273. package/dist/chunk-QDAQ4CJU.js +0 -345
  274. package/dist/chunk-QDAQ4CJU.js.map +0 -1
  275. package/dist/chunk-RMH72RZI.js.map +0 -1
  276. package/dist/chunk-SWKWQZXM.js +0 -117
  277. package/dist/chunk-SWKWQZXM.js.map +0 -1
  278. package/dist/chunk-TJBP2K4T.js.map +0 -1
  279. package/dist/chunk-TLCEAQDQ.js +0 -542
  280. package/dist/chunk-TLCEAQDQ.js.map +0 -1
  281. package/dist/chunk-UZXBISGO.js.map +0 -1
  282. package/dist/chunk-WGBAKP3Q.js +0 -3220
  283. package/dist/chunk-WGBAKP3Q.js.map +0 -1
  284. package/dist/compare/index.cjs +0 -5475
  285. package/dist/compare/index.cjs.map +0 -1
  286. package/dist/compare/index.d.cts +0 -114
  287. package/dist/compare/index.d.ts +0 -114
  288. package/dist/compare/index.js +0 -731
  289. package/dist/compare/index.js.map +0 -1
  290. package/dist/core/commands/formatting-commands.cjs +0 -828
  291. package/dist/core/commands/formatting-commands.cjs.map +0 -1
  292. package/dist/core/commands/formatting-commands.d.cts +0 -63
  293. package/dist/core/commands/formatting-commands.d.ts +0 -63
  294. package/dist/core/commands/formatting-commands.js +0 -37
  295. package/dist/core/commands/formatting-commands.js.map +0 -1
  296. package/dist/core/commands/image-commands.cjs +0 -2023
  297. package/dist/core/commands/image-commands.cjs.map +0 -1
  298. package/dist/core/commands/image-commands.d.cts +0 -58
  299. package/dist/core/commands/image-commands.d.ts +0 -58
  300. package/dist/core/commands/image-commands.js +0 -18
  301. package/dist/core/commands/image-commands.js.map +0 -1
  302. package/dist/core/commands/section-layout-commands.cjs.map +0 -1
  303. package/dist/core/commands/section-layout-commands.d.cts +0 -62
  304. package/dist/core/commands/section-layout-commands.d.ts +0 -62
  305. package/dist/core/commands/section-layout-commands.js +0 -21
  306. package/dist/core/commands/section-layout-commands.js.map +0 -1
  307. package/dist/core/commands/style-commands.cjs.map +0 -1
  308. package/dist/core/commands/style-commands.d.cts +0 -13
  309. package/dist/core/commands/style-commands.d.ts +0 -13
  310. package/dist/core/commands/style-commands.js +0 -9
  311. package/dist/core/commands/style-commands.js.map +0 -1
  312. package/dist/core/commands/table-structure-commands.cjs +0 -1883
  313. package/dist/core/commands/table-structure-commands.cjs.map +0 -1
  314. package/dist/core/commands/table-structure-commands.d.cts +0 -59
  315. package/dist/core/commands/table-structure-commands.d.ts +0 -59
  316. package/dist/core/commands/table-structure-commands.js +0 -12
  317. package/dist/core/commands/table-structure-commands.js.map +0 -1
  318. package/dist/core/commands/text-commands.cjs +0 -2391
  319. package/dist/core/commands/text-commands.cjs.map +0 -1
  320. package/dist/core/commands/text-commands.d.cts +0 -24
  321. package/dist/core/commands/text-commands.d.ts +0 -24
  322. package/dist/core/commands/text-commands.js +0 -28
  323. package/dist/core/commands/text-commands.js.map +0 -1
  324. package/dist/core/selection/mapping.cjs +0 -200
  325. package/dist/core/selection/mapping.cjs.map +0 -1
  326. package/dist/core/selection/mapping.d.cts +0 -2
  327. package/dist/core/selection/mapping.d.ts +0 -2
  328. package/dist/core/selection/mapping.js +0 -31
  329. package/dist/core/selection/mapping.js.map +0 -1
  330. package/dist/core/state/editor-state.cjs +0 -2278
  331. package/dist/core/state/editor-state.cjs.map +0 -1
  332. package/dist/core/state/editor-state.d.cts +0 -2
  333. package/dist/core/state/editor-state.d.ts +0 -2
  334. package/dist/core/state/editor-state.js +0 -26
  335. package/dist/core/state/editor-state.js.map +0 -1
  336. package/dist/index.cjs +0 -38553
  337. package/dist/index.cjs.map +0 -1
  338. package/dist/index.d.cts +0 -15
  339. package/dist/index.d.ts +0 -15
  340. package/dist/index.js +0 -7856
  341. package/dist/index.js.map +0 -1
  342. package/dist/io/docx-session.cjs +0 -16236
  343. package/dist/io/docx-session.cjs.map +0 -1
  344. package/dist/io/docx-session.d.cts +0 -21
  345. package/dist/io/docx-session.d.ts +0 -21
  346. package/dist/io/docx-session.js +0 -18
  347. package/dist/io/docx-session.js.map +0 -1
  348. package/dist/legal/index.cjs +0 -3900
  349. package/dist/legal/index.cjs.map +0 -1
  350. package/dist/legal/index.d.cts +0 -86
  351. package/dist/legal/index.d.ts +0 -86
  352. package/dist/legal/index.js +0 -616
  353. package/dist/legal/index.js.map +0 -1
  354. package/dist/public-types-7ZL_94cz.d.ts +0 -1573
  355. package/dist/public-types-CeMaDueh.d.cts +0 -1573
  356. package/dist/public-types.cjs +0 -19
  357. package/dist/public-types.cjs.map +0 -1
  358. package/dist/public-types.d.cts +0 -2
  359. package/dist/public-types.d.ts +0 -2
  360. package/dist/public-types.js +0 -1
  361. package/dist/public-types.js.map +0 -1
  362. package/dist/runtime/document-runtime.cjs +0 -11140
  363. package/dist/runtime/document-runtime.cjs.map +0 -1
  364. package/dist/runtime/document-runtime.d.cts +0 -231
  365. package/dist/runtime/document-runtime.d.ts +0 -231
  366. package/dist/runtime/document-runtime.js +0 -21
  367. package/dist/runtime/document-runtime.js.map +0 -1
  368. package/dist/structural-helpers-CilgOVhh.d.cts +0 -10
  369. package/dist/structural-helpers-q0Gd-eBN.d.ts +0 -10
  370. package/dist/ui-tailwind/editor-surface/search-plugin.cjs +0 -313
  371. package/dist/ui-tailwind/editor-surface/search-plugin.cjs.map +0 -1
  372. package/dist/ui-tailwind/editor-surface/search-plugin.d.cts +0 -67
  373. package/dist/ui-tailwind/editor-surface/search-plugin.d.ts +0 -67
  374. package/dist/ui-tailwind/editor-surface/search-plugin.js +0 -23
  375. package/dist/ui-tailwind/editor-surface/search-plugin.js.map +0 -1
  376. package/dist/ui-tailwind/index.cjs +0 -4833
  377. package/dist/ui-tailwind/index.cjs.map +0 -1
  378. package/dist/ui-tailwind/index.d.cts +0 -617
  379. package/dist/ui-tailwind/index.d.ts +0 -617
  380. package/dist/ui-tailwind/index.js +0 -575
  381. package/dist/ui-tailwind/index.js.map +0 -1
@@ -0,0 +1,752 @@
1
+ import type {
2
+ EditorStoryTarget,
3
+ FormattingStateSnapshot,
4
+ SuggestionEntrySnapshot,
5
+ TrackedChangeEntrySnapshot,
6
+ WorkflowScopeSnapshot,
7
+ } from "../../api/public-types";
8
+ import { storyTargetsEqual } from "../../core/selection/mapping";
9
+ import type { SelectionToolResolverInput } from "./selection-tool-context";
10
+ import type {
11
+ ActiveSelectionToolModel,
12
+ BlockedExplainerSelectionToolModel,
13
+ CommentThreadSelectionToolModel,
14
+ FormattingInlineSelectionToolModel,
15
+ SelectionToolBadge,
16
+ StructureContextSelectionToolModel,
17
+ SuggestionReviewSelectionToolModel,
18
+ WorkflowTaskSelectionToolModel,
19
+ } from "./selection-tool-types";
20
+
21
+ export function resolveActiveSelectionTool(
22
+ input: SelectionToolResolverInput,
23
+ ): ActiveSelectionToolModel | null {
24
+ const suggestionTool = buildSuggestionReviewSelectionToolModel(input);
25
+ const commentTool = buildCommentThreadSelectionToolModel(input);
26
+ const workflowTool = buildWorkflowTaskSelectionToolModel(input);
27
+ const structureTool = buildStructureContextSelectionToolModel(input);
28
+ // Semantic targets beat generic range formatting so structure tools do not
29
+ // leak preview text when the user is acting on a table, image, or object.
30
+ const formattingTool = buildFormattingInlineSelectionToolModel(input);
31
+ const blockedTool = buildBlockedExplainerSelectionToolModel(input);
32
+ const resolvedTool =
33
+ suggestionTool ??
34
+ commentTool ??
35
+ workflowTool ??
36
+ structureTool ??
37
+ formattingTool ??
38
+ blockedTool;
39
+
40
+ return resolvedTool;
41
+ }
42
+
43
+ export function buildFormattingInlineSelectionToolModel(
44
+ input: SelectionToolResolverInput,
45
+ ): FormattingInlineSelectionToolModel | null {
46
+ const {
47
+ snapshot,
48
+ viewState,
49
+ capabilities,
50
+ documentNavigation,
51
+ styleCatalog,
52
+ formattingState,
53
+ workflowScopeSnapshot,
54
+ interactionGuardSnapshot,
55
+ addCommentDisabledReason,
56
+ } = input;
57
+
58
+ if (
59
+ !snapshot.surface ||
60
+ snapshot.selection.isCollapsed ||
61
+ snapshot.selection.activeRange.kind !== "range" ||
62
+ viewState.viewMode === "view"
63
+ ) {
64
+ return null;
65
+ }
66
+
67
+ const previewText = summarizeSelectionPreview(snapshot);
68
+ if (!previewText) {
69
+ return null;
70
+ }
71
+
72
+ const workflowPosture = resolveSelectionWorkflowPosture(
73
+ snapshot,
74
+ viewState.activeStory,
75
+ workflowScopeSnapshot,
76
+ interactionGuardSnapshot,
77
+ );
78
+ const targetAccess = getSelectionTargetAccess(input, workflowPosture);
79
+ const formattingCapability = getCommandCapability(input, "formatting", workflowPosture);
80
+
81
+ if (!capabilities.canEdit && !formattingCapability.supported) {
82
+ return null;
83
+ }
84
+ if (targetAccess === "comment-only" || targetAccess === "view-only" || targetAccess === "blocked") {
85
+ return null;
86
+ }
87
+ if (Boolean(input.preferListStructureContext) && Boolean(viewState.activeListContext?.isOrdered)) {
88
+ return null;
89
+ }
90
+ const badges = [
91
+ createSelectionToolStoryBadge(viewState.activeStory),
92
+ createSelectionToolWorkflowBadge(workflowPosture),
93
+ viewState.workspaceMode === "page" && documentNavigation.pageCount > 0
94
+ ? { label: `Page ${documentNavigation.activePageIndex + 1}` as const }
95
+ : null,
96
+ formattingState ? createSelectionToolStyleBadge(styleCatalog.paragraphs, formattingState) : null,
97
+ createSelectionToolListBadge(viewState.activeListContext),
98
+ ].filter((badge): badge is SelectionToolBadge => Boolean(badge));
99
+
100
+ const canToggleFormatting = formattingCapability.supported;
101
+ const canAddComment = capabilities.canAddComment;
102
+ const disabledReason = resolveToolDisabledReason(
103
+ canToggleFormatting ? undefined : formattingCapability.blockedReason,
104
+ workflowPosture.disabledReason,
105
+ canAddComment ? undefined : addCommentDisabledReason,
106
+ );
107
+
108
+ return {
109
+ kind: "formatting-inline",
110
+ previewText,
111
+ badges,
112
+ canToggleFormatting,
113
+ boldActive: formattingState?.bold ?? false,
114
+ italicActive: formattingState?.italic ?? false,
115
+ underlineActive: formattingState?.underline ?? false,
116
+ canAddComment,
117
+ ...(disabledReason ? { disabledReason } : {}),
118
+ };
119
+ }
120
+
121
+ export function buildSuggestionReviewSelectionToolModel(
122
+ input: SelectionToolResolverInput,
123
+ ): SuggestionReviewSelectionToolModel | null {
124
+ const {
125
+ snapshot,
126
+ viewState,
127
+ capabilities,
128
+ workflowScopeSnapshot,
129
+ interactionGuardSnapshot,
130
+ suggestionsSnapshot,
131
+ activeRevisionId,
132
+ suppressedSuggestionRevisionId,
133
+ addCommentDisabledReason,
134
+ activeTableContext,
135
+ activeImageContext,
136
+ activeObjectContext,
137
+ activeListContext,
138
+ preferListStructureContext,
139
+ } = input;
140
+
141
+ if (
142
+ !snapshot.surface ||
143
+ !capabilities.canEdit ||
144
+ viewState.viewMode === "view"
145
+ ) {
146
+ return null;
147
+ }
148
+ if (
149
+ activeTableContext ||
150
+ activeImageContext ||
151
+ activeObjectContext ||
152
+ (preferListStructureContext && activeListContext)
153
+ ) {
154
+ return null;
155
+ }
156
+
157
+ const activeRange =
158
+ !snapshot.selection.isCollapsed && snapshot.selection.activeRange.kind === "range"
159
+ ? snapshot.selection.activeRange
160
+ : null;
161
+ const selectionFrom = activeRange
162
+ ? Math.min(activeRange.from, activeRange.to)
163
+ : null;
164
+ const selectionTo = activeRange
165
+ ? Math.max(activeRange.from, activeRange.to)
166
+ : null;
167
+ const matchingScope = findSelectionMatchingScope(
168
+ activeRange,
169
+ viewState.activeStory,
170
+ workflowScopeSnapshot,
171
+ );
172
+
173
+ const candidateSuggestions = suggestionsSnapshot?.suggestions.filter((suggestion) =>
174
+ storyTargetsEqual(suggestion.storyTarget, viewState.activeStory) &&
175
+ suggestion.status === "active" &&
176
+ suggestion.actionability === "actionable" &&
177
+ suggestion.anchor.kind === "range" &&
178
+ (
179
+ (
180
+ activeRevisionId !== undefined &&
181
+ suggestion.changeIds.includes(activeRevisionId) &&
182
+ selectionMatchesSuggestion(snapshot.selection, suggestion)
183
+ ) ||
184
+ (
185
+ selectionFrom !== null &&
186
+ selectionTo !== null &&
187
+ rangesOverlap(
188
+ selectionFrom,
189
+ selectionTo,
190
+ suggestion.anchor.from,
191
+ suggestion.anchor.to,
192
+ )
193
+ )
194
+ )
195
+ ) ?? [];
196
+
197
+ const focusedSuggestion = (
198
+ activeRevisionId
199
+ ? candidateSuggestions.find((suggestion) => suggestion.changeIds.includes(activeRevisionId))
200
+ : null
201
+ ) ?? candidateSuggestions[0];
202
+
203
+ if (!focusedSuggestion || focusedSuggestion.suggestionId === suppressedSuggestionRevisionId) {
204
+ return null;
205
+ }
206
+
207
+ const badges = [
208
+ createSelectionToolStoryBadge(viewState.activeStory),
209
+ matchingScope?.label
210
+ ? {
211
+ label: matchingScope.label,
212
+ tone: "accent" as const,
213
+ }
214
+ : workflowScopeSnapshot?.activeWorkItem?.title
215
+ ? {
216
+ label: workflowScopeSnapshot.activeWorkItem.title,
217
+ tone: "accent" as const,
218
+ }
219
+ : null,
220
+ ].filter((badge): badge is SelectionToolBadge => Boolean(badge));
221
+ const workflowPosture = resolveSelectionWorkflowPosture(
222
+ snapshot,
223
+ viewState.activeStory,
224
+ workflowScopeSnapshot,
225
+ interactionGuardSnapshot,
226
+ );
227
+ const targetAccess = getSelectionTargetAccess(input, workflowPosture);
228
+ const canReviewSuggestion = workflowPosture.mode === "edit" || workflowPosture.mode === "suggest";
229
+ const canAddComment = targetAccess === "view-only" || targetAccess === "blocked"
230
+ ? false
231
+ : capabilities.canAddComment;
232
+ const disabledReason = resolveToolDisabledReason(
233
+ workflowPosture.disabledReason,
234
+ canAddComment ? undefined : addCommentDisabledReason,
235
+ );
236
+
237
+ return {
238
+ kind: "suggestion-review",
239
+ suggestionId: focusedSuggestion.suggestionId,
240
+ changeIds: focusedSuggestion.changeIds,
241
+ kindLabel: getSuggestionKindLabel(focusedSuggestion.kind),
242
+ previewText:
243
+ focusedSuggestion.excerpt ??
244
+ focusedSuggestion.detail ??
245
+ "Suggested change",
246
+ badges,
247
+ canAccept: canReviewSuggestion && capabilities.canAcceptChange && focusedSuggestion.canAccept,
248
+ canReject: canReviewSuggestion && capabilities.canRejectChange && focusedSuggestion.canReject,
249
+ canEditSuggestion: canReviewSuggestion && focusedSuggestion.editable,
250
+ canAddComment,
251
+ ...(disabledReason ? { disabledReason } : {}),
252
+ };
253
+ }
254
+
255
+ function selectionMatchesSuggestion(
256
+ selection: SelectionToolResolverInput["snapshot"]["selection"],
257
+ suggestion: SuggestionEntrySnapshot,
258
+ ): boolean {
259
+ if (suggestion.anchor.kind !== "range" || selection.activeRange.kind !== "range") {
260
+ return false;
261
+ }
262
+ if (selection.isCollapsed) {
263
+ const point = selection.activeRange.from;
264
+ return point >= suggestion.anchor.from && point <= suggestion.anchor.to;
265
+ }
266
+ const selectionFrom = Math.min(selection.activeRange.from, selection.activeRange.to);
267
+ const selectionTo = Math.max(selection.activeRange.from, selection.activeRange.to);
268
+ return rangesOverlap(
269
+ selectionFrom,
270
+ selectionTo,
271
+ suggestion.anchor.from,
272
+ suggestion.anchor.to,
273
+ );
274
+ }
275
+
276
+ function findSelectionMatchingScope(
277
+ activeRange: SelectionToolResolverInput["snapshot"]["selection"]["activeRange"] | null,
278
+ activeStory: EditorStoryTarget,
279
+ workflowScopeSnapshot?: WorkflowScopeSnapshot | null,
280
+ ) {
281
+ if (!activeRange || activeRange.kind !== "range" || !workflowScopeSnapshot) {
282
+ return null;
283
+ }
284
+ return workflowScopeSnapshot.scopes.find((scope) => {
285
+ const scopeStoryTarget = scope.storyTarget ?? { kind: "main" as const };
286
+ if (!storyTargetsEqual(scopeStoryTarget, activeStory)) {
287
+ return false;
288
+ }
289
+ if (scope.anchor.kind === "detached") {
290
+ return false;
291
+ }
292
+ const scopeFrom = scope.anchor.kind === "range" ? scope.anchor.from : scope.anchor.at;
293
+ const scopeTo = scope.anchor.kind === "range" ? scope.anchor.to : scope.anchor.at;
294
+ return activeRange.from >= scopeFrom && activeRange.to <= scopeTo;
295
+ }) ?? null;
296
+ }
297
+
298
+ export function buildStructureContextSelectionToolModel(
299
+ input: SelectionToolResolverInput,
300
+ ): StructureContextSelectionToolModel | null {
301
+ const workflowPosture = resolveSelectionWorkflowPosture(
302
+ input.snapshot,
303
+ input.viewState.activeStory,
304
+ input.workflowScopeSnapshot,
305
+ input.interactionGuardSnapshot,
306
+ );
307
+ const structureCapability = getCommandCapability(input, "structure", workflowPosture);
308
+ const canMutate = structureCapability.supported;
309
+ const badges = [
310
+ createSelectionToolStoryBadge(input.viewState.activeStory),
311
+ createSelectionToolWorkflowBadge(workflowPosture),
312
+ ].filter((badge): badge is SelectionToolBadge => Boolean(badge));
313
+ const disabledReason = resolveToolDisabledReason(
314
+ canMutate ? undefined : structureCapability.blockedReason,
315
+ canMutate ? undefined : workflowPosture.disabledReason,
316
+ );
317
+
318
+ if (!structureCapability.supported) {
319
+ return null;
320
+ }
321
+
322
+ if (input.activeImageContext) {
323
+ return {
324
+ kind: "structure-context",
325
+ structureKind: "image",
326
+ badges,
327
+ activeImage: input.activeImageContext,
328
+ canMutate,
329
+ ...(disabledReason ? { disabledReason } : {}),
330
+ };
331
+ }
332
+
333
+ if (input.activeObjectContext) {
334
+ return {
335
+ kind: "structure-context",
336
+ structureKind: "object",
337
+ badges,
338
+ activeObject: input.activeObjectContext,
339
+ canMutate,
340
+ ...(disabledReason ? { disabledReason } : {}),
341
+ };
342
+ }
343
+
344
+ const isTableContext = Boolean(
345
+ input.formattingState?.breadcrumb.some(
346
+ (item) => item.kind === "table" || item.kind === "table_cell" || item.kind === "table_row",
347
+ ),
348
+ );
349
+ const isTableTextRangeContext =
350
+ input.activeTableContext?.selectionKind === "text" &&
351
+ !input.snapshot.selection.isCollapsed;
352
+ if (isTableTextRangeContext) {
353
+ return null;
354
+ }
355
+ if (isTableContext) {
356
+ return {
357
+ kind: "structure-context",
358
+ structureKind: "table",
359
+ badges,
360
+ tableStyles: input.styleCatalog.tables,
361
+ activeTable: input.activeTableContext ?? null,
362
+ canMutate,
363
+ ...(disabledReason ? { disabledReason } : {}),
364
+ };
365
+ }
366
+
367
+ if (Boolean(input.preferListStructureContext) && Boolean(input.viewState.activeListContext?.isOrdered)) {
368
+ return {
369
+ kind: "structure-context",
370
+ structureKind: "list",
371
+ badges,
372
+ activeListContext: input.viewState.activeListContext,
373
+ canMutate,
374
+ ...(disabledReason ? { disabledReason } : {}),
375
+ };
376
+ }
377
+
378
+ return null;
379
+ }
380
+
381
+ export function buildWorkflowTaskSelectionToolModel(
382
+ input: SelectionToolResolverInput,
383
+ ): WorkflowTaskSelectionToolModel | null {
384
+ const workflowPosture = resolveSelectionWorkflowPosture(
385
+ input.snapshot,
386
+ input.viewState.activeStory,
387
+ input.workflowScopeSnapshot,
388
+ input.interactionGuardSnapshot,
389
+ );
390
+ const targetAccess = getSelectionTargetAccess(input, workflowPosture);
391
+ const activeWorkItem = input.workflowScopeSnapshot?.activeWorkItem;
392
+ if (!activeWorkItem || targetAccess === "direct-edit" || targetAccess === "blocked") {
393
+ return null;
394
+ }
395
+
396
+ return {
397
+ kind: "workflow-task",
398
+ badges: [
399
+ { label: activeWorkItem.title, tone: "accent" },
400
+ ...(
401
+ workflowPosture.mode === "suggest"
402
+ ? [{ label: "Suggest", tone: "accent" as const }]
403
+ : workflowPosture.mode === "comment"
404
+ ? [{ label: "Comment only", tone: "accent" as const }]
405
+ : workflowPosture.mode === "view"
406
+ ? [{ label: "View only" }]
407
+ : [{ label: "Blocked" }]
408
+ ),
409
+ ],
410
+ workflowTitle: activeWorkItem.title,
411
+ workflowDetail: activeWorkItem.description,
412
+ ...(workflowPosture.disabledReason ? { disabledReason: workflowPosture.disabledReason } : {}),
413
+ };
414
+ }
415
+
416
+ export function buildCommentThreadSelectionToolModel(
417
+ input: SelectionToolResolverInput,
418
+ ): CommentThreadSelectionToolModel | null {
419
+ const workflowPosture = resolveSelectionWorkflowPosture(
420
+ input.snapshot,
421
+ input.viewState.activeStory,
422
+ input.workflowScopeSnapshot,
423
+ input.interactionGuardSnapshot,
424
+ );
425
+ const targetAccess = getSelectionTargetAccess(input, workflowPosture);
426
+ if (!input.activeCommentThread || !input.snapshot.selection.isCollapsed) {
427
+ return null;
428
+ }
429
+
430
+ const canAddComment = targetAccess !== "view-only" &&
431
+ targetAccess !== "blocked" &&
432
+ input.capabilities.canAddComment;
433
+ return {
434
+ kind: "comment-thread",
435
+ badges: [
436
+ { label: input.activeCommentThread.status === "open" ? "Comment" : "Resolved comment", tone: "accent" },
437
+ ],
438
+ previewText: input.activeCommentThread.excerpt,
439
+ thread: input.activeCommentThread,
440
+ canAddComment,
441
+ ...(workflowPosture.disabledReason ? { disabledReason: workflowPosture.disabledReason } : {}),
442
+ };
443
+ }
444
+
445
+ export function buildBlockedExplainerSelectionToolModel(
446
+ input: SelectionToolResolverInput,
447
+ ): BlockedExplainerSelectionToolModel | null {
448
+ const workflowPosture = resolveSelectionWorkflowPosture(
449
+ input.snapshot,
450
+ input.viewState.activeStory,
451
+ input.workflowScopeSnapshot,
452
+ input.interactionGuardSnapshot,
453
+ );
454
+
455
+ if (!workflowPosture.disabledReason) {
456
+ return null;
457
+ }
458
+
459
+ if (workflowPosture.mode === "edit") {
460
+ return null;
461
+ }
462
+
463
+ return {
464
+ kind: "blocked-explainer",
465
+ badges: [
466
+ createSelectionToolStoryBadge(input.viewState.activeStory),
467
+ createSelectionToolWorkflowBadge(workflowPosture),
468
+ ].filter((badge): badge is SelectionToolBadge => Boolean(badge)),
469
+ disabledReason: workflowPosture.disabledReason,
470
+ };
471
+ }
472
+
473
+ function createSelectionToolStoryBadge(target: EditorStoryTarget): SelectionToolBadge | null {
474
+ if (target.kind === "main") {
475
+ return null;
476
+ }
477
+
478
+ return {
479
+ label:
480
+ target.kind === "header"
481
+ ? target.variant === "default"
482
+ ? "Header"
483
+ : `Header ${target.variant}`
484
+ : target.kind === "footer"
485
+ ? target.variant === "default"
486
+ ? "Footer"
487
+ : `Footer ${target.variant}`
488
+ : target.kind === "footnote"
489
+ ? "Footnote"
490
+ : "Endnote",
491
+ tone: "accent",
492
+ };
493
+ }
494
+
495
+ function createSelectionToolStyleBadge(
496
+ paragraphStyles: Array<{ styleId: string; displayName: string; isDefault: boolean }>,
497
+ formattingState: FormattingStateSnapshot,
498
+ ): SelectionToolBadge | null {
499
+ if (!formattingState.paragraphStyleId) {
500
+ return null;
501
+ }
502
+
503
+ const styleEntry = paragraphStyles.find(
504
+ (entry) => entry.styleId === formattingState.paragraphStyleId,
505
+ );
506
+ if (!styleEntry || styleEntry.isDefault) {
507
+ return null;
508
+ }
509
+
510
+ return { label: styleEntry.displayName };
511
+ }
512
+
513
+ function createSelectionToolListBadge(
514
+ activeListContext: SelectionToolResolverInput["viewState"]["activeListContext"],
515
+ ): SelectionToolBadge | null {
516
+ if (!activeListContext) {
517
+ return null;
518
+ }
519
+
520
+ return {
521
+ label: activeListContext.isOrdered ? "Numbered list" : "Bulleted list",
522
+ };
523
+ }
524
+
525
+ function createSelectionToolWorkflowBadge(
526
+ posture: ReturnType<typeof resolveSelectionWorkflowPosture>,
527
+ ): SelectionToolBadge | null {
528
+ switch (posture.mode) {
529
+ case "suggest":
530
+ return { label: "Suggest", tone: "accent" };
531
+ case "comment":
532
+ return { label: "Comment only", tone: "accent" };
533
+ case "view":
534
+ return { label: "View only" };
535
+ case "blocked":
536
+ return { label: "Blocked" };
537
+ default:
538
+ return null;
539
+ }
540
+ }
541
+
542
+ function getSelectionTargetAccess(
543
+ input: SelectionToolResolverInput,
544
+ workflowPosture: ReturnType<typeof resolveSelectionWorkflowPosture>,
545
+ ): NonNullable<NonNullable<SelectionToolResolverInput["interactionGuardSnapshot"]>["targetAccess"]> {
546
+ if (input.interactionGuardSnapshot?.targetAccess) {
547
+ return input.interactionGuardSnapshot.targetAccess;
548
+ }
549
+
550
+ switch (workflowPosture.mode) {
551
+ case "suggest":
552
+ return "suggest";
553
+ case "comment":
554
+ return "comment-only";
555
+ case "view":
556
+ return "view-only";
557
+ case "blocked":
558
+ return "blocked";
559
+ default:
560
+ return "direct-edit";
561
+ }
562
+ }
563
+
564
+ function getCommandCapability(
565
+ input: SelectionToolResolverInput,
566
+ family: "text" | "formatting" | "structure",
567
+ workflowPosture: ReturnType<typeof resolveSelectionWorkflowPosture>,
568
+ ): {
569
+ supported: boolean;
570
+ blockedReason?: string;
571
+ } {
572
+ const explicitCapability = input.interactionGuardSnapshot?.commandCapabilities?.find(
573
+ (entry) => entry.family === family,
574
+ );
575
+ if (explicitCapability) {
576
+ return {
577
+ supported: explicitCapability.supported,
578
+ blockedReason: explicitCapability.blockedReasons[0]?.message,
579
+ };
580
+ }
581
+
582
+ const targetAccess = getSelectionTargetAccess(input, workflowPosture);
583
+ const supported = Boolean(input.capabilities.canEdit) && targetAccess === "direct-edit";
584
+ return {
585
+ supported,
586
+ blockedReason: supported ? undefined : input.interactionGuardSnapshot?.disabledReason,
587
+ };
588
+ }
589
+
590
+ function resolveToolDisabledReason(
591
+ ...reasons: Array<string | null | undefined>
592
+ ): string | undefined {
593
+ return reasons.find((reason): reason is string => typeof reason === "string" && reason.length > 0);
594
+ }
595
+
596
+ export function resolveSelectionWorkflowPosture(
597
+ snapshot: SelectionToolResolverInput["snapshot"],
598
+ activeStory: EditorStoryTarget,
599
+ workflowScopeSnapshot?: WorkflowScopeSnapshot | null,
600
+ interactionGuardSnapshot?: SelectionToolResolverInput["interactionGuardSnapshot"],
601
+ ): {
602
+ mode: "edit" | "suggest" | "comment" | "view" | "blocked";
603
+ disabledReason?: string;
604
+ } {
605
+ const blockedReasons =
606
+ interactionGuardSnapshot?.blockedReasons ??
607
+ workflowScopeSnapshot?.blockedReasons ??
608
+ [];
609
+ const blockingReason = blockedReasons[0];
610
+ if (blockingReason) {
611
+ if (blockingReason.code === "workflow_comment_only") {
612
+ return {
613
+ mode: "comment",
614
+ disabledReason: blockingReason.message,
615
+ };
616
+ }
617
+ if (blockingReason.code === "workflow_view_only") {
618
+ return {
619
+ mode: "view",
620
+ disabledReason: blockingReason.message,
621
+ };
622
+ }
623
+ return {
624
+ mode: "blocked",
625
+ disabledReason: blockingReason.message,
626
+ };
627
+ }
628
+
629
+ if (interactionGuardSnapshot) {
630
+ if (interactionGuardSnapshot.effectiveMode === "suggest") {
631
+ return {
632
+ mode: "suggest",
633
+ disabledReason:
634
+ interactionGuardSnapshot.disabledReason ??
635
+ "Suggestion authoring is active here; direct formatting changes are blocked.",
636
+ };
637
+ }
638
+ if (interactionGuardSnapshot.effectiveMode === "comment") {
639
+ return {
640
+ mode: "comment",
641
+ disabledReason: interactionGuardSnapshot.disabledReason,
642
+ };
643
+ }
644
+ if (interactionGuardSnapshot.effectiveMode === "view") {
645
+ return {
646
+ mode: "view",
647
+ disabledReason: interactionGuardSnapshot.disabledReason,
648
+ };
649
+ }
650
+ if (interactionGuardSnapshot.effectiveMode === "blocked") {
651
+ return {
652
+ mode: "blocked",
653
+ disabledReason: interactionGuardSnapshot.disabledReason,
654
+ };
655
+ }
656
+ }
657
+
658
+ const activeRange =
659
+ !snapshot.selection.isCollapsed && snapshot.selection.activeRange.kind === "range"
660
+ ? snapshot.selection.activeRange
661
+ : null;
662
+ const matchingScope = activeRange && workflowScopeSnapshot
663
+ ? workflowScopeSnapshot.scopes.find((scope) => {
664
+ const scopeStoryTarget = scope.storyTarget ?? { kind: "main" as const };
665
+ if (!storyTargetsEqual(scopeStoryTarget, activeStory)) {
666
+ return false;
667
+ }
668
+ if (scope.anchor.kind === "detached") {
669
+ return false;
670
+ }
671
+ const scopeFrom = scope.anchor.kind === "range" ? scope.anchor.from : scope.anchor.at;
672
+ const scopeTo = scope.anchor.kind === "range" ? scope.anchor.to : scope.anchor.at;
673
+ return activeRange.from >= scopeFrom && activeRange.to <= scopeTo;
674
+ })
675
+ : null;
676
+
677
+ if (matchingScope?.mode === "suggest") {
678
+ return {
679
+ mode: "suggest",
680
+ disabledReason: "Suggestion authoring is active here; direct formatting changes are blocked.",
681
+ };
682
+ }
683
+ if (matchingScope?.mode === "comment") {
684
+ return {
685
+ mode: "comment",
686
+ disabledReason: `Scope "${matchingScope.label ?? matchingScope.scopeId}" allows comments only.`,
687
+ };
688
+ }
689
+ if (matchingScope?.mode === "view") {
690
+ return {
691
+ mode: "view",
692
+ disabledReason: `Scope "${matchingScope.label ?? matchingScope.scopeId}" is view-only.`,
693
+ };
694
+ }
695
+ return { mode: "edit" };
696
+ }
697
+
698
+ function summarizeSelectionPreview(snapshot: SelectionToolResolverInput["snapshot"]): string | null {
699
+ if (!snapshot.surface || snapshot.selection.isCollapsed) {
700
+ return null;
701
+ }
702
+
703
+ const range = snapshot.selection.activeRange;
704
+ if (range.kind !== "range") {
705
+ return "Selected range";
706
+ }
707
+
708
+ const preview = snapshot.surface.plainText
709
+ .slice(range.from, range.to)
710
+ .replace(/\s+/g, " ")
711
+ .trim();
712
+
713
+ if (!preview) {
714
+ return "Selected range";
715
+ }
716
+
717
+ return preview.length > 48 ? `${preview.slice(0, 45)}...` : preview;
718
+ }
719
+
720
+ function rangesOverlap(
721
+ leftFrom: number,
722
+ leftTo: number,
723
+ rightFrom: number,
724
+ rightTo: number,
725
+ ): boolean {
726
+ return leftFrom < rightTo && rightFrom < leftTo;
727
+ }
728
+
729
+ function getSuggestionKindLabel(
730
+ kind: TrackedChangeEntrySnapshot["kind"] | SuggestionEntrySnapshot["kind"],
731
+ ): string {
732
+ switch (kind) {
733
+ case "insertion":
734
+ return "Suggested insertion";
735
+ case "deletion":
736
+ return "Suggested deletion";
737
+ case "replacement":
738
+ return "Suggested replacement";
739
+ case "formatting":
740
+ case "formatting-change":
741
+ return "Suggested formatting change";
742
+ case "property-change":
743
+ case "paragraph-property-change":
744
+ return "Suggested property change";
745
+ case "move":
746
+ return "Suggested move";
747
+ case "structural-change":
748
+ return "Suggested structural change";
749
+ case "object-change":
750
+ return "Suggested object change";
751
+ }
752
+ }