@beyondwork/docx-react-component 1.0.0

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 (690) hide show
  1. package/.codex/config.toml +5 -0
  2. package/.corepack/v1/pnpm/10.30.3/.corepack +1 -0
  3. package/.corepack/v1/pnpm/10.30.3/LICENSE +22 -0
  4. package/.corepack/v1/pnpm/10.30.3/README.md +240 -0
  5. package/.corepack/v1/pnpm/10.30.3/dist/node-gyp-bin/node-gyp +6 -0
  6. package/.corepack/v1/pnpm/10.30.3/dist/node-gyp-bin/node-gyp.cmd +5 -0
  7. package/.corepack/v1/pnpm/10.30.3/dist/pnpm.cjs +195400 -0
  8. package/.corepack/v1/pnpm/10.30.3/dist/pnpmrc +2 -0
  9. package/.corepack/v1/pnpm/10.30.3/dist/reflink.darwin-arm64-2HJ4WGO6.node +0 -0
  10. package/.corepack/v1/pnpm/10.30.3/dist/reflink.darwin-x64-3G3H6IW4.node +0 -0
  11. package/.corepack/v1/pnpm/10.30.3/dist/reflink.win32-arm64-msvc-Q6BARPPB.node +0 -0
  12. package/.corepack/v1/pnpm/10.30.3/dist/reflink.win32-x64-msvc-J2TZHRQI.node +0 -0
  13. package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.bash +31 -0
  14. package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.fish +22 -0
  15. package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.ps1 +193 -0
  16. package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.zsh +27 -0
  17. package/.corepack/v1/pnpm/10.30.3/dist/vendor/fastlist-0.3.0-x64.exe +0 -0
  18. package/.corepack/v1/pnpm/10.30.3/dist/vendor/fastlist-0.3.0-x86.exe +0 -0
  19. package/.corepack/v1/pnpm/10.30.3/dist/worker.js +10119 -0
  20. package/.corepack/v1/pnpm/10.30.3/package.json +192 -0
  21. package/.cursor/mcp.json +7 -0
  22. package/.github/workflows/ci.yml +35 -0
  23. package/.mcp.json +7 -0
  24. package/.openclaw/workspace-state.json +4 -0
  25. package/.pnpmrc.json +1 -0
  26. package/.wave-launch.sh +7 -0
  27. package/.workspace-marker +1 -0
  28. package/AGENTS.md +78 -0
  29. package/CHANGELOG.md +177 -0
  30. package/DESIGN.md +929 -0
  31. package/HEARTBEAT.md +7 -0
  32. package/IDENTITY.md +23 -0
  33. package/LICENSE.md +31 -0
  34. package/README.md +180 -0
  35. package/SOUL.md +36 -0
  36. package/TOOLS.md +40 -0
  37. package/USER.md +17 -0
  38. package/docs/README.md +107 -0
  39. package/docs/agents/wave-cont-eval-role.md +36 -0
  40. package/docs/agents/wave-cont-qa-role.md +52 -0
  41. package/docs/agents/wave-deploy-verifier-role.md +34 -0
  42. package/docs/agents/wave-design-role.md +47 -0
  43. package/docs/agents/wave-documentation-role.md +34 -0
  44. package/docs/agents/wave-infra-role.md +34 -0
  45. package/docs/agents/wave-integration-role.md +37 -0
  46. package/docs/agents/wave-launcher-role.md +41 -0
  47. package/docs/agents/wave-orchestrator-role.md +52 -0
  48. package/docs/agents/wave-planner-role.md +39 -0
  49. package/docs/agents/wave-security-role.md +40 -0
  50. package/docs/architecture/docx/README.md +10 -0
  51. package/docs/architecture/future/README.md +8 -0
  52. package/docs/architecture/ooxml-upgrade-analysis.md +134 -0
  53. package/docs/architecture/platform/shared-openxml-editor-platform.md +153 -0
  54. package/docs/architecture/xlsx/canonical-workbook-model-and-commands.md +187 -0
  55. package/docs/architecture/xlsx/spreadsheet-editor-frontend-architecture.md +150 -0
  56. package/docs/comment-redline-overview.md +350 -0
  57. package/docs/concepts/context7-vs-skills.md +118 -0
  58. package/docs/concepts/operating-modes.md +91 -0
  59. package/docs/concepts/runtime-agnostic-orchestration.md +111 -0
  60. package/docs/concepts/what-is-a-wave.md +217 -0
  61. package/docs/context7/bundles.json +222 -0
  62. package/docs/context7/planner-agent/README.md +28 -0
  63. package/docs/context7/planner-agent/manifest.json +83 -0
  64. package/docs/context7/planner-agent/papers/cooperbench-why-coding-agents-cannot-be-your-teammates-yet.md +3283 -0
  65. package/docs/context7/planner-agent/papers/dova-deliberation-first-multi-agent-orchestration-for-autonomous-research-automation.md +1699 -0
  66. package/docs/context7/planner-agent/papers/dpbench-large-language-models-struggle-with-simultaneous-coordination.md +2251 -0
  67. package/docs/context7/planner-agent/papers/incremental-planning-to-control-a-blackboard-based-problem-solver.md +1729 -0
  68. package/docs/context7/planner-agent/papers/silo-bench-a-scalable-environment-for-evaluating-distributed-coordination-in-multi-agent-llm-systems.md +3747 -0
  69. package/docs/context7/planner-agent/papers/todoevolve-learning-to-architect-agent-planning-systems.md +1675 -0
  70. package/docs/context7/planner-agent/papers/verified-multi-agent-orchestration-a-plan-execute-verify-replan-framework-for-complex-query-resolution.md +1173 -0
  71. package/docs/context7/planner-agent/papers/why-do-multi-agent-llm-systems-fail.md +5211 -0
  72. package/docs/context7/planner-agent/topics/planning-and-orchestration.md +24 -0
  73. package/docs/evals/arm-templates/README.md +13 -0
  74. package/docs/evals/arm-templates/full-wave.json +15 -0
  75. package/docs/evals/arm-templates/single-agent.json +15 -0
  76. package/docs/evals/benchmark-catalog.json +670 -0
  77. package/docs/evals/cases/README.md +47 -0
  78. package/docs/evals/cases/wave-blackboard-inbox-targeting.json +73 -0
  79. package/docs/evals/cases/wave-contradiction-conflict.json +104 -0
  80. package/docs/evals/cases/wave-expert-routing-preservation.json +69 -0
  81. package/docs/evals/cases/wave-hidden-profile-private-evidence.json +81 -0
  82. package/docs/evals/cases/wave-premature-closure-guard.json +71 -0
  83. package/docs/evals/cases/wave-silo-cross-agent-state.json +77 -0
  84. package/docs/evals/cases/wave-simultaneous-lockstep.json +92 -0
  85. package/docs/evals/external-benchmarks.json +85 -0
  86. package/docs/evals/external-command-config.sample.json +9 -0
  87. package/docs/evals/external-command-config.swe-bench-pro.json +8 -0
  88. package/docs/evals/pilots/README.md +47 -0
  89. package/docs/evals/pilots/swe-bench-pro-public-full-wave-review-10.json +64 -0
  90. package/docs/evals/pilots/swe-bench-pro-public-pilot.json +111 -0
  91. package/docs/evals/wave-benchmark-program.md +302 -0
  92. package/docs/guides/planner.md +220 -0
  93. package/docs/guides/recommendations-0.8.9.md +133 -0
  94. package/docs/guides/signal-wrappers.md +165 -0
  95. package/docs/guides/terminal-surfaces.md +96 -0
  96. package/docs/image copy.png +0 -0
  97. package/docs/image.png +0 -0
  98. package/docs/images/image.png +0 -0
  99. package/docs/legal-feedback-architecture.md +498 -0
  100. package/docs/plans/component-cutover-matrix.json +1072 -0
  101. package/docs/plans/component-cutover-matrix.md +307 -0
  102. package/docs/plans/context7-wave-orchestrator.md +155 -0
  103. package/docs/plans/current-state.md +198 -0
  104. package/docs/plans/docx/README.md +9 -0
  105. package/docs/plans/examples/wave-benchmark-improvement.md +108 -0
  106. package/docs/plans/examples/wave-example-live-proof.md +435 -0
  107. package/docs/plans/master-plan.md +224 -0
  108. package/docs/plans/migration.md +538 -0
  109. package/docs/plans/operations/README.md +7 -0
  110. package/docs/plans/operations/wave-10-word-certification.md +87 -0
  111. package/docs/plans/operations/wave-8-railway-staging.md +153 -0
  112. package/docs/plans/operations/wave-9-manual-certification.md +73 -0
  113. package/docs/plans/platform/README.md +9 -0
  114. package/docs/plans/reference/legal-checklist-coverage.md +258 -0
  115. package/docs/plans/wave-orchestrator.md +423 -0
  116. package/docs/plans/waves/README.md +75 -0
  117. package/docs/plans/waves/completed/wave-0.md +195 -0
  118. package/docs/plans/waves/completed/wave-1.md +379 -0
  119. package/docs/plans/waves/completed/wave-10.md +670 -0
  120. package/docs/plans/waves/completed/wave-11.md +335 -0
  121. package/docs/plans/waves/completed/wave-12.md +417 -0
  122. package/docs/plans/waves/completed/wave-13.md +316 -0
  123. package/docs/plans/waves/completed/wave-14.md +319 -0
  124. package/docs/plans/waves/completed/wave-15.md +321 -0
  125. package/docs/plans/waves/completed/wave-16.md +316 -0
  126. package/docs/plans/waves/completed/wave-17.md +331 -0
  127. package/docs/plans/waves/completed/wave-18.md +328 -0
  128. package/docs/plans/waves/completed/wave-2.md +438 -0
  129. package/docs/plans/waves/completed/wave-3.md +435 -0
  130. package/docs/plans/waves/completed/wave-4.md +430 -0
  131. package/docs/plans/waves/completed/wave-5.md +430 -0
  132. package/docs/plans/waves/completed/wave-6.md +430 -0
  133. package/docs/plans/waves/completed/wave-7.md +526 -0
  134. package/docs/plans/waves/completed/wave-8.md +596 -0
  135. package/docs/plans/waves/completed/wave-9.md +552 -0
  136. package/docs/plans/waves/deferred/README.md +14 -0
  137. package/docs/plans/waves/deferred/encrypted-intake-contracts.md +282 -0
  138. package/docs/plans/waves/deferred/legal-feedback-wave-expansion.md +308 -0
  139. package/docs/plans/waves/deferred/wave-encrypted-intake.md +451 -0
  140. package/docs/plans/waves/design/README.md +5 -0
  141. package/docs/plans/waves/design/wave-1-a1.md +309 -0
  142. package/docs/plans/waves/reviews/README.md +5 -0
  143. package/docs/plans/waves/reviews/wave-0-cont-qa.md +151 -0
  144. package/docs/plans/waves/reviews/wave-1-cont-qa.md +46 -0
  145. package/docs/plans/waves/reviews/wave-10-accessibility-and-design.md +51 -0
  146. package/docs/plans/waves/reviews/wave-10-cont-qa.md +24 -0
  147. package/docs/plans/waves/reviews/wave-10-dashboard-proof.md +46 -0
  148. package/docs/plans/waves/reviews/wave-10-performance-signoff.md +55 -0
  149. package/docs/plans/waves/reviews/wave-10-regression-proof.md +23 -0
  150. package/docs/plans/waves/reviews/wave-10-release-audit.md +31 -0
  151. package/docs/plans/waves/reviews/wave-10-service-proof.md +83 -0
  152. package/docs/plans/waves/reviews/wave-10-word-certification.md +31 -0
  153. package/docs/plans/waves/reviews/wave-18-ai-contract-closure.md +277 -0
  154. package/docs/plans/waves/reviews/wave-18-cont-qa.md +255 -0
  155. package/docs/plans/waves/reviews/wave-18-parity-proof.md +271 -0
  156. package/docs/plans/waves/reviews/wave-19-cont-qa.md +59 -0
  157. package/docs/plans/waves/reviews/wave-2-cont-qa.md +72 -0
  158. package/docs/plans/waves/reviews/wave-20-cont-qa.md +60 -0
  159. package/docs/plans/waves/reviews/wave-25-cont-qa.md +48 -0
  160. package/docs/plans/waves/reviews/wave-28-cont-qa.md +46 -0
  161. package/docs/plans/waves/reviews/wave-29-cont-qa.md +53 -0
  162. package/docs/plans/waves/reviews/wave-3-cont-qa.md +53 -0
  163. package/docs/plans/waves/reviews/wave-3-core-proof.md +77 -0
  164. package/docs/plans/waves/reviews/wave-3-validator-proof.md +73 -0
  165. package/docs/plans/waves/reviews/wave-32-cont-qa.md +43 -0
  166. package/docs/plans/waves/reviews/wave-33-cont-qa.md +526 -0
  167. package/docs/plans/waves/reviews/wave-34-cont-qa.md +100 -0
  168. package/docs/plans/waves/reviews/wave-35-cont-qa.md +145 -0
  169. package/docs/plans/waves/reviews/wave-4-cont-qa.md +47 -0
  170. package/docs/plans/waves/reviews/wave-4-structure-proof.md +69 -0
  171. package/docs/plans/waves/reviews/wave-5-comment-proof.md +158 -0
  172. package/docs/plans/waves/reviews/wave-5-cont-qa.md +68 -0
  173. package/docs/plans/waves/reviews/wave-6-cont-qa.md +416 -0
  174. package/docs/plans/waves/reviews/wave-6-redline-proof.md +130 -0
  175. package/docs/plans/waves/reviews/wave-7-cont-qa.md +82 -0
  176. package/docs/plans/waves/reviews/wave-7-ooxml-compliance.md +85 -0
  177. package/docs/plans/waves/reviews/wave-7-preservation-proof.md +119 -0
  178. package/docs/plans/waves/reviews/wave-7-trust-ux.md +87 -0
  179. package/docs/plans/waves/reviews/wave-8-accessibility-and-design.md +128 -0
  180. package/docs/plans/waves/reviews/wave-8-cont-qa.md +92 -0
  181. package/docs/plans/waves/reviews/wave-8-live-proof.md +140 -0
  182. package/docs/plans/waves/reviews/wave-8-security.md +47 -0
  183. package/docs/plans/waves/reviews/wave-9-editor-embedding.md +39 -0
  184. package/docs/plans/waves/reviews/wave-9-fixture-runner.md +56 -0
  185. package/docs/plans/waves/reviews/wave-9-live-proof.md +105 -0
  186. package/docs/plans/waves/reviews/wave-9-usability-and-performance.md +152 -0
  187. package/docs/plans/waves/specs/README.md +5 -0
  188. package/docs/plans/waves/specs/wave-1-component-boundaries.md +322 -0
  189. package/docs/plans/waves/specs/wave-1-ooxml-contracts.md +323 -0
  190. package/docs/plans/waves/specs/wave-1-review-and-ui-contracts.md +339 -0
  191. package/docs/plans/waves/specs/wave-1-runtime-contracts.md +509 -0
  192. package/docs/plans/waves/wave-19.md +341 -0
  193. package/docs/plans/waves/wave-20.md +308 -0
  194. package/docs/plans/waves/wave-21.md +289 -0
  195. package/docs/plans/waves/wave-22.md +221 -0
  196. package/docs/plans/waves/wave-23.md +295 -0
  197. package/docs/plans/waves/wave-24.md +286 -0
  198. package/docs/plans/waves/wave-25.md +313 -0
  199. package/docs/plans/waves/wave-26.md +300 -0
  200. package/docs/plans/waves/wave-27.md +299 -0
  201. package/docs/plans/waves/wave-28.md +368 -0
  202. package/docs/plans/waves/wave-29.md +303 -0
  203. package/docs/plans/waves/wave-30.md +307 -0
  204. package/docs/plans/waves/wave-31.md +231 -0
  205. package/docs/plans/waves/wave-32.md +152 -0
  206. package/docs/plans/waves/wave-33.md +147 -0
  207. package/docs/plans/waves/wave-34.md +148 -0
  208. package/docs/plans/waves/wave-35.md +141 -0
  209. package/docs/plans/waves/wave-36.md +146 -0
  210. package/docs/plans/xlsx/README.md +14 -0
  211. package/docs/plans/xlsx/xlsx-fixture-corpus-and-certification-plan.md +126 -0
  212. package/docs/reference/cli-reference.md +600 -0
  213. package/docs/reference/coordination-and-closure.md +487 -0
  214. package/docs/reference/deep-research-report (15).md +25 -0
  215. package/docs/reference/docx/README.md +10 -0
  216. package/docs/reference/legal-checklist.md +445 -0
  217. package/docs/reference/live-proof-waves.md +199 -0
  218. package/docs/reference/ooxml-compliance.md +129 -0
  219. package/docs/reference/ooxml-feature-parity-matrix.md +172 -0
  220. package/docs/reference/platform/shared-ooxml-platform-guidance.md +77 -0
  221. package/docs/reference/prototype-agent-prompt-legal-fidelity.md +155 -0
  222. package/docs/reference/public-api.md +456 -0
  223. package/docs/reference/repository-guidance.md +58 -0
  224. package/docs/reference/runtime-config/README.md +182 -0
  225. package/docs/reference/runtime-config/claude.md +110 -0
  226. package/docs/reference/runtime-config/codex.md +82 -0
  227. package/docs/reference/runtime-config/opencode.md +93 -0
  228. package/docs/reference/sample-waves.md +105 -0
  229. package/docs/reference/skills.md +237 -0
  230. package/docs/reference/templates/AGENTS.md +78 -0
  231. package/docs/reference/templates/HEARTBEAT.md +7 -0
  232. package/docs/reference/templates/IDENTITY.md +23 -0
  233. package/docs/reference/templates/SOUL.md +36 -0
  234. package/docs/reference/templates/TOOLS.md +40 -0
  235. package/docs/reference/templates/USER.md +17 -0
  236. package/docs/reference/wave-control.md +184 -0
  237. package/docs/reference/wave-planning-lessons.md +167 -0
  238. package/docs/reference/word-review-editor-frontend-architecture.md +479 -0
  239. package/docs/reference/word-review-editor-ux-guide.md +253 -0
  240. package/docs/reference/xlsx/xlsx-ooxml-compliance.md +137 -0
  241. package/docs/research/agent-context-sources.md +178 -0
  242. package/docs/research/coordination-failure-review.md +290 -0
  243. package/docs/research/docx-react-component/Canonical Document Schema Specification for a React-based Word-compatible Editor.md +2317 -0
  244. package/docs/research/docx-react-component/Feature Compatibility Matrix for a React Word Compatible Legal Editor v1.md +219 -0
  245. package/docs/research/docx-react-component/React Component Architecture and Front-End Structure Specification for a Word-Compatible Legal Review Editor.md +1112 -0
  246. package/docs/research/docx-react-component/document_compatibility_and_testing_spec.md +751 -0
  247. package/docs/research/xlsx/raw/README.md +13 -0
  248. package/docs/roadmap.md +174 -0
  249. package/docs/superpowers/plans/2026-03-28-harness-control-bar.md +677 -0
  250. package/docs/superpowers/specs/2026-03-28-harness-control-bar-design.md +274 -0
  251. package/docs/xlsx-react/README.md +38 -0
  252. package/docs/xlsx-react/agent-llm-interaction-layer-docx-xlsx.md +621 -0
  253. package/docs/xlsx-react/canonical-workbook-model-and-commands.md +948 -0
  254. package/docs/xlsx-react/shared-openxml-editor-platform-docx-xlsx.md +228 -0
  255. package/docs/xlsx-react/spreadsheet-editor-component-architecture.md +809 -0
  256. package/docs/xlsx-react/spreadsheet-editor-frontend-architecture.md +537 -0
  257. package/docs/xlsx-react/spreadsheet-editor-ux-guide.md +520 -0
  258. package/docs/xlsx-react/xlsx-editor-research-pack.md +871 -0
  259. package/docs/xlsx-react/xlsx-fixture-corpus-and-certification-plan.md +436 -0
  260. package/docs/xlsx-react/xlsx-ooxml-compliance.md +320 -0
  261. package/examples/README.md +16 -0
  262. package/memory/MEMORY.md +24 -0
  263. package/package.json +63 -0
  264. package/pnpm-workspace.yaml +4 -0
  265. package/scripts/check-no-authored-js.sh +13 -0
  266. package/scripts/context7-api-check.sh +65 -0
  267. package/scripts/context7-export-env.sh +42 -0
  268. package/scripts/run-context7-mcp.sh +8 -0
  269. package/scripts/run-workspace-tests.sh +15 -0
  270. package/scripts/start-wave-10-local.sh +189 -0
  271. package/scripts/wave-agent-attach.sh +47 -0
  272. package/scripts/wave-auto-answer.sh +118 -0
  273. package/scripts/wave-dashboard-attach.sh +13 -0
  274. package/scripts/wave-launch.sh +273 -0
  275. package/scripts/wave-overnight-supervisor.sh +145 -0
  276. package/scripts/wave-status.sh +379 -0
  277. package/scripts/wave-watch.sh +231 -0
  278. package/services/README.md +17 -0
  279. package/services/openxml-validator/Dockerfile +29 -0
  280. package/services/openxml-validator/OpenXmlValidator.Api.csproj +12 -0
  281. package/services/openxml-validator/Program.cs +436 -0
  282. package/services/openxml-validator/README.md +152 -0
  283. package/services/openxml-validator/railway.json +16 -0
  284. package/services/react-word-editor/.tmp-a4/src/api/public-types.ts +318 -0
  285. package/services/react-word-editor/.tmp-a4/src/ui/WordReviewEditor.tsx +1302 -0
  286. package/services/react-word-editor/.tmp-a4/src/ui/editor-surface/editor-surface.tsx +546 -0
  287. package/services/react-word-editor/.tmp-a4/test/ui/word-review-editor.test.tsx +146 -0
  288. package/services/react-word-editor/.tmp-a4-build/src/api/public-types.js +2 -0
  289. package/services/react-word-editor/.tmp-a4-build/src/ui/WordReviewEditor.js +818 -0
  290. package/services/react-word-editor/.tmp-a4-build/src/ui/editor-surface/editor-surface.js +229 -0
  291. package/services/react-word-editor/.tmp-a4-build/test/ui/word-review-editor.test.js +121 -0
  292. package/services/react-word-editor/.tmp-wave-4-a3-tsconfig.json +21 -0
  293. package/services/react-word-editor/.tmp-wave-4-a3-tsconfig.tsbuildinfo +1 -0
  294. package/services/react-word-editor/Dockerfile +26 -0
  295. package/services/react-word-editor/README.md +254 -0
  296. package/services/react-word-editor/app/api/certification/route.ts +79 -0
  297. package/services/react-word-editor/app/api/demo-sessions/route.ts +109 -0
  298. package/services/react-word-editor/app/api/deploy-health/route.ts +23 -0
  299. package/services/react-word-editor/app/api/exports/[exportId]/route.ts +34 -0
  300. package/services/react-word-editor/app/api/exports/route.ts +81 -0
  301. package/services/react-word-editor/app/api/fixtures/[fixtureId]/run/route.ts +100 -0
  302. package/services/react-word-editor/app/api/health/route.ts +70 -0
  303. package/services/react-word-editor/app/api/runs/[runId]/route.ts +36 -0
  304. package/services/react-word-editor/app/api/scenarios/[scenarioId]/run/route.ts +85 -0
  305. package/services/react-word-editor/app/api/sessions/[sessionId]/route.ts +199 -0
  306. package/services/react-word-editor/app/api/sessions/[sessionId]/source/route.ts +45 -0
  307. package/services/react-word-editor/app/api/uploads/route.ts +70 -0
  308. package/services/react-word-editor/app/api/validate/route.ts +310 -0
  309. package/services/react-word-editor/app/certification/[runId]/page.tsx +14 -0
  310. package/services/react-word-editor/app/certification/page.tsx +32 -0
  311. package/services/react-word-editor/app/dashboard/page.tsx +7 -0
  312. package/services/react-word-editor/app/demo/page.tsx +30 -0
  313. package/services/react-word-editor/app/demo/prototype-client.tsx +1080 -0
  314. package/services/react-word-editor/app/editor/[sessionId]/page.tsx +33 -0
  315. package/services/react-word-editor/app/fixtures/page.tsx +7 -0
  316. package/services/react-word-editor/app/globals.css +121 -0
  317. package/services/react-word-editor/app/layout.tsx +32 -0
  318. package/services/react-word-editor/app/page.tsx +30 -0
  319. package/services/react-word-editor/app/runs/[runId]/page.tsx +34 -0
  320. package/services/react-word-editor/app/wave-10-word-review/page.tsx +7 -0
  321. package/services/react-word-editor/components/harness-control-bar.tsx +289 -0
  322. package/services/react-word-editor/components/harness-editor-session-client.tsx +1214 -0
  323. package/services/react-word-editor/components/harness-workspace-page.tsx +715 -0
  324. package/services/react-word-editor/components/reduced-motion-toggle.tsx +79 -0
  325. package/services/react-word-editor/components/workspace-certification-panel.tsx +307 -0
  326. package/services/react-word-editor/lib/certification-bundle.ts +796 -0
  327. package/services/react-word-editor/lib/certification-store.ts +661 -0
  328. package/services/react-word-editor/lib/demo-fixtures.test.mjs +195 -0
  329. package/services/react-word-editor/lib/demo-fixtures.ts +1519 -0
  330. package/services/react-word-editor/lib/editor-session-summary.test.mjs +68 -0
  331. package/services/react-word-editor/lib/editor-session-summary.ts +14 -0
  332. package/services/react-word-editor/lib/editor-session.ts +228 -0
  333. package/services/react-word-editor/lib/exports-route.test.mjs +32 -0
  334. package/services/react-word-editor/lib/harness-client.ts +347 -0
  335. package/services/react-word-editor/lib/harness-config.json +30 -0
  336. package/services/react-word-editor/lib/harness-config.test.mjs +31 -0
  337. package/services/react-word-editor/lib/harness-config.ts +21 -0
  338. package/services/react-word-editor/lib/harness-editor-datastore.test.mjs +220 -0
  339. package/services/react-word-editor/lib/harness-editor-datastore.ts +161 -0
  340. package/services/react-word-editor/lib/private-mode.test.mjs +42 -0
  341. package/services/react-word-editor/lib/private-mode.ts +61 -0
  342. package/services/react-word-editor/lib/regression-report.test.mjs +352 -0
  343. package/services/react-word-editor/lib/regression-report.ts +896 -0
  344. package/services/react-word-editor/lib/run-artifacts.ts +934 -0
  345. package/services/react-word-editor/lib/run-history.ts +755 -0
  346. package/services/react-word-editor/lib/scenario-artifacts.test.mjs +41 -0
  347. package/services/react-word-editor/lib/scenario-artifacts.ts +44 -0
  348. package/services/react-word-editor/lib/storage.ts +953 -0
  349. package/services/react-word-editor/lib/validator-client.test.mjs +54 -0
  350. package/services/react-word-editor/lib/validator-client.ts +95 -0
  351. package/services/react-word-editor/lib/workspace-navigation.ts +79 -0
  352. package/services/react-word-editor/middleware.ts +35 -0
  353. package/services/react-word-editor/next-env.d.ts +6 -0
  354. package/services/react-word-editor/next.config.mjs +15 -0
  355. package/services/react-word-editor/package.json +38 -0
  356. package/services/react-word-editor/postcss.config.mjs +8 -0
  357. package/services/react-word-editor/railway.json +21 -0
  358. package/services/react-word-editor/scripts/wave-10-certification.mjs +101 -0
  359. package/services/react-word-editor/scripts/wave-9-live-usability-pilot.mjs +911 -0
  360. package/services/react-word-editor/tsconfig.json +39 -0
  361. package/services/react-word-editor/tsconfig.tsbuildinfo +1 -0
  362. package/skills/README.md +48 -0
  363. package/skills/domain-docx-compatibility/SKILL.md +44 -0
  364. package/skills/domain-docx-compatibility/skill.json +19 -0
  365. package/skills/domain-editor-architecture/SKILL.md +49 -0
  366. package/skills/domain-editor-architecture/skill.json +19 -0
  367. package/skills/domain-legal-review/SKILL.md +39 -0
  368. package/skills/domain-legal-review/skill.json +19 -0
  369. package/skills/provider-aws/SKILL.md +117 -0
  370. package/skills/provider-aws/adapters/claude.md +1 -0
  371. package/skills/provider-aws/adapters/codex.md +1 -0
  372. package/skills/provider-aws/references/service-verification.md +39 -0
  373. package/skills/provider-aws/skill.json +54 -0
  374. package/skills/provider-custom-deploy/SKILL.md +64 -0
  375. package/skills/provider-custom-deploy/skill.json +50 -0
  376. package/skills/provider-docker-compose/SKILL.md +96 -0
  377. package/skills/provider-docker-compose/adapters/local.md +1 -0
  378. package/skills/provider-docker-compose/skill.json +53 -0
  379. package/skills/provider-github-release/SKILL.md +121 -0
  380. package/skills/provider-github-release/adapters/claude.md +1 -0
  381. package/skills/provider-github-release/adapters/codex.md +1 -0
  382. package/skills/provider-github-release/skill.json +55 -0
  383. package/skills/provider-kubernetes/SKILL.md +143 -0
  384. package/skills/provider-kubernetes/adapters/claude.md +1 -0
  385. package/skills/provider-kubernetes/adapters/codex.md +1 -0
  386. package/skills/provider-kubernetes/references/kubectl-patterns.md +58 -0
  387. package/skills/provider-kubernetes/skill.json +52 -0
  388. package/skills/provider-railway/SKILL.md +123 -0
  389. package/skills/provider-railway/adapters/claude.md +1 -0
  390. package/skills/provider-railway/adapters/codex.md +1 -0
  391. package/skills/provider-railway/adapters/local.md +1 -0
  392. package/skills/provider-railway/adapters/opencode.md +1 -0
  393. package/skills/provider-railway/references/verification-commands.md +39 -0
  394. package/skills/provider-railway/skill.json +71 -0
  395. package/skills/provider-ssh-manual/SKILL.md +97 -0
  396. package/skills/provider-ssh-manual/skill.json +54 -0
  397. package/skills/repo-coding-rules/SKILL.md +55 -0
  398. package/skills/repo-coding-rules/skill.json +34 -0
  399. package/skills/role-cont-eval/SKILL.md +91 -0
  400. package/skills/role-cont-eval/adapters/codex.md +1 -0
  401. package/skills/role-cont-eval/skill.json +36 -0
  402. package/skills/role-cont-qa/SKILL.md +100 -0
  403. package/skills/role-cont-qa/adapters/claude.md +1 -0
  404. package/skills/role-cont-qa/skill.json +36 -0
  405. package/skills/role-deploy/SKILL.md +97 -0
  406. package/skills/role-deploy/skill.json +36 -0
  407. package/skills/role-design/SKILL.md +50 -0
  408. package/skills/role-design/skill.json +36 -0
  409. package/skills/role-documentation/SKILL.md +76 -0
  410. package/skills/role-documentation/skill.json +36 -0
  411. package/skills/role-implementation/SKILL.md +45 -0
  412. package/skills/role-implementation/skill.json +36 -0
  413. package/skills/role-infra/SKILL.md +81 -0
  414. package/skills/role-infra/skill.json +36 -0
  415. package/skills/role-integration/SKILL.md +91 -0
  416. package/skills/role-integration/skill.json +36 -0
  417. package/skills/role-planner/SKILL.md +39 -0
  418. package/skills/role-planner/skill.json +21 -0
  419. package/skills/role-research/SKILL.md +65 -0
  420. package/skills/role-research/skill.json +36 -0
  421. package/skills/role-security/SKILL.md +60 -0
  422. package/skills/role-security/skill.json +36 -0
  423. package/skills/runtime-claude/SKILL.md +66 -0
  424. package/skills/runtime-claude/skill.json +36 -0
  425. package/skills/runtime-codex/SKILL.md +58 -0
  426. package/skills/runtime-codex/skill.json +36 -0
  427. package/skills/runtime-local/SKILL.md +46 -0
  428. package/skills/runtime-local/skill.json +36 -0
  429. package/skills/runtime-opencode/SKILL.md +58 -0
  430. package/skills/runtime-opencode/skill.json +36 -0
  431. package/skills/signal-hygiene/SKILL.md +51 -0
  432. package/skills/signal-hygiene/skill.json +20 -0
  433. package/skills/tui-design/SKILL.md +77 -0
  434. package/skills/tui-design/references/tui-design.md +259 -0
  435. package/skills/tui-design/skill.json +36 -0
  436. package/skills/wave-core/SKILL.md +141 -0
  437. package/skills/wave-core/references/marker-syntax.md +70 -0
  438. package/skills/wave-core/skill.json +35 -0
  439. package/src/README.md +85 -0
  440. package/src/api/README.md +22 -0
  441. package/src/api/public-types.ts +525 -0
  442. package/src/component-inventory.md +99 -0
  443. package/src/core/README.md +10 -0
  444. package/src/core/commands/README.md +3 -0
  445. package/src/core/commands/formatting-commands.ts +161 -0
  446. package/src/core/commands/image-commands.ts +144 -0
  447. package/src/core/commands/index.ts +1013 -0
  448. package/src/core/commands/list-commands.ts +370 -0
  449. package/src/core/commands/review-commands.ts +108 -0
  450. package/src/core/commands/text-commands.ts +119 -0
  451. package/src/core/schema/README.md +3 -0
  452. package/src/core/schema/text-schema.ts +512 -0
  453. package/src/core/selection/README.md +3 -0
  454. package/src/core/selection/mapping.ts +238 -0
  455. package/src/core/selection/review-anchors.ts +94 -0
  456. package/src/core/state/README.md +3 -0
  457. package/src/core/state/editor-state.ts +580 -0
  458. package/src/core/state/text-transaction.ts +276 -0
  459. package/src/formats/xlsx/io/parse-shared-strings.ts +41 -0
  460. package/src/formats/xlsx/io/parse-sheet.ts +289 -0
  461. package/src/formats/xlsx/io/parse-styles.ts +57 -0
  462. package/src/formats/xlsx/io/parse-workbook.ts +75 -0
  463. package/src/formats/xlsx/io/xlsx-session.ts +306 -0
  464. package/src/formats/xlsx/model/cell.ts +189 -0
  465. package/src/formats/xlsx/model/sheet.ts +244 -0
  466. package/src/formats/xlsx/model/styles.ts +118 -0
  467. package/src/formats/xlsx/model/workbook.ts +449 -0
  468. package/src/io/README.md +10 -0
  469. package/src/io/docx-session.ts +1763 -0
  470. package/src/io/export/README.md +3 -0
  471. package/src/io/export/export-session.ts +165 -0
  472. package/src/io/export/minimal-docx.ts +115 -0
  473. package/src/io/export/reattach-preserved-parts.ts +54 -0
  474. package/src/io/export/serialize-comments.ts +876 -0
  475. package/src/io/export/serialize-footnotes.ts +217 -0
  476. package/src/io/export/serialize-headers-footers.ts +200 -0
  477. package/src/io/export/serialize-main-document.ts +982 -0
  478. package/src/io/export/serialize-numbering.ts +97 -0
  479. package/src/io/export/serialize-revisions.ts +389 -0
  480. package/src/io/export/serialize-runtime-revisions.ts +265 -0
  481. package/src/io/export/serialize-tables.ts +147 -0
  482. package/src/io/export/split-review-boundaries.ts +194 -0
  483. package/src/io/normalize/README.md +3 -0
  484. package/src/io/normalize/normalize-text.ts +437 -0
  485. package/src/io/ooxml/README.md +3 -0
  486. package/src/io/ooxml/parse-comments.ts +779 -0
  487. package/src/io/ooxml/parse-complex-content.ts +287 -0
  488. package/src/io/ooxml/parse-fields.ts +438 -0
  489. package/src/io/ooxml/parse-footnotes.ts +403 -0
  490. package/src/io/ooxml/parse-headers-footers.ts +483 -0
  491. package/src/io/ooxml/parse-inline-media.ts +431 -0
  492. package/src/io/ooxml/parse-main-document.ts +1846 -0
  493. package/src/io/ooxml/parse-numbering.ts +425 -0
  494. package/src/io/ooxml/parse-revisions.ts +658 -0
  495. package/src/io/ooxml/parse-shapes.ts +271 -0
  496. package/src/io/ooxml/parse-tables.ts +568 -0
  497. package/src/io/ooxml/parse-theme.ts +314 -0
  498. package/src/io/ooxml/part-manifest.ts +136 -0
  499. package/src/io/ooxml/revision-boundaries.ts +351 -0
  500. package/src/io/opc/README.md +3 -0
  501. package/src/io/opc/corrupt-package.ts +166 -0
  502. package/src/io/opc/docx-package.ts +74 -0
  503. package/src/io/opc/package-reader.ts +320 -0
  504. package/src/io/opc/package-writer.ts +273 -0
  505. package/src/model/README.md +3 -0
  506. package/src/model/canonical-document.ts +1911 -0
  507. package/src/model/cds-1.0.0.ts +196 -0
  508. package/src/model/snapshot.ts +393 -0
  509. package/src/preservation/README.md +3 -0
  510. package/src/preservation/markup-compatibility.ts +48 -0
  511. package/src/preservation/opaque-fragment-store.ts +89 -0
  512. package/src/preservation/opaque-region.ts +233 -0
  513. package/src/preservation/package-preservation.ts +120 -0
  514. package/src/preservation/preserved-part-manifest.ts +56 -0
  515. package/src/preservation/relationship-retention.ts +57 -0
  516. package/src/preservation/store.ts +185 -0
  517. package/src/review/README.md +16 -0
  518. package/src/review/store/README.md +3 -0
  519. package/src/review/store/comment-anchors.ts +70 -0
  520. package/src/review/store/comment-remapping.ts +154 -0
  521. package/src/review/store/comment-store.ts +331 -0
  522. package/src/review/store/comment-thread.ts +109 -0
  523. package/src/review/store/revision-actions.ts +394 -0
  524. package/src/review/store/revision-store.ts +303 -0
  525. package/src/review/store/revision-types.ts +168 -0
  526. package/src/review/store/runtime-comment-store.ts +43 -0
  527. package/src/runtime/README.md +3 -0
  528. package/src/runtime/ai-action-policy.ts +764 -0
  529. package/src/runtime/document-runtime.ts +969 -0
  530. package/src/runtime/read-only-diagnostics-runtime.ts +232 -0
  531. package/src/runtime/review-runtime.ts +44 -0
  532. package/src/runtime/revision-runtime.ts +107 -0
  533. package/src/runtime/session-capabilities.ts +138 -0
  534. package/src/runtime/surface-projection.ts +570 -0
  535. package/src/runtime/table-commands.ts +84 -0
  536. package/src/runtime/table-schema.ts +125 -0
  537. package/src/ui/README.md +30 -0
  538. package/src/ui/WordReviewEditor.tsx +1283 -0
  539. package/src/ui/comments/README.md +3 -0
  540. package/src/ui/compatibility/README.md +3 -0
  541. package/src/ui/editor-surface/README.md +3 -0
  542. package/src/ui/headless/comment-decoration-model.ts +124 -0
  543. package/src/ui/headless/revision-decoration-model.ts +128 -0
  544. package/src/ui/headless/selection-helpers.ts +34 -0
  545. package/src/ui/headless/use-editor-keyboard.ts +98 -0
  546. package/src/ui/review/README.md +3 -0
  547. package/src/ui/shared/revision-filters.ts +31 -0
  548. package/src/ui/status/README.md +3 -0
  549. package/src/ui/theme/README.md +3 -0
  550. package/src/ui/toolbar/README.md +3 -0
  551. package/src/ui-tailwind/chrome/tw-alert-banner.tsx +48 -0
  552. package/src/ui-tailwind/chrome/tw-selection-toolbar.tsx +44 -0
  553. package/src/ui-tailwind/chrome/tw-unsaved-modal.tsx +58 -0
  554. package/src/ui-tailwind/chrome/use-before-unload.ts +20 -0
  555. package/src/ui-tailwind/editor-surface/pm-command-bridge.ts +139 -0
  556. package/src/ui-tailwind/editor-surface/pm-decorations.ts +98 -0
  557. package/src/ui-tailwind/editor-surface/pm-position-map.ts +123 -0
  558. package/src/ui-tailwind/editor-surface/pm-schema.ts +452 -0
  559. package/src/ui-tailwind/editor-surface/pm-state-from-snapshot.ts +327 -0
  560. package/src/ui-tailwind/editor-surface/search-plugin.ts +157 -0
  561. package/src/ui-tailwind/editor-surface/tw-caret.tsx +12 -0
  562. package/src/ui-tailwind/editor-surface/tw-editor-surface.tsx +150 -0
  563. package/src/ui-tailwind/editor-surface/tw-inline-token.tsx +118 -0
  564. package/src/ui-tailwind/editor-surface/tw-opaque-block.tsx +52 -0
  565. package/src/ui-tailwind/editor-surface/tw-paragraph-block.tsx +151 -0
  566. package/src/ui-tailwind/editor-surface/tw-prosemirror-surface.tsx +215 -0
  567. package/src/ui-tailwind/editor-surface/tw-segment-view.tsx +111 -0
  568. package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +108 -0
  569. package/src/ui-tailwind/index.ts +61 -0
  570. package/src/ui-tailwind/review/tw-comment-sidebar.tsx +276 -0
  571. package/src/ui-tailwind/review/tw-health-panel.tsx +120 -0
  572. package/src/ui-tailwind/review/tw-review-rail.tsx +120 -0
  573. package/src/ui-tailwind/review/tw-revision-sidebar.tsx +164 -0
  574. package/src/ui-tailwind/status/tw-status-bar.tsx +58 -0
  575. package/src/ui-tailwind/theme/editor-theme.css +190 -0
  576. package/src/ui-tailwind/toolbar/tw-toolbar-icon-button.tsx +48 -0
  577. package/src/ui-tailwind/toolbar/tw-toolbar.tsx +231 -0
  578. package/src/ui-tailwind/tw-review-workspace.tsx +140 -0
  579. package/src/validation/README.md +3 -0
  580. package/src/validation/compatibility-engine.ts +317 -0
  581. package/src/validation/compatibility-report.ts +160 -0
  582. package/src/validation/diagnostics.ts +203 -0
  583. package/src/validation/import-diagnostics.ts +128 -0
  584. package/src/validation/low-priority-word-surfaces.ts +373 -0
  585. package/test/README.md +16 -0
  586. package/test/core/formatting-commands.test.ts +285 -0
  587. package/test/core/image-commands.test.ts +298 -0
  588. package/test/core/mapping.test.ts +186 -0
  589. package/test/core/text-commands.test.ts +176 -0
  590. package/test/fixtures/docx/F01-basic-contract.docx +0 -0
  591. package/test/fixtures/docx/F01-basic-contract.md +33 -0
  592. package/test/fixtures/docx/F02-headings-styles.docx +0 -0
  593. package/test/fixtures/docx/F02-headings-styles.md +33 -0
  594. package/test/fixtures/docx/F03-legal-outline-numbering.docx +0 -0
  595. package/test/fixtures/docx/F03-legal-outline-numbering.md +34 -0
  596. package/test/fixtures/docx/F04-restart-numbering-schedules.docx +0 -0
  597. package/test/fixtures/docx/F04-restart-numbering-schedules.md +33 -0
  598. package/test/fixtures/docx/F05-table-heavy-agreement.docx +0 -0
  599. package/test/fixtures/docx/F05-table-heavy-agreement.md +34 -0
  600. package/test/fixtures/docx/F06-merged-cells-signature-table.docx +0 -0
  601. package/test/fixtures/docx/F06-merged-cells-signature-table.md +34 -0
  602. package/test/fixtures/docx/F07-inline-images-exhibit.docx +0 -0
  603. package/test/fixtures/docx/F07-inline-images-exhibit.md +34 -0
  604. package/test/fixtures/docx/F08-hyperlinks.docx +0 -0
  605. package/test/fixtures/docx/F08-hyperlinks.md +33 -0
  606. package/test/fixtures/docx/F09-comments-single-paragraph.docx +0 -0
  607. package/test/fixtures/docx/F09-comments-single-paragraph.md +33 -0
  608. package/test/fixtures/docx/F10-threaded-comments-resolve.docx +0 -0
  609. package/test/fixtures/docx/F10-threaded-comments-resolve.md +33 -0
  610. package/test/fixtures/docx/F11-redlines-basic.docx +0 -0
  611. package/test/fixtures/docx/F11-redlines-basic.md +33 -0
  612. package/test/fixtures/docx/F12-redlines-paragraph-joins-splits.docx +0 -0
  613. package/test/fixtures/docx/F12-redlines-paragraph-joins-splits.md +33 -0
  614. package/test/fixtures/docx/F13-comments-on-deleted-text.docx +0 -0
  615. package/test/fixtures/docx/F13-comments-on-deleted-text.md +33 -0
  616. package/test/fixtures/docx/F14-revisions-in-tables-and-lists.docx +0 -0
  617. package/test/fixtures/docx/F14-revisions-in-tables-and-lists.md +33 -0
  618. package/test/fixtures/docx/F15-sections-headers-footers.docx +0 -0
  619. package/test/fixtures/docx/F15-sections-headers-footers.md +33 -0
  620. package/test/fixtures/docx/F16-footnotes-endnotes.docx +0 -0
  621. package/test/fixtures/docx/F16-footnotes-endnotes.md +33 -0
  622. package/test/fixtures/docx/F17-fields-and-toc.docx +0 -0
  623. package/test/fixtures/docx/F17-fields-and-toc.md +33 -0
  624. package/test/fixtures/docx/F18-content-controls-template.docx +0 -0
  625. package/test/fixtures/docx/F18-content-controls-template.md +33 -0
  626. package/test/fixtures/docx/F19-custom-xml-doc-assembly.docx +0 -0
  627. package/test/fixtures/docx/F19-custom-xml-doc-assembly.md +35 -0
  628. package/test/fixtures/docx/F20-unknown-ooxml-and-alternatecontent.docx +0 -0
  629. package/test/fixtures/docx/F20-unknown-ooxml-and-alternatecontent.md +33 -0
  630. package/test/fixtures/docx/F21-malformed-broken-docx.docx +0 -0
  631. package/test/fixtures/docx/F21-malformed-broken-docx.md +33 -0
  632. package/test/fixtures/docx/README.md +74 -0
  633. package/test/fixtures/docx/certification-manifest.json +104 -0
  634. package/test/fixtures/docx/fixtures.manifest.json +196 -0
  635. package/test/fixtures/encrypted-docx/README.md +27 -0
  636. package/test/fixtures/encrypted-docx/certification-manifest.json +9 -0
  637. package/test/fixtures/encrypted-docx/fixtures.manifest.json +47 -0
  638. package/test/fixtures/scenarios/docx/README.md +25 -0
  639. package/test/fixtures/scenarios/docx/S01-sow-template.docx +0 -0
  640. package/test/fixtures/scenarios/docx/S01-sow-template.md +30 -0
  641. package/test/fixtures/scenarios/docx/S02-bw-partner-user-licence-agreement-redlines.docx +0 -0
  642. package/test/fixtures/scenarios/docx/S02-bw-partner-user-licence-agreement-redlines.md +32 -0
  643. package/test/fixtures/scenarios/docx/scenario-manifest.json +53 -0
  644. package/test/formats/xlsx/io/xlsx-import.test.ts +766 -0
  645. package/test/formats/xlsx/model/workbook.test.ts +669 -0
  646. package/test/helpers/dom-setup.ts +124 -0
  647. package/test/io/comment-roundtrip.test.ts +272 -0
  648. package/test/io/complex-content-roundtrip.test.ts +632 -0
  649. package/test/io/docx-compatibility-regression.test.ts +199 -0
  650. package/test/io/docx-session.test.ts +1495 -0
  651. package/test/io/footnotes-roundtrip.test.ts +318 -0
  652. package/test/io/headers-footers-roundtrip.test.ts +547 -0
  653. package/test/io/numbering-roundtrip.test.ts +234 -0
  654. package/test/io/package-reader.test.ts +199 -0
  655. package/test/io/paragraph-properties-roundtrip.test.ts +129 -0
  656. package/test/io/preserved-package-roundtrip.test.ts +365 -0
  657. package/test/io/property-completeness.test.ts +292 -0
  658. package/test/io/revision-roundtrip.test.ts +347 -0
  659. package/test/io/structural-blocks.test.ts +202 -0
  660. package/test/io/table-media-roundtrip.test.ts +448 -0
  661. package/test/io/table-properties-roundtrip.test.ts +569 -0
  662. package/test/io/table-roundtrip.test.ts +302 -0
  663. package/test/io/text-roundtrip.test.ts +344 -0
  664. package/test/model/canonical-document.test.ts +285 -0
  665. package/test/preservation/opaque-fragment-store.test.ts +121 -0
  666. package/test/preservation/package-preservation.test.ts +395 -0
  667. package/test/preservation/store.test.ts +84 -0
  668. package/test/review/comment-remapping.test.ts +220 -0
  669. package/test/review/comment-store.test.ts +180 -0
  670. package/test/review/move-revisions.test.ts +143 -0
  671. package/test/review/property-change-revisions.test.ts +225 -0
  672. package/test/review/revision-actions.test.ts +330 -0
  673. package/test/review/revision-store.test.ts +193 -0
  674. package/test/runtime/session-capabilities.test.ts +260 -0
  675. package/test/runtime/table-commands.test.ts +356 -0
  676. package/test/runtime/table-schema.test.ts +221 -0
  677. package/test/runtime/tracked-changes-toggle.test.ts +107 -0
  678. package/test/ui/comment-review-surface.test.tsx +114 -0
  679. package/test/ui/reduced-motion-toggle.test.tsx +137 -0
  680. package/test/ui/word-review-editor.imported-scenarios.test.tsx +169 -0
  681. package/test/ui/word-review-editor.interaction.test.tsx +1198 -0
  682. package/test/ui/word-review-editor.test.js +188 -0
  683. package/test/ui/word-review-editor.test.tsx +280 -0
  684. package/test/ui-tailwind/search-plugin.test.ts +286 -0
  685. package/test/validation/compatibility-engine.test.ts +336 -0
  686. package/test/validation/compatibility-report.test.ts +189 -0
  687. package/test/validation/low-priority-word-surfaces.test.ts +282 -0
  688. package/test/validation/malformed-doc.test.ts +113 -0
  689. package/test-results/.last-run.json +4 -0
  690. package/wave.config.json +406 -0
