@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,168 @@
1
+ /**
2
+ * ProseMirror search/highlight plugin for the editor surface.
3
+ *
4
+ * Provides in-document search with optional case sensitivity and regex support.
5
+ * Matched ranges are shown as inline decorations with a configurable highlight color.
6
+ *
7
+ * Usage:
8
+ * 1. Include createSearchPlugin() in the editor plugin list.
9
+ * 2. Call performSearch(state, query, options) to compute matches.
10
+ * 3. Dispatch the results into the plugin state via searchPluginKey meta.
11
+ * 4. Call clearSearch(state, dispatch) to remove all highlights.
12
+ */
13
+
14
+ import { Plugin, PluginKey } from "prosemirror-state";
15
+ import type { EditorState, Transaction } from "prosemirror-state";
16
+ import { Decoration, DecorationSet } from "prosemirror-view";
17
+
18
+ import type {
19
+ SearchOptions as PublicSearchOptions,
20
+ } from "../../api/public-types";
21
+ import {
22
+ buildSearchPattern,
23
+ createSearchExcerpt,
24
+ findSearchMatches,
25
+ searchSecondaryStories,
26
+ type SecondaryStorySearchResult,
27
+ type SearchTextOptions,
28
+ } from "../../core/search/search-text.ts";
29
+
30
+ // ---------------------------------------------------------------------------
31
+ // Public types
32
+ // ---------------------------------------------------------------------------
33
+
34
+ export interface SearchResult {
35
+ from: number;
36
+ to: number;
37
+ text: string;
38
+ index: number;
39
+ }
40
+
41
+ export interface SearchOptions extends PublicSearchOptions, SearchTextOptions {
42
+ regex?: boolean;
43
+ highlightColor?: string;
44
+ }
45
+
46
+ export const DEFAULT_SEARCH_HIGHLIGHT_COLOR = "#fde68a";
47
+
48
+ // ---------------------------------------------------------------------------
49
+ // Plugin state
50
+ // ---------------------------------------------------------------------------
51
+
52
+ interface SearchPluginState {
53
+ results: SearchResult[];
54
+ highlightColor: string | null;
55
+ }
56
+
57
+ export const searchPluginKey = new PluginKey<SearchPluginState>("search");
58
+
59
+ // ---------------------------------------------------------------------------
60
+ // Plugin factory
61
+ // ---------------------------------------------------------------------------
62
+
63
+ export function createSearchPlugin(): Plugin<SearchPluginState> {
64
+ return new Plugin<SearchPluginState>({
65
+ key: searchPluginKey,
66
+
67
+ state: {
68
+ init(): SearchPluginState {
69
+ return { results: [], highlightColor: null };
70
+ },
71
+
72
+ apply(tr: Transaction, pluginState: SearchPluginState): SearchPluginState {
73
+ const meta = tr.getMeta(searchPluginKey) as SearchPluginState | undefined;
74
+ if (meta) return meta;
75
+ return pluginState;
76
+ },
77
+ },
78
+
79
+ props: {
80
+ decorations(state: EditorState): DecorationSet {
81
+ const pluginState = searchPluginKey.getState(state);
82
+ if (
83
+ !pluginState ||
84
+ pluginState.results.length === 0 ||
85
+ !pluginState.highlightColor
86
+ ) {
87
+ return DecorationSet.empty;
88
+ }
89
+
90
+ const decos = pluginState.results.map((r) =>
91
+ Decoration.inline(r.from, r.to, {
92
+ style: `background-color: ${pluginState.highlightColor}; border-radius: 2px;`,
93
+ class: "search-highlight",
94
+ }),
95
+ );
96
+
97
+ return DecorationSet.create(state.doc, decos);
98
+ },
99
+ },
100
+ });
101
+ }
102
+
103
+ // ---------------------------------------------------------------------------
104
+ // Search computation
105
+ // ---------------------------------------------------------------------------
106
+
107
+ /**
108
+ * Scan all text nodes in the document for matches of `query`.
109
+ * Returns an array of match ranges; does not dispatch anything.
110
+ * To apply highlights, set the results on the plugin via:
111
+ * view.dispatch(view.state.tr.setMeta(searchPluginKey, { results, highlightColor }))
112
+ */
113
+ export function performSearch(
114
+ state: EditorState,
115
+ query: string,
116
+ options: SearchOptions = {},
117
+ ): SearchResult[] {
118
+ const pattern = buildSearchPattern(query, options);
119
+ if (!pattern) return [];
120
+
121
+ const results: SearchResult[] = [];
122
+
123
+ state.doc.descendants((node, pos) => {
124
+ if (!node.isText || !node.text) return;
125
+
126
+ let match: RegExpExecArray | null;
127
+ pattern.lastIndex = 0;
128
+ while ((match = pattern.exec(node.text)) !== null) {
129
+ results.push({
130
+ from: pos + match.index,
131
+ to: pos + match.index + match[0].length,
132
+ text: match[0],
133
+ index: results.length,
134
+ });
135
+
136
+ if (match[0].length === 0) {
137
+ pattern.lastIndex += 1;
138
+ }
139
+ }
140
+ });
141
+
142
+ return results;
143
+ }
144
+
145
+ // ---------------------------------------------------------------------------
146
+ // Clear helper (ProseMirror Command signature)
147
+ // ---------------------------------------------------------------------------
148
+
149
+ /**
150
+ * ProseMirror command that clears all search highlights from the plugin state.
151
+ */
152
+ export function clearSearch(
153
+ state: EditorState,
154
+ dispatch?: (tr: Transaction) => void,
155
+ ): boolean {
156
+ if (!dispatch) return true;
157
+ dispatch(
158
+ state.tr.setMeta(searchPluginKey, { results: [], highlightColor: null }),
159
+ );
160
+ return true;
161
+ }
162
+
163
+ export {
164
+ createSearchExcerpt,
165
+ findSearchMatches,
166
+ searchSecondaryStories,
167
+ };
168
+ export type { SecondaryStorySearchResult };
@@ -0,0 +1,63 @@
1
+ import type {
2
+ EditorStoryTarget,
3
+ EditorSurfaceSnapshot,
4
+ } from "../../api/public-types.ts";
5
+
6
+ const surfaceIdentityMap = new WeakMap<EditorSurfaceSnapshot, number>();
7
+ let nextSurfaceIdentity = 0;
8
+
9
+ function getSurfaceIdentity(surface: EditorSurfaceSnapshot): number {
10
+ const cached = surfaceIdentityMap.get(surface);
11
+ if (cached !== undefined) {
12
+ return cached;
13
+ }
14
+ const identity = nextSurfaceIdentity;
15
+ nextSurfaceIdentity += 1;
16
+ surfaceIdentityMap.set(surface, identity);
17
+ return identity;
18
+ }
19
+
20
+ export function createSurfaceDocumentBuildKey(input: {
21
+ surface: EditorSurfaceSnapshot | null | undefined;
22
+ activeStory: EditorStoryTarget;
23
+ mediaPreviewKey: string;
24
+ showUnsupportedObjectPreviews?: boolean;
25
+ }): string {
26
+ return JSON.stringify({
27
+ surfaceIdentity:
28
+ input.surface === undefined || input.surface === null
29
+ ? "loading"
30
+ : getSurfaceIdentity(input.surface),
31
+ activeStory: input.activeStory,
32
+ mediaPreviewKey: input.mediaPreviewKey,
33
+ showUnsupportedObjectPreviews: input.showUnsupportedObjectPreviews ?? false,
34
+ });
35
+ }
36
+
37
+ export function createSurfaceDecorationKey(input: {
38
+ markupDisplay: string;
39
+ showTrackedChanges: boolean;
40
+ canEdit: boolean;
41
+ activeCommentId?: string;
42
+ activeRevisionId?: string;
43
+ workflowScopeSignature?: string;
44
+ workflowCandidateSignature?: string;
45
+ workflowBlockedSignature?: string;
46
+ workflowMetadataSignature?: string;
47
+ activeWorkflowWorkItemId?: string | null;
48
+ activeWorkflowScopeIds?: readonly string[];
49
+ }): string {
50
+ return JSON.stringify({
51
+ markupDisplay: input.markupDisplay,
52
+ showTrackedChanges: input.showTrackedChanges,
53
+ canEdit: input.canEdit,
54
+ activeCommentId: input.activeCommentId ?? null,
55
+ activeRevisionId: input.activeRevisionId ?? null,
56
+ workflowScopeSignature: input.workflowScopeSignature ?? null,
57
+ workflowCandidateSignature: input.workflowCandidateSignature ?? null,
58
+ workflowBlockedSignature: input.workflowBlockedSignature ?? null,
59
+ workflowMetadataSignature: input.workflowMetadataSignature ?? null,
60
+ activeWorkflowWorkItemId: input.activeWorkflowWorkItemId ?? null,
61
+ activeWorkflowScopeIds: input.activeWorkflowScopeIds ?? [],
62
+ });
63
+ }
@@ -0,0 +1,12 @@
1
+ import React from "react";
2
+
3
+ import type { SelectionSnapshot } from "../../api/public-types";
4
+
5
+ export function renderTwCaret(selection: SelectionSnapshot, position: number) {
6
+ return selection.isCollapsed && selection.anchor === position ? (
7
+ <span
8
+ aria-hidden="true"
9
+ className="inline-block w-0.5 h-[1.2em] bg-accent rounded-full align-middle animate-wre-blink"
10
+ />
11
+ ) : null;
12
+ }
@@ -0,0 +1,150 @@
1
+ import React, { type FocusEventHandler, useMemo } from "react";
2
+
3
+ import type {
4
+ EditorUser,
5
+ RuntimeRenderSnapshot,
6
+ SelectionSnapshot,
7
+ } from "../../api/public-types";
8
+ import {
9
+ createCommentDecorationModel,
10
+ type MarkupDisplay,
11
+ } from "../../ui/headless/comment-decoration-model";
12
+ import { createRevisionDecorationModel } from "../../ui/headless/revision-decoration-model";
13
+ import { createEditorKeyboardHandler } from "../../ui/headless/use-editor-keyboard";
14
+ import { TwOpaqueBlock } from "./tw-opaque-block";
15
+ import { TwParagraphBlock } from "./tw-paragraph-block";
16
+
17
+ export interface TwEditorSurfaceProps {
18
+ currentUser: EditorUser;
19
+ snapshot: RuntimeRenderSnapshot;
20
+ reviewMode: "editing" | "review";
21
+ markupDisplay: MarkupDisplay;
22
+ activeRevisionId?: string;
23
+ /** When false, revision decorations are suppressed in the document surface. */
24
+ showTrackedChanges?: boolean;
25
+ onFocus: FocusEventHandler<HTMLDivElement>;
26
+ onBlur: FocusEventHandler<HTMLDivElement>;
27
+ onSelectionChange?: (selection: SelectionSnapshot) => void;
28
+ onInsertText?: (text: string) => void;
29
+ onDeleteBackward?: () => void;
30
+ onDeleteForward?: () => void;
31
+ onInsertTab?: () => void;
32
+ onInsertHardBreak?: () => void;
33
+ onSplitParagraph?: () => void;
34
+ onCommentActivated?: (commentId: string) => void;
35
+ onRevisionActivated?: (revisionId: string) => void;
36
+ }
37
+
38
+ export function TwEditorSurface(props: TwEditorSurfaceProps) {
39
+ const { currentUser, markupDisplay, onBlur, onFocus, snapshot } = props;
40
+ const surface = snapshot.surface;
41
+
42
+ const canEdit = Boolean(
43
+ surface && snapshot.isReady && !snapshot.readOnly && !snapshot.fatalError,
44
+ );
45
+
46
+ const commentDecorations = useMemo(
47
+ () => createCommentDecorationModel(snapshot.comments),
48
+ [snapshot.comments],
49
+ );
50
+
51
+ const showTrackedChanges = props.showTrackedChanges !== false;
52
+ const revisionDecorations = useMemo(
53
+ () => showTrackedChanges
54
+ ? createRevisionDecorationModel(snapshot.trackedChanges, props.activeRevisionId)
55
+ : undefined,
56
+ [snapshot.trackedChanges, props.activeRevisionId, showTrackedChanges],
57
+ );
58
+
59
+ const handleKeyDown = useMemo(() => {
60
+ if (!surface) return undefined;
61
+ return createEditorKeyboardHandler(
62
+ {
63
+ selection: snapshot.selection,
64
+ storySize: surface.storySize,
65
+ canEdit,
66
+ },
67
+ {
68
+ onSelectionChange: props.onSelectionChange,
69
+ onInsertText: props.onInsertText,
70
+ onDeleteBackward: props.onDeleteBackward,
71
+ onDeleteForward: props.onDeleteForward,
72
+ onInsertTab: props.onInsertTab,
73
+ onInsertHardBreak: props.onInsertHardBreak,
74
+ onSplitParagraph: props.onSplitParagraph,
75
+ },
76
+ );
77
+ }, [surface, snapshot.selection, canEdit, props]);
78
+
79
+ const fontClass = markupDisplay === "clean"
80
+ ? "font-[family-name:var(--font-legal-sans)]"
81
+ : "font-[family-name:var(--font-legal-serif)]";
82
+
83
+ return (
84
+ <section aria-label="Document canvas" className="min-w-0">
85
+ <div
86
+ aria-label="Document surface"
87
+ aria-multiline="true"
88
+ role="textbox"
89
+ tabIndex={0}
90
+ onFocus={onFocus}
91
+ onBlur={onBlur}
92
+ onKeyDown={handleKeyDown}
93
+ className={`bg-transparent border-none px-12 py-10 outline-offset-1 ${fontClass}`}
94
+ >
95
+ {/* Document header */}
96
+ <div className="flex justify-between gap-4 items-start mb-5">
97
+ <div>
98
+ <h1 className="font-[family-name:var(--font-legal-serif)] text-2xl leading-tight text-primary">
99
+ {snapshot.sourceLabel ?? "Document draft"}
100
+ </h1>
101
+ <p className="text-xs text-secondary mt-2">
102
+ {snapshot.sessionId} · {snapshot.documentStats.storyLength} chars
103
+ </p>
104
+ </div>
105
+ <span className="text-xs text-secondary whitespace-nowrap">
106
+ Reviewer {currentUser.displayName}
107
+ </span>
108
+ </div>
109
+
110
+ {/* Document blocks */}
111
+ <div className="space-y-4 pl-10 relative">
112
+ {surface ? (
113
+ surface.blocks.map((block) =>
114
+ block.kind === "paragraph" ? (
115
+ <TwParagraphBlock
116
+ key={block.blockId}
117
+ block={block}
118
+ selection={snapshot.selection}
119
+ markupDisplay={markupDisplay}
120
+ commentDecorations={commentDecorations}
121
+ revisionDecorations={revisionDecorations}
122
+ onSelectionChange={props.onSelectionChange}
123
+ onCommentActivated={props.onCommentActivated}
124
+ onRevisionActivated={props.onRevisionActivated}
125
+ />
126
+ ) : block.kind === "opaque_block" ? (
127
+ <TwOpaqueBlock
128
+ key={block.blockId}
129
+ block={block}
130
+ selection={snapshot.selection}
131
+ onSelectionChange={props.onSelectionChange}
132
+ />
133
+ ) : null,
134
+ )
135
+ ) : (
136
+ <p className="text-sm text-secondary leading-relaxed">
137
+ Loading the review surface. Compatibility cues and edit actions appear here once
138
+ the runtime snapshot is ready.
139
+ </p>
140
+ )}
141
+ {snapshot.fatalError ? (
142
+ <p className="text-sm text-danger">
143
+ Fatal runtime error: {snapshot.fatalError.message}
144
+ </p>
145
+ ) : null}
146
+ </div>
147
+ </div>
148
+ </section>
149
+ );
150
+ }
@@ -0,0 +1,129 @@
1
+ import React from "react";
2
+
3
+ import type { SelectionSnapshot, SurfaceInlineSegment } from "../../api/public-types";
4
+ import type { CommentDecorationModel } from "../../ui/headless/comment-decoration-model";
5
+ import { getCommentHighlightClass, type MarkupDisplay } from "../../ui/headless/comment-decoration-model";
6
+ import { createSelectionSnapshot, selectionTouchesRange } from "../../ui/headless/selection-helpers";
7
+ import { renderTwCaret } from "./tw-caret";
8
+
9
+ export interface TwInlineTokenProps {
10
+ segment: SurfaceInlineSegment;
11
+ selection: SelectionSnapshot;
12
+ markupDisplay: MarkupDisplay;
13
+ commentDecorations?: CommentDecorationModel;
14
+ onSelectionChange?: (selection: SelectionSnapshot) => void;
15
+ }
16
+
17
+ const focusRingClass =
18
+ "focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent focus-visible:ring-offset-2 focus-visible:ring-offset-canvas";
19
+
20
+ export function TwInlineToken(props: TwInlineTokenProps) {
21
+ const { segment, selection } = props;
22
+ const selected = selectionTouchesRange(selection, segment.from, segment.to);
23
+ const commentClass = getCommentHighlightClass(
24
+ props.commentDecorations,
25
+ segment.from,
26
+ segment.to,
27
+ props.markupDisplay,
28
+ );
29
+
30
+ const showSymbols = props.markupDisplay !== "clean";
31
+
32
+ if (segment.kind === "tab") {
33
+ return (
34
+ <button
35
+ type="button"
36
+ tabIndex={-1}
37
+ onMouseDown={(e) => {
38
+ e.preventDefault();
39
+ props.onSelectionChange?.(createSelectionSnapshot(segment.from, segment.to));
40
+ }}
41
+ className={`inline-flex items-center border-none bg-transparent cursor-text ${commentClass} ${selected ? "bg-surface-hover" : ""} ${focusRingClass} ${showSymbols ? "mx-0.5 text-tertiary/50" : "w-8"}`}
42
+ title="Tab character"
43
+ >
44
+ {renderTwCaret(selection, segment.from)}
45
+ {showSymbols ? <span className="text-xs">→</span> : <span className="w-8" />}
46
+ {renderTwCaret(selection, segment.to)}
47
+ </button>
48
+ );
49
+ }
50
+
51
+ if (segment.kind === "hard_break") {
52
+ return (
53
+ <>
54
+ <button
55
+ type="button"
56
+ tabIndex={-1}
57
+ onMouseDown={(e) => {
58
+ e.preventDefault();
59
+ props.onSelectionChange?.(createSelectionSnapshot(segment.from, segment.to));
60
+ }}
61
+ className={`inline-flex items-center border-none bg-transparent cursor-text ${commentClass} ${selected ? "bg-surface-hover" : ""} ${focusRingClass} ${showSymbols ? "mx-0.5 text-tertiary/40" : ""}`}
62
+ title="Line break"
63
+ >
64
+ {renderTwCaret(selection, segment.from)}
65
+ {showSymbols ? <span className="text-xs">↵</span> : null}
66
+ {renderTwCaret(selection, segment.to)}
67
+ </button>
68
+ <br />
69
+ </>
70
+ );
71
+ }
72
+
73
+ if (segment.kind === "image") {
74
+ const isMissing = segment.state === "missing";
75
+ return (
76
+ <button
77
+ type="button"
78
+ tabIndex={-1}
79
+ onMouseDown={(e) => {
80
+ e.preventDefault();
81
+ props.onSelectionChange?.(createSelectionSnapshot(segment.from, segment.to));
82
+ }}
83
+ className={`inline-flex items-center gap-1 mx-0.5 px-1.5 py-0.5 rounded text-xs border-none cursor-pointer ${commentClass} ${
84
+ isMissing ? "text-danger bg-delete-soft" : "text-secondary bg-surface"
85
+ } ${selected ? "ring-1 ring-accent/30" : ""} ${focusRingClass}`}
86
+ title={segment.detail ?? segment.altText ?? "Inline image"}
87
+ >
88
+ {renderTwCaret(selection, segment.from)}
89
+ <span>📷</span>
90
+ {segment.altText ?? (isMissing ? "Missing image" : "Image")}
91
+ {renderTwCaret(selection, segment.to)}
92
+ </button>
93
+ );
94
+ }
95
+
96
+ // opaque_inline
97
+ if (segment.kind === "opaque_inline") {
98
+ if (segment.presentation === "quiet-marker") {
99
+ return (
100
+ <span
101
+ aria-label={segment.label}
102
+ title={segment.detail}
103
+ className="inline-block h-0 w-0 overflow-hidden align-baseline"
104
+ data-inline-presentation="quiet-marker"
105
+ />
106
+ );
107
+ }
108
+
109
+ return (
110
+ <button
111
+ type="button"
112
+ tabIndex={-1}
113
+ onMouseDown={(e) => {
114
+ e.preventDefault();
115
+ props.onSelectionChange?.(createSelectionSnapshot(segment.from, segment.to));
116
+ }}
117
+ className={`inline-flex items-center gap-1 mx-0.5 px-1.5 py-0.5 rounded text-xs text-comment bg-warning-soft border-none cursor-pointer ${commentClass} ${selected ? "ring-1 ring-accent/30" : ""} ${focusRingClass}`}
118
+ title={segment.detail}
119
+ >
120
+ {renderTwCaret(selection, segment.from)}
121
+ <span>🔒</span>
122
+ {segment.label}
123
+ {renderTwCaret(selection, segment.to)}
124
+ </button>
125
+ );
126
+ }
127
+
128
+ return null;
129
+ }
@@ -0,0 +1,58 @@
1
+ import React from "react";
2
+
3
+ import type { SelectionSnapshot, SurfaceBlockSnapshot } from "../../api/public-types";
4
+ import { selectionTouchesRange, createSelectionSnapshot } from "../../ui/headless/selection-helpers";
5
+
6
+ export interface TwOpaqueBlockProps {
7
+ block: Extract<SurfaceBlockSnapshot, { kind: "opaque_block" }>;
8
+ selection: SelectionSnapshot;
9
+ onSelectionChange?: (selection: SelectionSnapshot) => void;
10
+ workflowTargeted?: boolean;
11
+ }
12
+
13
+ const focusRingClass =
14
+ "focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent focus-visible:ring-offset-2 focus-visible:ring-offset-canvas";
15
+
16
+ export function TwOpaqueBlock(props: TwOpaqueBlockProps) {
17
+ const { block, selection, workflowTargeted } = props;
18
+ const selected = selectionTouchesRange(selection, block.from, block.to);
19
+
20
+ return (
21
+ <div className="group relative">
22
+ <div className="absolute -left-10 top-1 flex flex-col items-end gap-0 select-none w-8 text-right">
23
+ <div className="flex items-center gap-0.5">
24
+ <span className="text-[10px] text-tertiary/60 font-medium">Lock</span>
25
+ <span className="inline-block h-1 w-1 rounded-full" style={{ backgroundColor: "var(--color-comment)" }} />
26
+ </div>
27
+ </div>
28
+ <button
29
+ type="button"
30
+ tabIndex={-1}
31
+ onMouseDown={(e) => {
32
+ e.preventDefault();
33
+ props.onSelectionChange?.(createSelectionSnapshot(block.from, block.to));
34
+ }}
35
+ className={[
36
+ "w-full text-left border-l-2 border-dashed border-warning/30 pl-4 py-2 rounded-r bg-warning-soft/20",
37
+ "cursor-pointer transition-colors",
38
+ selected ? "ring-1 ring-accent/30 bg-warning-soft/40" : "hover:bg-warning-soft/30",
39
+ focusRingClass,
40
+ ].join(" ")}
41
+ >
42
+ <div className="flex items-center gap-1.5 text-xs text-tertiary mb-1">
43
+ <span>🔒</span>
44
+ <span>{block.label}</span>
45
+ <span className="inline-flex items-center px-1.5 py-0.5 rounded-full text-[10px] font-semibold text-comment bg-warning-soft">
46
+ preserve-only
47
+ </span>
48
+ {workflowTargeted && (
49
+ <span className="inline-flex items-center px-1.5 py-0.5 rounded-full text-[10px] font-semibold text-amber-700 bg-amber-100">
50
+ workflow-targeted
51
+ </span>
52
+ )}
53
+ </div>
54
+ <p className="text-sm text-secondary">{block.detail}</p>
55
+ </button>
56
+ </div>
57
+ );
58
+ }