@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,824 @@
1
+ import type {
2
+ CompatibilityReport,
3
+ EditorAnchorProjection,
4
+ EditorStoryTarget,
5
+ EditorWarning,
6
+ InteractionGuardSnapshot,
7
+ ReviewWorkSnapshot,
8
+ RuntimeContextAnalyticsActionHint,
9
+ RuntimeContextAnalyticsBadge,
10
+ RuntimeContextAnalyticsProvenance,
11
+ RuntimeContextAnalyticsQuery,
12
+ RuntimeContextAnalyticsSnapshot,
13
+ RuntimeRenderSnapshot,
14
+ SuggestionsSnapshot,
15
+ WorkflowBlockedCommandReason,
16
+ WorkflowMarkupItem,
17
+ WorkflowMarkupSnapshot,
18
+ WorkflowOverlay,
19
+ WorkflowScope,
20
+ WorkflowScopeSnapshot,
21
+ WorkflowWorkItem,
22
+ } from "../api/public-types";
23
+ import { MAIN_STORY_TARGET, storyTargetsEqual } from "../core/selection/mapping.ts";
24
+
25
+ const DERIVED_FROM: RuntimeContextAnalyticsProvenance["derivedFrom"] = [
26
+ "render",
27
+ "workflowScope",
28
+ "interactionGuard",
29
+ "workflowMarkup",
30
+ "reviewWork",
31
+ "warnings",
32
+ "compatibility",
33
+ ];
34
+
35
+ const UNAVAILABLE_FIELDS: RuntimeContextAnalyticsProvenance["unavailable"] = [
36
+ "tasksRemaining",
37
+ "approvalsRemaining",
38
+ "overdueItems",
39
+ "tags",
40
+ "assignee",
41
+ "dueDate",
42
+ "lastTouchedAt",
43
+ ];
44
+
45
+ type AnalyticsInput = {
46
+ query?: RuntimeContextAnalyticsQuery;
47
+ renderSnapshot: RuntimeRenderSnapshot;
48
+ workflowOverlay: WorkflowOverlay | null;
49
+ workflowScopeSnapshot: WorkflowScopeSnapshot | null;
50
+ interactionGuardSnapshot: InteractionGuardSnapshot;
51
+ workflowMarkupSnapshot: WorkflowMarkupSnapshot;
52
+ suggestionsSnapshot: SuggestionsSnapshot;
53
+ reviewWorkSnapshot: ReviewWorkSnapshot;
54
+ warnings: EditorWarning[];
55
+ compatibility: CompatibilityReport;
56
+ };
57
+
58
+ export function createRuntimeContextAnalyticsSnapshot(
59
+ input: AnalyticsInput,
60
+ ): RuntimeContextAnalyticsSnapshot | null {
61
+ const scopeKind = input.query?.scopeKind ?? "selection";
62
+ switch (scopeKind) {
63
+ case "document":
64
+ return buildDocumentAnalytics(input);
65
+ case "workflow_scope":
66
+ return buildWorkflowScopeAnalytics(
67
+ input,
68
+ input.query?.scopeId ?? input.interactionGuardSnapshot.matchedScopeId,
69
+ );
70
+ case "work_item":
71
+ return buildWorkItemAnalytics(
72
+ input,
73
+ input.query?.workItemId ??
74
+ input.workflowOverlay?.activeWorkItemId ??
75
+ input.workflowScopeSnapshot?.activeWorkItemId ??
76
+ undefined,
77
+ );
78
+ case "selection":
79
+ default:
80
+ return buildSelectionAnalytics(input);
81
+ }
82
+ }
83
+
84
+ export function resolveCurrentContextAnalyticsQuery(input: {
85
+ workflowScopeSnapshot: WorkflowScopeSnapshot | null;
86
+ interactionGuardSnapshot: InteractionGuardSnapshot;
87
+ }): RuntimeContextAnalyticsQuery | undefined {
88
+ const activeWorkItemId = input.workflowScopeSnapshot?.activeWorkItemId ?? null;
89
+ if (activeWorkItemId) {
90
+ return {
91
+ scopeKind: "work_item",
92
+ workItemId: activeWorkItemId,
93
+ };
94
+ }
95
+ if (input.interactionGuardSnapshot.matchedScopeId) {
96
+ return {
97
+ scopeKind: "workflow_scope",
98
+ scopeId: input.interactionGuardSnapshot.matchedScopeId,
99
+ };
100
+ }
101
+ return undefined;
102
+ }
103
+
104
+ export function runtimeContextAnalyticsSnapshotsEqual(
105
+ left: RuntimeContextAnalyticsSnapshot | null,
106
+ right: RuntimeContextAnalyticsSnapshot | null,
107
+ ): boolean {
108
+ if (left === right) {
109
+ return true;
110
+ }
111
+ if (!left || !right) {
112
+ return false;
113
+ }
114
+ return (
115
+ left.scopeKind === right.scopeKind &&
116
+ left.scopeId === right.scopeId &&
117
+ left.workItemId === right.workItemId &&
118
+ analyticsBadgesEqual(left.badges, right.badges) &&
119
+ analyticsActionHintsEqual(left.nextActions, right.nextActions) &&
120
+ analyticsCountsEqual(left.counts, right.counts) &&
121
+ analyticsStateEqual(left.state, right.state) &&
122
+ analyticsProvenanceEqual(left.provenance, right.provenance)
123
+ );
124
+ }
125
+
126
+ function buildDocumentAnalytics(input: AnalyticsInput): RuntimeContextAnalyticsSnapshot {
127
+ const warningCount = getDocumentWarningCount(input.compatibility, input.warnings);
128
+ const errorCount = getDocumentErrorCount(input.compatibility);
129
+ const exportReadiness = getExportReadiness({
130
+ compatibility: input.compatibility,
131
+ warnings: warningCount,
132
+ errors: errorCount,
133
+ workflowMarkup: input.workflowMarkupSnapshot,
134
+ });
135
+ const openWorkItems = countOpenWorkItems(input.workflowOverlay, input.workflowScopeSnapshot);
136
+ const unresolvedComments = input.reviewWorkSnapshot.openCommentCount;
137
+ const pendingRevisions = input.reviewWorkSnapshot.actionableRevisionCount;
138
+ const pendingSuggestions = countActiveSuggestions(input.suggestionsSnapshot);
139
+ const badges = [
140
+ openWorkItems > 0 ? createCountBadge("open-work-items", openWorkItems, "open work item") : null,
141
+ unresolvedComments > 0
142
+ ? createCountBadge("document-comments", unresolvedComments, "unresolved comment")
143
+ : null,
144
+ pendingSuggestions > 0
145
+ ? createCountBadge("document-suggestions", pendingSuggestions, "pending suggestion")
146
+ : pendingRevisions > 0
147
+ ? createCountBadge("document-revisions", pendingRevisions, "pending change")
148
+ : null,
149
+ createExportBadge(exportReadiness),
150
+ ].filter((badge): badge is RuntimeContextAnalyticsBadge => Boolean(badge));
151
+ const nextActions = [
152
+ openWorkItems > 0
153
+ ? createActionHint(
154
+ "open-work-items",
155
+ `Continue ${openWorkItems} open ${pluralize(openWorkItems, "work item")}`,
156
+ "next_step",
157
+ )
158
+ : null,
159
+ unresolvedComments > 0
160
+ ? createActionHint(
161
+ "resolve-comments",
162
+ `Resolve ${unresolvedComments} ${pluralize(unresolvedComments, "comment")}`,
163
+ "next_step",
164
+ )
165
+ : null,
166
+ pendingSuggestions > 0
167
+ ? createActionHint(
168
+ "review-suggestions",
169
+ `Review ${pendingSuggestions} pending ${pluralize(pendingSuggestions, "suggestion")}`,
170
+ "next_step",
171
+ )
172
+ : pendingRevisions > 0
173
+ ? createActionHint(
174
+ "review-revisions",
175
+ `Review ${pendingRevisions} pending ${pluralize(pendingRevisions, "change")}`,
176
+ "next_step",
177
+ )
178
+ : null,
179
+ exportReadiness === "blocked"
180
+ ? createActionHint("export-blocked", "Export is currently blocked", "blocker")
181
+ : exportReadiness === "warning"
182
+ ? createActionHint("export-warning", "Export has compatibility warnings", "warning")
183
+ : null,
184
+ ].filter((action): action is RuntimeContextAnalyticsActionHint => Boolean(action));
185
+
186
+ return {
187
+ scopeKind: "document",
188
+ badges,
189
+ nextActions,
190
+ counts: {
191
+ unresolvedComments,
192
+ pendingRevisions,
193
+ pendingSuggestions,
194
+ warnings: warningCount,
195
+ errors: errorCount,
196
+ openWorkItems,
197
+ exportBlockers: input.compatibility.blockExport ? 1 : 0,
198
+ },
199
+ state: {
200
+ isBlocked: exportReadiness === "blocked",
201
+ completionState:
202
+ openWorkItems === 0 && unresolvedComments === 0 && pendingRevisions === 0 && exportReadiness === "ready"
203
+ ? "ready"
204
+ : "in_progress",
205
+ exportReadiness,
206
+ },
207
+ provenance: createProvenance(),
208
+ };
209
+ }
210
+
211
+ function buildSelectionAnalytics(input: AnalyticsInput): RuntimeContextAnalyticsSnapshot {
212
+ const selectionAnchor = getSelectionAnchor(input.renderSnapshot);
213
+ const activeStory = input.renderSnapshot.activeStory;
214
+ const matchingScope = findMatchingScopeForSelection(
215
+ input.workflowOverlay,
216
+ input.workflowScopeSnapshot,
217
+ input.interactionGuardSnapshot,
218
+ selectionAnchor,
219
+ activeStory,
220
+ );
221
+ const matchingItems = input.workflowMarkupSnapshot.items.filter((item) =>
222
+ markupMatchesAnchor(item, selectionAnchor, activeStory),
223
+ );
224
+ const unresolvedComments = countUnresolvedComments(matchingItems);
225
+ const pendingRevisions = countPendingRevisions(matchingItems);
226
+ const pendingSuggestions = countPendingSuggestionsForMarkupItems(
227
+ matchingItems,
228
+ input.suggestionsSnapshot,
229
+ );
230
+ const blockedCommands = input.interactionGuardSnapshot.blockedReasons.length;
231
+ const badges = [
232
+ createModeBadge(input.interactionGuardSnapshot.effectiveMode),
233
+ unresolvedComments > 0
234
+ ? createCountBadge("selection-comments", unresolvedComments, "comment")
235
+ : null,
236
+ pendingSuggestions > 0
237
+ ? createCountBadge("selection-suggestions", pendingSuggestions, "pending suggestion")
238
+ : pendingRevisions > 0
239
+ ? createCountBadge("selection-revisions", pendingRevisions, "pending change")
240
+ : null,
241
+ ].filter((badge): badge is RuntimeContextAnalyticsBadge => Boolean(badge));
242
+
243
+ return {
244
+ scopeKind: "selection",
245
+ ...(matchingScope?.scopeId ? { scopeId: matchingScope.scopeId } : {}),
246
+ ...(matchingScope?.workItemId ? { workItemId: matchingScope.workItemId } : {}),
247
+ badges,
248
+ nextActions: createAttentionHints({
249
+ unresolvedComments,
250
+ pendingRevisions,
251
+ pendingSuggestions,
252
+ blockedReasons: input.interactionGuardSnapshot.blockedReasons,
253
+ }),
254
+ counts: {
255
+ unresolvedComments,
256
+ pendingRevisions,
257
+ pendingSuggestions,
258
+ blockedCommands,
259
+ },
260
+ state: {
261
+ effectiveMode: input.interactionGuardSnapshot.effectiveMode,
262
+ isBlocked: input.interactionGuardSnapshot.effectiveMode === "blocked",
263
+ ...(input.interactionGuardSnapshot.disabledReason
264
+ ? { disabledReason: input.interactionGuardSnapshot.disabledReason }
265
+ : {}),
266
+ completionState:
267
+ unresolvedComments === 0 &&
268
+ pendingRevisions === 0 &&
269
+ input.interactionGuardSnapshot.blockedReasons.length === 0
270
+ ? "ready"
271
+ : "in_progress",
272
+ },
273
+ provenance: createProvenance(),
274
+ };
275
+ }
276
+
277
+ function buildWorkflowScopeAnalytics(
278
+ input: AnalyticsInput,
279
+ scopeId: string | undefined,
280
+ ): RuntimeContextAnalyticsSnapshot | null {
281
+ if (!scopeId) {
282
+ return null;
283
+ }
284
+ const scope = getAllScopes(input.workflowOverlay, input.workflowScopeSnapshot).find(
285
+ (entry) => entry.scopeId === scopeId,
286
+ );
287
+ if (!scope) {
288
+ return null;
289
+ }
290
+ const matchingItems = input.workflowMarkupSnapshot.items.filter((item) =>
291
+ markupMatchesScope(item, scope),
292
+ );
293
+ const blockedReasons = (input.workflowScopeSnapshot?.blockedReasons ?? []).filter(
294
+ (reason) => reason.scopeId === scope.scopeId,
295
+ );
296
+ const unresolvedComments = countUnresolvedComments(matchingItems);
297
+ const pendingRevisions = countPendingRevisions(matchingItems);
298
+ const effectiveMode = deriveBlockedAwareMode(scope.mode, blockedReasons);
299
+
300
+ return {
301
+ scopeKind: "workflow_scope",
302
+ scopeId: scope.scopeId,
303
+ ...(scope.workItemId ? { workItemId: scope.workItemId } : {}),
304
+ badges: [
305
+ scope.label ? { key: "scope-label", label: scope.label, tone: "accent" } : null,
306
+ createModeBadge(effectiveMode),
307
+ unresolvedComments > 0 ? createCountBadge("scope-comments", unresolvedComments, "comment") : null,
308
+ pendingRevisions > 0 ? createCountBadge("scope-revisions", pendingRevisions, "pending change") : null,
309
+ ].filter((badge): badge is RuntimeContextAnalyticsBadge => Boolean(badge)),
310
+ nextActions: createAttentionHints({
311
+ unresolvedComments,
312
+ pendingRevisions,
313
+ blockedReasons,
314
+ }),
315
+ counts: {
316
+ unresolvedComments,
317
+ pendingRevisions,
318
+ blockedCommands: blockedReasons.length,
319
+ },
320
+ state: {
321
+ effectiveMode,
322
+ isBlocked: effectiveMode === "blocked",
323
+ ...(blockedReasons[0]?.message ? { disabledReason: blockedReasons[0].message } : {}),
324
+ completionState:
325
+ unresolvedComments === 0 && pendingRevisions === 0 && blockedReasons.length === 0
326
+ ? "ready"
327
+ : "in_progress",
328
+ },
329
+ provenance: createProvenance(),
330
+ };
331
+ }
332
+
333
+ function buildWorkItemAnalytics(
334
+ input: AnalyticsInput,
335
+ workItemId: string | undefined,
336
+ ): RuntimeContextAnalyticsSnapshot | null {
337
+ if (!workItemId) {
338
+ return null;
339
+ }
340
+ const workItem = getWorkflowWorkItem(input.workflowOverlay, input.workflowScopeSnapshot, workItemId);
341
+ const scopes = getAllScopes(input.workflowOverlay, input.workflowScopeSnapshot).filter(
342
+ (scope) => scope.workItemId === workItemId,
343
+ );
344
+ if (!workItem && scopes.length === 0) {
345
+ return null;
346
+ }
347
+ const matchingItems = input.workflowMarkupSnapshot.items.filter((item) =>
348
+ scopes.some((scope) => markupMatchesScope(item, scope)),
349
+ );
350
+ const blockedReasons = (input.workflowScopeSnapshot?.blockedReasons ?? []).filter(
351
+ (reason) => reason.workItemId === workItemId,
352
+ );
353
+ const unresolvedComments = countUnresolvedComments(matchingItems);
354
+ const pendingRevisions = countPendingRevisions(matchingItems);
355
+ const effectiveMode =
356
+ scopes.length === 1
357
+ ? deriveBlockedAwareMode(scopes[0]!.mode, blockedReasons)
358
+ : undefined;
359
+
360
+ return {
361
+ scopeKind: "work_item",
362
+ workItemId,
363
+ ...(scopes.length === 1 ? { scopeId: scopes[0]!.scopeId } : {}),
364
+ badges: [
365
+ workItem?.title ? { key: "work-item-title", label: workItem.title, tone: "accent" } : null,
366
+ effectiveMode ? createModeBadge(effectiveMode) : null,
367
+ unresolvedComments > 0
368
+ ? createCountBadge("work-item-comments", unresolvedComments, "comment")
369
+ : null,
370
+ pendingRevisions > 0
371
+ ? createCountBadge("work-item-revisions", pendingRevisions, "pending change")
372
+ : null,
373
+ ].filter((badge): badge is RuntimeContextAnalyticsBadge => Boolean(badge)),
374
+ nextActions: createAttentionHints({
375
+ unresolvedComments,
376
+ pendingRevisions,
377
+ blockedReasons,
378
+ }),
379
+ counts: {
380
+ unresolvedComments,
381
+ pendingRevisions,
382
+ blockedCommands: blockedReasons.length,
383
+ },
384
+ state: {
385
+ ...(effectiveMode ? { effectiveMode } : {}),
386
+ isBlocked: blockedReasons.length > 0,
387
+ ...(blockedReasons[0]?.message ? { disabledReason: blockedReasons[0].message } : {}),
388
+ completionState: resolveCompletionState(workItem, unresolvedComments, pendingRevisions, blockedReasons.length > 0),
389
+ },
390
+ provenance: createProvenance(),
391
+ };
392
+ }
393
+
394
+ function getSelectionAnchor(snapshot: RuntimeRenderSnapshot): EditorAnchorProjection {
395
+ if (!snapshot.selection.isCollapsed) {
396
+ return snapshot.selection.activeRange;
397
+ }
398
+ return {
399
+ kind: "node",
400
+ at: snapshot.selection.head,
401
+ assoc: 1,
402
+ };
403
+ }
404
+
405
+ function findMatchingScopeForSelection(
406
+ workflowOverlay: WorkflowOverlay | null,
407
+ workflowScopeSnapshot: WorkflowScopeSnapshot | null,
408
+ interactionGuardSnapshot: InteractionGuardSnapshot,
409
+ selectionAnchor: EditorAnchorProjection,
410
+ activeStory: EditorStoryTarget,
411
+ ): WorkflowScope | null {
412
+ if (interactionGuardSnapshot.matchedScopeId) {
413
+ return (
414
+ getAllScopes(workflowOverlay, workflowScopeSnapshot).find(
415
+ (scope) => scope.scopeId === interactionGuardSnapshot.matchedScopeId,
416
+ ) ?? null
417
+ );
418
+ }
419
+ return (
420
+ getAllScopes(workflowOverlay, workflowScopeSnapshot).find((scope) =>
421
+ anchorsOverlap(scope.anchor, scope.storyTarget, selectionAnchor, activeStory),
422
+ ) ?? null
423
+ );
424
+ }
425
+
426
+ function getAllScopes(
427
+ workflowOverlay: WorkflowOverlay | null,
428
+ workflowScopeSnapshot: WorkflowScopeSnapshot | null,
429
+ ): WorkflowScope[] {
430
+ return workflowOverlay?.scopes ?? workflowScopeSnapshot?.scopes ?? [];
431
+ }
432
+
433
+ function getWorkflowWorkItem(
434
+ workflowOverlay: WorkflowOverlay | null,
435
+ workflowScopeSnapshot: WorkflowScopeSnapshot | null,
436
+ workItemId: string,
437
+ ): WorkflowWorkItem | null {
438
+ if (workflowOverlay?.workItems) {
439
+ return workflowOverlay.workItems.find((item) => item.workItemId === workItemId) ?? null;
440
+ }
441
+ if (workflowScopeSnapshot?.activeWorkItem?.workItemId === workItemId) {
442
+ return workflowScopeSnapshot.activeWorkItem;
443
+ }
444
+ return null;
445
+ }
446
+
447
+ function countOpenWorkItems(
448
+ workflowOverlay: WorkflowOverlay | null,
449
+ workflowScopeSnapshot: WorkflowScopeSnapshot | null,
450
+ ): number {
451
+ if (workflowOverlay?.workItems?.length) {
452
+ return workflowOverlay.workItems.filter((item) => item.status !== "done").length;
453
+ }
454
+ const uniqueWorkItemIds = new Set(
455
+ (workflowScopeSnapshot?.scopes ?? [])
456
+ .map((scope) => scope.workItemId)
457
+ .filter((workItemId): workItemId is string => Boolean(workItemId)),
458
+ );
459
+ return uniqueWorkItemIds.size;
460
+ }
461
+
462
+ function markupMatchesScope(item: WorkflowMarkupItem, scope: WorkflowScope): boolean {
463
+ return anchorsOverlap(item.anchor, item.storyTarget, scope.anchor, scope.storyTarget);
464
+ }
465
+
466
+ function markupMatchesAnchor(
467
+ item: WorkflowMarkupItem,
468
+ anchor: EditorAnchorProjection,
469
+ storyTarget: EditorStoryTarget,
470
+ ): boolean {
471
+ return anchorsOverlap(item.anchor, item.storyTarget, anchor, storyTarget);
472
+ }
473
+
474
+ function anchorsOverlap(
475
+ left: EditorAnchorProjection,
476
+ leftStoryTarget: EditorStoryTarget | undefined,
477
+ right: EditorAnchorProjection,
478
+ rightStoryTarget: EditorStoryTarget | undefined,
479
+ ): boolean {
480
+ if (
481
+ !storyTargetsEqual(leftStoryTarget ?? MAIN_STORY_TARGET, rightStoryTarget ?? MAIN_STORY_TARGET)
482
+ ) {
483
+ return false;
484
+ }
485
+ const leftRange = toComparableRange(left);
486
+ const rightRange = toComparableRange(right);
487
+ if (!leftRange || !rightRange) {
488
+ return false;
489
+ }
490
+ return leftRange.from <= rightRange.to && rightRange.from <= leftRange.to;
491
+ }
492
+
493
+ function toComparableRange(
494
+ anchor: EditorAnchorProjection,
495
+ ): { from: number; to: number } | null {
496
+ if (anchor.kind === "detached") {
497
+ return null;
498
+ }
499
+ if (anchor.kind === "range") {
500
+ return {
501
+ from: Math.min(anchor.from, anchor.to),
502
+ to: Math.max(anchor.from, anchor.to),
503
+ };
504
+ }
505
+ return {
506
+ from: anchor.at,
507
+ to: anchor.at,
508
+ };
509
+ }
510
+
511
+ function countUnresolvedComments(items: readonly WorkflowMarkupItem[]): number {
512
+ return items.filter((item) => item.kind === "comment" && item.status === "open").length;
513
+ }
514
+
515
+ function countPendingRevisions(items: readonly WorkflowMarkupItem[]): number {
516
+ return items.filter(
517
+ (item) =>
518
+ item.kind === "revision" &&
519
+ item.status === "active" &&
520
+ item.actionability === "actionable",
521
+ ).length;
522
+ }
523
+
524
+ function getDocumentWarningCount(
525
+ compatibility: CompatibilityReport,
526
+ warnings: readonly EditorWarning[],
527
+ ): number {
528
+ return (
529
+ warnings.length +
530
+ compatibility.featureEntries.filter((entry) => entry.featureClass !== "supported-roundtrip").length
531
+ );
532
+ }
533
+
534
+ function getDocumentErrorCount(compatibility: CompatibilityReport): number {
535
+ return compatibility.errors.length;
536
+ }
537
+
538
+ function getExportReadiness(input: {
539
+ compatibility: CompatibilityReport;
540
+ workflowMarkup: WorkflowMarkupSnapshot;
541
+ warnings: number;
542
+ errors: number;
543
+ }): "ready" | "warning" | "blocked" {
544
+ if (input.compatibility.blockExport || input.errors > 0) {
545
+ return "blocked";
546
+ }
547
+ if (
548
+ input.warnings > 0 ||
549
+ input.workflowMarkup.opaqueFragments.length > 0 ||
550
+ input.workflowMarkup.protectedRanges.some((item) => item.enforced)
551
+ ) {
552
+ return "warning";
553
+ }
554
+ return "ready";
555
+ }
556
+
557
+ function createAttentionHints(input: {
558
+ unresolvedComments: number;
559
+ pendingRevisions: number;
560
+ pendingSuggestions?: number;
561
+ blockedReasons: readonly WorkflowBlockedCommandReason[];
562
+ }): RuntimeContextAnalyticsActionHint[] {
563
+ return [
564
+ input.blockedReasons[0]
565
+ ? createActionHint("blocked", input.blockedReasons[0].message, "blocker")
566
+ : null,
567
+ input.pendingSuggestions && input.pendingSuggestions > 0
568
+ ? createActionHint(
569
+ "review-suggestions",
570
+ `Review ${input.pendingSuggestions} pending ${pluralize(input.pendingSuggestions, "suggestion")}`,
571
+ "next_step",
572
+ )
573
+ : input.pendingRevisions > 0
574
+ ? createActionHint(
575
+ "review-revisions",
576
+ `Review ${input.pendingRevisions} pending ${pluralize(input.pendingRevisions, "change")}`,
577
+ "next_step",
578
+ )
579
+ : null,
580
+ input.unresolvedComments > 0
581
+ ? createActionHint(
582
+ "resolve-comments",
583
+ `Resolve ${input.unresolvedComments} ${pluralize(input.unresolvedComments, "comment")}`,
584
+ "next_step",
585
+ )
586
+ : null,
587
+ ].filter((action): action is RuntimeContextAnalyticsActionHint => Boolean(action));
588
+ }
589
+
590
+ function createModeBadge(
591
+ mode: "edit" | "suggest" | "comment" | "view" | "blocked",
592
+ ): RuntimeContextAnalyticsBadge | null {
593
+ switch (mode) {
594
+ case "suggest":
595
+ return { key: "mode", label: "Suggest", tone: "accent" };
596
+ case "comment":
597
+ return { key: "mode", label: "Comment only", tone: "accent" };
598
+ case "view":
599
+ return { key: "mode", label: "View only" };
600
+ case "blocked":
601
+ return { key: "mode", label: "Blocked", tone: "danger" };
602
+ case "edit":
603
+ default:
604
+ return null;
605
+ }
606
+ }
607
+
608
+ function createCountBadge(
609
+ key: string,
610
+ count: number,
611
+ singularLabel: string,
612
+ ): RuntimeContextAnalyticsBadge {
613
+ return {
614
+ key,
615
+ label: `${count} ${pluralize(count, singularLabel)}`,
616
+ };
617
+ }
618
+
619
+ function createExportBadge(
620
+ exportReadiness: "ready" | "warning" | "blocked",
621
+ ): RuntimeContextAnalyticsBadge {
622
+ return {
623
+ key: "export",
624
+ label:
625
+ exportReadiness === "blocked"
626
+ ? "Export blocked"
627
+ : exportReadiness === "warning"
628
+ ? "Export warnings"
629
+ : "Export ready",
630
+ tone:
631
+ exportReadiness === "blocked"
632
+ ? "danger"
633
+ : exportReadiness === "warning"
634
+ ? "warning"
635
+ : "success",
636
+ };
637
+ }
638
+
639
+ function createActionHint(
640
+ key: string,
641
+ label: string,
642
+ kind: RuntimeContextAnalyticsActionHint["kind"],
643
+ ): RuntimeContextAnalyticsActionHint {
644
+ return { key, label, kind };
645
+ }
646
+
647
+ function countActiveSuggestions(snapshot: SuggestionsSnapshot): number {
648
+ return snapshot.suggestions.filter(
649
+ (suggestion) =>
650
+ suggestion.status === "active" &&
651
+ suggestion.actionability === "actionable",
652
+ ).length;
653
+ }
654
+
655
+ function countPendingSuggestionsForMarkupItems(
656
+ items: readonly WorkflowMarkupItem[],
657
+ suggestionsSnapshot: SuggestionsSnapshot,
658
+ ): number {
659
+ if (items.length === 0) {
660
+ return 0;
661
+ }
662
+ const revisionIds = new Set(
663
+ items
664
+ .filter((item): item is Extract<WorkflowMarkupItem, { kind: "revision" }> => item.kind === "revision")
665
+ .filter((item) => item.status === "active" && item.actionability === "actionable")
666
+ .map((item) => item.revisionId),
667
+ );
668
+ if (revisionIds.size === 0) {
669
+ return 0;
670
+ }
671
+ return suggestionsSnapshot.suggestions.filter(
672
+ (suggestion) =>
673
+ suggestion.status === "active" &&
674
+ suggestion.actionability === "actionable" &&
675
+ suggestion.changeIds.some((changeId) => revisionIds.has(changeId)),
676
+ ).length;
677
+ }
678
+
679
+ function createProvenance(): RuntimeContextAnalyticsProvenance {
680
+ return {
681
+ kind: "runtime-derived",
682
+ derivedFrom: [...DERIVED_FROM],
683
+ unavailable: [...UNAVAILABLE_FIELDS],
684
+ };
685
+ }
686
+
687
+ function deriveBlockedAwareMode(
688
+ mode: WorkflowScope["mode"],
689
+ blockedReasons: readonly WorkflowBlockedCommandReason[],
690
+ ): "edit" | "suggest" | "comment" | "view" | "blocked" {
691
+ const primaryBlockedReason = blockedReasons[0];
692
+ if (!primaryBlockedReason) {
693
+ return mode;
694
+ }
695
+ if (primaryBlockedReason.code === "workflow_comment_only") {
696
+ return "comment";
697
+ }
698
+ if (primaryBlockedReason.code === "workflow_view_only") {
699
+ return "view";
700
+ }
701
+ return "blocked";
702
+ }
703
+
704
+ function resolveCompletionState(
705
+ workItem: WorkflowWorkItem | null,
706
+ unresolvedComments: number,
707
+ pendingRevisions: number,
708
+ isBlocked: boolean,
709
+ ): "not_started" | "in_progress" | "ready" | "done" {
710
+ if (workItem?.status === "done") {
711
+ return "done";
712
+ }
713
+ if (isBlocked || unresolvedComments > 0 || pendingRevisions > 0 || workItem?.status === "active") {
714
+ return "in_progress";
715
+ }
716
+ if (workItem?.status === "pending") {
717
+ return "not_started";
718
+ }
719
+ return "ready";
720
+ }
721
+
722
+ function pluralize(count: number, singular: string): string {
723
+ return count === 1 ? singular : `${singular}s`;
724
+ }
725
+
726
+ function analyticsBadgesEqual(
727
+ left: readonly RuntimeContextAnalyticsBadge[],
728
+ right: readonly RuntimeContextAnalyticsBadge[],
729
+ ): boolean {
730
+ if (left.length !== right.length) {
731
+ return false;
732
+ }
733
+ for (let index = 0; index < left.length; index += 1) {
734
+ const leftBadge = left[index]!;
735
+ const rightBadge = right[index]!;
736
+ if (
737
+ leftBadge.key !== rightBadge.key ||
738
+ leftBadge.label !== rightBadge.label ||
739
+ leftBadge.value !== rightBadge.value ||
740
+ leftBadge.tone !== rightBadge.tone ||
741
+ leftBadge.priority !== rightBadge.priority
742
+ ) {
743
+ return false;
744
+ }
745
+ }
746
+ return true;
747
+ }
748
+
749
+ function analyticsActionHintsEqual(
750
+ left: readonly RuntimeContextAnalyticsActionHint[],
751
+ right: readonly RuntimeContextAnalyticsActionHint[],
752
+ ): boolean {
753
+ if (left.length !== right.length) {
754
+ return false;
755
+ }
756
+ for (let index = 0; index < left.length; index += 1) {
757
+ const leftAction = left[index]!;
758
+ const rightAction = right[index]!;
759
+ if (
760
+ leftAction.key !== rightAction.key ||
761
+ leftAction.label !== rightAction.label ||
762
+ leftAction.kind !== rightAction.kind
763
+ ) {
764
+ return false;
765
+ }
766
+ }
767
+ return true;
768
+ }
769
+
770
+ function analyticsCountsEqual(
771
+ left: RuntimeContextAnalyticsSnapshot["counts"],
772
+ right: RuntimeContextAnalyticsSnapshot["counts"],
773
+ ): boolean {
774
+ return (
775
+ left.tasksRemaining === right.tasksRemaining &&
776
+ left.unresolvedComments === right.unresolvedComments &&
777
+ left.pendingRevisions === right.pendingRevisions &&
778
+ left.pendingSuggestions === right.pendingSuggestions &&
779
+ left.blockedCommands === right.blockedCommands &&
780
+ left.approvalsRemaining === right.approvalsRemaining &&
781
+ left.overdueItems === right.overdueItems &&
782
+ left.warnings === right.warnings &&
783
+ left.errors === right.errors &&
784
+ left.openWorkItems === right.openWorkItems &&
785
+ left.exportBlockers === right.exportBlockers
786
+ );
787
+ }
788
+
789
+ function analyticsStateEqual(
790
+ left: RuntimeContextAnalyticsSnapshot["state"],
791
+ right: RuntimeContextAnalyticsSnapshot["state"],
792
+ ): boolean {
793
+ return (
794
+ left.effectiveMode === right.effectiveMode &&
795
+ left.isBlocked === right.isBlocked &&
796
+ left.disabledReason === right.disabledReason &&
797
+ left.completionState === right.completionState &&
798
+ left.exportReadiness === right.exportReadiness &&
799
+ left.lastTouchedAt === right.lastTouchedAt
800
+ );
801
+ }
802
+
803
+ function analyticsProvenanceEqual(
804
+ left: RuntimeContextAnalyticsProvenance,
805
+ right: RuntimeContextAnalyticsProvenance,
806
+ ): boolean {
807
+ return (
808
+ left.kind === right.kind &&
809
+ stringArraysEqual(left.derivedFrom, right.derivedFrom) &&
810
+ stringArraysEqual(left.unavailable, right.unavailable)
811
+ );
812
+ }
813
+
814
+ function stringArraysEqual(left: readonly string[], right: readonly string[]): boolean {
815
+ if (left.length !== right.length) {
816
+ return false;
817
+ }
818
+ for (let index = 0; index < left.length; index += 1) {
819
+ if (left[index] !== right[index]) {
820
+ return false;
821
+ }
822
+ }
823
+ return true;
824
+ }