@beyondwork/docx-react-component 1.0.0 → 1.0.2

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 (560) hide show
  1. package/README.md +44 -104
  2. package/package.json +66 -15
  3. package/src/api/public-types.ts +1 -1
  4. package/src/compare/diff-engine.ts +530 -0
  5. package/src/compare/export-redlines.ts +162 -0
  6. package/src/compare/snapshot.ts +37 -0
  7. package/src/core/commands/index.ts +1 -1
  8. package/src/core/state/editor-state.ts +2 -2
  9. package/src/index.ts +45 -0
  10. package/src/legal/bookmarks.ts +196 -0
  11. package/src/legal/cross-references.ts +356 -0
  12. package/src/legal/defined-terms.ts +203 -0
  13. package/src/runtime/document-runtime.ts +3 -5
  14. package/src/runtime/table-commands.ts +4 -1
  15. package/src/runtime/table-schema.ts +17 -2
  16. package/src/runtime/virtualized-rendering.ts +258 -0
  17. package/src/ui/WordReviewEditor.tsx +256 -35
  18. package/src/ui-tailwind/editor-surface/tw-editor-surface.tsx +2 -2
  19. package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +16 -2
  20. package/.codex/config.toml +0 -5
  21. package/.corepack/v1/pnpm/10.30.3/.corepack +0 -1
  22. package/.corepack/v1/pnpm/10.30.3/LICENSE +0 -22
  23. package/.corepack/v1/pnpm/10.30.3/README.md +0 -240
  24. package/.corepack/v1/pnpm/10.30.3/dist/node-gyp-bin/node-gyp +0 -6
  25. package/.corepack/v1/pnpm/10.30.3/dist/node-gyp-bin/node-gyp.cmd +0 -5
  26. package/.corepack/v1/pnpm/10.30.3/dist/pnpm.cjs +0 -195400
  27. package/.corepack/v1/pnpm/10.30.3/dist/pnpmrc +0 -2
  28. package/.corepack/v1/pnpm/10.30.3/dist/reflink.darwin-arm64-2HJ4WGO6.node +0 -0
  29. package/.corepack/v1/pnpm/10.30.3/dist/reflink.darwin-x64-3G3H6IW4.node +0 -0
  30. package/.corepack/v1/pnpm/10.30.3/dist/reflink.win32-arm64-msvc-Q6BARPPB.node +0 -0
  31. package/.corepack/v1/pnpm/10.30.3/dist/reflink.win32-x64-msvc-J2TZHRQI.node +0 -0
  32. package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.bash +0 -31
  33. package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.fish +0 -22
  34. package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.ps1 +0 -193
  35. package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.zsh +0 -27
  36. package/.corepack/v1/pnpm/10.30.3/dist/vendor/fastlist-0.3.0-x64.exe +0 -0
  37. package/.corepack/v1/pnpm/10.30.3/dist/vendor/fastlist-0.3.0-x86.exe +0 -0
  38. package/.corepack/v1/pnpm/10.30.3/dist/worker.js +0 -10119
  39. package/.corepack/v1/pnpm/10.30.3/package.json +0 -192
  40. package/.cursor/mcp.json +0 -7
  41. package/.github/workflows/ci.yml +0 -35
  42. package/.mcp.json +0 -7
  43. package/.openclaw/workspace-state.json +0 -4
  44. package/.pnpmrc.json +0 -1
  45. package/.wave-launch.sh +0 -7
  46. package/.workspace-marker +0 -1
  47. package/AGENTS.md +0 -78
  48. package/CHANGELOG.md +0 -177
  49. package/DESIGN.md +0 -929
  50. package/HEARTBEAT.md +0 -7
  51. package/IDENTITY.md +0 -23
  52. package/SOUL.md +0 -36
  53. package/TOOLS.md +0 -40
  54. package/USER.md +0 -17
  55. package/docs/README.md +0 -107
  56. package/docs/agents/wave-cont-eval-role.md +0 -36
  57. package/docs/agents/wave-cont-qa-role.md +0 -52
  58. package/docs/agents/wave-deploy-verifier-role.md +0 -34
  59. package/docs/agents/wave-design-role.md +0 -47
  60. package/docs/agents/wave-documentation-role.md +0 -34
  61. package/docs/agents/wave-infra-role.md +0 -34
  62. package/docs/agents/wave-integration-role.md +0 -37
  63. package/docs/agents/wave-launcher-role.md +0 -41
  64. package/docs/agents/wave-orchestrator-role.md +0 -52
  65. package/docs/agents/wave-planner-role.md +0 -39
  66. package/docs/agents/wave-security-role.md +0 -40
  67. package/docs/architecture/docx/README.md +0 -10
  68. package/docs/architecture/future/README.md +0 -8
  69. package/docs/architecture/ooxml-upgrade-analysis.md +0 -134
  70. package/docs/architecture/platform/shared-openxml-editor-platform.md +0 -153
  71. package/docs/architecture/xlsx/canonical-workbook-model-and-commands.md +0 -187
  72. package/docs/architecture/xlsx/spreadsheet-editor-frontend-architecture.md +0 -150
  73. package/docs/comment-redline-overview.md +0 -350
  74. package/docs/concepts/context7-vs-skills.md +0 -118
  75. package/docs/concepts/operating-modes.md +0 -91
  76. package/docs/concepts/runtime-agnostic-orchestration.md +0 -111
  77. package/docs/concepts/what-is-a-wave.md +0 -217
  78. package/docs/context7/bundles.json +0 -222
  79. package/docs/context7/planner-agent/README.md +0 -28
  80. package/docs/context7/planner-agent/manifest.json +0 -83
  81. package/docs/context7/planner-agent/papers/cooperbench-why-coding-agents-cannot-be-your-teammates-yet.md +0 -3283
  82. package/docs/context7/planner-agent/papers/dova-deliberation-first-multi-agent-orchestration-for-autonomous-research-automation.md +0 -1699
  83. package/docs/context7/planner-agent/papers/dpbench-large-language-models-struggle-with-simultaneous-coordination.md +0 -2251
  84. package/docs/context7/planner-agent/papers/incremental-planning-to-control-a-blackboard-based-problem-solver.md +0 -1729
  85. package/docs/context7/planner-agent/papers/silo-bench-a-scalable-environment-for-evaluating-distributed-coordination-in-multi-agent-llm-systems.md +0 -3747
  86. package/docs/context7/planner-agent/papers/todoevolve-learning-to-architect-agent-planning-systems.md +0 -1675
  87. package/docs/context7/planner-agent/papers/verified-multi-agent-orchestration-a-plan-execute-verify-replan-framework-for-complex-query-resolution.md +0 -1173
  88. package/docs/context7/planner-agent/papers/why-do-multi-agent-llm-systems-fail.md +0 -5211
  89. package/docs/context7/planner-agent/topics/planning-and-orchestration.md +0 -24
  90. package/docs/evals/arm-templates/README.md +0 -13
  91. package/docs/evals/arm-templates/full-wave.json +0 -15
  92. package/docs/evals/arm-templates/single-agent.json +0 -15
  93. package/docs/evals/benchmark-catalog.json +0 -670
  94. package/docs/evals/cases/README.md +0 -47
  95. package/docs/evals/cases/wave-blackboard-inbox-targeting.json +0 -73
  96. package/docs/evals/cases/wave-contradiction-conflict.json +0 -104
  97. package/docs/evals/cases/wave-expert-routing-preservation.json +0 -69
  98. package/docs/evals/cases/wave-hidden-profile-private-evidence.json +0 -81
  99. package/docs/evals/cases/wave-premature-closure-guard.json +0 -71
  100. package/docs/evals/cases/wave-silo-cross-agent-state.json +0 -77
  101. package/docs/evals/cases/wave-simultaneous-lockstep.json +0 -92
  102. package/docs/evals/external-benchmarks.json +0 -85
  103. package/docs/evals/external-command-config.sample.json +0 -9
  104. package/docs/evals/external-command-config.swe-bench-pro.json +0 -8
  105. package/docs/evals/pilots/README.md +0 -47
  106. package/docs/evals/pilots/swe-bench-pro-public-full-wave-review-10.json +0 -64
  107. package/docs/evals/pilots/swe-bench-pro-public-pilot.json +0 -111
  108. package/docs/evals/wave-benchmark-program.md +0 -302
  109. package/docs/guides/planner.md +0 -220
  110. package/docs/guides/recommendations-0.8.9.md +0 -133
  111. package/docs/guides/signal-wrappers.md +0 -165
  112. package/docs/guides/terminal-surfaces.md +0 -96
  113. package/docs/image copy.png +0 -0
  114. package/docs/image.png +0 -0
  115. package/docs/images/image.png +0 -0
  116. package/docs/legal-feedback-architecture.md +0 -498
  117. package/docs/plans/component-cutover-matrix.json +0 -1072
  118. package/docs/plans/component-cutover-matrix.md +0 -307
  119. package/docs/plans/context7-wave-orchestrator.md +0 -155
  120. package/docs/plans/current-state.md +0 -198
  121. package/docs/plans/docx/README.md +0 -9
  122. package/docs/plans/examples/wave-benchmark-improvement.md +0 -108
  123. package/docs/plans/examples/wave-example-live-proof.md +0 -435
  124. package/docs/plans/master-plan.md +0 -224
  125. package/docs/plans/migration.md +0 -538
  126. package/docs/plans/operations/README.md +0 -7
  127. package/docs/plans/operations/wave-10-word-certification.md +0 -87
  128. package/docs/plans/operations/wave-8-railway-staging.md +0 -153
  129. package/docs/plans/operations/wave-9-manual-certification.md +0 -73
  130. package/docs/plans/platform/README.md +0 -9
  131. package/docs/plans/reference/legal-checklist-coverage.md +0 -258
  132. package/docs/plans/wave-orchestrator.md +0 -423
  133. package/docs/plans/waves/README.md +0 -75
  134. package/docs/plans/waves/completed/wave-0.md +0 -195
  135. package/docs/plans/waves/completed/wave-1.md +0 -379
  136. package/docs/plans/waves/completed/wave-10.md +0 -670
  137. package/docs/plans/waves/completed/wave-11.md +0 -335
  138. package/docs/plans/waves/completed/wave-12.md +0 -417
  139. package/docs/plans/waves/completed/wave-13.md +0 -316
  140. package/docs/plans/waves/completed/wave-14.md +0 -319
  141. package/docs/plans/waves/completed/wave-15.md +0 -321
  142. package/docs/plans/waves/completed/wave-16.md +0 -316
  143. package/docs/plans/waves/completed/wave-17.md +0 -331
  144. package/docs/plans/waves/completed/wave-18.md +0 -328
  145. package/docs/plans/waves/completed/wave-2.md +0 -438
  146. package/docs/plans/waves/completed/wave-3.md +0 -435
  147. package/docs/plans/waves/completed/wave-4.md +0 -430
  148. package/docs/plans/waves/completed/wave-5.md +0 -430
  149. package/docs/plans/waves/completed/wave-6.md +0 -430
  150. package/docs/plans/waves/completed/wave-7.md +0 -526
  151. package/docs/plans/waves/completed/wave-8.md +0 -596
  152. package/docs/plans/waves/completed/wave-9.md +0 -552
  153. package/docs/plans/waves/deferred/README.md +0 -14
  154. package/docs/plans/waves/deferred/encrypted-intake-contracts.md +0 -282
  155. package/docs/plans/waves/deferred/legal-feedback-wave-expansion.md +0 -308
  156. package/docs/plans/waves/deferred/wave-encrypted-intake.md +0 -451
  157. package/docs/plans/waves/design/README.md +0 -5
  158. package/docs/plans/waves/design/wave-1-a1.md +0 -309
  159. package/docs/plans/waves/reviews/README.md +0 -5
  160. package/docs/plans/waves/reviews/wave-0-cont-qa.md +0 -151
  161. package/docs/plans/waves/reviews/wave-1-cont-qa.md +0 -46
  162. package/docs/plans/waves/reviews/wave-10-accessibility-and-design.md +0 -51
  163. package/docs/plans/waves/reviews/wave-10-cont-qa.md +0 -24
  164. package/docs/plans/waves/reviews/wave-10-dashboard-proof.md +0 -46
  165. package/docs/plans/waves/reviews/wave-10-performance-signoff.md +0 -55
  166. package/docs/plans/waves/reviews/wave-10-regression-proof.md +0 -23
  167. package/docs/plans/waves/reviews/wave-10-release-audit.md +0 -31
  168. package/docs/plans/waves/reviews/wave-10-service-proof.md +0 -83
  169. package/docs/plans/waves/reviews/wave-10-word-certification.md +0 -31
  170. package/docs/plans/waves/reviews/wave-18-ai-contract-closure.md +0 -277
  171. package/docs/plans/waves/reviews/wave-18-cont-qa.md +0 -255
  172. package/docs/plans/waves/reviews/wave-18-parity-proof.md +0 -271
  173. package/docs/plans/waves/reviews/wave-19-cont-qa.md +0 -59
  174. package/docs/plans/waves/reviews/wave-2-cont-qa.md +0 -72
  175. package/docs/plans/waves/reviews/wave-20-cont-qa.md +0 -60
  176. package/docs/plans/waves/reviews/wave-25-cont-qa.md +0 -48
  177. package/docs/plans/waves/reviews/wave-28-cont-qa.md +0 -46
  178. package/docs/plans/waves/reviews/wave-29-cont-qa.md +0 -53
  179. package/docs/plans/waves/reviews/wave-3-cont-qa.md +0 -53
  180. package/docs/plans/waves/reviews/wave-3-core-proof.md +0 -77
  181. package/docs/plans/waves/reviews/wave-3-validator-proof.md +0 -73
  182. package/docs/plans/waves/reviews/wave-32-cont-qa.md +0 -43
  183. package/docs/plans/waves/reviews/wave-33-cont-qa.md +0 -526
  184. package/docs/plans/waves/reviews/wave-34-cont-qa.md +0 -100
  185. package/docs/plans/waves/reviews/wave-35-cont-qa.md +0 -145
  186. package/docs/plans/waves/reviews/wave-4-cont-qa.md +0 -47
  187. package/docs/plans/waves/reviews/wave-4-structure-proof.md +0 -69
  188. package/docs/plans/waves/reviews/wave-5-comment-proof.md +0 -158
  189. package/docs/plans/waves/reviews/wave-5-cont-qa.md +0 -68
  190. package/docs/plans/waves/reviews/wave-6-cont-qa.md +0 -416
  191. package/docs/plans/waves/reviews/wave-6-redline-proof.md +0 -130
  192. package/docs/plans/waves/reviews/wave-7-cont-qa.md +0 -82
  193. package/docs/plans/waves/reviews/wave-7-ooxml-compliance.md +0 -85
  194. package/docs/plans/waves/reviews/wave-7-preservation-proof.md +0 -119
  195. package/docs/plans/waves/reviews/wave-7-trust-ux.md +0 -87
  196. package/docs/plans/waves/reviews/wave-8-accessibility-and-design.md +0 -128
  197. package/docs/plans/waves/reviews/wave-8-cont-qa.md +0 -92
  198. package/docs/plans/waves/reviews/wave-8-live-proof.md +0 -140
  199. package/docs/plans/waves/reviews/wave-8-security.md +0 -47
  200. package/docs/plans/waves/reviews/wave-9-editor-embedding.md +0 -39
  201. package/docs/plans/waves/reviews/wave-9-fixture-runner.md +0 -56
  202. package/docs/plans/waves/reviews/wave-9-live-proof.md +0 -105
  203. package/docs/plans/waves/reviews/wave-9-usability-and-performance.md +0 -152
  204. package/docs/plans/waves/specs/README.md +0 -5
  205. package/docs/plans/waves/specs/wave-1-component-boundaries.md +0 -322
  206. package/docs/plans/waves/specs/wave-1-ooxml-contracts.md +0 -323
  207. package/docs/plans/waves/specs/wave-1-review-and-ui-contracts.md +0 -339
  208. package/docs/plans/waves/specs/wave-1-runtime-contracts.md +0 -509
  209. package/docs/plans/waves/wave-19.md +0 -341
  210. package/docs/plans/waves/wave-20.md +0 -308
  211. package/docs/plans/waves/wave-21.md +0 -289
  212. package/docs/plans/waves/wave-22.md +0 -221
  213. package/docs/plans/waves/wave-23.md +0 -295
  214. package/docs/plans/waves/wave-24.md +0 -286
  215. package/docs/plans/waves/wave-25.md +0 -313
  216. package/docs/plans/waves/wave-26.md +0 -300
  217. package/docs/plans/waves/wave-27.md +0 -299
  218. package/docs/plans/waves/wave-28.md +0 -368
  219. package/docs/plans/waves/wave-29.md +0 -303
  220. package/docs/plans/waves/wave-30.md +0 -307
  221. package/docs/plans/waves/wave-31.md +0 -231
  222. package/docs/plans/waves/wave-32.md +0 -152
  223. package/docs/plans/waves/wave-33.md +0 -147
  224. package/docs/plans/waves/wave-34.md +0 -148
  225. package/docs/plans/waves/wave-35.md +0 -141
  226. package/docs/plans/waves/wave-36.md +0 -146
  227. package/docs/plans/xlsx/README.md +0 -14
  228. package/docs/plans/xlsx/xlsx-fixture-corpus-and-certification-plan.md +0 -126
  229. package/docs/reference/cli-reference.md +0 -600
  230. package/docs/reference/coordination-and-closure.md +0 -487
  231. package/docs/reference/deep-research-report (15).md +0 -25
  232. package/docs/reference/docx/README.md +0 -10
  233. package/docs/reference/legal-checklist.md +0 -445
  234. package/docs/reference/live-proof-waves.md +0 -199
  235. package/docs/reference/ooxml-compliance.md +0 -129
  236. package/docs/reference/ooxml-feature-parity-matrix.md +0 -172
  237. package/docs/reference/platform/shared-ooxml-platform-guidance.md +0 -77
  238. package/docs/reference/prototype-agent-prompt-legal-fidelity.md +0 -155
  239. package/docs/reference/public-api.md +0 -456
  240. package/docs/reference/repository-guidance.md +0 -58
  241. package/docs/reference/runtime-config/README.md +0 -182
  242. package/docs/reference/runtime-config/claude.md +0 -110
  243. package/docs/reference/runtime-config/codex.md +0 -82
  244. package/docs/reference/runtime-config/opencode.md +0 -93
  245. package/docs/reference/sample-waves.md +0 -105
  246. package/docs/reference/skills.md +0 -237
  247. package/docs/reference/templates/AGENTS.md +0 -78
  248. package/docs/reference/templates/HEARTBEAT.md +0 -7
  249. package/docs/reference/templates/IDENTITY.md +0 -23
  250. package/docs/reference/templates/SOUL.md +0 -36
  251. package/docs/reference/templates/TOOLS.md +0 -40
  252. package/docs/reference/templates/USER.md +0 -17
  253. package/docs/reference/wave-control.md +0 -184
  254. package/docs/reference/wave-planning-lessons.md +0 -167
  255. package/docs/reference/word-review-editor-frontend-architecture.md +0 -479
  256. package/docs/reference/word-review-editor-ux-guide.md +0 -253
  257. package/docs/reference/xlsx/xlsx-ooxml-compliance.md +0 -137
  258. package/docs/research/agent-context-sources.md +0 -178
  259. package/docs/research/coordination-failure-review.md +0 -290
  260. package/docs/research/docx-react-component/Canonical Document Schema Specification for a React-based Word-compatible Editor.md +0 -2317
  261. package/docs/research/docx-react-component/Feature Compatibility Matrix for a React Word Compatible Legal Editor v1.md +0 -219
  262. package/docs/research/docx-react-component/React Component Architecture and Front-End Structure Specification for a Word-Compatible Legal Review Editor.md +0 -1112
  263. package/docs/research/docx-react-component/document_compatibility_and_testing_spec.md +0 -751
  264. package/docs/research/xlsx/raw/README.md +0 -13
  265. package/docs/roadmap.md +0 -174
  266. package/docs/superpowers/plans/2026-03-28-harness-control-bar.md +0 -677
  267. package/docs/superpowers/specs/2026-03-28-harness-control-bar-design.md +0 -274
  268. package/docs/xlsx-react/README.md +0 -38
  269. package/docs/xlsx-react/agent-llm-interaction-layer-docx-xlsx.md +0 -621
  270. package/docs/xlsx-react/canonical-workbook-model-and-commands.md +0 -948
  271. package/docs/xlsx-react/shared-openxml-editor-platform-docx-xlsx.md +0 -228
  272. package/docs/xlsx-react/spreadsheet-editor-component-architecture.md +0 -809
  273. package/docs/xlsx-react/spreadsheet-editor-frontend-architecture.md +0 -537
  274. package/docs/xlsx-react/spreadsheet-editor-ux-guide.md +0 -520
  275. package/docs/xlsx-react/xlsx-editor-research-pack.md +0 -871
  276. package/docs/xlsx-react/xlsx-fixture-corpus-and-certification-plan.md +0 -436
  277. package/docs/xlsx-react/xlsx-ooxml-compliance.md +0 -320
  278. package/examples/README.md +0 -16
  279. package/memory/MEMORY.md +0 -24
  280. package/pnpm-workspace.yaml +0 -4
  281. package/scripts/check-no-authored-js.sh +0 -13
  282. package/scripts/context7-api-check.sh +0 -65
  283. package/scripts/context7-export-env.sh +0 -42
  284. package/scripts/run-context7-mcp.sh +0 -8
  285. package/scripts/run-workspace-tests.sh +0 -15
  286. package/scripts/start-wave-10-local.sh +0 -189
  287. package/scripts/wave-agent-attach.sh +0 -47
  288. package/scripts/wave-auto-answer.sh +0 -118
  289. package/scripts/wave-dashboard-attach.sh +0 -13
  290. package/scripts/wave-launch.sh +0 -273
  291. package/scripts/wave-overnight-supervisor.sh +0 -145
  292. package/scripts/wave-status.sh +0 -379
  293. package/scripts/wave-watch.sh +0 -231
  294. package/services/README.md +0 -17
  295. package/services/openxml-validator/Dockerfile +0 -29
  296. package/services/openxml-validator/OpenXmlValidator.Api.csproj +0 -12
  297. package/services/openxml-validator/Program.cs +0 -436
  298. package/services/openxml-validator/README.md +0 -152
  299. package/services/openxml-validator/railway.json +0 -16
  300. package/services/react-word-editor/.tmp-a4/src/api/public-types.ts +0 -318
  301. package/services/react-word-editor/.tmp-a4/src/ui/WordReviewEditor.tsx +0 -1302
  302. package/services/react-word-editor/.tmp-a4/src/ui/editor-surface/editor-surface.tsx +0 -546
  303. package/services/react-word-editor/.tmp-a4/test/ui/word-review-editor.test.tsx +0 -146
  304. package/services/react-word-editor/.tmp-a4-build/src/api/public-types.js +0 -2
  305. package/services/react-word-editor/.tmp-a4-build/src/ui/WordReviewEditor.js +0 -818
  306. package/services/react-word-editor/.tmp-a4-build/src/ui/editor-surface/editor-surface.js +0 -229
  307. package/services/react-word-editor/.tmp-a4-build/test/ui/word-review-editor.test.js +0 -121
  308. package/services/react-word-editor/.tmp-wave-4-a3-tsconfig.json +0 -21
  309. package/services/react-word-editor/.tmp-wave-4-a3-tsconfig.tsbuildinfo +0 -1
  310. package/services/react-word-editor/Dockerfile +0 -26
  311. package/services/react-word-editor/README.md +0 -254
  312. package/services/react-word-editor/app/api/certification/route.ts +0 -79
  313. package/services/react-word-editor/app/api/demo-sessions/route.ts +0 -109
  314. package/services/react-word-editor/app/api/deploy-health/route.ts +0 -23
  315. package/services/react-word-editor/app/api/exports/[exportId]/route.ts +0 -34
  316. package/services/react-word-editor/app/api/exports/route.ts +0 -81
  317. package/services/react-word-editor/app/api/fixtures/[fixtureId]/run/route.ts +0 -100
  318. package/services/react-word-editor/app/api/health/route.ts +0 -70
  319. package/services/react-word-editor/app/api/runs/[runId]/route.ts +0 -36
  320. package/services/react-word-editor/app/api/scenarios/[scenarioId]/run/route.ts +0 -85
  321. package/services/react-word-editor/app/api/sessions/[sessionId]/route.ts +0 -199
  322. package/services/react-word-editor/app/api/sessions/[sessionId]/source/route.ts +0 -45
  323. package/services/react-word-editor/app/api/uploads/route.ts +0 -70
  324. package/services/react-word-editor/app/api/validate/route.ts +0 -310
  325. package/services/react-word-editor/app/certification/[runId]/page.tsx +0 -14
  326. package/services/react-word-editor/app/certification/page.tsx +0 -32
  327. package/services/react-word-editor/app/dashboard/page.tsx +0 -7
  328. package/services/react-word-editor/app/demo/page.tsx +0 -30
  329. package/services/react-word-editor/app/demo/prototype-client.tsx +0 -1080
  330. package/services/react-word-editor/app/editor/[sessionId]/page.tsx +0 -33
  331. package/services/react-word-editor/app/fixtures/page.tsx +0 -7
  332. package/services/react-word-editor/app/globals.css +0 -121
  333. package/services/react-word-editor/app/layout.tsx +0 -32
  334. package/services/react-word-editor/app/page.tsx +0 -30
  335. package/services/react-word-editor/app/runs/[runId]/page.tsx +0 -34
  336. package/services/react-word-editor/app/wave-10-word-review/page.tsx +0 -7
  337. package/services/react-word-editor/components/harness-control-bar.tsx +0 -289
  338. package/services/react-word-editor/components/harness-editor-session-client.tsx +0 -1214
  339. package/services/react-word-editor/components/harness-workspace-page.tsx +0 -715
  340. package/services/react-word-editor/components/reduced-motion-toggle.tsx +0 -79
  341. package/services/react-word-editor/components/workspace-certification-panel.tsx +0 -307
  342. package/services/react-word-editor/lib/certification-bundle.ts +0 -796
  343. package/services/react-word-editor/lib/certification-store.ts +0 -661
  344. package/services/react-word-editor/lib/demo-fixtures.test.mjs +0 -195
  345. package/services/react-word-editor/lib/demo-fixtures.ts +0 -1519
  346. package/services/react-word-editor/lib/editor-session-summary.test.mjs +0 -68
  347. package/services/react-word-editor/lib/editor-session-summary.ts +0 -14
  348. package/services/react-word-editor/lib/editor-session.ts +0 -228
  349. package/services/react-word-editor/lib/exports-route.test.mjs +0 -32
  350. package/services/react-word-editor/lib/harness-client.ts +0 -347
  351. package/services/react-word-editor/lib/harness-config.json +0 -30
  352. package/services/react-word-editor/lib/harness-config.test.mjs +0 -31
  353. package/services/react-word-editor/lib/harness-config.ts +0 -21
  354. package/services/react-word-editor/lib/harness-editor-datastore.test.mjs +0 -220
  355. package/services/react-word-editor/lib/harness-editor-datastore.ts +0 -161
  356. package/services/react-word-editor/lib/private-mode.test.mjs +0 -42
  357. package/services/react-word-editor/lib/private-mode.ts +0 -61
  358. package/services/react-word-editor/lib/regression-report.test.mjs +0 -352
  359. package/services/react-word-editor/lib/regression-report.ts +0 -896
  360. package/services/react-word-editor/lib/run-artifacts.ts +0 -934
  361. package/services/react-word-editor/lib/run-history.ts +0 -755
  362. package/services/react-word-editor/lib/scenario-artifacts.test.mjs +0 -41
  363. package/services/react-word-editor/lib/scenario-artifacts.ts +0 -44
  364. package/services/react-word-editor/lib/storage.ts +0 -953
  365. package/services/react-word-editor/lib/validator-client.test.mjs +0 -54
  366. package/services/react-word-editor/lib/validator-client.ts +0 -95
  367. package/services/react-word-editor/lib/workspace-navigation.ts +0 -79
  368. package/services/react-word-editor/middleware.ts +0 -35
  369. package/services/react-word-editor/next-env.d.ts +0 -6
  370. package/services/react-word-editor/next.config.mjs +0 -15
  371. package/services/react-word-editor/package.json +0 -38
  372. package/services/react-word-editor/postcss.config.mjs +0 -8
  373. package/services/react-word-editor/railway.json +0 -21
  374. package/services/react-word-editor/scripts/wave-10-certification.mjs +0 -101
  375. package/services/react-word-editor/scripts/wave-9-live-usability-pilot.mjs +0 -911
  376. package/services/react-word-editor/tsconfig.json +0 -39
  377. package/services/react-word-editor/tsconfig.tsbuildinfo +0 -1
  378. package/skills/README.md +0 -48
  379. package/skills/domain-docx-compatibility/SKILL.md +0 -44
  380. package/skills/domain-docx-compatibility/skill.json +0 -19
  381. package/skills/domain-editor-architecture/SKILL.md +0 -49
  382. package/skills/domain-editor-architecture/skill.json +0 -19
  383. package/skills/domain-legal-review/SKILL.md +0 -39
  384. package/skills/domain-legal-review/skill.json +0 -19
  385. package/skills/provider-aws/SKILL.md +0 -117
  386. package/skills/provider-aws/adapters/claude.md +0 -1
  387. package/skills/provider-aws/adapters/codex.md +0 -1
  388. package/skills/provider-aws/references/service-verification.md +0 -39
  389. package/skills/provider-aws/skill.json +0 -54
  390. package/skills/provider-custom-deploy/SKILL.md +0 -64
  391. package/skills/provider-custom-deploy/skill.json +0 -50
  392. package/skills/provider-docker-compose/SKILL.md +0 -96
  393. package/skills/provider-docker-compose/adapters/local.md +0 -1
  394. package/skills/provider-docker-compose/skill.json +0 -53
  395. package/skills/provider-github-release/SKILL.md +0 -121
  396. package/skills/provider-github-release/adapters/claude.md +0 -1
  397. package/skills/provider-github-release/adapters/codex.md +0 -1
  398. package/skills/provider-github-release/skill.json +0 -55
  399. package/skills/provider-kubernetes/SKILL.md +0 -143
  400. package/skills/provider-kubernetes/adapters/claude.md +0 -1
  401. package/skills/provider-kubernetes/adapters/codex.md +0 -1
  402. package/skills/provider-kubernetes/references/kubectl-patterns.md +0 -58
  403. package/skills/provider-kubernetes/skill.json +0 -52
  404. package/skills/provider-railway/SKILL.md +0 -123
  405. package/skills/provider-railway/adapters/claude.md +0 -1
  406. package/skills/provider-railway/adapters/codex.md +0 -1
  407. package/skills/provider-railway/adapters/local.md +0 -1
  408. package/skills/provider-railway/adapters/opencode.md +0 -1
  409. package/skills/provider-railway/references/verification-commands.md +0 -39
  410. package/skills/provider-railway/skill.json +0 -71
  411. package/skills/provider-ssh-manual/SKILL.md +0 -97
  412. package/skills/provider-ssh-manual/skill.json +0 -54
  413. package/skills/repo-coding-rules/SKILL.md +0 -55
  414. package/skills/repo-coding-rules/skill.json +0 -34
  415. package/skills/role-cont-eval/SKILL.md +0 -91
  416. package/skills/role-cont-eval/adapters/codex.md +0 -1
  417. package/skills/role-cont-eval/skill.json +0 -36
  418. package/skills/role-cont-qa/SKILL.md +0 -100
  419. package/skills/role-cont-qa/adapters/claude.md +0 -1
  420. package/skills/role-cont-qa/skill.json +0 -36
  421. package/skills/role-deploy/SKILL.md +0 -97
  422. package/skills/role-deploy/skill.json +0 -36
  423. package/skills/role-design/SKILL.md +0 -50
  424. package/skills/role-design/skill.json +0 -36
  425. package/skills/role-documentation/SKILL.md +0 -76
  426. package/skills/role-documentation/skill.json +0 -36
  427. package/skills/role-implementation/SKILL.md +0 -45
  428. package/skills/role-implementation/skill.json +0 -36
  429. package/skills/role-infra/SKILL.md +0 -81
  430. package/skills/role-infra/skill.json +0 -36
  431. package/skills/role-integration/SKILL.md +0 -91
  432. package/skills/role-integration/skill.json +0 -36
  433. package/skills/role-planner/SKILL.md +0 -39
  434. package/skills/role-planner/skill.json +0 -21
  435. package/skills/role-research/SKILL.md +0 -65
  436. package/skills/role-research/skill.json +0 -36
  437. package/skills/role-security/SKILL.md +0 -60
  438. package/skills/role-security/skill.json +0 -36
  439. package/skills/runtime-claude/SKILL.md +0 -66
  440. package/skills/runtime-claude/skill.json +0 -36
  441. package/skills/runtime-codex/SKILL.md +0 -58
  442. package/skills/runtime-codex/skill.json +0 -36
  443. package/skills/runtime-local/SKILL.md +0 -46
  444. package/skills/runtime-local/skill.json +0 -36
  445. package/skills/runtime-opencode/SKILL.md +0 -58
  446. package/skills/runtime-opencode/skill.json +0 -36
  447. package/skills/signal-hygiene/SKILL.md +0 -51
  448. package/skills/signal-hygiene/skill.json +0 -20
  449. package/skills/tui-design/SKILL.md +0 -77
  450. package/skills/tui-design/references/tui-design.md +0 -259
  451. package/skills/tui-design/skill.json +0 -36
  452. package/skills/wave-core/SKILL.md +0 -141
  453. package/skills/wave-core/references/marker-syntax.md +0 -70
  454. package/skills/wave-core/skill.json +0 -35
  455. package/test/README.md +0 -16
  456. package/test/core/formatting-commands.test.ts +0 -285
  457. package/test/core/image-commands.test.ts +0 -298
  458. package/test/core/mapping.test.ts +0 -186
  459. package/test/core/text-commands.test.ts +0 -176
  460. package/test/fixtures/docx/F01-basic-contract.docx +0 -0
  461. package/test/fixtures/docx/F01-basic-contract.md +0 -33
  462. package/test/fixtures/docx/F02-headings-styles.docx +0 -0
  463. package/test/fixtures/docx/F02-headings-styles.md +0 -33
  464. package/test/fixtures/docx/F03-legal-outline-numbering.docx +0 -0
  465. package/test/fixtures/docx/F03-legal-outline-numbering.md +0 -34
  466. package/test/fixtures/docx/F04-restart-numbering-schedules.docx +0 -0
  467. package/test/fixtures/docx/F04-restart-numbering-schedules.md +0 -33
  468. package/test/fixtures/docx/F05-table-heavy-agreement.docx +0 -0
  469. package/test/fixtures/docx/F05-table-heavy-agreement.md +0 -34
  470. package/test/fixtures/docx/F06-merged-cells-signature-table.docx +0 -0
  471. package/test/fixtures/docx/F06-merged-cells-signature-table.md +0 -34
  472. package/test/fixtures/docx/F07-inline-images-exhibit.docx +0 -0
  473. package/test/fixtures/docx/F07-inline-images-exhibit.md +0 -34
  474. package/test/fixtures/docx/F08-hyperlinks.docx +0 -0
  475. package/test/fixtures/docx/F08-hyperlinks.md +0 -33
  476. package/test/fixtures/docx/F09-comments-single-paragraph.docx +0 -0
  477. package/test/fixtures/docx/F09-comments-single-paragraph.md +0 -33
  478. package/test/fixtures/docx/F10-threaded-comments-resolve.docx +0 -0
  479. package/test/fixtures/docx/F10-threaded-comments-resolve.md +0 -33
  480. package/test/fixtures/docx/F11-redlines-basic.docx +0 -0
  481. package/test/fixtures/docx/F11-redlines-basic.md +0 -33
  482. package/test/fixtures/docx/F12-redlines-paragraph-joins-splits.docx +0 -0
  483. package/test/fixtures/docx/F12-redlines-paragraph-joins-splits.md +0 -33
  484. package/test/fixtures/docx/F13-comments-on-deleted-text.docx +0 -0
  485. package/test/fixtures/docx/F13-comments-on-deleted-text.md +0 -33
  486. package/test/fixtures/docx/F14-revisions-in-tables-and-lists.docx +0 -0
  487. package/test/fixtures/docx/F14-revisions-in-tables-and-lists.md +0 -33
  488. package/test/fixtures/docx/F15-sections-headers-footers.docx +0 -0
  489. package/test/fixtures/docx/F15-sections-headers-footers.md +0 -33
  490. package/test/fixtures/docx/F16-footnotes-endnotes.docx +0 -0
  491. package/test/fixtures/docx/F16-footnotes-endnotes.md +0 -33
  492. package/test/fixtures/docx/F17-fields-and-toc.docx +0 -0
  493. package/test/fixtures/docx/F17-fields-and-toc.md +0 -33
  494. package/test/fixtures/docx/F18-content-controls-template.docx +0 -0
  495. package/test/fixtures/docx/F18-content-controls-template.md +0 -33
  496. package/test/fixtures/docx/F19-custom-xml-doc-assembly.docx +0 -0
  497. package/test/fixtures/docx/F19-custom-xml-doc-assembly.md +0 -35
  498. package/test/fixtures/docx/F20-unknown-ooxml-and-alternatecontent.docx +0 -0
  499. package/test/fixtures/docx/F20-unknown-ooxml-and-alternatecontent.md +0 -33
  500. package/test/fixtures/docx/F21-malformed-broken-docx.docx +0 -0
  501. package/test/fixtures/docx/F21-malformed-broken-docx.md +0 -33
  502. package/test/fixtures/docx/README.md +0 -74
  503. package/test/fixtures/docx/certification-manifest.json +0 -104
  504. package/test/fixtures/docx/fixtures.manifest.json +0 -196
  505. package/test/fixtures/encrypted-docx/README.md +0 -27
  506. package/test/fixtures/encrypted-docx/certification-manifest.json +0 -9
  507. package/test/fixtures/encrypted-docx/fixtures.manifest.json +0 -47
  508. package/test/fixtures/scenarios/docx/README.md +0 -25
  509. package/test/fixtures/scenarios/docx/S01-sow-template.docx +0 -0
  510. package/test/fixtures/scenarios/docx/S01-sow-template.md +0 -30
  511. package/test/fixtures/scenarios/docx/S02-bw-partner-user-licence-agreement-redlines.docx +0 -0
  512. package/test/fixtures/scenarios/docx/S02-bw-partner-user-licence-agreement-redlines.md +0 -32
  513. package/test/fixtures/scenarios/docx/scenario-manifest.json +0 -53
  514. package/test/formats/xlsx/io/xlsx-import.test.ts +0 -766
  515. package/test/formats/xlsx/model/workbook.test.ts +0 -669
  516. package/test/helpers/dom-setup.ts +0 -124
  517. package/test/io/comment-roundtrip.test.ts +0 -272
  518. package/test/io/complex-content-roundtrip.test.ts +0 -632
  519. package/test/io/docx-compatibility-regression.test.ts +0 -199
  520. package/test/io/docx-session.test.ts +0 -1495
  521. package/test/io/footnotes-roundtrip.test.ts +0 -318
  522. package/test/io/headers-footers-roundtrip.test.ts +0 -547
  523. package/test/io/numbering-roundtrip.test.ts +0 -234
  524. package/test/io/package-reader.test.ts +0 -199
  525. package/test/io/paragraph-properties-roundtrip.test.ts +0 -129
  526. package/test/io/preserved-package-roundtrip.test.ts +0 -365
  527. package/test/io/property-completeness.test.ts +0 -292
  528. package/test/io/revision-roundtrip.test.ts +0 -347
  529. package/test/io/structural-blocks.test.ts +0 -202
  530. package/test/io/table-media-roundtrip.test.ts +0 -448
  531. package/test/io/table-properties-roundtrip.test.ts +0 -569
  532. package/test/io/table-roundtrip.test.ts +0 -302
  533. package/test/io/text-roundtrip.test.ts +0 -344
  534. package/test/model/canonical-document.test.ts +0 -285
  535. package/test/preservation/opaque-fragment-store.test.ts +0 -121
  536. package/test/preservation/package-preservation.test.ts +0 -395
  537. package/test/preservation/store.test.ts +0 -84
  538. package/test/review/comment-remapping.test.ts +0 -220
  539. package/test/review/comment-store.test.ts +0 -180
  540. package/test/review/move-revisions.test.ts +0 -143
  541. package/test/review/property-change-revisions.test.ts +0 -225
  542. package/test/review/revision-actions.test.ts +0 -330
  543. package/test/review/revision-store.test.ts +0 -193
  544. package/test/runtime/session-capabilities.test.ts +0 -260
  545. package/test/runtime/table-commands.test.ts +0 -356
  546. package/test/runtime/table-schema.test.ts +0 -221
  547. package/test/runtime/tracked-changes-toggle.test.ts +0 -107
  548. package/test/ui/comment-review-surface.test.tsx +0 -114
  549. package/test/ui/reduced-motion-toggle.test.tsx +0 -137
  550. package/test/ui/word-review-editor.imported-scenarios.test.tsx +0 -169
  551. package/test/ui/word-review-editor.interaction.test.tsx +0 -1198
  552. package/test/ui/word-review-editor.test.js +0 -188
  553. package/test/ui/word-review-editor.test.tsx +0 -280
  554. package/test/ui-tailwind/search-plugin.test.ts +0 -286
  555. package/test/validation/compatibility-engine.test.ts +0 -336
  556. package/test/validation/compatibility-report.test.ts +0 -189
  557. package/test/validation/low-priority-word-surfaces.test.ts +0 -282
  558. package/test/validation/malformed-doc.test.ts +0 -113
  559. package/test-results/.last-run.json +0 -4
  560. package/wave.config.json +0 -406