@@ -0,0 +1,1283 @@
1
+ import React, {
2
+ forwardRef,
3
+ useEffect,
4
+ useImperativeHandle,
5
+ useMemo,
6
+ useRef,
7
+ useState,
8
+ useSyncExternalStore,
9
+ } from "react";
10
+
11
+ import type {
12
+ AutosaveState,
13
+ EditorDatastoreAdapter,
14
+ CompatibilityReport,
15
+ EditorError,
16
+ EditorWarning,
17
+ ExportDocxOptions,
18
+ PersistedEditorSnapshot,
19
+ RuntimeRenderSnapshot,
20
+ SelectionSnapshot as PublicSelectionSnapshot,
21
+ ExportResult,
22
+ WordReviewEditorEvent,
23
+ WordReviewEditorProps,
24
+ WordReviewEditorRef,
25
+ } from "../api/public-types";
26
+ import {
27
+ createDetachedAnchor,
28
+ createNodeAnchor,
29
+ createRangeAnchor,
30
+ } from "../core/selection/mapping.ts";
31
+ import { createCanonicalDocumentId } from "../core/state/editor-state.ts";
32
+ import {
33
+ createDocumentRuntime,
34
+ type DocumentRuntime,
35
+ } from "../runtime/document-runtime.ts";
36
+ import { loadDocxEditorSession } from "../io/docx-session.ts";
37
+ import { exportSnapshotToMinimalDocx } from "../io/export/minimal-docx.ts";
38
+ import { DOCX_MIME_TYPE } from "../io/opc/docx-package.ts";
39
+ import { deriveCapabilities } from "../runtime/session-capabilities";
40
+ import { TwProseMirrorSurface } from "../ui-tailwind/editor-surface/tw-prosemirror-surface";
41
+ import { TwReviewWorkspace } from "../ui-tailwind/tw-review-workspace";
42
+ import type { ReviewRailTab } from "../ui-tailwind/review/tw-review-rail";
43
+ import type { ViewMode } from "../ui-tailwind/toolbar/tw-toolbar";
44
+ import type { MarkupDisplay } from "./headless/comment-decoration-model";
45
+
46
+ interface ResolvedSource {
47
+ source: "docx" | "snapshot" | "datastore";
48
+ sourceLabel?: string;
49
+ initialDocx?: Uint8Array | ArrayBuffer;
50
+ initialSnapshot?: PersistedEditorSnapshot;
51
+ }
52
+
53
+ interface CreateRuntimeArgs {
54
+ documentId: string;
55
+ readOnly: boolean;
56
+ source: ResolvedSource;
57
+ datastore?: EditorDatastoreAdapter;
58
+ currentUserId?: string;
59
+ }
60
+
61
+ interface RuntimeLifecycleHandlers {
62
+ onEvent?: (event: WordReviewEditorEvent) => void;
63
+ onWarning?: (warning: EditorWarning) => void;
64
+ onError?: (error: EditorError) => void;
65
+ }
66
+
67
+ interface WordReviewEditorRuntime extends DocumentRuntime {
68
+ getFatalError?(): EditorError | undefined;
69
+ dispose?(): void;
70
+ }
71
+
72
+ export function __createWordReviewEditorRefBridge(
73
+ runtime: WordReviewEditorRuntime,
74
+ ): WordReviewEditorRef {
75
+ return {
76
+ focus: () => runtime.focus(),
77
+ blur: () => runtime.blur(),
78
+ undo: () => runtime.undo(),
79
+ redo: () => runtime.redo(),
80
+ addComment: (params) => runtime.addComment(params),
81
+ openComment: (commentId) => runtime.openComment(commentId),
82
+ resolveComment: (commentId) => runtime.resolveComment(commentId),
83
+ reopenComment: (commentId) => runtime.reopenComment(commentId),
84
+ addCommentReply: (commentId, body) => runtime.addCommentReply(commentId, body),
85
+ editCommentBody: (commentId, body) => runtime.editCommentBody(commentId, body),
86
+ acceptChange: (changeId) => runtime.acceptChange(changeId),
87
+ rejectChange: (changeId) => runtime.rejectChange(changeId),
88
+ acceptAllChanges: () => runtime.acceptAllChanges(),
89
+ rejectAllChanges: () => runtime.rejectAllChanges(),
90
+ exportDocx: (options) => runtime.exportDocx(options),
91
+ getSnapshot: () => runtime.getPersistedSnapshot(),
92
+ getCompatibilityReport: () => runtime.getCompatibilityReport(),
93
+ getWarnings: () => runtime.getWarnings(),
94
+ };
95
+ }
96
+
97
+ export async function __resolveWordReviewEditorSource(
98
+ props: Pick<
99
+ WordReviewEditorProps,
100
+ | "documentId"
101
+ | "datastore"
102
+ | "externalDocSource"
103
+ | "initialDocx"
104
+ | "initialSnapshot"
105
+ | "initialSourceLabel"
106
+ >,
107
+ ): Promise<ResolvedSource> {
108
+ const explicitInitialCount =
109
+ Number(Boolean(props.initialDocx)) + Number(Boolean(props.initialSnapshot));
110
+ if (explicitInitialCount > 1) {
111
+ throw new Error("Provide exactly one of initialDocx or initialSnapshot.");
112
+ }
113
+
114
+ if (props.externalDocSource) {
115
+ return props.externalDocSource.kind === "docx"
116
+ ? {
117
+ source: "docx",
118
+ initialDocx: props.externalDocSource.bytes,
119
+ sourceLabel: props.externalDocSource.sourceLabel,
120
+ }
121
+ : {
122
+ source: "snapshot",
123
+ initialSnapshot: props.externalDocSource.snapshot,
124
+ sourceLabel: props.externalDocSource.sourceLabel,
125
+ };
126
+ }
127
+
128
+ if (props.initialSnapshot) {
129
+ return {
130
+ source: "snapshot",
131
+ initialSnapshot: props.initialSnapshot,
132
+ sourceLabel: props.initialSourceLabel,
133
+ };
134
+ }
135
+
136
+ if (props.initialDocx) {
137
+ return {
138
+ source: "docx",
139
+ initialDocx: props.initialDocx,
140
+ sourceLabel: props.initialSourceLabel,
141
+ };
142
+ }
143
+
144
+ if (!props.datastore) {
145
+ throw new Error(
146
+ `WordReviewEditor ${props.documentId} needs initialDocx, initialSnapshot, or datastore.load().`,
147
+ );
148
+ }
149
+
150
+ const loadResult = await props.datastore.load({
151
+ documentId: props.documentId,
152
+ });
153
+
154
+ if (!loadResult.source) {
155
+ throw new Error(`Datastore did not return a loadable source for ${props.documentId}.`);
156
+ }
157
+
158
+ return loadResult.source.kind === "docx"
159
+ ? {
160
+ source: "datastore",
161
+ initialDocx: loadResult.source.bytes,
162
+ sourceLabel: loadResult.source.sourceLabel,
163
+ }
164
+ : {
165
+ source: "datastore",
166
+ initialSnapshot: loadResult.source.snapshot,
167
+ sourceLabel: loadResult.source.sourceLabel,
168
+ };
169
+ }
170
+
171
+ export function __createFallbackRuntime(args: CreateRuntimeArgs): WordReviewEditorRuntime {
172
+ return createRuntime(args);
173
+ }
174
+
175
+ function createRuntime(
176
+ args: CreateRuntimeArgs,
177
+ handlers: RuntimeLifecycleHandlers = {},
178
+ ): WordReviewEditorRuntime {
179
+ const docxSession = args.source.initialDocx
180
+ ? loadDocxEditorSession({
181
+ documentId: args.documentId,
182
+ sourceLabel: args.source.sourceLabel,
183
+ bytes: args.source.initialDocx,
184
+ editorBuild: "dev",
185
+ })
186
+ : undefined;
187
+ const initialSnapshot =
188
+ args.source.initialSnapshot ??
189
+ docxSession?.initialSnapshot ??
190
+ createFallbackPersistedSnapshot(
191
+ args.documentId,
192
+ args.source.sourceLabel ?? "Generated shell snapshot",
193
+ );
194
+
195
+ return createDocumentRuntime({
196
+ documentId: args.documentId,
197
+ initialSnapshot,
198
+ sourceKind: args.source.source,
199
+ sourceLabel: args.source.sourceLabel,
200
+ readOnly: args.readOnly || docxSession?.readOnly,
201
+ editorBuild: initialSnapshot.editorBuild,
202
+ fatalError: docxSession?.fatalError,
203
+ exportDocx: async (snapshot, options) =>
204
+ docxSession
205
+ ? docxSession.exportDocx(snapshot, options)
206
+ : {
207
+ bytes: exportSnapshotToMinimalDocx(snapshot),
208
+ mimeType: DOCX_MIME_TYPE,
209
+ fileName: options?.fileName ?? `${args.documentId}.docx`,
210
+ },
211
+ onWarning: handlers.onWarning,
212
+ onError: handlers.onError,
213
+ defaultAuthorId: args.currentUserId,
214
+ });
215
+ }
216
+
217
+ export const WordReviewEditor = forwardRef<WordReviewEditorRef, WordReviewEditorProps>(
218
+ function WordReviewEditor(props, ref) {
219
+ const {
220
+ currentUser,
221
+ datastore,
222
+ documentId,
223
+ externalDocSource,
224
+ externalDocumentRevision,
225
+ initialDocx,
226
+ initialSnapshot,
227
+ initialSourceLabel,
228
+ markupDisplay = "simple",
229
+ onError,
230
+ onEvent,
231
+ onWarning,
232
+ readOnly = false,
233
+ reviewMode = "review",
234
+ } = props;
235
+
236
+ const [runtime, setRuntime] = useState<WordReviewEditorRuntime | null>(null);
237
+ const [loadError, setLoadError] = useState<EditorError | null>(null);
238
+ const [viewMode, setViewMode] = useState<ViewMode>("canvas");
239
+ const liveMarkupDisplay: MarkupDisplay = viewMode === "document" ? "all" : "clean";
240
+ const [activeRailTab, setActiveRailTab] = useState<ReviewRailTab>("comments");
241
+ const [showTrackedChanges, setShowTrackedChanges] = useState(false);
242
+ const [activeRevisionId, setActiveRevisionId] = useState<string | undefined>();
243
+ const runtimeRef = useRef<WordReviewEditorRuntime | null>(null);
244
+ const autosaveTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null);
245
+ const lastSavedRevisionTokenRef = useRef<string | null>(null);
246
+ const datastoreRef = useRef(datastore);
247
+ const onEventRef = useRef(onEvent);
248
+ const onWarningRef = useRef(onWarning);
249
+ const onErrorRef = useRef(onError);
250
+ const initialSourceRef = useRef<{
251
+ documentId: string;
252
+ initialDocx?: Uint8Array | ArrayBuffer;
253
+ initialSnapshot?: PersistedEditorSnapshot;
254
+ initialSourceLabel?: string;
255
+ } | null>(null);
256
+
257
+ if (!initialSourceRef.current || initialSourceRef.current.documentId !== documentId) {
258
+ initialSourceRef.current = {
259
+ documentId,
260
+ initialDocx,
261
+ initialSnapshot,
262
+ initialSourceLabel,
263
+ };
264
+ }
265
+
266
+ const stableInitialSource = initialSourceRef.current;
267
+ const sourceReloadKey = externalDocSource
268
+ ? `external:${externalDocSource.kind}:${externalDocumentRevision ?? "static"}`
269
+ : stableInitialSource?.initialSnapshot
270
+ ? "initial-snapshot"
271
+ : stableInitialSource?.initialDocx
272
+ ? "initial-docx"
273
+ : "datastore";
274
+
275
+ useEffect(() => {
276
+ datastoreRef.current = datastore;
277
+ onEventRef.current = onEvent;
278
+ onWarningRef.current = onWarning;
279
+ onErrorRef.current = onError;
280
+ }, [datastore, onError, onEvent, onWarning]);
281
+
282
+ useEffect(() => {
283
+ let cancelled = false;
284
+
285
+ async function loadRuntime(): Promise<void> {
286
+ setLoadError(null);
287
+
288
+ try {
289
+ const source = await __resolveWordReviewEditorSource({
290
+ documentId,
291
+ datastore: datastoreRef.current,
292
+ externalDocSource,
293
+ initialDocx: stableInitialSource?.initialDocx,
294
+ initialSnapshot: stableInitialSource?.initialSnapshot,
295
+ initialSourceLabel: stableInitialSource?.initialSourceLabel,
296
+ });
297
+
298
+ if (cancelled) {
299
+ return;
300
+ }
301
+
302
+ runtimeRef.current?.dispose?.();
303
+ const nextRuntime = createRuntime(
304
+ {
305
+ documentId,
306
+ readOnly,
307
+ source,
308
+ datastore: datastoreRef.current,
309
+ currentUserId: currentUser.userId,
310
+ },
311
+ {
312
+ onWarning: onWarningRef.current,
313
+ onError: onErrorRef.current,
314
+ },
315
+ );
316
+ emitEditorEvent({
317
+ datastore: datastoreRef.current,
318
+ onEvent: onEventRef.current,
319
+ event: createReadyEvent(nextRuntime, source.source),
320
+ });
321
+ runtimeRef.current = nextRuntime;
322
+ setRuntime(nextRuntime);
323
+ } catch (error) {
324
+ if (cancelled) {
325
+ return;
326
+ }
327
+
328
+ const normalized = normalizeEditorError(error);
329
+ setLoadError(normalized);
330
+ onErrorRef.current?.(normalized);
331
+ emitEditorEvent({
332
+ datastore: datastoreRef.current,
333
+ onEvent: onEventRef.current,
334
+ event: {
335
+ type: "error",
336
+ documentId,
337
+ error: normalized,
338
+ },
339
+ });
340
+ }
341
+ }
342
+
343
+ void loadRuntime();
344
+
345
+ return () => {
346
+ cancelled = true;
347
+ };
348
+ }, [
349
+ documentId,
350
+ readOnly,
351
+ sourceReloadKey,
352
+ ]);
353
+
354
+ useEffect(() => {
355
+ if (!runtime?.subscribeToEvents) {
356
+ return;
357
+ }
358
+
359
+ return runtime.subscribeToEvents((event) => {
360
+ emitEditorEvent({
361
+ datastore: datastoreRef.current,
362
+ onEvent: onEventRef.current,
363
+ event,
364
+ });
365
+ });
366
+ }, [runtime]);
367
+
368
+ useEffect(() => {
369
+ return () => {
370
+ if (autosaveTimerRef.current) {
371
+ clearTimeout(autosaveTimerRef.current);
372
+ autosaveTimerRef.current = null;
373
+ }
374
+ runtimeRef.current?.dispose?.();
375
+ runtimeRef.current = null;
376
+ };
377
+ }, []);
378
+
379
+ const optimisticRuntime = useMemo(
380
+ () =>
381
+ __createFallbackRuntime({
382
+ documentId,
383
+ readOnly,
384
+ currentUserId: currentUser.userId,
385
+ source: {
386
+ source: "snapshot",
387
+ initialSnapshot:
388
+ initialSnapshot ?? createFallbackPersistedSnapshot(documentId, initialSourceLabel),
389
+ sourceLabel: guessSourceLabel(initialSourceLabel, initialSnapshot, externalDocSource),
390
+ },
391
+ datastore: datastoreRef.current,
392
+ }),
393
+ [
394
+ currentUser.userId,
395
+ documentId,
396
+ initialSnapshot,
397
+ initialSourceLabel,
398
+ readOnly,
399
+ externalDocSource?.kind,
400
+ externalDocSource?.sourceLabel,
401
+ ],
402
+ );
403
+
404
+ const fallbackSnapshot = useMemo(
405
+ () =>
406
+ loadError
407
+ ? createErrorSnapshot(documentId, loadError)
408
+ : createLoadingSnapshot(
409
+ documentId,
410
+ readOnly,
411
+ guessSourceLabel(initialSourceLabel, initialSnapshot, externalDocSource),
412
+ ),
413
+ [
414
+ documentId,
415
+ externalDocSource,
416
+ initialSnapshot,
417
+ initialSourceLabel,
418
+ loadError,
419
+ readOnly,
420
+ ],
421
+ );
422
+
423
+ const snapshot = useSyncExternalStore(
424
+ (listener) => runtime?.subscribe(listener) ?? (() => undefined),
425
+ () => runtime?.getRenderSnapshot() ?? fallbackSnapshot,
426
+ () => runtime?.getRenderSnapshot() ?? fallbackSnapshot,
427
+ );
428
+
429
+ const activeRuntime = runtime ?? optimisticRuntime;
430
+
431
+ useImperativeHandle(
432
+ ref,
433
+ () => ({
434
+ focus: () => activeRuntime.focus(),
435
+ blur: () => activeRuntime.blur(),
436
+ undo: () => activeRuntime.undo(),
437
+ redo: () => activeRuntime.redo(),
438
+ addComment: (params) =>
439
+ activeRuntime.addComment({
440
+ ...params,
441
+ authorId: params.authorId ?? currentUser.userId,
442
+ }),
443
+ openComment: (commentId) => activeRuntime.openComment(commentId),
444
+ resolveComment: (commentId) => activeRuntime.resolveComment(commentId),
445
+ reopenComment: (commentId) => activeRuntime.reopenComment(commentId),
446
+ addCommentReply: (commentId, body) =>
447
+ activeRuntime.addCommentReply(commentId, body, currentUser.userId),
448
+ editCommentBody: (commentId, body) =>
449
+ activeRuntime.editCommentBody(commentId, body),
450
+ acceptChange: (changeId) => activeRuntime.acceptChange(changeId),
451
+ rejectChange: (changeId) => activeRuntime.rejectChange(changeId),
452
+ acceptAllChanges: () => activeRuntime.acceptAllChanges(),
453
+ rejectAllChanges: () => activeRuntime.rejectAllChanges(),
454
+ exportDocx: (options) =>
455
+ runtime
456
+ ? persistAndExport({
457
+ datastore: datastoreRef.current,
458
+ documentId,
459
+ runtime,
460
+ onError: onErrorRef.current,
461
+ onEvent: onEventRef.current,
462
+ options,
463
+ lastSavedRevisionTokenRef,
464
+ autosaveTimerRef,
465
+ })
466
+ : rejectExportWhileLoading({
467
+ documentId,
468
+ datastore: datastoreRef.current,
469
+ onError: onErrorRef.current,
470
+ onEvent: onEventRef.current,
471
+ }),
472
+ getSnapshot: () => activeRuntime.getPersistedSnapshot(),
473
+ getCompatibilityReport: () => activeRuntime.getCompatibilityReport(),
474
+ getWarnings: () => activeRuntime.getWarnings(),
475
+ }),
476
+ [activeRuntime, currentUser.userId, documentId, runtime],
477
+ );
478
+
479
+ useEffect(() => {
480
+ if (!datastoreRef.current || props.autosave?.enabled === false || !runtime || readOnly) {
481
+ if (autosaveTimerRef.current) {
482
+ clearTimeout(autosaveTimerRef.current);
483
+ autosaveTimerRef.current = null;
484
+ }
485
+ return;
486
+ }
487
+
488
+ if (!snapshot.isReady || !snapshot.isDirty) {
489
+ return;
490
+ }
491
+
492
+ if (lastSavedRevisionTokenRef.current === snapshot.revisionToken) {
493
+ return;
494
+ }
495
+
496
+ if (autosaveTimerRef.current) {
497
+ clearTimeout(autosaveTimerRef.current);
498
+ }
499
+
500
+ const debounceMs = props.autosave?.debounceMs ?? 800;
501
+ if (debounceMs <= 0) {
502
+ void persistSnapshot({
503
+ datastore: datastoreRef.current,
504
+ documentId,
505
+ runtime,
506
+ isAutosave: true,
507
+ onError: onErrorRef.current,
508
+ onEvent: onEventRef.current,
509
+ lastSavedRevisionTokenRef,
510
+ });
511
+ return;
512
+ }
513
+
514
+ autosaveTimerRef.current = setTimeout(() => {
515
+ void persistSnapshot({
516
+ datastore: datastoreRef.current,
517
+ documentId,
518
+ runtime,
519
+ isAutosave: true,
520
+ onError: onErrorRef.current,
521
+ onEvent: onEventRef.current,
522
+ lastSavedRevisionTokenRef,
523
+ });
524
+ }, debounceMs);
525
+
526
+ return () => {
527
+ if (autosaveTimerRef.current) {
528
+ clearTimeout(autosaveTimerRef.current);
529
+ autosaveTimerRef.current = null;
530
+ }
531
+ };
532
+ }, [
533
+ documentId,
534
+ props.autosave?.debounceMs,
535
+ props.autosave?.enabled,
536
+ readOnly,
537
+ runtime,
538
+ snapshot.isDirty,
539
+ snapshot.isReady,
540
+ snapshot.revisionToken,
541
+ ]);
542
+
543
+ // Auto-select the most relevant rail tab based on content.
544
+ // Health tab was moved to toolbar popover — rail has only comments/changes.
545
+ useEffect(() => {
546
+ if (
547
+ activeRailTab === "comments" &&
548
+ snapshot.comments.totalCount === 0 &&
549
+ snapshot.trackedChanges.totalCount > 0
550
+ ) {
551
+ setActiveRailTab("changes");
552
+ return;
553
+ }
554
+ }, [
555
+ activeRailTab,
556
+ loadError,
557
+ snapshot.comments.totalCount,
558
+ snapshot.compatibility.blockExport,
559
+ snapshot.fatalError,
560
+ snapshot.trackedChanges.totalCount,
561
+ ]);
562
+
563
+ function focusAnchor(anchor: PublicSelectionSnapshot["activeRange"]): void {
564
+ if (anchor.kind === "detached") {
565
+ return;
566
+ }
567
+
568
+ activeRuntime.dispatch({
569
+ type: "selection.set",
570
+ selection: toRuntimeSelectionSnapshot(createSelectionFromAnchor(anchor)),
571
+ });
572
+ }
573
+
574
+ function addReviewComment(): void {
575
+ activeRuntime.addComment({
576
+ anchor: snapshot.selection.activeRange,
577
+ body: "New review comment",
578
+ authorId: currentUser.userId,
579
+ });
580
+ setActiveRailTab("comments");
581
+ }
582
+
583
+ function exportCurrentDocument(): void {
584
+ void (runtime
585
+ ? persistAndExport({
586
+ datastore: datastoreRef.current,
587
+ documentId,
588
+ runtime,
589
+ onError: onErrorRef.current,
590
+ onEvent: onEventRef.current,
591
+ lastSavedRevisionTokenRef,
592
+ autosaveTimerRef,
593
+ })
594
+ : rejectExportWhileLoading({
595
+ documentId,
596
+ datastore: datastoreRef.current,
597
+ onError: onErrorRef.current,
598
+ onEvent: onEventRef.current,
599
+ }));
600
+ }
601
+
602
+ const selectionPreview = summarizeSelectionPreview(snapshot);
603
+ const capabilities = deriveCapabilities(snapshot, reviewMode);
604
+
605
+ const dispatchSelection = (selection: PublicSelectionSnapshot) =>
606
+ activeRuntime.dispatch({
607
+ type: "selection.set",
608
+ selection: toRuntimeSelectionSnapshot(selection),
609
+ });
610
+
611
+ const editorCallbacks = {
612
+ onFocus: () => activeRuntime.focus(),
613
+ onBlur: () => activeRuntime.blur(),
614
+ onSelectionChange: dispatchSelection,
615
+ onInsertText: (text: string) => activeRuntime.dispatch({ type: "text.insert", text }),
616
+ onDeleteBackward: () => activeRuntime.dispatch({ type: "text.delete-backward" }),
617
+ onDeleteForward: () => activeRuntime.dispatch({ type: "text.delete-forward" }),
618
+ onInsertTab: () => activeRuntime.dispatch({ type: "text.insert-tab" }),
619
+ onInsertHardBreak: () => activeRuntime.dispatch({ type: "text.insert-hard-break" }),
620
+ onSplitParagraph: () => activeRuntime.dispatch({ type: "paragraph.split" }),
621
+ };
622
+
623
+ const reviewCallbacks = {
624
+ onUndo: () => activeRuntime.undo(),
625
+ onRedo: () => activeRuntime.redo(),
626
+ onAddComment: addReviewComment,
627
+ onExport: exportCurrentDocument,
628
+ onOpenComment: (thread: typeof snapshot.comments.threads[number]) => {
629
+ activeRuntime.openComment(thread.commentId);
630
+ focusAnchor(thread.anchor);
631
+ setActiveRailTab("comments");
632
+ },
633
+ onResolveComment: (commentId: string) => {
634
+ activeRuntime.resolveComment(commentId);
635
+ setActiveRailTab("comments");
636
+ },
637
+ onReopenComment: (commentId: string) => {
638
+ activeRuntime.reopenComment(commentId);
639
+ setActiveRailTab("comments");
640
+ },
641
+ onAddReply: (commentId: string, body: string) => {
642
+ activeRuntime.addCommentReply(commentId, body, currentUser.userId);
643
+ },
644
+ onEditBody: (commentId: string, body: string) => {
645
+ activeRuntime.editCommentBody(commentId, body);
646
+ },
647
+ onOpenRevision: (revision: typeof snapshot.trackedChanges.revisions[number]) => {
648
+ setActiveRevisionId(revision.revisionId);
649
+ focusAnchor(revision.anchor);
650
+ setActiveRailTab("changes");
651
+ },
652
+ onAcceptRevision: (revisionId: string) => {
653
+ activeRuntime.acceptChange(revisionId);
654
+ setActiveRailTab("changes");
655
+ },
656
+ onRejectRevision: (revisionId: string) => {
657
+ activeRuntime.rejectChange(revisionId);
658
+ setActiveRailTab("changes");
659
+ },
660
+ onAcceptAllChanges: () => {
661
+ activeRuntime.acceptAllChanges();
662
+ setActiveRailTab("changes");
663
+ },
664
+ onRejectAllChanges: () => {
665
+ activeRuntime.rejectAllChanges();
666
+ setActiveRailTab("changes");
667
+ },
668
+ };
669
+
670
+ return (
671
+ <TwReviewWorkspace
672
+ snapshot={snapshot}
673
+ currentUserId={currentUser.userId}
674
+ capabilities={capabilities}
675
+ reviewMode={reviewMode}
676
+ viewMode={viewMode}
677
+ activeRailTab={activeRailTab}
678
+ activeCommentId={snapshot.comments.activeCommentId}
679
+ activeRevisionId={activeRevisionId}
680
+ showTrackedChanges={showTrackedChanges}
681
+ selectionPreview={selectionPreview}
682
+ onViewModeChange={setViewMode}
683
+ onActiveRailTabChange={setActiveRailTab}
684
+ onShowTrackedChangesChange={setShowTrackedChanges}
685
+ {...reviewCallbacks}
686
+ document={
687
+ <TwProseMirrorSurface
688
+ currentUser={currentUser}
689
+ snapshot={snapshot}
690
+ reviewMode={reviewMode}
691
+ markupDisplay={liveMarkupDisplay}
692
+ activeRevisionId={activeRevisionId}
693
+ showTrackedChanges={showTrackedChanges}
694
+ {...editorCallbacks}
695
+ onCommentActivated={(commentId) => {
696
+ activeRuntime.openComment(commentId);
697
+ setActiveRailTab("comments");
698
+ }}
699
+ onRevisionActivated={(revisionId) => {
700
+ setActiveRevisionId(revisionId);
701
+ setActiveRailTab("changes");
702
+ }}
703
+ />
704
+ }
705
+ />
706
+ );
707
+ },
708
+ );
709
+
710
+ function normalizeEditorError(error: unknown): EditorError {
711
+ if (
712
+ typeof error === "object" &&
713
+ error !== null &&
714
+ "errorId" in error &&
715
+ "code" in error &&
716
+ "message" in error
717
+ ) {
718
+ return error as EditorError;
719
+ }
720
+
721
+ return {
722
+ errorId: "word-review-editor-load",
723
+ code: "internal_invariant",
724
+ message: error instanceof Error ? error.message : "Unknown editor load failure.",
725
+ isFatal: true,
726
+ source: "runtime",
727
+ };
728
+ }
729
+
730
+ function guessSourceLabel(
731
+ initialSourceLabel?: string,
732
+ initialSnapshot?: PersistedEditorSnapshot,
733
+ externalDocSource?: WordReviewEditorProps["externalDocSource"],
734
+ ): string | undefined {
735
+ return (
736
+ externalDocSource?.sourceLabel ??
737
+ initialSourceLabel ??
738
+ initialSnapshot?.editorBuild ??
739
+ undefined
740
+ );
741
+ }
742
+
743
+ function createLoadingSnapshot(
744
+ documentId: string,
745
+ readOnly: boolean,
746
+ sourceLabel?: string,
747
+ ): RuntimeRenderSnapshot {
748
+ return {
749
+ documentId,
750
+ sessionId: `${documentId}-loading`,
751
+ sourceLabel,
752
+ revisionToken: `${documentId}:loading`,
753
+ isReady: false,
754
+ isDirty: false,
755
+ readOnly,
756
+ selection: collapsedSelection(),
757
+ documentStats: {
758
+ storyLength: 0,
759
+ commentCount: 0,
760
+ revisionCount: 0,
761
+ opaqueFragmentCount: 0,
762
+ },
763
+ comments: {
764
+ openCommentIds: [],
765
+ resolvedCommentIds: [],
766
+ detachedCommentIds: [],
767
+ totalCount: 0,
768
+ threads: [],
769
+ },
770
+ trackedChanges: {
771
+ pendingChangeIds: [],
772
+ acceptedChangeIds: [],
773
+ rejectedChangeIds: [],
774
+ detachedChangeIds: [],
775
+ actionableChangeIds: [],
776
+ preserveOnlyChangeIds: [],
777
+ totalCount: 0,
778
+ revisions: [],
779
+ },
780
+ compatibility: {
781
+ blockExport: false,
782
+ blockExportReasons: [],
783
+ warningCount: 0,
784
+ errorCount: 0,
785
+ featureEntries: [],
786
+ },
787
+ warnings: [],
788
+ commandState: {
789
+ canUndo: false,
790
+ canRedo: false,
791
+ readOnly,
792
+ },
793
+ };
794
+ }
795
+
796
+ function createErrorSnapshot(documentId: string, error: EditorError): RuntimeRenderSnapshot {
797
+ return {
798
+ ...createLoadingSnapshot(documentId, true),
799
+ isReady: true,
800
+ sessionId: `${documentId}-error`,
801
+ revisionToken: `${documentId}:error`,
802
+ compatibility: {
803
+ blockExport: true,
804
+ blockExportReasons: [error.message],
805
+ warningCount: 0,
806
+ errorCount: 1,
807
+ featureEntries: [],
808
+ },
809
+ fatalError: error,
810
+ };
811
+ }
812
+
813
+ async function persistAndExport(input: {
814
+ datastore?: EditorDatastoreAdapter;
815
+ documentId: string;
816
+ runtime: WordReviewEditorRuntime;
817
+ onError?: (error: EditorError) => void;
818
+ onEvent?: (event: WordReviewEditorEvent) => void;
819
+ options?: ExportDocxOptions;
820
+ lastSavedRevisionTokenRef: React.MutableRefObject<string | null>;
821
+ autosaveTimerRef: React.MutableRefObject<ReturnType<typeof setTimeout> | null>;
822
+ }): Promise<ExportResult> {
823
+ if (input.autosaveTimerRef.current) {
824
+ clearTimeout(input.autosaveTimerRef.current);
825
+ input.autosaveTimerRef.current = null;
826
+ }
827
+
828
+ await persistSnapshot({
829
+ datastore: input.datastore,
830
+ documentId: input.documentId,
831
+ runtime: input.runtime,
832
+ isAutosave: false,
833
+ onError: input.onError,
834
+ onEvent: input.onEvent,
835
+ lastSavedRevisionTokenRef: input.lastSavedRevisionTokenRef,
836
+ });
837
+
838
+ const result = await input.runtime.exportDocx(input.options);
839
+
840
+ if (!input.datastore) {
841
+ return result;
842
+ }
843
+
844
+ try {
845
+ await input.datastore.saveExport({
846
+ documentId: input.documentId,
847
+ result,
848
+ });
849
+ } catch (error) {
850
+ const normalized = normalizeDatastoreError(error, {
851
+ message: "Export persisted bytes could not be stored.",
852
+ details: {
853
+ operation: "saveExport",
854
+ },
855
+ });
856
+ input.onError?.(normalized);
857
+ emitEditorEvent({
858
+ datastore: input.datastore,
859
+ onEvent: input.onEvent,
860
+ event: {
861
+ type: "error",
862
+ documentId: input.documentId,
863
+ error: normalized,
864
+ },
865
+ });
866
+ }
867
+
868
+ return result;
869
+ }
870
+
871
+ function rejectExportWhileLoading(input: {
872
+ documentId: string;
873
+ datastore?: EditorDatastoreAdapter;
874
+ onError?: (error: EditorError) => void;
875
+ onEvent?: (event: WordReviewEditorEvent) => void;
876
+ }): Promise<never> {
877
+ const error: EditorError = {
878
+ errorId: "word-review-editor-loading-export",
879
+ code: "internal_invariant",
880
+ message: "WordReviewEditor is still loading and cannot export yet.",
881
+ isFatal: false,
882
+ source: "runtime",
883
+ };
884
+ input.onError?.(error);
885
+ emitEditorEvent({
886
+ datastore: input.datastore,
887
+ onEvent: input.onEvent,
888
+ event: {
889
+ type: "error",
890
+ documentId: input.documentId,
891
+ error,
892
+ },
893
+ });
894
+ return Promise.reject(error);
895
+ }
896
+
897
+ async function persistSnapshot(input: {
898
+ datastore?: EditorDatastoreAdapter;
899
+ documentId: string;
900
+ runtime: WordReviewEditorRuntime;
901
+ isAutosave: boolean;
902
+ onError?: (error: EditorError) => void;
903
+ onEvent?: (event: WordReviewEditorEvent) => void;
904
+ lastSavedRevisionTokenRef: React.MutableRefObject<string | null>;
905
+ }): Promise<void> {
906
+ if (!input.datastore) {
907
+ return;
908
+ }
909
+
910
+ const snapshot = input.runtime.getPersistedSnapshot();
911
+ const revisionToken = input.runtime.getRenderSnapshot().revisionToken;
912
+
913
+ if (input.isAutosave) {
914
+ emitEditorEvent({
915
+ datastore: input.datastore,
916
+ onEvent: input.onEvent,
917
+ event: {
918
+ type: "autosave_state",
919
+ documentId: input.documentId,
920
+ state: {
921
+ status: "saving",
922
+ } satisfies AutosaveState,
923
+ },
924
+ });
925
+ }
926
+
927
+ try {
928
+ const result = await input.datastore.saveSnapshot({
929
+ documentId: input.documentId,
930
+ snapshot,
931
+ isAutosave: input.isAutosave,
932
+ });
933
+ const savedSnapshot: PersistedEditorSnapshot = {
934
+ ...snapshot,
935
+ savedAt: result.savedAt,
936
+ };
937
+ input.lastSavedRevisionTokenRef.current = revisionToken;
938
+ emitEditorEvent({
939
+ datastore: input.datastore,
940
+ onEvent: input.onEvent,
941
+ event: {
942
+ type: "snapshot_saved",
943
+ documentId: input.documentId,
944
+ snapshot: savedSnapshot,
945
+ isAutosave: input.isAutosave,
946
+ },
947
+ });
948
+ if (input.isAutosave) {
949
+ emitEditorEvent({
950
+ datastore: input.datastore,
951
+ onEvent: input.onEvent,
952
+ event: {
953
+ type: "autosave_state",
954
+ documentId: input.documentId,
955
+ state: {
956
+ status: "saved",
957
+ savedAt: result.savedAt,
958
+ } satisfies AutosaveState,
959
+ },
960
+ });
961
+ }
962
+ } catch (error) {
963
+ const normalized = normalizeDatastoreError(error, {
964
+ message: input.isAutosave
965
+ ? "Autosave failed while storing the editor snapshot."
966
+ : "Snapshot save failed while preparing the export checkpoint.",
967
+ details: {
968
+ operation: "saveSnapshot",
969
+ isAutosave: input.isAutosave,
970
+ },
971
+ });
972
+ input.onError?.(normalized);
973
+ emitEditorEvent({
974
+ datastore: input.datastore,
975
+ onEvent: input.onEvent,
976
+ event: {
977
+ type: "error",
978
+ documentId: input.documentId,
979
+ error: normalized,
980
+ },
981
+ });
982
+ if (input.isAutosave) {
983
+ emitEditorEvent({
984
+ datastore: input.datastore,
985
+ onEvent: input.onEvent,
986
+ event: {
987
+ type: "autosave_state",
988
+ documentId: input.documentId,
989
+ state: {
990
+ status: "error",
991
+ error: normalized,
992
+ } satisfies AutosaveState,
993
+ },
994
+ });
995
+ }
996
+ if (!input.isAutosave) {
997
+ throw normalized;
998
+ }
999
+ }
1000
+ }
1001
+
1002
+ function emitEditorEvent(input: {
1003
+ datastore?: EditorDatastoreAdapter;
1004
+ onEvent?: (event: WordReviewEditorEvent) => void;
1005
+ event: WordReviewEditorEvent;
1006
+ }): void {
1007
+ input.onEvent?.(input.event);
1008
+ input.datastore?.logEvent?.({
1009
+ type: input.event.type,
1010
+ documentId: input.event.documentId,
1011
+ detail: summarizeEventDetail(input.event),
1012
+ });
1013
+ }
1014
+
1015
+ function summarizeEventDetail(
1016
+ event: WordReviewEditorEvent,
1017
+ ): Record<string, unknown> | undefined {
1018
+ switch (event.type) {
1019
+ case "dirty_changed":
1020
+ return { isDirty: event.isDirty };
1021
+ case "comment_added":
1022
+ return { commentId: event.commentId };
1023
+ case "comment_resolved":
1024
+ return { commentId: event.commentId };
1025
+ case "change_accepted":
1026
+ case "change_rejected":
1027
+ return { changeId: event.changeId };
1028
+ case "warning_added":
1029
+ return { warningId: event.warning.warningId, code: event.warning.code };
1030
+ case "warning_cleared":
1031
+ return { warningId: event.warningId, code: event.code };
1032
+ case "error":
1033
+ return { errorId: event.error.errorId, code: event.error.code };
1034
+ case "autosave_state":
1035
+ return { status: event.state.status };
1036
+ case "snapshot_saved":
1037
+ return { isAutosave: event.isAutosave, savedAt: event.snapshot.savedAt };
1038
+ case "export_completed":
1039
+ return { fileName: event.result.fileName };
1040
+ case "selection_changed":
1041
+ return {
1042
+ anchor: event.selection.anchor,
1043
+ head: event.selection.head,
1044
+ };
1045
+ case "ready":
1046
+ return {
1047
+ source: event.source,
1048
+ blockExport: event.compatibility.blockExport,
1049
+ };
1050
+ }
1051
+ }
1052
+
1053
+ function createReadyEvent(
1054
+ runtime: Pick<WordReviewEditorRuntime, "getCompatibilityReport" | "getRenderSnapshot">,
1055
+ source: "docx" | "snapshot" | "datastore" | "canonical",
1056
+ ): Extract<WordReviewEditorEvent, { type: "ready" }> {
1057
+ const snapshot = runtime.getRenderSnapshot();
1058
+ return {
1059
+ type: "ready",
1060
+ documentId: snapshot.documentId,
1061
+ sessionId: snapshot.sessionId,
1062
+ source,
1063
+ stats: snapshot.documentStats,
1064
+ compatibility: runtime.getCompatibilityReport(),
1065
+ };
1066
+ }
1067
+
1068
+ function normalizeDatastoreError(
1069
+ error: unknown,
1070
+ fallback: {
1071
+ message: string;
1072
+ details?: Record<string, unknown>;
1073
+ },
1074
+ ): EditorError {
1075
+ if (
1076
+ typeof error === "object" &&
1077
+ error !== null &&
1078
+ "errorId" in error &&
1079
+ "code" in error &&
1080
+ "message" in error
1081
+ ) {
1082
+ return error as EditorError;
1083
+ }
1084
+
1085
+ return {
1086
+ errorId: "word-review-editor-datastore",
1087
+ code: "datastore_failed",
1088
+ message: error instanceof Error ? error.message : fallback.message,
1089
+ isFatal: false,
1090
+ source: "datastore",
1091
+ details: fallback.details,
1092
+ };
1093
+ }
1094
+
1095
+ function createFallbackSnapshot(args: CreateRuntimeArgs): RuntimeRenderSnapshot {
1096
+ const warnings = args.source.initialSnapshot?.warningLog ?? [];
1097
+ const compatibility = args.source.initialSnapshot?.compatibility ?? emptyCompatibilityReport();
1098
+
1099
+ return {
1100
+ ...createLoadingSnapshot(args.documentId, args.readOnly, args.source.sourceLabel),
1101
+ sessionId: `${args.documentId}-session`,
1102
+ revisionToken: `${args.documentId}:0`,
1103
+ isReady: true,
1104
+ documentStats: {
1105
+ storyLength: estimateStoryLength(args.source.initialSnapshot),
1106
+ commentCount: 0,
1107
+ revisionCount: 0,
1108
+ opaqueFragmentCount: 0,
1109
+ },
1110
+ compatibility: {
1111
+ blockExport: compatibility.blockExport,
1112
+ blockExportReasons: [],
1113
+ warningCount: compatibility.warnings.length,
1114
+ errorCount: compatibility.errors.length,
1115
+ featureEntries: compatibility.featureEntries,
1116
+ },
1117
+ warnings,
1118
+ };
1119
+ }
1120
+
1121
+ function createFallbackPersistedSnapshot(
1122
+ documentId: string,
1123
+ label = "Generated shell snapshot",
1124
+ ): PersistedEditorSnapshot {
1125
+ const docId = createCanonicalDocumentId(documentId);
1126
+ return {
1127
+ snapshotVersion: "persisted-editor-snapshot/1",
1128
+ schemaVersion: "cds/1.0.0",
1129
+ documentId,
1130
+ docId,
1131
+ createdAt: "1970-01-01T00:00:00.000Z",
1132
+ updatedAt: "1970-01-01T00:00:00.000Z",
1133
+ savedAt: "1970-01-01T00:00:00.000Z",
1134
+ editorBuild: label,
1135
+ canonicalDocument: {
1136
+ schemaVersion: "cds/1.0.0",
1137
+ docId,
1138
+ createdAt: "1970-01-01T00:00:00.000Z",
1139
+ updatedAt: "1970-01-01T00:00:00.000Z",
1140
+ metadata: {
1141
+ customProperties: {},
1142
+ },
1143
+ styles: {
1144
+ paragraphs: {},
1145
+ characters: {},
1146
+ tables: {},
1147
+ },
1148
+ numbering: {
1149
+ abstractDefinitions: {},
1150
+ instances: {},
1151
+ },
1152
+ media: {
1153
+ items: {},
1154
+ },
1155
+ content: {
1156
+ type: "doc",
1157
+ children: [{ type: "paragraph", children: [] }],
1158
+ },
1159
+ review: {
1160
+ comments: {},
1161
+ revisions: {},
1162
+ },
1163
+ preservation: {
1164
+ opaqueFragments: {},
1165
+ packageParts: {},
1166
+ },
1167
+ diagnostics: {
1168
+ warnings: [],
1169
+ errors: [],
1170
+ },
1171
+ },
1172
+ compatibility: emptyCompatibilityReport(),
1173
+ warningLog: [],
1174
+ };
1175
+ }
1176
+
1177
+ function emptyCompatibilityReport(): CompatibilityReport {
1178
+ return {
1179
+ reportVersion: "compatibility-report/1",
1180
+ generatedAt: "1970-01-01T00:00:00.000Z",
1181
+ blockExport: false,
1182
+ featureEntries: [],
1183
+ warnings: [],
1184
+ errors: [],
1185
+ };
1186
+ }
1187
+
1188
+ function toRuntimeSelectionSnapshot(selection: PublicSelectionSnapshot) {
1189
+ return {
1190
+ anchor: selection.anchor,
1191
+ head: selection.head,
1192
+ isCollapsed: selection.isCollapsed,
1193
+ activeRange:
1194
+ selection.activeRange.kind === "range"
1195
+ ? createRangeAnchor(
1196
+ selection.activeRange.from,
1197
+ selection.activeRange.to,
1198
+ selection.activeRange.assoc,
1199
+ )
1200
+ : selection.activeRange.kind === "node"
1201
+ ? createNodeAnchor(selection.activeRange.at, selection.activeRange.assoc)
1202
+ : createDetachedAnchor(
1203
+ selection.activeRange.lastKnownRange,
1204
+ selection.activeRange.reason,
1205
+ ),
1206
+ };
1207
+ }
1208
+
1209
+ function createSelectionFromAnchor(
1210
+ anchor: PublicSelectionSnapshot["activeRange"],
1211
+ ): PublicSelectionSnapshot {
1212
+ switch (anchor.kind) {
1213
+ case "range":
1214
+ return {
1215
+ anchor: anchor.from,
1216
+ head: anchor.to,
1217
+ isCollapsed: anchor.from === anchor.to,
1218
+ activeRange: anchor,
1219
+ };
1220
+ case "node":
1221
+ return {
1222
+ anchor: anchor.at,
1223
+ head: anchor.at,
1224
+ isCollapsed: true,
1225
+ activeRange: anchor,
1226
+ };
1227
+ case "detached":
1228
+ return {
1229
+ anchor: anchor.lastKnownRange.from,
1230
+ head: anchor.lastKnownRange.to,
1231
+ isCollapsed: anchor.lastKnownRange.from === anchor.lastKnownRange.to,
1232
+ activeRange: anchor,
1233
+ };
1234
+ }
1235
+ }
1236
+
1237
+ function estimateStoryLength(snapshot?: PersistedEditorSnapshot): number {
1238
+ const content = snapshot?.canonicalDocument.content;
1239
+ return Array.isArray(content) ? content.length : 0;
1240
+ }
1241
+
1242
+ function collapsedSelection(): RuntimeRenderSnapshot["selection"] {
1243
+ return {
1244
+ anchor: 0,
1245
+ head: 0,
1246
+ isCollapsed: true,
1247
+ activeRange: {
1248
+ kind: "range",
1249
+ from: 0,
1250
+ to: 0,
1251
+ assoc: {
1252
+ start: -1,
1253
+ end: 1,
1254
+ },
1255
+ },
1256
+ };
1257
+ }
1258
+
1259
+ function toUint8Array(bytes: Uint8Array | ArrayBuffer): Uint8Array {
1260
+ return bytes instanceof Uint8Array ? new Uint8Array(bytes) : new Uint8Array(bytes);
1261
+ }
1262
+
1263
+ function summarizeSelectionPreview(snapshot: RuntimeRenderSnapshot): string | null {
1264
+ if (!snapshot.surface || snapshot.selection.isCollapsed) {
1265
+ return null;
1266
+ }
1267
+
1268
+ const range = snapshot.selection.activeRange;
1269
+ if (range.kind !== "range") {
1270
+ return "Selected range";
1271
+ }
1272
+
1273
+ const preview = snapshot.surface.plainText
1274
+ .slice(range.from, range.to)
1275
+ .replace(/\s+/g, " ")
1276
+ .trim();
1277
+
1278
+ if (!preview) {
1279
+ return "Selected range";
1280
+ }
1281
+
1282
+ return preview.length > 48 ? `${preview.slice(0, 45)}...` : preview;
1283
+ }