@@ -1,1302 +0,0 @@
1
- import React, {
2
- forwardRef,
3
- useEffect,
4
- useImperativeHandle,
5
- useRef,
6
- useSyncExternalStore,
7
- } from "react";
8
-
9
- import type {
10
- AddCommentParams,
11
- CompatibilityReport,
12
- DocumentStats,
13
- EditorAnchorProjection,
14
- EditorDatastoreAdapter,
15
- EditorError,
16
- EditorWarning,
17
- ExportDocxOptions,
18
- ExportResult,
19
- ExternalDocumentSource,
20
- PersistedEditorSnapshot,
21
- SelectionSnapshot,
22
- WordReviewEditorEvent,
23
- WordReviewEditorProps,
24
- WordReviewEditorRef,
25
- } from "../api/public-types";
26
- import {
27
- EditorSurface,
28
- type EditorSurfaceModel,
29
- } from "./editor-surface/editor-surface";
30
-
31
- type Unsubscribe = () => void;
32
-
33
- interface RuntimeViewportDelegate {
34
- focus(): void;
35
- blur(): void;
36
- }
37
-
38
- interface RuntimeCommentThread {
39
- commentId: string;
40
- body: string;
41
- status: "open" | "resolved" | "detached";
42
- anchor: EditorAnchorProjection;
43
- }
44
-
45
- interface RuntimeTrackedChange {
46
- changeId: string;
47
- kind: "insertion" | "deletion";
48
- status: "active" | "accepted" | "rejected" | "detached";
49
- }
50
-
51
- interface RuntimeCommandStateSnapshot {
52
- canUndo: boolean;
53
- canRedo: boolean;
54
- canAddComment: boolean;
55
- canAcceptAllChanges: boolean;
56
- canRejectAllChanges: boolean;
57
- canExport: boolean;
58
- canSaveSnapshot: boolean;
59
- }
60
-
61
- interface RuntimeCompatibilityPanelSnapshot {
62
- report: CompatibilityReport;
63
- blockExport: boolean;
64
- preserveOnlyCount: number;
65
- fatalCount: number;
66
- }
67
-
68
- interface RuntimeCommentSidebarSnapshot {
69
- commentIdsInOrder: string[];
70
- threads: Record<string, RuntimeCommentThread>;
71
- openCommentId?: string;
72
- }
73
-
74
- interface RuntimeTrackedChangesSnapshot {
75
- changeIdsInOrder: string[];
76
- changes: Record<string, RuntimeTrackedChange>;
77
- }
78
-
79
- interface RuntimeRenderSnapshot {
80
- documentId: string;
81
- sessionId: string;
82
- sourceLabel?: string;
83
- revisionToken: string;
84
- isReady: boolean;
85
- isDirty: boolean;
86
- readOnly: boolean;
87
- selection: SelectionSnapshot;
88
- documentStats: DocumentStats;
89
- comments: RuntimeCommentSidebarSnapshot;
90
- trackedChanges: RuntimeTrackedChangesSnapshot;
91
- compatibility: RuntimeCompatibilityPanelSnapshot;
92
- warnings: EditorWarning[];
93
- fatalError?: EditorError;
94
- commandState: RuntimeCommandStateSnapshot;
95
- }
96
-
97
- interface RuntimeOptions {
98
- documentId: string;
99
- readOnly: boolean;
100
- initialSource?: ExternalDocumentSource;
101
- initialSourceLabel?: string;
102
- datastore?: EditorDatastoreAdapter;
103
- onEvent?: (event: WordReviewEditorEvent) => void;
104
- onWarning?: (warning: EditorWarning) => void;
105
- onError?: (error: EditorError) => void;
106
- }
107
-
108
- interface RuntimeInitializeOptions {
109
- documentId: string;
110
- initialSource?: ExternalDocumentSource;
111
- datastore?: EditorDatastoreAdapter;
112
- }
113
-
114
- interface RuntimeSourceDescriptor {
115
- source: ExternalDocumentSource;
116
- sourceKind: "docx" | "snapshot" | "datastore";
117
- }
118
-
119
- interface WordReviewEditorRuntime {
120
- subscribe(listener: () => void): Unsubscribe;
121
- getRenderSnapshot(): RuntimeRenderSnapshot;
122
- initialize(options: RuntimeInitializeOptions): Promise<void>;
123
- replaceExternalSource(
124
- revisionToken: string,
125
- source: ExternalDocumentSource,
126
- ): Promise<void>;
127
- focus(): void;
128
- blur(): void;
129
- undo(): void;
130
- redo(): void;
131
- addComment(params: AddCommentParams): string;
132
- openComment(commentId: string): void;
133
- resolveComment(commentId: string): void;
134
- acceptChange(changeId: string): void;
135
- rejectChange(changeId: string): void;
136
- acceptAllChanges(): void;
137
- rejectAllChanges(): void;
138
- saveSnapshot(isAutosave: boolean): Promise<PersistedEditorSnapshot>;
139
- getPersistedSnapshot(): PersistedEditorSnapshot;
140
- getCompatibilityReport(): CompatibilityReport;
141
- getWarnings(): EditorWarning[];
142
- exportDocx(options?: ExportDocxOptions): Promise<ExportResult>;
143
- attachViewport(delegate: RuntimeViewportDelegate | null): void;
144
- destroy(): void;
145
- }
146
-
147
- let runtimeInstanceCounter = 0;
148
- let commentCounter = 0;
149
-
150
- function createNowIsoString() {
151
- return new Date().toISOString();
152
- }
153
-
154
- function createSessionId(documentId: string) {
155
- runtimeInstanceCounter += 1;
156
- return `wre-${documentId}-${runtimeInstanceCounter}`;
157
- }
158
-
159
- function toUint8Array(
160
- bytes: Uint8Array | ArrayBuffer,
161
- ): Uint8Array<ArrayBufferLike> {
162
- return bytes instanceof Uint8Array ? bytes : new Uint8Array(bytes);
163
- }
164
-
165
- function createEmptyAnchorProjection(): EditorAnchorProjection {
166
- return {
167
- kind: "range",
168
- from: 0,
169
- to: 0,
170
- assoc: { start: 1, end: 1 },
171
- };
172
- }
173
-
174
- function createEmptySelectionSnapshot(): SelectionSnapshot {
175
- return {
176
- anchor: 0,
177
- head: 0,
178
- isCollapsed: true,
179
- activeRange: createEmptyAnchorProjection(),
180
- };
181
- }
182
-
183
- function createCompatibilityReport(overrides?: Partial<CompatibilityReport>): CompatibilityReport {
184
- return {
185
- reportVersion: "compatibility-report/1",
186
- generatedAt: createNowIsoString(),
187
- blockExport: false,
188
- featureEntries: [],
189
- warnings: [],
190
- errors: [],
191
- ...overrides,
192
- };
193
- }
194
-
195
- function createEmptyCanonicalDocument(documentId: string) {
196
- return {
197
- schemaVersion: "cds/1.0.0" as const,
198
- docId: `doc-${documentId}`,
199
- metadata: {},
200
- styles: {},
201
- numbering: {},
202
- media: {},
203
- content: [],
204
- review: {
205
- comments: {},
206
- revisions: {},
207
- },
208
- preservation: {},
209
- diagnostics: {},
210
- };
211
- }
212
-
213
- function createPersistedSnapshot(
214
- documentId: string,
215
- compatibility: CompatibilityReport,
216
- ): PersistedEditorSnapshot {
217
- const now = createNowIsoString();
218
-
219
- return {
220
- snapshotVersion: "persisted-editor-snapshot/1",
221
- schemaVersion: "cds/1.0.0",
222
- documentId,
223
- docId: `doc-${documentId}`,
224
- createdAt: now,
225
- updatedAt: now,
226
- savedAt: now,
227
- editorBuild: "wave-2-shell",
228
- canonicalDocument: createEmptyCanonicalDocument(documentId),
229
- compatibility,
230
- warningLog: compatibility.warnings,
231
- };
232
- }
233
-
234
- function createRuntimeError(
235
- code: EditorError["code"],
236
- message: string,
237
- source: EditorError["source"],
238
- isFatal: boolean,
239
- details?: Record<string, unknown>,
240
- ): EditorError {
241
- return {
242
- errorId: `${code}-${Math.random().toString(36).slice(2, 10)}`,
243
- code,
244
- message,
245
- isFatal,
246
- source,
247
- details,
248
- };
249
- }
250
-
251
- function buildDocumentStats(
252
- source: ExternalDocumentSource | undefined,
253
- commentCount: number,
254
- changeCount: number,
255
- ): DocumentStats {
256
- if (!source) {
257
- return {
258
- wordCount: 0,
259
- paragraphCount: 0,
260
- commentCount,
261
- changeCount,
262
- };
263
- }
264
-
265
- if (source.kind === "snapshot") {
266
- const content = source.snapshot.canonicalDocument.content;
267
- const paragraphCount = Array.isArray(content) ? content.length : 0;
268
-
269
- return {
270
- wordCount: Math.max(32, source.snapshot.warningLog.length * 8),
271
- paragraphCount: Math.max(1, paragraphCount),
272
- commentCount,
273
- changeCount,
274
- };
275
- }
276
-
277
- const byteLength = toUint8Array(source.bytes).byteLength;
278
-
279
- return {
280
- wordCount: Math.max(32, Math.floor(byteLength / 6)),
281
- paragraphCount: Math.max(1, Math.floor(byteLength / 320)),
282
- commentCount,
283
- changeCount,
284
- };
285
- }
286
-
287
- function buildCompatibilityPanelSnapshot(
288
- report: CompatibilityReport,
289
- ): RuntimeCompatibilityPanelSnapshot {
290
- return {
291
- report,
292
- blockExport: report.blockExport,
293
- preserveOnlyCount: report.featureEntries.filter(
294
- (entry) => entry.featureClass === "preserve-only",
295
- ).length,
296
- fatalCount: report.featureEntries.filter(
297
- (entry) => entry.featureClass === "unsupported-fatal",
298
- ).length,
299
- };
300
- }
301
-
302
- function createBaseRenderSnapshot(
303
- documentId: string,
304
- readOnly: boolean,
305
- ): RuntimeRenderSnapshot {
306
- const compatibility = createCompatibilityReport();
307
-
308
- return {
309
- documentId,
310
- sessionId: createSessionId(documentId),
311
- revisionToken: "initial",
312
- isReady: false,
313
- isDirty: false,
314
- readOnly,
315
- selection: createEmptySelectionSnapshot(),
316
- documentStats: {
317
- wordCount: 0,
318
- paragraphCount: 0,
319
- commentCount: 0,
320
- changeCount: 0,
321
- },
322
- comments: {
323
- commentIdsInOrder: [],
324
- threads: {},
325
- },
326
- trackedChanges: {
327
- changeIdsInOrder: [],
328
- changes: {},
329
- },
330
- compatibility: buildCompatibilityPanelSnapshot(compatibility),
331
- warnings: [],
332
- commandState: {
333
- canUndo: false,
334
- canRedo: false,
335
- canAddComment: !readOnly,
336
- canAcceptAllChanges: false,
337
- canRejectAllChanges: false,
338
- canExport: true,
339
- canSaveSnapshot: false,
340
- },
341
- };
342
- }
343
-
344
- function resolveInitialExplicitSource(
345
- props: Pick<
346
- WordReviewEditorProps,
347
- "initialDocx" | "initialSnapshot" | "initialSourceLabel"
348
- >,
349
- ): ExternalDocumentSource | undefined {
350
- if (props.initialSnapshot) {
351
- return {
352
- kind: "snapshot",
353
- snapshot: props.initialSnapshot,
354
- sourceLabel: props.initialSourceLabel,
355
- };
356
- }
357
-
358
- if (props.initialDocx) {
359
- return {
360
- kind: "docx",
361
- bytes: props.initialDocx,
362
- sourceLabel: props.initialSourceLabel,
363
- };
364
- }
365
-
366
- return undefined;
367
- }
368
-
369
- function buildEditorSurfaceModel(
370
- snapshot: RuntimeRenderSnapshot,
371
- reviewMode: NonNullable<WordReviewEditorProps["reviewMode"]>,
372
- markupDisplay: NonNullable<WordReviewEditorProps["markupDisplay"]>,
373
- ): EditorSurfaceModel {
374
- const features = snapshot.compatibility.report.featureEntries.map((entry) => ({
375
- featureEntryId: entry.featureEntryId,
376
- featureClass: entry.featureClass,
377
- message: entry.message,
378
- }));
379
-
380
- const comments = snapshot.comments.commentIdsInOrder.map((commentId) => {
381
- const thread = snapshot.comments.threads[commentId];
382
-
383
- return {
384
- commentId,
385
- preview: thread.body.slice(0, 72) || "Untitled comment",
386
- status: thread.status,
387
- isActive: snapshot.comments.openCommentId === commentId,
388
- };
389
- });
390
-
391
- const trackedChanges = snapshot.trackedChanges.changeIdsInOrder.map(
392
- (changeId) => {
393
- const change = snapshot.trackedChanges.changes[changeId];
394
-
395
- return {
396
- changeId,
397
- label: `${change.kind} ${changeId}`,
398
- status: change.status,
399
- };
400
- },
401
- );
402
-
403
- const warningSummary =
404
- snapshot.warnings.length === 0
405
- ? "No active runtime warnings"
406
- : `${snapshot.warnings.length} active warning${
407
- snapshot.warnings.length === 1 ? "" : "s"
408
- }`;
409
-
410
- const compatibilitySummary = snapshot.compatibility.blockExport
411
- ? "Export is blocked pending compatibility remediation"
412
- : snapshot.compatibility.report.featureEntries.length === 0
413
- ? "No compatibility findings in the current snapshot"
414
- : `${snapshot.compatibility.report.featureEntries.length} compatibility entr${
415
- snapshot.compatibility.report.featureEntries.length === 1 ? "y" : "ies"
416
- } tracked`;
417
-
418
- const activeRange = snapshot.selection.activeRange;
419
- const selectionLabel =
420
- activeRange.kind === "range"
421
- ? `${activeRange.from} → ${activeRange.to}`
422
- : activeRange.kind === "node"
423
- ? `Node at ${activeRange.at}`
424
- : `Detached (${activeRange.reason})`;
425
-
426
- return {
427
- documentId: snapshot.documentId,
428
- sessionId: snapshot.sessionId,
429
- sourceLabel: snapshot.sourceLabel,
430
- isReady: snapshot.isReady,
431
- isDirty: snapshot.isDirty,
432
- readOnly: snapshot.readOnly,
433
- reviewMode,
434
- markupDisplay,
435
- documentStats: snapshot.documentStats,
436
- selectionLabel,
437
- warningSummary,
438
- compatibilitySummary,
439
- warnings: snapshot.warnings.map((warning) => ({
440
- warningId: warning.warningId,
441
- severity: warning.severity,
442
- code: warning.code,
443
- message: warning.message,
444
- })),
445
- comments,
446
- trackedChanges,
447
- features,
448
- };
449
- }
450
-
451
- export function __unstable_createWordReviewEditorRuntime(
452
- options: RuntimeOptions,
453
- ): WordReviewEditorRuntime {
454
- let listeners = new Set<() => void>();
455
- let viewportDelegate: RuntimeViewportDelegate | null = null;
456
- let renderSnapshot = createBaseRenderSnapshot(options.documentId, options.readOnly);
457
- let persistedSnapshot = createPersistedSnapshot(
458
- options.documentId,
459
- renderSnapshot.compatibility.report,
460
- );
461
- let lastExport: Uint8Array<ArrayBufferLike> = new Uint8Array();
462
- let destroyed = false;
463
-
464
- const notify = () => {
465
- if (destroyed) {
466
- return;
467
- }
468
-
469
- listeners.forEach((listener) => listener());
470
- };
471
-
472
- const emitEvent = (event: WordReviewEditorEvent) => {
473
- options.onEvent?.(event);
474
- options.datastore?.logEvent?.({
475
- type: event.type,
476
- documentId: event.documentId,
477
- });
478
- };
479
-
480
- const emitError = (error: EditorError) => {
481
- options.onError?.(error);
482
- emitEvent({
483
- type: "error",
484
- documentId: renderSnapshot.documentId,
485
- error,
486
- });
487
- };
488
-
489
- const setDirty = (isDirty: boolean) => {
490
- if (renderSnapshot.isDirty === isDirty) {
491
- return;
492
- }
493
-
494
- renderSnapshot = {
495
- ...renderSnapshot,
496
- isDirty,
497
- };
498
- emitEvent({
499
- type: "dirty_changed",
500
- documentId: renderSnapshot.documentId,
501
- isDirty,
502
- });
503
- };
504
-
505
- const setRenderSnapshot = (
506
- nextSnapshot:
507
- | RuntimeRenderSnapshot
508
- | ((current: RuntimeRenderSnapshot) => RuntimeRenderSnapshot),
509
- ) => {
510
- renderSnapshot =
511
- typeof nextSnapshot === "function" ? nextSnapshot(renderSnapshot) : nextSnapshot;
512
-
513
- persistedSnapshot = {
514
- ...persistedSnapshot,
515
- documentId: renderSnapshot.documentId,
516
- updatedAt: createNowIsoString(),
517
- warningLog: renderSnapshot.warnings,
518
- compatibility: renderSnapshot.compatibility.report,
519
- };
520
-
521
- notify();
522
- };
523
-
524
- const applySource = (
525
- descriptor: RuntimeSourceDescriptor | undefined,
526
- revisionToken: string,
527
- ) => {
528
- const source = descriptor?.source;
529
- const sourceLabel = source?.sourceLabel ?? options.initialSourceLabel;
530
- const compatibility =
531
- source?.kind === "snapshot"
532
- ? source.snapshot.compatibility
533
- : createCompatibilityReport();
534
- const warnings =
535
- source?.kind === "snapshot" ? source.snapshot.warningLog : compatibility.warnings;
536
- const comments: RuntimeCommentSidebarSnapshot = {
537
- commentIdsInOrder: [],
538
- threads: {},
539
- };
540
- const trackedChanges: RuntimeTrackedChangesSnapshot = {
541
- changeIdsInOrder: [],
542
- changes: {},
543
- };
544
-
545
- const nextSnapshot: RuntimeRenderSnapshot = {
546
- ...renderSnapshot,
547
- documentId: options.documentId,
548
- sessionId: createSessionId(options.documentId),
549
- sourceLabel,
550
- revisionToken,
551
- isReady: true,
552
- isDirty: false,
553
- readOnly: options.readOnly,
554
- documentStats: buildDocumentStats(source, 0, 0),
555
- comments,
556
- trackedChanges,
557
- compatibility: buildCompatibilityPanelSnapshot(compatibility),
558
- warnings,
559
- fatalError: undefined,
560
- commandState: {
561
- canUndo: false,
562
- canRedo: false,
563
- canAddComment: !options.readOnly,
564
- canAcceptAllChanges: false,
565
- canRejectAllChanges: false,
566
- canExport: !compatibility.blockExport,
567
- canSaveSnapshot: Boolean(options.datastore),
568
- },
569
- };
570
-
571
- persistedSnapshot =
572
- source?.kind === "snapshot"
573
- ? {
574
- ...source.snapshot,
575
- documentId: options.documentId,
576
- }
577
- : createPersistedSnapshot(options.documentId, compatibility);
578
-
579
- if (source?.kind === "docx") {
580
- lastExport = toUint8Array(source.bytes);
581
- } else {
582
- lastExport = new Uint8Array();
583
- }
584
-
585
- setRenderSnapshot(nextSnapshot);
586
- warnings.forEach((warning) => options.onWarning?.(warning));
587
- emitEvent({
588
- type: "ready",
589
- documentId: options.documentId,
590
- sessionId: nextSnapshot.sessionId,
591
- source: descriptor?.sourceKind ?? "datastore",
592
- stats: nextSnapshot.documentStats,
593
- compatibility: nextSnapshot.compatibility.report,
594
- });
595
- };
596
-
597
- const handleLoadFailure = (message: string, details?: Record<string, unknown>) => {
598
- const error = createRuntimeError(
599
- "datastore_failed",
600
- message,
601
- "datastore",
602
- true,
603
- details,
604
- );
605
-
606
- setRenderSnapshot((current) => ({
607
- ...current,
608
- isReady: true,
609
- fatalError: error,
610
- commandState: {
611
- ...current.commandState,
612
- canAddComment: false,
613
- canExport: false,
614
- canSaveSnapshot: false,
615
- },
616
- }));
617
- emitError(error);
618
- };
619
-
620
- return {
621
- subscribe(listener) {
622
- listeners.add(listener);
623
-
624
- return () => {
625
- listeners.delete(listener);
626
- };
627
- },
628
-
629
- getRenderSnapshot() {
630
- return renderSnapshot;
631
- },
632
-
633
- async initialize(runtimeOptions) {
634
- if (destroyed) {
635
- return;
636
- }
637
-
638
- if (
639
- runtimeOptions.initialSource &&
640
- options.datastore &&
641
- options.datastore.logEvent
642
- ) {
643
- options.datastore.logEvent({
644
- type: "runtime_initialized",
645
- documentId: runtimeOptions.documentId,
646
- });
647
- }
648
-
649
- if (runtimeOptions.initialSource) {
650
- applySource(
651
- {
652
- source: runtimeOptions.initialSource,
653
- sourceKind: runtimeOptions.initialSource.kind,
654
- },
655
- "initial-load",
656
- );
657
- return;
658
- }
659
-
660
- if (!runtimeOptions.datastore) {
661
- applySource(undefined, "empty-load");
662
- return;
663
- }
664
-
665
- setRenderSnapshot((current) => ({
666
- ...current,
667
- isReady: false,
668
- }));
669
-
670
- try {
671
- const loaded = await runtimeOptions.datastore.load({
672
- documentId: runtimeOptions.documentId,
673
- });
674
-
675
- applySource(
676
- loaded.source
677
- ? {
678
- source: loaded.source,
679
- sourceKind: "datastore",
680
- }
681
- : undefined,
682
- "datastore-load",
683
- );
684
- } catch (error) {
685
- handleLoadFailure("Datastore load failed.", {
686
- cause: error instanceof Error ? error.message : String(error),
687
- });
688
- }
689
- },
690
-
691
- async replaceExternalSource(revisionToken, source) {
692
- applySource(
693
- {
694
- source,
695
- sourceKind: source.kind,
696
- },
697
- revisionToken,
698
- );
699
- },
700
-
701
- focus() {
702
- viewportDelegate?.focus();
703
- },
704
-
705
- blur() {
706
- viewportDelegate?.blur();
707
- },
708
-
709
- undo() {},
710
-
711
- redo() {},
712
-
713
- addComment(params) {
714
- commentCounter += 1;
715
- const commentId = `comment-${commentCounter}`;
716
- const nextThreads = {
717
- ...renderSnapshot.comments.threads,
718
- [commentId]: {
719
- commentId,
720
- body: params.body,
721
- status: "open" as const,
722
- anchor: params.anchor,
723
- },
724
- };
725
- const nextCommentIds = [...renderSnapshot.comments.commentIdsInOrder, commentId];
726
- const nextStats = {
727
- ...renderSnapshot.documentStats,
728
- commentCount: nextCommentIds.length,
729
- };
730
-
731
- setRenderSnapshot((current) => ({
732
- ...current,
733
- comments: {
734
- commentIdsInOrder: nextCommentIds,
735
- threads: nextThreads,
736
- openCommentId: commentId,
737
- },
738
- documentStats: nextStats,
739
- }));
740
- setDirty(true);
741
- emitEvent({
742
- type: "comment_added",
743
- documentId: renderSnapshot.documentId,
744
- commentId,
745
- anchor: params.anchor,
746
- });
747
-
748
- return commentId;
749
- },
750
-
751
- openComment(commentId) {
752
- if (!renderSnapshot.comments.threads[commentId]) {
753
- return;
754
- }
755
-
756
- setRenderSnapshot((current) => ({
757
- ...current,
758
- comments: {
759
- ...current.comments,
760
- openCommentId: commentId,
761
- },
762
- }));
763
- },
764
-
765
- resolveComment(commentId) {
766
- const thread = renderSnapshot.comments.threads[commentId];
767
- if (!thread) {
768
- return;
769
- }
770
-
771
- setRenderSnapshot((current) => ({
772
- ...current,
773
- comments: {
774
- ...current.comments,
775
- threads: {
776
- ...current.comments.threads,
777
- [commentId]: {
778
- ...thread,
779
- status: "resolved",
780
- },
781
- },
782
- },
783
- }));
784
- setDirty(true);
785
- emitEvent({
786
- type: "comment_resolved",
787
- documentId: renderSnapshot.documentId,
788
- commentId,
789
- });
790
- },
791
-
792
- acceptChange(changeId) {
793
- const change = renderSnapshot.trackedChanges.changes[changeId];
794
- if (!change) {
795
- return;
796
- }
797
-
798
- setRenderSnapshot((current) => ({
799
- ...current,
800
- trackedChanges: {
801
- ...current.trackedChanges,
802
- changes: {
803
- ...current.trackedChanges.changes,
804
- [changeId]: {
805
- ...change,
806
- status: "accepted",
807
- },
808
- },
809
- },
810
- }));
811
- setDirty(true);
812
- emitEvent({
813
- type: "change_accepted",
814
- documentId: renderSnapshot.documentId,
815
- changeId,
816
- });
817
- },
818
-
819
- rejectChange(changeId) {
820
- const change = renderSnapshot.trackedChanges.changes[changeId];
821
- if (!change) {
822
- return;
823
- }
824
-
825
- setRenderSnapshot((current) => ({
826
- ...current,
827
- trackedChanges: {
828
- ...current.trackedChanges,
829
- changes: {
830
- ...current.trackedChanges.changes,
831
- [changeId]: {
832
- ...change,
833
- status: "rejected",
834
- },
835
- },
836
- },
837
- }));
838
- setDirty(true);
839
- emitEvent({
840
- type: "change_rejected",
841
- documentId: renderSnapshot.documentId,
842
- changeId,
843
- });
844
- },
845
-
846
- acceptAllChanges() {
847
- const nextChanges: RuntimeTrackedChangesSnapshot["changes"] = {};
848
-
849
- for (const [changeId, change] of Object.entries(
850
- renderSnapshot.trackedChanges.changes,
851
- )) {
852
- nextChanges[changeId] = { ...change, status: "accepted" };
853
- }
854
-
855
- setRenderSnapshot((current) => ({
856
- ...current,
857
- trackedChanges: {
858
- ...current.trackedChanges,
859
- changes: nextChanges,
860
- },
861
- }));
862
- setDirty(true);
863
- },
864
-
865
- rejectAllChanges() {
866
- const nextChanges: RuntimeTrackedChangesSnapshot["changes"] = {};
867
-
868
- for (const [changeId, change] of Object.entries(
869
- renderSnapshot.trackedChanges.changes,
870
- )) {
871
- nextChanges[changeId] = { ...change, status: "rejected" };
872
- }
873
-
874
- setRenderSnapshot((current) => ({
875
- ...current,
876
- trackedChanges: {
877
- ...current.trackedChanges,
878
- changes: nextChanges,
879
- },
880
- }));
881
- setDirty(true);
882
- },
883
-
884
- async saveSnapshot(isAutosave) {
885
- const snapshot = {
886
- ...persistedSnapshot,
887
- updatedAt: createNowIsoString(),
888
- savedAt: createNowIsoString(),
889
- warningLog: renderSnapshot.warnings,
890
- compatibility: renderSnapshot.compatibility.report,
891
- };
892
-
893
- persistedSnapshot = snapshot;
894
-
895
- if (options.datastore) {
896
- try {
897
- await options.datastore.saveSnapshot({
898
- documentId: renderSnapshot.documentId,
899
- snapshot,
900
- isAutosave,
901
- });
902
- } catch (error) {
903
- emitError(
904
- createRuntimeError(
905
- "datastore_failed",
906
- "Snapshot save failed.",
907
- "datastore",
908
- false,
909
- {
910
- cause: error instanceof Error ? error.message : String(error),
911
- },
912
- ),
913
- );
914
- }
915
- }
916
-
917
- emitEvent({
918
- type: "snapshot_saved",
919
- documentId: renderSnapshot.documentId,
920
- snapshot,
921
- isAutosave,
922
- });
923
- setDirty(false);
924
-
925
- return snapshot;
926
- },
927
-
928
- getPersistedSnapshot() {
929
- return persistedSnapshot;
930
- },
931
-
932
- getCompatibilityReport() {
933
- return renderSnapshot.compatibility.report;
934
- },
935
-
936
- getWarnings() {
937
- return renderSnapshot.warnings;
938
- },
939
-
940
- async exportDocx(_options) {
941
- const result: ExportResult = {
942
- docx: lastExport,
943
- compatibilityReport: renderSnapshot.compatibility.report,
944
- warnings: renderSnapshot.warnings,
945
- };
946
-
947
- if (options.datastore) {
948
- try {
949
- await options.datastore.saveExport({
950
- documentId: renderSnapshot.documentId,
951
- docx: result.docx,
952
- compatibilityReport: result.compatibilityReport,
953
- });
954
- } catch (error) {
955
- emitError(
956
- createRuntimeError(
957
- "datastore_failed",
958
- "Export persistence failed.",
959
- "datastore",
960
- false,
961
- {
962
- cause: error instanceof Error ? error.message : String(error),
963
- },
964
- ),
965
- );
966
- }
967
- }
968
-
969
- emitEvent({
970
- type: "export_completed",
971
- documentId: renderSnapshot.documentId,
972
- result,
973
- });
974
-
975
- return result;
976
- },
977
-
978
- attachViewport(delegate) {
979
- viewportDelegate = delegate;
980
- },
981
-
982
- destroy() {
983
- destroyed = true;
984
- listeners.clear();
985
- viewportDelegate = null;
986
- },
987
- };
988
- }
989
-
990
- function ToolbarButton({
991
- label,
992
- disabled,
993
- onClick,
994
- }: {
995
- label: string;
996
- disabled?: boolean;
997
- onClick: () => void;
998
- }) {
999
- return (
1000
- <button
1001
- type="button"
1002
- onClick={onClick}
1003
- disabled={disabled}
1004
- style={{
1005
- padding: "8px 12px",
1006
- borderRadius: "12px",
1007
- border: "1px solid rgba(23, 31, 38, 0.08)",
1008
- background: disabled ? "rgba(241, 238, 229, 0.7)" : "rgba(255, 255, 255, 0.9)",
1009
- color: disabled ? "#8b949b" : "#18212a",
1010
- fontSize: "13px",
1011
- cursor: disabled ? "not-allowed" : "pointer",
1012
- }}
1013
- >
1014
- {label}
1015
- </button>
1016
- );
1017
- }
1018
-
1019
- export const WordReviewEditor = forwardRef<WordReviewEditorRef, WordReviewEditorProps>(
1020
- function WordReviewEditor(
1021
- {
1022
- documentId,
1023
- currentUser,
1024
- initialDocx,
1025
- initialSnapshot,
1026
- initialSourceLabel,
1027
- externalDocumentRevision,
1028
- externalDocSource,
1029
- readOnly = false,
1030
- reviewMode = "review",
1031
- markupDisplay = "all",
1032
- datastore,
1033
- autosave,
1034
- onEvent,
1035
- onWarning,
1036
- onError,
1037
- },
1038
- ref,
1039
- ) {
1040
- const initialSourceStateRef = useRef<{
1041
- initialized: boolean;
1042
- value?: ExternalDocumentSource;
1043
- }>({
1044
- initialized: false,
1045
- value: undefined,
1046
- });
1047
- const runtimeRef = useRef<WordReviewEditorRuntime | null>(null);
1048
- const canvasRef = useRef<HTMLDivElement | null>(null);
1049
- const previousRevisionRef = useRef(externalDocumentRevision);
1050
-
1051
- if (!initialSourceStateRef.current.initialized) {
1052
- initialSourceStateRef.current = {
1053
- initialized: true,
1054
- value: resolveInitialExplicitSource({
1055
- initialDocx,
1056
- initialSnapshot,
1057
- initialSourceLabel,
1058
- }),
1059
- };
1060
- }
1061
-
1062
- const initialSource = initialSourceStateRef.current.value;
1063
-
1064
- if (!runtimeRef.current) {
1065
- runtimeRef.current = __unstable_createWordReviewEditorRuntime({
1066
- documentId,
1067
- readOnly,
1068
- initialSource,
1069
- initialSourceLabel,
1070
- datastore,
1071
- onEvent,
1072
- onWarning,
1073
- onError,
1074
- });
1075
- }
1076
-
1077
- const runtime = runtimeRef.current;
1078
- const snapshot = useSyncExternalStore(
1079
- runtime.subscribe,
1080
- runtime.getRenderSnapshot,
1081
- runtime.getRenderSnapshot,
1082
- );
1083
-
1084
- useEffect(() => {
1085
- void runtime.initialize({
1086
- documentId,
1087
- initialSource,
1088
- datastore,
1089
- });
1090
-
1091
- return () => {
1092
- runtime.destroy();
1093
- };
1094
- }, [runtime]);
1095
-
1096
- useEffect(() => {
1097
- runtime.attachViewport({
1098
- focus() {
1099
- canvasRef.current?.focus();
1100
- },
1101
- blur() {
1102
- canvasRef.current?.blur();
1103
- },
1104
- });
1105
-
1106
- return () => {
1107
- runtime.attachViewport(null);
1108
- };
1109
- }, [runtime]);
1110
-
1111
- useEffect(() => {
1112
- if (
1113
- externalDocumentRevision &&
1114
- externalDocSource &&
1115
- previousRevisionRef.current !== externalDocumentRevision
1116
- ) {
1117
- previousRevisionRef.current = externalDocumentRevision;
1118
- void runtime.replaceExternalSource(
1119
- externalDocumentRevision,
1120
- externalDocSource,
1121
- );
1122
- }
1123
- }, [externalDocumentRevision, externalDocSource, runtime]);
1124
-
1125
- useEffect(() => {
1126
- if (!autosave?.enabled || !snapshot.isDirty) {
1127
- return;
1128
- }
1129
-
1130
- const timeoutId = window.setTimeout(() => {
1131
- void runtime.saveSnapshot(true);
1132
- }, autosave.debounceMs);
1133
-
1134
- return () => {
1135
- window.clearTimeout(timeoutId);
1136
- };
1137
- }, [autosave, runtime, snapshot.isDirty]);
1138
-
1139
- useImperativeHandle(
1140
- ref,
1141
- (): WordReviewEditorRef => ({
1142
- focus: () => runtime.focus(),
1143
- blur: () => runtime.blur(),
1144
- undo: () => runtime.undo(),
1145
- redo: () => runtime.redo(),
1146
- addComment: (params) => runtime.addComment(params),
1147
- openComment: (commentId) => runtime.openComment(commentId),
1148
- resolveComment: (commentId) => runtime.resolveComment(commentId),
1149
- acceptChange: (changeId) => runtime.acceptChange(changeId),
1150
- rejectChange: (changeId) => runtime.rejectChange(changeId),
1151
- acceptAllChanges: () => runtime.acceptAllChanges(),
1152
- rejectAllChanges: () => runtime.rejectAllChanges(),
1153
- exportDocx: (options) => runtime.exportDocx(options),
1154
- getSnapshot: () => runtime.getPersistedSnapshot(),
1155
- getCompatibilityReport: () => runtime.getCompatibilityReport(),
1156
- getWarnings: () => runtime.getWarnings(),
1157
- }),
1158
- [runtime],
1159
- );
1160
-
1161
- const surfaceModel = buildEditorSurfaceModel(
1162
- snapshot,
1163
- reviewMode,
1164
- markupDisplay,
1165
- );
1166
-
1167
- return (
1168
- <section
1169
- aria-label="Word Review Editor"
1170
- data-document-id={documentId}
1171
- data-session-id={snapshot.sessionId}
1172
- style={{
1173
- minHeight: "100%",
1174
- background:
1175
- "radial-gradient(circle at top left, rgba(227, 232, 225, 0.8), transparent 34%), linear-gradient(180deg, #eef1ea 0%, #e4e8df 100%)",
1176
- color: "#18212a",
1177
- padding: "24px",
1178
- borderRadius: "28px",
1179
- }}
1180
- >
1181
- <div
1182
- style={{
1183
- display: "grid",
1184
- gap: "18px",
1185
- }}
1186
- >
1187
- <header
1188
- style={{
1189
- display: "flex",
1190
- flexWrap: "wrap",
1191
- justifyContent: "space-between",
1192
- gap: "14px",
1193
- alignItems: "center",
1194
- padding: "12px 14px",
1195
- borderRadius: "18px",
1196
- border: "1px solid rgba(23, 31, 38, 0.07)",
1197
- background: "rgba(250, 249, 246, 0.82)",
1198
- backdropFilter: "blur(10px)",
1199
- }}
1200
- >
1201
- <div style={{ display: "flex", gap: "8px", flexWrap: "wrap" }}>
1202
- <ToolbarButton
1203
- label="Undo"
1204
- disabled={!snapshot.commandState.canUndo}
1205
- onClick={() => runtime.undo()}
1206
- />
1207
- <ToolbarButton
1208
- label="Redo"
1209
- disabled={!snapshot.commandState.canRedo}
1210
- onClick={() => runtime.redo()}
1211
- />
1212
- <ToolbarButton
1213
- label="Add comment"
1214
- disabled={!snapshot.commandState.canAddComment}
1215
- onClick={() =>
1216
- runtime.addComment({
1217
- anchor: snapshot.selection.activeRange,
1218
- body: `Comment from ${currentUser.displayName}`,
1219
- })
1220
- }
1221
- />
1222
- <ToolbarButton
1223
- label="Accept all"
1224
- disabled={!snapshot.commandState.canAcceptAllChanges}
1225
- onClick={() => runtime.acceptAllChanges()}
1226
- />
1227
- <ToolbarButton
1228
- label="Reject all"
1229
- disabled={!snapshot.commandState.canRejectAllChanges}
1230
- onClick={() => runtime.rejectAllChanges()}
1231
- />
1232
- </div>
1233
-
1234
- <div style={{ display: "flex", gap: "8px", flexWrap: "wrap" }}>
1235
- <ToolbarButton
1236
- label="Save snapshot"
1237
- disabled={!snapshot.commandState.canSaveSnapshot}
1238
- onClick={() => {
1239
- void runtime.saveSnapshot(false);
1240
- }}
1241
- />
1242
- <ToolbarButton
1243
- label="Export docx"
1244
- disabled={!snapshot.commandState.canExport}
1245
- onClick={() => {
1246
- void runtime.exportDocx({
1247
- includeCompatibilityReport: true,
1248
- });
1249
- }}
1250
- />
1251
- </div>
1252
- </header>
1253
-
1254
- {snapshot.fatalError ? (
1255
- <div
1256
- role="alert"
1257
- style={{
1258
- padding: "16px 18px",
1259
- borderRadius: "18px",
1260
- background: "rgba(242, 223, 223, 0.92)",
1261
- border: "1px solid rgba(142, 59, 59, 0.25)",
1262
- color: "#5f2626",
1263
- }}
1264
- >
1265
- {snapshot.fatalError.message}
1266
- </div>
1267
- ) : null}
1268
-
1269
- <EditorSurface
1270
- canvasRef={canvasRef}
1271
- model={surfaceModel}
1272
- onOpenComment={(commentId) => runtime.openComment(commentId)}
1273
- />
1274
-
1275
- <footer
1276
- style={{
1277
- display: "flex",
1278
- flexWrap: "wrap",
1279
- justifyContent: "space-between",
1280
- gap: "12px",
1281
- color: "#58646f",
1282
- fontSize: "13px",
1283
- padding: "0 4px",
1284
- }}
1285
- >
1286
- <span>
1287
- {snapshot.isReady ? "Ready" : "Loading"} •{" "}
1288
- {snapshot.isDirty ? "Unsaved changes" : "Clean"} • session{" "}
1289
- {snapshot.sessionId}
1290
- </span>
1291
- <span>
1292
- {reviewMode} review • {markupDisplay} markup •{" "}
1293
- {snapshot.compatibility.blockExport ? "export blocked" : "export allowed"}
1294
- </span>
1295
- </footer>
1296
- </div>
1297
- </section>
1298
- );
1299
- },
1300
- );
1301
-
1302
- export default WordReviewEditor;