@beyondwork/docx-react-component 1.0.0 → 1.0.1

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 (704) hide show
  1. package/dist/chunk-32W6IVQE.js +7725 -0
  2. package/dist/chunk-32W6IVQE.js.map +1 -0
  3. package/dist/index.cjs +23722 -0
  4. package/dist/index.cjs.map +1 -0
  5. package/dist/index.d.cts +7 -0
  6. package/dist/index.d.ts +7 -0
  7. package/dist/index.js +16011 -0
  8. package/dist/index.js.map +1 -0
  9. package/dist/public-types-DqCURAz8.d.cts +1152 -0
  10. package/dist/public-types-DqCURAz8.d.ts +1152 -0
  11. package/dist/tailwind.cjs +8295 -0
  12. package/dist/tailwind.cjs.map +1 -0
  13. package/dist/tailwind.d.cts +323 -0
  14. package/dist/tailwind.d.ts +323 -0
  15. package/dist/tailwind.js +553 -0
  16. package/dist/tailwind.js.map +1 -0
  17. package/package.json +52 -31
  18. package/.codex/config.toml +0 -5
  19. package/.corepack/v1/pnpm/10.30.3/.corepack +0 -1
  20. package/.corepack/v1/pnpm/10.30.3/LICENSE +0 -22
  21. package/.corepack/v1/pnpm/10.30.3/README.md +0 -240
  22. package/.corepack/v1/pnpm/10.30.3/dist/node-gyp-bin/node-gyp +0 -6
  23. package/.corepack/v1/pnpm/10.30.3/dist/node-gyp-bin/node-gyp.cmd +0 -5
  24. package/.corepack/v1/pnpm/10.30.3/dist/pnpm.cjs +0 -195400
  25. package/.corepack/v1/pnpm/10.30.3/dist/pnpmrc +0 -2
  26. package/.corepack/v1/pnpm/10.30.3/dist/reflink.darwin-arm64-2HJ4WGO6.node +0 -0
  27. package/.corepack/v1/pnpm/10.30.3/dist/reflink.darwin-x64-3G3H6IW4.node +0 -0
  28. package/.corepack/v1/pnpm/10.30.3/dist/reflink.win32-arm64-msvc-Q6BARPPB.node +0 -0
  29. package/.corepack/v1/pnpm/10.30.3/dist/reflink.win32-x64-msvc-J2TZHRQI.node +0 -0
  30. package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.bash +0 -31
  31. package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.fish +0 -22
  32. package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.ps1 +0 -193
  33. package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.zsh +0 -27
  34. package/.corepack/v1/pnpm/10.30.3/dist/vendor/fastlist-0.3.0-x64.exe +0 -0
  35. package/.corepack/v1/pnpm/10.30.3/dist/vendor/fastlist-0.3.0-x86.exe +0 -0
  36. package/.corepack/v1/pnpm/10.30.3/dist/worker.js +0 -10119
  37. package/.corepack/v1/pnpm/10.30.3/package.json +0 -192
  38. package/.cursor/mcp.json +0 -7
  39. package/.github/workflows/ci.yml +0 -35
  40. package/.mcp.json +0 -7
  41. package/.openclaw/workspace-state.json +0 -4
  42. package/.pnpmrc.json +0 -1
  43. package/.wave-launch.sh +0 -7
  44. package/.workspace-marker +0 -1
  45. package/AGENTS.md +0 -78
  46. package/CHANGELOG.md +0 -177
  47. package/DESIGN.md +0 -929
  48. package/HEARTBEAT.md +0 -7
  49. package/IDENTITY.md +0 -23
  50. package/SOUL.md +0 -36
  51. package/TOOLS.md +0 -40
  52. package/USER.md +0 -17
  53. package/docs/README.md +0 -107
  54. package/docs/agents/wave-cont-eval-role.md +0 -36
  55. package/docs/agents/wave-cont-qa-role.md +0 -52
  56. package/docs/agents/wave-deploy-verifier-role.md +0 -34
  57. package/docs/agents/wave-design-role.md +0 -47
  58. package/docs/agents/wave-documentation-role.md +0 -34
  59. package/docs/agents/wave-infra-role.md +0 -34
  60. package/docs/agents/wave-integration-role.md +0 -37
  61. package/docs/agents/wave-launcher-role.md +0 -41
  62. package/docs/agents/wave-orchestrator-role.md +0 -52
  63. package/docs/agents/wave-planner-role.md +0 -39
  64. package/docs/agents/wave-security-role.md +0 -40
  65. package/docs/architecture/docx/README.md +0 -10
  66. package/docs/architecture/future/README.md +0 -8
  67. package/docs/architecture/ooxml-upgrade-analysis.md +0 -134
  68. package/docs/architecture/platform/shared-openxml-editor-platform.md +0 -153
  69. package/docs/architecture/xlsx/canonical-workbook-model-and-commands.md +0 -187
  70. package/docs/architecture/xlsx/spreadsheet-editor-frontend-architecture.md +0 -150
  71. package/docs/comment-redline-overview.md +0 -350
  72. package/docs/concepts/context7-vs-skills.md +0 -118
  73. package/docs/concepts/operating-modes.md +0 -91
  74. package/docs/concepts/runtime-agnostic-orchestration.md +0 -111
  75. package/docs/concepts/what-is-a-wave.md +0 -217
  76. package/docs/context7/bundles.json +0 -222
  77. package/docs/context7/planner-agent/README.md +0 -28
  78. package/docs/context7/planner-agent/manifest.json +0 -83
  79. package/docs/context7/planner-agent/papers/cooperbench-why-coding-agents-cannot-be-your-teammates-yet.md +0 -3283
  80. package/docs/context7/planner-agent/papers/dova-deliberation-first-multi-agent-orchestration-for-autonomous-research-automation.md +0 -1699
  81. package/docs/context7/planner-agent/papers/dpbench-large-language-models-struggle-with-simultaneous-coordination.md +0 -2251
  82. package/docs/context7/planner-agent/papers/incremental-planning-to-control-a-blackboard-based-problem-solver.md +0 -1729
  83. package/docs/context7/planner-agent/papers/silo-bench-a-scalable-environment-for-evaluating-distributed-coordination-in-multi-agent-llm-systems.md +0 -3747
  84. package/docs/context7/planner-agent/papers/todoevolve-learning-to-architect-agent-planning-systems.md +0 -1675
  85. package/docs/context7/planner-agent/papers/verified-multi-agent-orchestration-a-plan-execute-verify-replan-framework-for-complex-query-resolution.md +0 -1173
  86. package/docs/context7/planner-agent/papers/why-do-multi-agent-llm-systems-fail.md +0 -5211
  87. package/docs/context7/planner-agent/topics/planning-and-orchestration.md +0 -24
  88. package/docs/evals/arm-templates/README.md +0 -13
  89. package/docs/evals/arm-templates/full-wave.json +0 -15
  90. package/docs/evals/arm-templates/single-agent.json +0 -15
  91. package/docs/evals/benchmark-catalog.json +0 -670
  92. package/docs/evals/cases/README.md +0 -47
  93. package/docs/evals/cases/wave-blackboard-inbox-targeting.json +0 -73
  94. package/docs/evals/cases/wave-contradiction-conflict.json +0 -104
  95. package/docs/evals/cases/wave-expert-routing-preservation.json +0 -69
  96. package/docs/evals/cases/wave-hidden-profile-private-evidence.json +0 -81
  97. package/docs/evals/cases/wave-premature-closure-guard.json +0 -71
  98. package/docs/evals/cases/wave-silo-cross-agent-state.json +0 -77
  99. package/docs/evals/cases/wave-simultaneous-lockstep.json +0 -92
  100. package/docs/evals/external-benchmarks.json +0 -85
  101. package/docs/evals/external-command-config.sample.json +0 -9
  102. package/docs/evals/external-command-config.swe-bench-pro.json +0 -8
  103. package/docs/evals/pilots/README.md +0 -47
  104. package/docs/evals/pilots/swe-bench-pro-public-full-wave-review-10.json +0 -64
  105. package/docs/evals/pilots/swe-bench-pro-public-pilot.json +0 -111
  106. package/docs/evals/wave-benchmark-program.md +0 -302
  107. package/docs/guides/planner.md +0 -220
  108. package/docs/guides/recommendations-0.8.9.md +0 -133
  109. package/docs/guides/signal-wrappers.md +0 -165
  110. package/docs/guides/terminal-surfaces.md +0 -96
  111. package/docs/image copy.png +0 -0
  112. package/docs/image.png +0 -0
  113. package/docs/images/image.png +0 -0
  114. package/docs/legal-feedback-architecture.md +0 -498
  115. package/docs/plans/component-cutover-matrix.json +0 -1072
  116. package/docs/plans/component-cutover-matrix.md +0 -307
  117. package/docs/plans/context7-wave-orchestrator.md +0 -155
  118. package/docs/plans/current-state.md +0 -198
  119. package/docs/plans/docx/README.md +0 -9
  120. package/docs/plans/examples/wave-benchmark-improvement.md +0 -108
  121. package/docs/plans/examples/wave-example-live-proof.md +0 -435
  122. package/docs/plans/master-plan.md +0 -224
  123. package/docs/plans/migration.md +0 -538
  124. package/docs/plans/operations/README.md +0 -7
  125. package/docs/plans/operations/wave-10-word-certification.md +0 -87
  126. package/docs/plans/operations/wave-8-railway-staging.md +0 -153
  127. package/docs/plans/operations/wave-9-manual-certification.md +0 -73
  128. package/docs/plans/platform/README.md +0 -9
  129. package/docs/plans/reference/legal-checklist-coverage.md +0 -258
  130. package/docs/plans/wave-orchestrator.md +0 -423
  131. package/docs/plans/waves/README.md +0 -75
  132. package/docs/plans/waves/completed/wave-0.md +0 -195
  133. package/docs/plans/waves/completed/wave-1.md +0 -379
  134. package/docs/plans/waves/completed/wave-10.md +0 -670
  135. package/docs/plans/waves/completed/wave-11.md +0 -335
  136. package/docs/plans/waves/completed/wave-12.md +0 -417
  137. package/docs/plans/waves/completed/wave-13.md +0 -316
  138. package/docs/plans/waves/completed/wave-14.md +0 -319
  139. package/docs/plans/waves/completed/wave-15.md +0 -321
  140. package/docs/plans/waves/completed/wave-16.md +0 -316
  141. package/docs/plans/waves/completed/wave-17.md +0 -331
  142. package/docs/plans/waves/completed/wave-18.md +0 -328
  143. package/docs/plans/waves/completed/wave-2.md +0 -438
  144. package/docs/plans/waves/completed/wave-3.md +0 -435
  145. package/docs/plans/waves/completed/wave-4.md +0 -430
  146. package/docs/plans/waves/completed/wave-5.md +0 -430
  147. package/docs/plans/waves/completed/wave-6.md +0 -430
  148. package/docs/plans/waves/completed/wave-7.md +0 -526
  149. package/docs/plans/waves/completed/wave-8.md +0 -596
  150. package/docs/plans/waves/completed/wave-9.md +0 -552
  151. package/docs/plans/waves/deferred/README.md +0 -14
  152. package/docs/plans/waves/deferred/encrypted-intake-contracts.md +0 -282
  153. package/docs/plans/waves/deferred/legal-feedback-wave-expansion.md +0 -308
  154. package/docs/plans/waves/deferred/wave-encrypted-intake.md +0 -451
  155. package/docs/plans/waves/design/README.md +0 -5
  156. package/docs/plans/waves/design/wave-1-a1.md +0 -309
  157. package/docs/plans/waves/reviews/README.md +0 -5
  158. package/docs/plans/waves/reviews/wave-0-cont-qa.md +0 -151
  159. package/docs/plans/waves/reviews/wave-1-cont-qa.md +0 -46
  160. package/docs/plans/waves/reviews/wave-10-accessibility-and-design.md +0 -51
  161. package/docs/plans/waves/reviews/wave-10-cont-qa.md +0 -24
  162. package/docs/plans/waves/reviews/wave-10-dashboard-proof.md +0 -46
  163. package/docs/plans/waves/reviews/wave-10-performance-signoff.md +0 -55
  164. package/docs/plans/waves/reviews/wave-10-regression-proof.md +0 -23
  165. package/docs/plans/waves/reviews/wave-10-release-audit.md +0 -31
  166. package/docs/plans/waves/reviews/wave-10-service-proof.md +0 -83
  167. package/docs/plans/waves/reviews/wave-10-word-certification.md +0 -31
  168. package/docs/plans/waves/reviews/wave-18-ai-contract-closure.md +0 -277
  169. package/docs/plans/waves/reviews/wave-18-cont-qa.md +0 -255
  170. package/docs/plans/waves/reviews/wave-18-parity-proof.md +0 -271
  171. package/docs/plans/waves/reviews/wave-19-cont-qa.md +0 -59
  172. package/docs/plans/waves/reviews/wave-2-cont-qa.md +0 -72
  173. package/docs/plans/waves/reviews/wave-20-cont-qa.md +0 -60
  174. package/docs/plans/waves/reviews/wave-25-cont-qa.md +0 -48
  175. package/docs/plans/waves/reviews/wave-28-cont-qa.md +0 -46
  176. package/docs/plans/waves/reviews/wave-29-cont-qa.md +0 -53
  177. package/docs/plans/waves/reviews/wave-3-cont-qa.md +0 -53
  178. package/docs/plans/waves/reviews/wave-3-core-proof.md +0 -77
  179. package/docs/plans/waves/reviews/wave-3-validator-proof.md +0 -73
  180. package/docs/plans/waves/reviews/wave-32-cont-qa.md +0 -43
  181. package/docs/plans/waves/reviews/wave-33-cont-qa.md +0 -526
  182. package/docs/plans/waves/reviews/wave-34-cont-qa.md +0 -100
  183. package/docs/plans/waves/reviews/wave-35-cont-qa.md +0 -145
  184. package/docs/plans/waves/reviews/wave-4-cont-qa.md +0 -47
  185. package/docs/plans/waves/reviews/wave-4-structure-proof.md +0 -69
  186. package/docs/plans/waves/reviews/wave-5-comment-proof.md +0 -158
  187. package/docs/plans/waves/reviews/wave-5-cont-qa.md +0 -68
  188. package/docs/plans/waves/reviews/wave-6-cont-qa.md +0 -416
  189. package/docs/plans/waves/reviews/wave-6-redline-proof.md +0 -130
  190. package/docs/plans/waves/reviews/wave-7-cont-qa.md +0 -82
  191. package/docs/plans/waves/reviews/wave-7-ooxml-compliance.md +0 -85
  192. package/docs/plans/waves/reviews/wave-7-preservation-proof.md +0 -119
  193. package/docs/plans/waves/reviews/wave-7-trust-ux.md +0 -87
  194. package/docs/plans/waves/reviews/wave-8-accessibility-and-design.md +0 -128
  195. package/docs/plans/waves/reviews/wave-8-cont-qa.md +0 -92
  196. package/docs/plans/waves/reviews/wave-8-live-proof.md +0 -140
  197. package/docs/plans/waves/reviews/wave-8-security.md +0 -47
  198. package/docs/plans/waves/reviews/wave-9-editor-embedding.md +0 -39
  199. package/docs/plans/waves/reviews/wave-9-fixture-runner.md +0 -56
  200. package/docs/plans/waves/reviews/wave-9-live-proof.md +0 -105
  201. package/docs/plans/waves/reviews/wave-9-usability-and-performance.md +0 -152
  202. package/docs/plans/waves/specs/README.md +0 -5
  203. package/docs/plans/waves/specs/wave-1-component-boundaries.md +0 -322
  204. package/docs/plans/waves/specs/wave-1-ooxml-contracts.md +0 -323
  205. package/docs/plans/waves/specs/wave-1-review-and-ui-contracts.md +0 -339
  206. package/docs/plans/waves/specs/wave-1-runtime-contracts.md +0 -509
  207. package/docs/plans/waves/wave-19.md +0 -341
  208. package/docs/plans/waves/wave-20.md +0 -308
  209. package/docs/plans/waves/wave-21.md +0 -289
  210. package/docs/plans/waves/wave-22.md +0 -221
  211. package/docs/plans/waves/wave-23.md +0 -295
  212. package/docs/plans/waves/wave-24.md +0 -286
  213. package/docs/plans/waves/wave-25.md +0 -313
  214. package/docs/plans/waves/wave-26.md +0 -300
  215. package/docs/plans/waves/wave-27.md +0 -299
  216. package/docs/plans/waves/wave-28.md +0 -368
  217. package/docs/plans/waves/wave-29.md +0 -303
  218. package/docs/plans/waves/wave-30.md +0 -307
  219. package/docs/plans/waves/wave-31.md +0 -231
  220. package/docs/plans/waves/wave-32.md +0 -152
  221. package/docs/plans/waves/wave-33.md +0 -147
  222. package/docs/plans/waves/wave-34.md +0 -148
  223. package/docs/plans/waves/wave-35.md +0 -141
  224. package/docs/plans/waves/wave-36.md +0 -146
  225. package/docs/plans/xlsx/README.md +0 -14
  226. package/docs/plans/xlsx/xlsx-fixture-corpus-and-certification-plan.md +0 -126
  227. package/docs/reference/cli-reference.md +0 -600
  228. package/docs/reference/coordination-and-closure.md +0 -487
  229. package/docs/reference/deep-research-report (15).md +0 -25
  230. package/docs/reference/docx/README.md +0 -10
  231. package/docs/reference/legal-checklist.md +0 -445
  232. package/docs/reference/live-proof-waves.md +0 -199
  233. package/docs/reference/ooxml-compliance.md +0 -129
  234. package/docs/reference/ooxml-feature-parity-matrix.md +0 -172
  235. package/docs/reference/platform/shared-ooxml-platform-guidance.md +0 -77
  236. package/docs/reference/prototype-agent-prompt-legal-fidelity.md +0 -155
  237. package/docs/reference/public-api.md +0 -456
  238. package/docs/reference/repository-guidance.md +0 -58
  239. package/docs/reference/runtime-config/README.md +0 -182
  240. package/docs/reference/runtime-config/claude.md +0 -110
  241. package/docs/reference/runtime-config/codex.md +0 -82
  242. package/docs/reference/runtime-config/opencode.md +0 -93
  243. package/docs/reference/sample-waves.md +0 -105
  244. package/docs/reference/skills.md +0 -237
  245. package/docs/reference/templates/AGENTS.md +0 -78
  246. package/docs/reference/templates/HEARTBEAT.md +0 -7
  247. package/docs/reference/templates/IDENTITY.md +0 -23
  248. package/docs/reference/templates/SOUL.md +0 -36
  249. package/docs/reference/templates/TOOLS.md +0 -40
  250. package/docs/reference/templates/USER.md +0 -17
  251. package/docs/reference/wave-control.md +0 -184
  252. package/docs/reference/wave-planning-lessons.md +0 -167
  253. package/docs/reference/word-review-editor-frontend-architecture.md +0 -479
  254. package/docs/reference/word-review-editor-ux-guide.md +0 -253
  255. package/docs/reference/xlsx/xlsx-ooxml-compliance.md +0 -137
  256. package/docs/research/agent-context-sources.md +0 -178
  257. package/docs/research/coordination-failure-review.md +0 -290
  258. package/docs/research/docx-react-component/Canonical Document Schema Specification for a React-based Word-compatible Editor.md +0 -2317
  259. package/docs/research/docx-react-component/Feature Compatibility Matrix for a React Word Compatible Legal Editor v1.md +0 -219
  260. package/docs/research/docx-react-component/React Component Architecture and Front-End Structure Specification for a Word-Compatible Legal Review Editor.md +0 -1112
  261. package/docs/research/docx-react-component/document_compatibility_and_testing_spec.md +0 -751
  262. package/docs/research/xlsx/raw/README.md +0 -13
  263. package/docs/roadmap.md +0 -174
  264. package/docs/superpowers/plans/2026-03-28-harness-control-bar.md +0 -677
  265. package/docs/superpowers/specs/2026-03-28-harness-control-bar-design.md +0 -274
  266. package/docs/xlsx-react/README.md +0 -38
  267. package/docs/xlsx-react/agent-llm-interaction-layer-docx-xlsx.md +0 -621
  268. package/docs/xlsx-react/canonical-workbook-model-and-commands.md +0 -948
  269. package/docs/xlsx-react/shared-openxml-editor-platform-docx-xlsx.md +0 -228
  270. package/docs/xlsx-react/spreadsheet-editor-component-architecture.md +0 -809
  271. package/docs/xlsx-react/spreadsheet-editor-frontend-architecture.md +0 -537
  272. package/docs/xlsx-react/spreadsheet-editor-ux-guide.md +0 -520
  273. package/docs/xlsx-react/xlsx-editor-research-pack.md +0 -871
  274. package/docs/xlsx-react/xlsx-fixture-corpus-and-certification-plan.md +0 -436
  275. package/docs/xlsx-react/xlsx-ooxml-compliance.md +0 -320
  276. package/examples/README.md +0 -16
  277. package/memory/MEMORY.md +0 -24
  278. package/pnpm-workspace.yaml +0 -4
  279. package/scripts/check-no-authored-js.sh +0 -13
  280. package/scripts/context7-api-check.sh +0 -65
  281. package/scripts/context7-export-env.sh +0 -42
  282. package/scripts/run-context7-mcp.sh +0 -8
  283. package/scripts/run-workspace-tests.sh +0 -15
  284. package/scripts/start-wave-10-local.sh +0 -189
  285. package/scripts/wave-agent-attach.sh +0 -47
  286. package/scripts/wave-auto-answer.sh +0 -118
  287. package/scripts/wave-dashboard-attach.sh +0 -13
  288. package/scripts/wave-launch.sh +0 -273
  289. package/scripts/wave-overnight-supervisor.sh +0 -145
  290. package/scripts/wave-status.sh +0 -379
  291. package/scripts/wave-watch.sh +0 -231
  292. package/services/README.md +0 -17
  293. package/services/openxml-validator/Dockerfile +0 -29
  294. package/services/openxml-validator/OpenXmlValidator.Api.csproj +0 -12
  295. package/services/openxml-validator/Program.cs +0 -436
  296. package/services/openxml-validator/README.md +0 -152
  297. package/services/openxml-validator/railway.json +0 -16
  298. package/services/react-word-editor/.tmp-a4/src/api/public-types.ts +0 -318
  299. package/services/react-word-editor/.tmp-a4/src/ui/WordReviewEditor.tsx +0 -1302
  300. package/services/react-word-editor/.tmp-a4/src/ui/editor-surface/editor-surface.tsx +0 -546
  301. package/services/react-word-editor/.tmp-a4/test/ui/word-review-editor.test.tsx +0 -146
  302. package/services/react-word-editor/.tmp-a4-build/src/api/public-types.js +0 -2
  303. package/services/react-word-editor/.tmp-a4-build/src/ui/WordReviewEditor.js +0 -818
  304. package/services/react-word-editor/.tmp-a4-build/src/ui/editor-surface/editor-surface.js +0 -229
  305. package/services/react-word-editor/.tmp-a4-build/test/ui/word-review-editor.test.js +0 -121
  306. package/services/react-word-editor/.tmp-wave-4-a3-tsconfig.json +0 -21
  307. package/services/react-word-editor/.tmp-wave-4-a3-tsconfig.tsbuildinfo +0 -1
  308. package/services/react-word-editor/Dockerfile +0 -26
  309. package/services/react-word-editor/README.md +0 -254
  310. package/services/react-word-editor/app/api/certification/route.ts +0 -79
  311. package/services/react-word-editor/app/api/demo-sessions/route.ts +0 -109
  312. package/services/react-word-editor/app/api/deploy-health/route.ts +0 -23
  313. package/services/react-word-editor/app/api/exports/[exportId]/route.ts +0 -34
  314. package/services/react-word-editor/app/api/exports/route.ts +0 -81
  315. package/services/react-word-editor/app/api/fixtures/[fixtureId]/run/route.ts +0 -100
  316. package/services/react-word-editor/app/api/health/route.ts +0 -70
  317. package/services/react-word-editor/app/api/runs/[runId]/route.ts +0 -36
  318. package/services/react-word-editor/app/api/scenarios/[scenarioId]/run/route.ts +0 -85
  319. package/services/react-word-editor/app/api/sessions/[sessionId]/route.ts +0 -199
  320. package/services/react-word-editor/app/api/sessions/[sessionId]/source/route.ts +0 -45
  321. package/services/react-word-editor/app/api/uploads/route.ts +0 -70
  322. package/services/react-word-editor/app/api/validate/route.ts +0 -310
  323. package/services/react-word-editor/app/certification/[runId]/page.tsx +0 -14
  324. package/services/react-word-editor/app/certification/page.tsx +0 -32
  325. package/services/react-word-editor/app/dashboard/page.tsx +0 -7
  326. package/services/react-word-editor/app/demo/page.tsx +0 -30
  327. package/services/react-word-editor/app/demo/prototype-client.tsx +0 -1080
  328. package/services/react-word-editor/app/editor/[sessionId]/page.tsx +0 -33
  329. package/services/react-word-editor/app/fixtures/page.tsx +0 -7
  330. package/services/react-word-editor/app/globals.css +0 -121
  331. package/services/react-word-editor/app/layout.tsx +0 -32
  332. package/services/react-word-editor/app/page.tsx +0 -30
  333. package/services/react-word-editor/app/runs/[runId]/page.tsx +0 -34
  334. package/services/react-word-editor/app/wave-10-word-review/page.tsx +0 -7
  335. package/services/react-word-editor/components/harness-control-bar.tsx +0 -289
  336. package/services/react-word-editor/components/harness-editor-session-client.tsx +0 -1214
  337. package/services/react-word-editor/components/harness-workspace-page.tsx +0 -715
  338. package/services/react-word-editor/components/reduced-motion-toggle.tsx +0 -79
  339. package/services/react-word-editor/components/workspace-certification-panel.tsx +0 -307
  340. package/services/react-word-editor/lib/certification-bundle.ts +0 -796
  341. package/services/react-word-editor/lib/certification-store.ts +0 -661
  342. package/services/react-word-editor/lib/demo-fixtures.test.mjs +0 -195
  343. package/services/react-word-editor/lib/demo-fixtures.ts +0 -1519
  344. package/services/react-word-editor/lib/editor-session-summary.test.mjs +0 -68
  345. package/services/react-word-editor/lib/editor-session-summary.ts +0 -14
  346. package/services/react-word-editor/lib/editor-session.ts +0 -228
  347. package/services/react-word-editor/lib/exports-route.test.mjs +0 -32
  348. package/services/react-word-editor/lib/harness-client.ts +0 -347
  349. package/services/react-word-editor/lib/harness-config.json +0 -30
  350. package/services/react-word-editor/lib/harness-config.test.mjs +0 -31
  351. package/services/react-word-editor/lib/harness-config.ts +0 -21
  352. package/services/react-word-editor/lib/harness-editor-datastore.test.mjs +0 -220
  353. package/services/react-word-editor/lib/harness-editor-datastore.ts +0 -161
  354. package/services/react-word-editor/lib/private-mode.test.mjs +0 -42
  355. package/services/react-word-editor/lib/private-mode.ts +0 -61
  356. package/services/react-word-editor/lib/regression-report.test.mjs +0 -352
  357. package/services/react-word-editor/lib/regression-report.ts +0 -896
  358. package/services/react-word-editor/lib/run-artifacts.ts +0 -934
  359. package/services/react-word-editor/lib/run-history.ts +0 -755
  360. package/services/react-word-editor/lib/scenario-artifacts.test.mjs +0 -41
  361. package/services/react-word-editor/lib/scenario-artifacts.ts +0 -44
  362. package/services/react-word-editor/lib/storage.ts +0 -953
  363. package/services/react-word-editor/lib/validator-client.test.mjs +0 -54
  364. package/services/react-word-editor/lib/validator-client.ts +0 -95
  365. package/services/react-word-editor/lib/workspace-navigation.ts +0 -79
  366. package/services/react-word-editor/middleware.ts +0 -35
  367. package/services/react-word-editor/next-env.d.ts +0 -6
  368. package/services/react-word-editor/next.config.mjs +0 -15
  369. package/services/react-word-editor/package.json +0 -38
  370. package/services/react-word-editor/postcss.config.mjs +0 -8
  371. package/services/react-word-editor/railway.json +0 -21
  372. package/services/react-word-editor/scripts/wave-10-certification.mjs +0 -101
  373. package/services/react-word-editor/scripts/wave-9-live-usability-pilot.mjs +0 -911
  374. package/services/react-word-editor/tsconfig.json +0 -39
  375. package/services/react-word-editor/tsconfig.tsbuildinfo +0 -1
  376. package/skills/README.md +0 -48
  377. package/skills/domain-docx-compatibility/SKILL.md +0 -44
  378. package/skills/domain-docx-compatibility/skill.json +0 -19
  379. package/skills/domain-editor-architecture/SKILL.md +0 -49
  380. package/skills/domain-editor-architecture/skill.json +0 -19
  381. package/skills/domain-legal-review/SKILL.md +0 -39
  382. package/skills/domain-legal-review/skill.json +0 -19
  383. package/skills/provider-aws/SKILL.md +0 -117
  384. package/skills/provider-aws/adapters/claude.md +0 -1
  385. package/skills/provider-aws/adapters/codex.md +0 -1
  386. package/skills/provider-aws/references/service-verification.md +0 -39
  387. package/skills/provider-aws/skill.json +0 -54
  388. package/skills/provider-custom-deploy/SKILL.md +0 -64
  389. package/skills/provider-custom-deploy/skill.json +0 -50
  390. package/skills/provider-docker-compose/SKILL.md +0 -96
  391. package/skills/provider-docker-compose/adapters/local.md +0 -1
  392. package/skills/provider-docker-compose/skill.json +0 -53
  393. package/skills/provider-github-release/SKILL.md +0 -121
  394. package/skills/provider-github-release/adapters/claude.md +0 -1
  395. package/skills/provider-github-release/adapters/codex.md +0 -1
  396. package/skills/provider-github-release/skill.json +0 -55
  397. package/skills/provider-kubernetes/SKILL.md +0 -143
  398. package/skills/provider-kubernetes/adapters/claude.md +0 -1
  399. package/skills/provider-kubernetes/adapters/codex.md +0 -1
  400. package/skills/provider-kubernetes/references/kubectl-patterns.md +0 -58
  401. package/skills/provider-kubernetes/skill.json +0 -52
  402. package/skills/provider-railway/SKILL.md +0 -123
  403. package/skills/provider-railway/adapters/claude.md +0 -1
  404. package/skills/provider-railway/adapters/codex.md +0 -1
  405. package/skills/provider-railway/adapters/local.md +0 -1
  406. package/skills/provider-railway/adapters/opencode.md +0 -1
  407. package/skills/provider-railway/references/verification-commands.md +0 -39
  408. package/skills/provider-railway/skill.json +0 -71
  409. package/skills/provider-ssh-manual/SKILL.md +0 -97
  410. package/skills/provider-ssh-manual/skill.json +0 -54
  411. package/skills/repo-coding-rules/SKILL.md +0 -55
  412. package/skills/repo-coding-rules/skill.json +0 -34
  413. package/skills/role-cont-eval/SKILL.md +0 -91
  414. package/skills/role-cont-eval/adapters/codex.md +0 -1
  415. package/skills/role-cont-eval/skill.json +0 -36
  416. package/skills/role-cont-qa/SKILL.md +0 -100
  417. package/skills/role-cont-qa/adapters/claude.md +0 -1
  418. package/skills/role-cont-qa/skill.json +0 -36
  419. package/skills/role-deploy/SKILL.md +0 -97
  420. package/skills/role-deploy/skill.json +0 -36
  421. package/skills/role-design/SKILL.md +0 -50
  422. package/skills/role-design/skill.json +0 -36
  423. package/skills/role-documentation/SKILL.md +0 -76
  424. package/skills/role-documentation/skill.json +0 -36
  425. package/skills/role-implementation/SKILL.md +0 -45
  426. package/skills/role-implementation/skill.json +0 -36
  427. package/skills/role-infra/SKILL.md +0 -81
  428. package/skills/role-infra/skill.json +0 -36
  429. package/skills/role-integration/SKILL.md +0 -91
  430. package/skills/role-integration/skill.json +0 -36
  431. package/skills/role-planner/SKILL.md +0 -39
  432. package/skills/role-planner/skill.json +0 -21
  433. package/skills/role-research/SKILL.md +0 -65
  434. package/skills/role-research/skill.json +0 -36
  435. package/skills/role-security/SKILL.md +0 -60
  436. package/skills/role-security/skill.json +0 -36
  437. package/skills/runtime-claude/SKILL.md +0 -66
  438. package/skills/runtime-claude/skill.json +0 -36
  439. package/skills/runtime-codex/SKILL.md +0 -58
  440. package/skills/runtime-codex/skill.json +0 -36
  441. package/skills/runtime-local/SKILL.md +0 -46
  442. package/skills/runtime-local/skill.json +0 -36
  443. package/skills/runtime-opencode/SKILL.md +0 -58
  444. package/skills/runtime-opencode/skill.json +0 -36
  445. package/skills/signal-hygiene/SKILL.md +0 -51
  446. package/skills/signal-hygiene/skill.json +0 -20
  447. package/skills/tui-design/SKILL.md +0 -77
  448. package/skills/tui-design/references/tui-design.md +0 -259
  449. package/skills/tui-design/skill.json +0 -36
  450. package/skills/wave-core/SKILL.md +0 -141
  451. package/skills/wave-core/references/marker-syntax.md +0 -70
  452. package/skills/wave-core/skill.json +0 -35
  453. package/src/README.md +0 -85
  454. package/src/api/README.md +0 -22
  455. package/src/api/public-types.ts +0 -525
  456. package/src/component-inventory.md +0 -99
  457. package/src/core/README.md +0 -10
  458. package/src/core/commands/README.md +0 -3
  459. package/src/core/commands/formatting-commands.ts +0 -161
  460. package/src/core/commands/image-commands.ts +0 -144
  461. package/src/core/commands/index.ts +0 -1013
  462. package/src/core/commands/list-commands.ts +0 -370
  463. package/src/core/commands/review-commands.ts +0 -108
  464. package/src/core/commands/text-commands.ts +0 -119
  465. package/src/core/schema/README.md +0 -3
  466. package/src/core/schema/text-schema.ts +0 -512
  467. package/src/core/selection/README.md +0 -3
  468. package/src/core/selection/mapping.ts +0 -238
  469. package/src/core/selection/review-anchors.ts +0 -94
  470. package/src/core/state/README.md +0 -3
  471. package/src/core/state/editor-state.ts +0 -580
  472. package/src/core/state/text-transaction.ts +0 -276
  473. package/src/formats/xlsx/io/parse-shared-strings.ts +0 -41
  474. package/src/formats/xlsx/io/parse-sheet.ts +0 -289
  475. package/src/formats/xlsx/io/parse-styles.ts +0 -57
  476. package/src/formats/xlsx/io/parse-workbook.ts +0 -75
  477. package/src/formats/xlsx/io/xlsx-session.ts +0 -306
  478. package/src/formats/xlsx/model/cell.ts +0 -189
  479. package/src/formats/xlsx/model/sheet.ts +0 -244
  480. package/src/formats/xlsx/model/styles.ts +0 -118
  481. package/src/formats/xlsx/model/workbook.ts +0 -449
  482. package/src/io/README.md +0 -10
  483. package/src/io/docx-session.ts +0 -1763
  484. package/src/io/export/README.md +0 -3
  485. package/src/io/export/export-session.ts +0 -165
  486. package/src/io/export/minimal-docx.ts +0 -115
  487. package/src/io/export/reattach-preserved-parts.ts +0 -54
  488. package/src/io/export/serialize-comments.ts +0 -876
  489. package/src/io/export/serialize-footnotes.ts +0 -217
  490. package/src/io/export/serialize-headers-footers.ts +0 -200
  491. package/src/io/export/serialize-main-document.ts +0 -982
  492. package/src/io/export/serialize-numbering.ts +0 -97
  493. package/src/io/export/serialize-revisions.ts +0 -389
  494. package/src/io/export/serialize-runtime-revisions.ts +0 -265
  495. package/src/io/export/serialize-tables.ts +0 -147
  496. package/src/io/export/split-review-boundaries.ts +0 -194
  497. package/src/io/normalize/README.md +0 -3
  498. package/src/io/normalize/normalize-text.ts +0 -437
  499. package/src/io/ooxml/README.md +0 -3
  500. package/src/io/ooxml/parse-comments.ts +0 -779
  501. package/src/io/ooxml/parse-complex-content.ts +0 -287
  502. package/src/io/ooxml/parse-fields.ts +0 -438
  503. package/src/io/ooxml/parse-footnotes.ts +0 -403
  504. package/src/io/ooxml/parse-headers-footers.ts +0 -483
  505. package/src/io/ooxml/parse-inline-media.ts +0 -431
  506. package/src/io/ooxml/parse-main-document.ts +0 -1846
  507. package/src/io/ooxml/parse-numbering.ts +0 -425
  508. package/src/io/ooxml/parse-revisions.ts +0 -658
  509. package/src/io/ooxml/parse-shapes.ts +0 -271
  510. package/src/io/ooxml/parse-tables.ts +0 -568
  511. package/src/io/ooxml/parse-theme.ts +0 -314
  512. package/src/io/ooxml/part-manifest.ts +0 -136
  513. package/src/io/ooxml/revision-boundaries.ts +0 -351
  514. package/src/io/opc/README.md +0 -3
  515. package/src/io/opc/corrupt-package.ts +0 -166
  516. package/src/io/opc/docx-package.ts +0 -74
  517. package/src/io/opc/package-reader.ts +0 -320
  518. package/src/io/opc/package-writer.ts +0 -273
  519. package/src/model/README.md +0 -3
  520. package/src/model/canonical-document.ts +0 -1911
  521. package/src/model/cds-1.0.0.ts +0 -196
  522. package/src/model/snapshot.ts +0 -393
  523. package/src/preservation/README.md +0 -3
  524. package/src/preservation/markup-compatibility.ts +0 -48
  525. package/src/preservation/opaque-fragment-store.ts +0 -89
  526. package/src/preservation/opaque-region.ts +0 -233
  527. package/src/preservation/package-preservation.ts +0 -120
  528. package/src/preservation/preserved-part-manifest.ts +0 -56
  529. package/src/preservation/relationship-retention.ts +0 -57
  530. package/src/preservation/store.ts +0 -185
  531. package/src/review/README.md +0 -16
  532. package/src/review/store/README.md +0 -3
  533. package/src/review/store/comment-anchors.ts +0 -70
  534. package/src/review/store/comment-remapping.ts +0 -154
  535. package/src/review/store/comment-store.ts +0 -331
  536. package/src/review/store/comment-thread.ts +0 -109
  537. package/src/review/store/revision-actions.ts +0 -394
  538. package/src/review/store/revision-store.ts +0 -303
  539. package/src/review/store/revision-types.ts +0 -168
  540. package/src/review/store/runtime-comment-store.ts +0 -43
  541. package/src/runtime/README.md +0 -3
  542. package/src/runtime/ai-action-policy.ts +0 -764
  543. package/src/runtime/document-runtime.ts +0 -969
  544. package/src/runtime/read-only-diagnostics-runtime.ts +0 -232
  545. package/src/runtime/review-runtime.ts +0 -44
  546. package/src/runtime/revision-runtime.ts +0 -107
  547. package/src/runtime/session-capabilities.ts +0 -138
  548. package/src/runtime/surface-projection.ts +0 -570
  549. package/src/runtime/table-commands.ts +0 -84
  550. package/src/runtime/table-schema.ts +0 -125
  551. package/src/ui/README.md +0 -30
  552. package/src/ui/WordReviewEditor.tsx +0 -1283
  553. package/src/ui/comments/README.md +0 -3
  554. package/src/ui/compatibility/README.md +0 -3
  555. package/src/ui/editor-surface/README.md +0 -3
  556. package/src/ui/headless/comment-decoration-model.ts +0 -124
  557. package/src/ui/headless/revision-decoration-model.ts +0 -128
  558. package/src/ui/headless/selection-helpers.ts +0 -34
  559. package/src/ui/headless/use-editor-keyboard.ts +0 -98
  560. package/src/ui/review/README.md +0 -3
  561. package/src/ui/shared/revision-filters.ts +0 -31
  562. package/src/ui/status/README.md +0 -3
  563. package/src/ui/theme/README.md +0 -3
  564. package/src/ui/toolbar/README.md +0 -3
  565. package/src/ui-tailwind/chrome/tw-alert-banner.tsx +0 -48
  566. package/src/ui-tailwind/chrome/tw-selection-toolbar.tsx +0 -44
  567. package/src/ui-tailwind/chrome/tw-unsaved-modal.tsx +0 -58
  568. package/src/ui-tailwind/chrome/use-before-unload.ts +0 -20
  569. package/src/ui-tailwind/editor-surface/pm-command-bridge.ts +0 -139
  570. package/src/ui-tailwind/editor-surface/pm-decorations.ts +0 -98
  571. package/src/ui-tailwind/editor-surface/pm-position-map.ts +0 -123
  572. package/src/ui-tailwind/editor-surface/pm-schema.ts +0 -452
  573. package/src/ui-tailwind/editor-surface/pm-state-from-snapshot.ts +0 -327
  574. package/src/ui-tailwind/editor-surface/search-plugin.ts +0 -157
  575. package/src/ui-tailwind/editor-surface/tw-caret.tsx +0 -12
  576. package/src/ui-tailwind/editor-surface/tw-editor-surface.tsx +0 -150
  577. package/src/ui-tailwind/editor-surface/tw-inline-token.tsx +0 -118
  578. package/src/ui-tailwind/editor-surface/tw-opaque-block.tsx +0 -52
  579. package/src/ui-tailwind/editor-surface/tw-paragraph-block.tsx +0 -151
  580. package/src/ui-tailwind/editor-surface/tw-prosemirror-surface.tsx +0 -215
  581. package/src/ui-tailwind/editor-surface/tw-segment-view.tsx +0 -111
  582. package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +0 -108
  583. package/src/ui-tailwind/index.ts +0 -61
  584. package/src/ui-tailwind/review/tw-comment-sidebar.tsx +0 -276
  585. package/src/ui-tailwind/review/tw-health-panel.tsx +0 -120
  586. package/src/ui-tailwind/review/tw-review-rail.tsx +0 -120
  587. package/src/ui-tailwind/review/tw-revision-sidebar.tsx +0 -164
  588. package/src/ui-tailwind/status/tw-status-bar.tsx +0 -58
  589. package/src/ui-tailwind/theme/editor-theme.css +0 -190
  590. package/src/ui-tailwind/toolbar/tw-toolbar-icon-button.tsx +0 -48
  591. package/src/ui-tailwind/toolbar/tw-toolbar.tsx +0 -231
  592. package/src/ui-tailwind/tw-review-workspace.tsx +0 -140
  593. package/src/validation/README.md +0 -3
  594. package/src/validation/compatibility-engine.ts +0 -317
  595. package/src/validation/compatibility-report.ts +0 -160
  596. package/src/validation/diagnostics.ts +0 -203
  597. package/src/validation/import-diagnostics.ts +0 -128
  598. package/src/validation/low-priority-word-surfaces.ts +0 -373
  599. package/test/README.md +0 -16
  600. package/test/core/formatting-commands.test.ts +0 -285
  601. package/test/core/image-commands.test.ts +0 -298
  602. package/test/core/mapping.test.ts +0 -186
  603. package/test/core/text-commands.test.ts +0 -176
  604. package/test/fixtures/docx/F01-basic-contract.docx +0 -0
  605. package/test/fixtures/docx/F01-basic-contract.md +0 -33
  606. package/test/fixtures/docx/F02-headings-styles.docx +0 -0
  607. package/test/fixtures/docx/F02-headings-styles.md +0 -33
  608. package/test/fixtures/docx/F03-legal-outline-numbering.docx +0 -0
  609. package/test/fixtures/docx/F03-legal-outline-numbering.md +0 -34
  610. package/test/fixtures/docx/F04-restart-numbering-schedules.docx +0 -0
  611. package/test/fixtures/docx/F04-restart-numbering-schedules.md +0 -33
  612. package/test/fixtures/docx/F05-table-heavy-agreement.docx +0 -0
  613. package/test/fixtures/docx/F05-table-heavy-agreement.md +0 -34
  614. package/test/fixtures/docx/F06-merged-cells-signature-table.docx +0 -0
  615. package/test/fixtures/docx/F06-merged-cells-signature-table.md +0 -34
  616. package/test/fixtures/docx/F07-inline-images-exhibit.docx +0 -0
  617. package/test/fixtures/docx/F07-inline-images-exhibit.md +0 -34
  618. package/test/fixtures/docx/F08-hyperlinks.docx +0 -0
  619. package/test/fixtures/docx/F08-hyperlinks.md +0 -33
  620. package/test/fixtures/docx/F09-comments-single-paragraph.docx +0 -0
  621. package/test/fixtures/docx/F09-comments-single-paragraph.md +0 -33
  622. package/test/fixtures/docx/F10-threaded-comments-resolve.docx +0 -0
  623. package/test/fixtures/docx/F10-threaded-comments-resolve.md +0 -33
  624. package/test/fixtures/docx/F11-redlines-basic.docx +0 -0
  625. package/test/fixtures/docx/F11-redlines-basic.md +0 -33
  626. package/test/fixtures/docx/F12-redlines-paragraph-joins-splits.docx +0 -0
  627. package/test/fixtures/docx/F12-redlines-paragraph-joins-splits.md +0 -33
  628. package/test/fixtures/docx/F13-comments-on-deleted-text.docx +0 -0
  629. package/test/fixtures/docx/F13-comments-on-deleted-text.md +0 -33
  630. package/test/fixtures/docx/F14-revisions-in-tables-and-lists.docx +0 -0
  631. package/test/fixtures/docx/F14-revisions-in-tables-and-lists.md +0 -33
  632. package/test/fixtures/docx/F15-sections-headers-footers.docx +0 -0
  633. package/test/fixtures/docx/F15-sections-headers-footers.md +0 -33
  634. package/test/fixtures/docx/F16-footnotes-endnotes.docx +0 -0
  635. package/test/fixtures/docx/F16-footnotes-endnotes.md +0 -33
  636. package/test/fixtures/docx/F17-fields-and-toc.docx +0 -0
  637. package/test/fixtures/docx/F17-fields-and-toc.md +0 -33
  638. package/test/fixtures/docx/F18-content-controls-template.docx +0 -0
  639. package/test/fixtures/docx/F18-content-controls-template.md +0 -33
  640. package/test/fixtures/docx/F19-custom-xml-doc-assembly.docx +0 -0
  641. package/test/fixtures/docx/F19-custom-xml-doc-assembly.md +0 -35
  642. package/test/fixtures/docx/F20-unknown-ooxml-and-alternatecontent.docx +0 -0
  643. package/test/fixtures/docx/F20-unknown-ooxml-and-alternatecontent.md +0 -33
  644. package/test/fixtures/docx/F21-malformed-broken-docx.docx +0 -0
  645. package/test/fixtures/docx/F21-malformed-broken-docx.md +0 -33
  646. package/test/fixtures/docx/README.md +0 -74
  647. package/test/fixtures/docx/certification-manifest.json +0 -104
  648. package/test/fixtures/docx/fixtures.manifest.json +0 -196
  649. package/test/fixtures/encrypted-docx/README.md +0 -27
  650. package/test/fixtures/encrypted-docx/certification-manifest.json +0 -9
  651. package/test/fixtures/encrypted-docx/fixtures.manifest.json +0 -47
  652. package/test/fixtures/scenarios/docx/README.md +0 -25
  653. package/test/fixtures/scenarios/docx/S01-sow-template.docx +0 -0
  654. package/test/fixtures/scenarios/docx/S01-sow-template.md +0 -30
  655. package/test/fixtures/scenarios/docx/S02-bw-partner-user-licence-agreement-redlines.docx +0 -0
  656. package/test/fixtures/scenarios/docx/S02-bw-partner-user-licence-agreement-redlines.md +0 -32
  657. package/test/fixtures/scenarios/docx/scenario-manifest.json +0 -53
  658. package/test/formats/xlsx/io/xlsx-import.test.ts +0 -766
  659. package/test/formats/xlsx/model/workbook.test.ts +0 -669
  660. package/test/helpers/dom-setup.ts +0 -124
  661. package/test/io/comment-roundtrip.test.ts +0 -272
  662. package/test/io/complex-content-roundtrip.test.ts +0 -632
  663. package/test/io/docx-compatibility-regression.test.ts +0 -199
  664. package/test/io/docx-session.test.ts +0 -1495
  665. package/test/io/footnotes-roundtrip.test.ts +0 -318
  666. package/test/io/headers-footers-roundtrip.test.ts +0 -547
  667. package/test/io/numbering-roundtrip.test.ts +0 -234
  668. package/test/io/package-reader.test.ts +0 -199
  669. package/test/io/paragraph-properties-roundtrip.test.ts +0 -129
  670. package/test/io/preserved-package-roundtrip.test.ts +0 -365
  671. package/test/io/property-completeness.test.ts +0 -292
  672. package/test/io/revision-roundtrip.test.ts +0 -347
  673. package/test/io/structural-blocks.test.ts +0 -202
  674. package/test/io/table-media-roundtrip.test.ts +0 -448
  675. package/test/io/table-properties-roundtrip.test.ts +0 -569
  676. package/test/io/table-roundtrip.test.ts +0 -302
  677. package/test/io/text-roundtrip.test.ts +0 -344
  678. package/test/model/canonical-document.test.ts +0 -285
  679. package/test/preservation/opaque-fragment-store.test.ts +0 -121
  680. package/test/preservation/package-preservation.test.ts +0 -395
  681. package/test/preservation/store.test.ts +0 -84
  682. package/test/review/comment-remapping.test.ts +0 -220
  683. package/test/review/comment-store.test.ts +0 -180
  684. package/test/review/move-revisions.test.ts +0 -143
  685. package/test/review/property-change-revisions.test.ts +0 -225
  686. package/test/review/revision-actions.test.ts +0 -330
  687. package/test/review/revision-store.test.ts +0 -193
  688. package/test/runtime/session-capabilities.test.ts +0 -260
  689. package/test/runtime/table-commands.test.ts +0 -356
  690. package/test/runtime/table-schema.test.ts +0 -221
  691. package/test/runtime/tracked-changes-toggle.test.ts +0 -107
  692. package/test/ui/comment-review-surface.test.tsx +0 -114
  693. package/test/ui/reduced-motion-toggle.test.tsx +0 -137
  694. package/test/ui/word-review-editor.imported-scenarios.test.tsx +0 -169
  695. package/test/ui/word-review-editor.interaction.test.tsx +0 -1198
  696. package/test/ui/word-review-editor.test.js +0 -188
  697. package/test/ui/word-review-editor.test.tsx +0 -280
  698. package/test/ui-tailwind/search-plugin.test.ts +0 -286
  699. package/test/validation/compatibility-engine.test.ts +0 -336
  700. package/test/validation/compatibility-report.test.ts +0 -189
  701. package/test/validation/low-priority-word-surfaces.test.ts +0 -282
  702. package/test/validation/malformed-doc.test.ts +0 -113
  703. package/test-results/.last-run.json +0 -4
  704. package/wave.config.json +0 -406
@@ -1,1763 +0,0 @@
1
- import type {
2
- CompatibilityReport as PublicCompatibilityReport,
3
- EditorError,
4
- EditorWarning as PublicEditorWarning,
5
- EditorAnchorProjection as PublicEditorAnchorProjection,
6
- ExportDocxOptions,
7
- ExportResult,
8
- PersistedEditorSnapshot,
9
- } from "../api/public-types.ts";
10
- import type {
11
- CanonicalDocumentEnvelope,
12
- CompatibilityFeatureEntry as InternalCompatibilityFeatureEntry,
13
- CompatibilityReport as InternalCompatibilityReport,
14
- CommentThreadRecord,
15
- EditorError as InternalEditorError,
16
- RevisionRecord as RuntimeRevisionRecord,
17
- EditorWarning as InternalEditorWarning,
18
- } from "../core/state/editor-state.ts";
19
- import { createCanonicalDocumentId } from "../core/state/editor-state.ts";
20
- import {
21
- createDetachedAnchor,
22
- type EditorAnchorProjection as InternalEditorAnchorProjection,
23
- } from "../core/selection/mapping.ts";
24
- import { DOCX_MIME_TYPE } from "./opc/docx-package.ts";
25
- import { readOpcPackage, type OpcPackage } from "./opc/package-reader.ts";
26
- import { parseMainDocumentXml } from "./ooxml/parse-main-document.ts";
27
- import { normalizeParsedTextDocument } from "./normalize/normalize-text.ts";
28
- import {
29
- normalizePartPath,
30
- resolveRelationshipTarget,
31
- type OpcRelationship,
32
- } from "./ooxml/part-manifest.ts";
33
- import {
34
- classifyCorruptPackageError,
35
- createBrokenRelationshipIssue,
36
- createMissingPartIssue,
37
- } from "./opc/corrupt-package.ts";
38
- import { createExportSession } from "./export/export-session.ts";
39
- import { serializeMainDocument } from "./export/serialize-main-document.ts";
40
- import { parseRevisionsFromDocumentXml, type ParsedRevisionsResult } from "./ooxml/parse-revisions.ts";
41
- import { parseCommentsFromOoxml } from "./ooxml/parse-comments.ts";
42
- import { parseNumberingXml } from "./ooxml/parse-numbering.ts";
43
- import {
44
- createCommentExportIdMap,
45
- serializeCommentAnchorsIntoDocumentXml,
46
- serializeMergedCommentsXml,
47
- } from "./export/serialize-comments.ts";
48
- import { splitDocumentAtReviewBoundaries } from "./export/split-review-boundaries.ts";
49
- import { serializeRuntimeRevisionsIntoDocumentXml } from "./export/serialize-runtime-revisions.ts";
50
- import { createCommentStoreFromRuntimeComments } from "../review/store/runtime-comment-store.ts";
51
- import type { CommentThread } from "../review/store/comment-store.ts";
52
- import type { RevisionRecord as ReviewRevisionRecord } from "../review/store/revision-types.ts";
53
- import { getRevisionActionability } from "../review/store/revision-types.ts";
54
- import { buildCompatibilityReport } from "../validation/compatibility-engine.ts";
55
- import {
56
- createPackageImportDiagnostics,
57
- createValidationImportDiagnostics,
58
- type ImportDiagnosticsResult,
59
- } from "../validation/import-diagnostics.ts";
60
- import type {
61
- FootnoteCollection,
62
- HeaderDocument,
63
- FooterDocument,
64
- MediaCatalog,
65
- NumberingCatalog,
66
- OpaqueFragmentRecord,
67
- PreservedPackagePart,
68
- SubPartsCatalog,
69
- } from "../model/canonical-document.ts";
70
- import { createCanonicalDocumentSignature } from "../model/canonical-document.ts";
71
- import type {
72
- CommentImportDiagnostic,
73
- ImportedCommentDefinition,
74
- ParsedCommentsResult,
75
- } from "./ooxml/parse-comments.ts";
76
- import { createReadOnlyDiagnosticsRuntime } from "../runtime/read-only-diagnostics-runtime.ts";
77
- import {
78
- WORD_NUMBERING_CONTENT_TYPE,
79
- serializeNumberingXml,
80
- } from "./export/serialize-numbering.ts";
81
- import {
82
- parseHeaderFooterReferences,
83
- parseHeaderXml,
84
- parseFooterXml,
85
- } from "./ooxml/parse-headers-footers.ts";
86
- import { parseFootnotesXml, parseEndnotesXml } from "./ooxml/parse-footnotes.ts";
87
- import { parseThemeXml } from "./ooxml/parse-theme.ts";
88
- import {
89
- serializeHeaderXml,
90
- serializeFooterXml,
91
- WORD_HEADER_CONTENT_TYPE,
92
- WORD_FOOTER_CONTENT_TYPE,
93
- } from "./export/serialize-headers-footers.ts";
94
- import {
95
- serializeFootnotesXml,
96
- serializeEndnotesXml,
97
- WORD_FOOTNOTES_CONTENT_TYPE,
98
- WORD_ENDNOTES_CONTENT_TYPE,
99
- } from "./export/serialize-footnotes.ts";
100
-
101
- const MAIN_DOCUMENT_PATH = "/word/document.xml";
102
- const NUMBERING_PART_PATH = "/word/numbering.xml";
103
- const COMMENTS_PART_PATH = "/word/comments.xml";
104
- const COMMENTS_EXTENDED_PART_PATH = "/word/commentsExtended.xml";
105
- const COMMENTS_IDS_PART_PATH = "/word/commentsIds.xml";
106
- const PEOPLE_PART_PATH = "/word/people.xml";
107
- const MAIN_DOCUMENT_CONTENT_TYPE =
108
- "application/vnd.openxmlformats-officedocument.wordprocessingml.document.main+xml";
109
- const NUMBERING_RELATIONSHIP_TYPE =
110
- "http://schemas.openxmlformats.org/officeDocument/2006/relationships/numbering";
111
- const COMMENTS_CONTENT_TYPE =
112
- "application/vnd.openxmlformats-officedocument.wordprocessingml.comments+xml";
113
- const COMMENTS_EXTENDED_CONTENT_TYPE =
114
- "application/vnd.openxmlformats-officedocument.wordprocessingml.commentsExtended+xml";
115
- const COMMENTS_IDS_CONTENT_TYPE =
116
- "application/vnd.openxmlformats-officedocument.wordprocessingml.commentsIds+xml";
117
- const PEOPLE_CONTENT_TYPE =
118
- "application/vnd.openxmlformats-officedocument.wordprocessingml.people+xml";
119
- const COMMENTS_RELATIONSHIP_TYPE =
120
- "http://schemas.openxmlformats.org/officeDocument/2006/relationships/comments";
121
- const COMMENTS_EXTENDED_RELATIONSHIP_TYPE =
122
- "http://schemas.microsoft.com/office/2011/relationships/commentsExtended";
123
- const COMMENTS_IDS_RELATIONSHIP_TYPE =
124
- "http://schemas.microsoft.com/office/2016/09/relationships/commentsIds";
125
- const PEOPLE_RELATIONSHIP_TYPE =
126
- "http://schemas.microsoft.com/office/2011/relationships/people";
127
- const HEADER_RELATIONSHIP_TYPE =
128
- "http://schemas.openxmlformats.org/officeDocument/2006/relationships/header";
129
- const FOOTER_RELATIONSHIP_TYPE =
130
- "http://schemas.openxmlformats.org/officeDocument/2006/relationships/footer";
131
- const FOOTNOTES_RELATIONSHIP_TYPE =
132
- "http://schemas.openxmlformats.org/officeDocument/2006/relationships/footnotes";
133
- const ENDNOTES_RELATIONSHIP_TYPE =
134
- "http://schemas.openxmlformats.org/officeDocument/2006/relationships/endnotes";
135
- const FOOTNOTES_PART_PATH = "/word/footnotes.xml";
136
- const ENDNOTES_PART_PATH = "/word/endnotes.xml";
137
-
138
- interface LoadDocxEditorSessionOptions {
139
- documentId: string;
140
- sourceLabel?: string;
141
- bytes: Uint8Array | ArrayBuffer;
142
- editorBuild: string;
143
- }
144
-
145
- export interface LoadedDocxEditorSession {
146
- initialSnapshot: PersistedEditorSnapshot;
147
- fatalError?: EditorError;
148
- readOnly: boolean;
149
- exportDocx: (
150
- snapshot: PersistedEditorSnapshot,
151
- options?: ExportDocxOptions,
152
- ) => Promise<ExportResult>;
153
- }
154
-
155
- interface ImportedDocxState {
156
- sourceBytes: Uint8Array;
157
- sourcePackage: OpcPackage;
158
- sourceDocumentRelationships: readonly OpcRelationship[];
159
- sourceDocumentAttributes: Record<string, string>;
160
- sourceNumberingPartPath?: string;
161
- sourceNumberingRelationshipId?: string;
162
- sourceCommentsPartPath?: string;
163
- sourceCommentsRelationshipId?: string;
164
- sourceCommentsRootTag?: string;
165
- sourceCommentsExtendedPartPath?: string;
166
- sourceCommentsExtendedRelationshipId?: string;
167
- sourceCommentsExtendedRootTag?: string;
168
- sourceCommentsIdsPartPath?: string;
169
- sourceCommentsIdsRelationshipId?: string;
170
- sourceCommentsIdsRootTag?: string;
171
- sourcePeoplePartPath?: string;
172
- sourcePeopleRelationshipId?: string;
173
- sourcePeopleRootTag?: string;
174
- sourcePeopleAuthors: readonly string[];
175
- preservedCommentDefinitions: readonly ImportedCommentDefinition[];
176
- blockingCommentDiagnostics: readonly CommentImportDiagnostic[];
177
- initialCanonicalSignature: string;
178
- sourceSubPartPaths: {
179
- headers: Array<{ partPath: string; relationshipId: string }>;
180
- footers: Array<{ partPath: string; relationshipId: string }>;
181
- footnotesPartPath?: string;
182
- footnotesRelationshipId?: string;
183
- endnotesPartPath?: string;
184
- endnotesRelationshipId?: string;
185
- themePartPath?: string;
186
- themeRelationshipId?: string;
187
- };
188
- }
189
-
190
- interface NormalizedImportedCommentsResult extends ParsedCommentsResult {
191
- preservedDefinitions: readonly ImportedCommentDefinition[];
192
- }
193
-
194
- const BLOCKING_COMMENT_DIAGNOSTIC_CODES = new Set<CommentImportDiagnostic["code"]>([
195
- "missing_comment_definition",
196
- "missing_anchor_reference",
197
- "multi_paragraph_anchor_preserve_only",
198
- "opaque_anchor_preserve_only",
199
- "preserve_only_revision_overlap",
200
- ]);
201
-
202
- export function loadDocxEditorSession(
203
- options: LoadDocxEditorSessionOptions,
204
- ): LoadedDocxEditorSession {
205
- const sourceBytes = toUint8Array(options.bytes);
206
- let sourcePackage: OpcPackage;
207
-
208
- try {
209
- sourcePackage = readOpcPackage(sourceBytes);
210
- } catch (error) {
211
- return createDiagnosticsSession(
212
- options,
213
- createPackageImportDiagnostics({
214
- issue: classifyCorruptPackageError(error),
215
- }),
216
- );
217
- }
218
-
219
- const brokenRelationshipIssues = collectBrokenInternalRelationshipIssues(sourcePackage);
220
- if (brokenRelationshipIssues.length > 0) {
221
- return createDiagnosticsSession(
222
- options,
223
- createPackageImportDiagnostics({
224
- issue: {
225
- ...brokenRelationshipIssues[0],
226
- message: summarizeBrokenRelationshipIssues(brokenRelationshipIssues),
227
- details: {
228
- issueCount: brokenRelationshipIssues.length,
229
- targets: brokenRelationshipIssues.map((issue) => issue.targetPartPath).filter(Boolean),
230
- },
231
- },
232
- }),
233
- );
234
- }
235
-
236
- const documentPart = sourcePackage.parts.get(MAIN_DOCUMENT_PATH);
237
- if (!documentPart) {
238
- return createDiagnosticsSession(
239
- options,
240
- createPackageImportDiagnostics({
241
- issue: createMissingPartIssue(MAIN_DOCUMENT_PATH),
242
- }),
243
- );
244
- }
245
-
246
- try {
247
- const sourceDocumentXml = decodeUtf8(documentPart.bytes);
248
- const importedRevisions = parseRevisionsFromDocumentXml(sourceDocumentXml);
249
- const numberingPartPath = resolveDocumentRelatedPartPath(
250
- sourcePackage,
251
- documentPart.relationships,
252
- NUMBERING_RELATIONSHIP_TYPE,
253
- NUMBERING_PART_PATH,
254
- );
255
- const parsedNumbering = numberingPartPath
256
- ? parseNumberingXml(
257
- decodeUtf8(sourcePackage.parts.get(numberingPartPath)?.bytes ?? new Uint8Array()),
258
- )
259
- : createEmptyNumberingCatalog();
260
- const mediaParts = collectInlineMediaParts(sourcePackage);
261
- const parsedDocument = parseMainDocumentXml(
262
- sourceDocumentXml,
263
- documentPart.relationships,
264
- mediaParts,
265
- MAIN_DOCUMENT_PATH,
266
- );
267
- const normalizedDocument = normalizeParsedTextDocument(
268
- parsedDocument,
269
- MAIN_DOCUMENT_PATH,
270
- );
271
- const commentsPartPath = resolveCommentsPartPath(sourcePackage, documentPart.relationships);
272
- const commentsExtendedPartPath = resolveDocumentRelatedPartPath(
273
- sourcePackage,
274
- documentPart.relationships,
275
- COMMENTS_EXTENDED_RELATIONSHIP_TYPE,
276
- COMMENTS_EXTENDED_PART_PATH,
277
- );
278
- const commentsIdsPartPath = resolveDocumentRelatedPartPath(
279
- sourcePackage,
280
- documentPart.relationships,
281
- COMMENTS_IDS_RELATIONSHIP_TYPE,
282
- COMMENTS_IDS_PART_PATH,
283
- );
284
- const peoplePartPath = resolveDocumentRelatedPartPath(
285
- sourcePackage,
286
- documentPart.relationships,
287
- PEOPLE_RELATIONSHIP_TYPE,
288
- PEOPLE_PART_PATH,
289
- );
290
- const parsedComments = commentsPartPath
291
- ? parseCommentsFromOoxml(
292
- sourceDocumentXml,
293
- {
294
- commentsXml: decodeUtf8(sourcePackage.parts.get(commentsPartPath)?.bytes ?? new Uint8Array()),
295
- commentsExtendedXml: decodeUtf8(
296
- sourcePackage.parts.get(commentsExtendedPartPath ?? "")?.bytes ?? new Uint8Array(),
297
- ),
298
- commentsIdsXml: decodeUtf8(
299
- sourcePackage.parts.get(commentsIdsPartPath ?? "")?.bytes ?? new Uint8Array(),
300
- ),
301
- peopleXml: decodeUtf8(
302
- sourcePackage.parts.get(peoplePartPath ?? "")?.bytes ?? new Uint8Array(),
303
- ),
304
- },
305
- )
306
- : {
307
- threads: [] as CommentThread[],
308
- diagnostics: [] as CommentImportDiagnostic[],
309
- definitions: [] as ImportedCommentDefinition[],
310
- sourceRootTag: undefined,
311
- sourceExtendedRootTag: undefined,
312
- sourceIdsRootTag: undefined,
313
- sourcePeopleRootTag: undefined,
314
- peopleAuthors: [] as string[],
315
- };
316
- const normalizedRevisions = normalizeImportedRevisionRecords(
317
- importedRevisions,
318
- normalizedDocument.content,
319
- normalizedDocument.preservation.opaqueFragments,
320
- );
321
- const normalizedComments = normalizeImportedCommentThreads(
322
- parsedComments,
323
- normalizedDocument.preservation.opaqueFragments,
324
- normalizedRevisions.revisions,
325
- );
326
- // ---- Parse sub-parts: headers, footers, footnotes, endnotes, theme ----
327
- const headerFooterRefs = parseHeaderFooterReferences(sourceDocumentXml);
328
- const parsedHeaders: HeaderDocument[] = [];
329
- const parsedFooters: FooterDocument[] = [];
330
- const sourceHeaderPaths: Array<{ partPath: string; relationshipId: string }> = [];
331
- const sourceFooterPaths: Array<{ partPath: string; relationshipId: string }> = [];
332
- const seenSubPartRelIds = new Set<string>();
333
-
334
- for (const ref of headerFooterRefs) {
335
- if (seenSubPartRelIds.has(ref.relationshipId)) {
336
- continue;
337
- }
338
- seenSubPartRelIds.add(ref.relationshipId);
339
-
340
- const relationship = documentPart.relationships.find(
341
- (r) => r.id === ref.relationshipId && r.targetMode === "internal",
342
- );
343
- if (!relationship) {
344
- continue;
345
- }
346
-
347
- const partPath = resolveRelationshipTarget(MAIN_DOCUMENT_PATH, relationship);
348
- const partBytes = sourcePackage.parts.get(partPath)?.bytes;
349
- if (!partBytes) {
350
- continue;
351
- }
352
-
353
- const xml = decodeUtf8(partBytes);
354
- if (ref.kind === "header") {
355
- const parsed = parseHeaderXml(xml);
356
- parsedHeaders.push({
357
- variant: ref.variant,
358
- partPath,
359
- relationshipId: ref.relationshipId,
360
- blocks: parsed.blocks,
361
- });
362
- sourceHeaderPaths.push({ partPath, relationshipId: ref.relationshipId });
363
- } else {
364
- const parsed = parseFooterXml(xml);
365
- parsedFooters.push({
366
- variant: ref.variant,
367
- partPath,
368
- relationshipId: ref.relationshipId,
369
- blocks: parsed.blocks,
370
- });
371
- sourceFooterPaths.push({ partPath, relationshipId: ref.relationshipId });
372
- }
373
- }
374
-
375
- const footnotesPartPath = resolveDocumentRelatedPartPath(
376
- sourcePackage,
377
- documentPart.relationships,
378
- FOOTNOTES_RELATIONSHIP_TYPE,
379
- FOOTNOTES_PART_PATH,
380
- );
381
- const footnotesRelationshipId = documentPart.relationships.find(
382
- (r) => r.type === FOOTNOTES_RELATIONSHIP_TYPE && r.targetMode === "internal",
383
- )?.id;
384
- const endnotesPartPath = resolveDocumentRelatedPartPath(
385
- sourcePackage,
386
- documentPart.relationships,
387
- ENDNOTES_RELATIONSHIP_TYPE,
388
- ENDNOTES_PART_PATH,
389
- );
390
- const endnotesRelationshipId = documentPart.relationships.find(
391
- (r) => r.type === ENDNOTES_RELATIONSHIP_TYPE && r.targetMode === "internal",
392
- )?.id;
393
-
394
- let footnoteCollection: FootnoteCollection | undefined;
395
- if (footnotesPartPath) {
396
- footnoteCollection = parseFootnotesXml(
397
- decodeUtf8(sourcePackage.parts.get(footnotesPartPath)?.bytes ?? new Uint8Array()),
398
- );
399
- }
400
- if (endnotesPartPath) {
401
- footnoteCollection = parseEndnotesXml(
402
- decodeUtf8(sourcePackage.parts.get(endnotesPartPath)?.bytes ?? new Uint8Array()),
403
- footnoteCollection,
404
- );
405
- }
406
-
407
- const themeRelationship = documentPart.relationships.find(
408
- (r) => r.type === "http://schemas.openxmlformats.org/officeDocument/2006/relationships/theme" &&
409
- r.targetMode === "internal",
410
- );
411
- const themePartPath = themeRelationship
412
- ? resolveRelationshipTarget(MAIN_DOCUMENT_PATH, themeRelationship)
413
- : undefined;
414
- const parsedTheme =
415
- themePartPath && sourcePackage.parts.has(themePartPath)
416
- ? parseThemeXml(
417
- decodeUtf8(sourcePackage.parts.get(themePartPath)?.bytes ?? new Uint8Array()),
418
- )
419
- : undefined;
420
-
421
- const subParts: SubPartsCatalog | undefined =
422
- parsedHeaders.length > 0 ||
423
- parsedFooters.length > 0 ||
424
- footnoteCollection !== undefined ||
425
- parsedTheme !== undefined
426
- ? {
427
- headers: parsedHeaders,
428
- footers: parsedFooters,
429
- ...(footnoteCollection !== undefined ? { footnoteCollection } : {}),
430
- ...(parsedTheme !== undefined ? { theme: parsedTheme } : {}),
431
- }
432
- : undefined;
433
-
434
- const timestamp = new Date().toISOString();
435
- const document = createImportedCanonicalDocument({
436
- documentId: options.documentId,
437
- timestamp,
438
- numbering: parsedNumbering,
439
- media: normalizedDocument.media,
440
- content: normalizedDocument.content,
441
- subParts,
442
- preservation: {
443
- ...normalizedDocument.preservation,
444
- packageParts: {
445
- ...normalizedDocument.preservation.packageParts,
446
- ...collectPreservedPackageParts(sourcePackage, [
447
- numberingPartPath,
448
- commentsPartPath,
449
- commentsExtendedPartPath,
450
- commentsIdsPartPath,
451
- peoplePartPath,
452
- ]),
453
- },
454
- },
455
- diagnostics: {
456
- warnings: [
457
- ...normalizedDocument.diagnostics.warnings,
458
- ...normalizedRevisions.diagnostics.map((diagnostic, index) => ({
459
- diagnosticId: `diagnostic:revision-import-${index + 1}`,
460
- warningId: `warning:revision-import-${diagnostic.revisionId}`,
461
- source: "review" as const,
462
- message: diagnostic.message,
463
- })),
464
- ...normalizedComments.diagnostics.map((diagnostic, index) => ({
465
- diagnosticId: `diagnostic:comment-import-${index + 1}`,
466
- warningId: `warning:comment-import-${diagnostic.commentId}`,
467
- source: "review" as const,
468
- message: diagnostic.message,
469
- })),
470
- ],
471
- errors: [],
472
- },
473
- review: {
474
- comments: toRuntimeCommentRecords(normalizedComments.threads),
475
- revisions: toRuntimeRevisionRecords(normalizedRevisions.revisions),
476
- },
477
- });
478
- const compatibility = buildCompatibilityReport({
479
- document,
480
- generatedAt: timestamp,
481
- });
482
- const snapshot = createImportedSnapshot({
483
- documentId: options.documentId,
484
- editorBuild: options.editorBuild,
485
- timestamp,
486
- document,
487
- compatibility: toPublicCompatibilityReport(compatibility),
488
- });
489
- const importedState: ImportedDocxState = {
490
- sourceBytes: new Uint8Array(sourceBytes),
491
- sourcePackage,
492
- sourceDocumentRelationships: documentPart.relationships,
493
- sourceDocumentAttributes: extractDocumentRootAttributes(sourceDocumentXml),
494
- sourceNumberingPartPath: numberingPartPath,
495
- sourceNumberingRelationshipId: documentPart.relationships.find(
496
- (relationship) =>
497
- relationship.type === NUMBERING_RELATIONSHIP_TYPE &&
498
- relationship.targetMode === "internal",
499
- )?.id,
500
- sourceCommentsPartPath: commentsPartPath,
501
- sourceCommentsRelationshipId: documentPart.relationships.find(
502
- (relationship) =>
503
- relationship.type === COMMENTS_RELATIONSHIP_TYPE &&
504
- relationship.targetMode === "internal",
505
- )?.id,
506
- sourceCommentsRootTag: normalizedComments.sourceRootTag,
507
- sourceCommentsExtendedPartPath: commentsExtendedPartPath,
508
- sourceCommentsExtendedRelationshipId: documentPart.relationships.find(
509
- (relationship) =>
510
- relationship.type === COMMENTS_EXTENDED_RELATIONSHIP_TYPE &&
511
- relationship.targetMode === "internal",
512
- )?.id,
513
- sourceCommentsExtendedRootTag: normalizedComments.sourceExtendedRootTag,
514
- sourceCommentsIdsPartPath: commentsIdsPartPath,
515
- sourceCommentsIdsRelationshipId: documentPart.relationships.find(
516
- (relationship) =>
517
- relationship.type === COMMENTS_IDS_RELATIONSHIP_TYPE &&
518
- relationship.targetMode === "internal",
519
- )?.id,
520
- sourceCommentsIdsRootTag: normalizedComments.sourceIdsRootTag,
521
- sourcePeoplePartPath: peoplePartPath,
522
- sourcePeopleRelationshipId: documentPart.relationships.find(
523
- (relationship) =>
524
- relationship.type === PEOPLE_RELATIONSHIP_TYPE &&
525
- relationship.targetMode === "internal",
526
- )?.id,
527
- sourcePeopleRootTag: normalizedComments.sourcePeopleRootTag,
528
- sourcePeopleAuthors: normalizedComments.peopleAuthors,
529
- preservedCommentDefinitions: normalizedComments.preservedDefinitions,
530
- blockingCommentDiagnostics: normalizedComments.diagnostics.filter((diagnostic) =>
531
- BLOCKING_COMMENT_DIAGNOSTIC_CODES.has(diagnostic.code),
532
- ),
533
- initialCanonicalSignature: serializeCanonicalDocumentForExport(document),
534
- sourceSubPartPaths: {
535
- headers: sourceHeaderPaths,
536
- footers: sourceFooterPaths,
537
- footnotesPartPath,
538
- footnotesRelationshipId,
539
- endnotesPartPath,
540
- endnotesRelationshipId,
541
- themePartPath,
542
- themeRelationshipId: themeRelationship?.id,
543
- },
544
- };
545
-
546
- return {
547
- initialSnapshot: snapshot,
548
- readOnly: false,
549
- exportDocx: async (nextSnapshot, exportOptions) =>
550
- exportDocxEditorSession(importedState, nextSnapshot, exportOptions),
551
- };
552
- } catch (error) {
553
- return createDiagnosticsSession(
554
- options,
555
- createImportDiagnosticsFromError(error),
556
- );
557
- }
558
- }
559
-
560
- function exportDocxEditorSession(
561
- state: ImportedDocxState,
562
- snapshot: PersistedEditorSnapshot,
563
- options?: ExportDocxOptions,
564
- ): ExportResult {
565
- if (snapshot.compatibility.blockExport) {
566
- throw new Error("DOCX export is blocked by the current compatibility report.");
567
- }
568
-
569
- const currentDocument = snapshot.canonicalDocument as CanonicalDocumentEnvelope;
570
- if (
571
- serializeCanonicalDocumentForExport(currentDocument) ===
572
- state.initialCanonicalSignature &&
573
- canReuseSourceBytesForCurrentDocument(state, currentDocument)
574
- ) {
575
- return {
576
- bytes: new Uint8Array(state.sourceBytes),
577
- mimeType: DOCX_MIME_TYPE,
578
- fileName: options?.fileName ?? `${snapshot.documentId}.docx`,
579
- };
580
- }
581
- if (state.blockingCommentDiagnostics.length > 0) {
582
- throw new Error(
583
- `DOCX export is blocked because ${state.blockingCommentDiagnostics.length} preserve-only comment anchors cannot be safely remapped after runtime edits.`,
584
- );
585
- }
586
- const currentRevisions = toReviewRevisionRecords(currentDocument.review.revisions);
587
- const actionableRevisions = currentRevisions.filter(
588
- (revision) => getRevisionActionability(revision) === "actionable",
589
- );
590
- const commentThreads = Object.values(
591
- createCommentStoreFromRuntimeComments(currentDocument.review.comments).threads,
592
- );
593
- const preservedCommentIds = new Set(
594
- state.preservedCommentDefinitions.map((definition) => definition.commentId),
595
- );
596
- const ownedCommentThreads = commentThreads.filter(
597
- (thread) => !preservedCommentIds.has(thread.commentId),
598
- );
599
- const serialized = serializeMainDocument(
600
- splitDocumentAtReviewBoundaries(
601
- currentDocument.content as never,
602
- ownedCommentThreads,
603
- actionableRevisions,
604
- ) as never,
605
- currentDocument.preservation as never,
606
- state.sourceDocumentRelationships,
607
- {
608
- documentAttributes: state.sourceDocumentAttributes,
609
- media: currentDocument.media as MediaCatalog,
610
- },
611
- );
612
- const revisionDocument = serializeRuntimeRevisionsIntoDocumentXml(
613
- serialized.documentXml,
614
- actionableRevisions,
615
- serialized.paragraphBoundaries,
616
- );
617
- if (revisionDocument.skippedRevisionIds.length > 0) {
618
- throw new Error(
619
- `DOCX export is blocked because ${revisionDocument.skippedRevisionIds.length} active revisions overlap unsupported serialization boundaries.`,
620
- );
621
- }
622
-
623
- const strippedDocumentXml = stripCommentMarkup(
624
- revisionDocument.documentXml,
625
- ownedCommentThreads.map((thread) => thread.commentId),
626
- );
627
- const exportCommentIds = createCommentExportIdMap(
628
- ownedCommentThreads,
629
- state.preservedCommentDefinitions,
630
- );
631
- const serializedComments = serializeMergedCommentsXml(ownedCommentThreads, {
632
- exportCommentIds,
633
- preservedDefinitions: state.preservedCommentDefinitions,
634
- sourceRootTag: state.sourceCommentsRootTag,
635
- sourceExtendedRootTag: state.sourceCommentsExtendedRootTag,
636
- sourceIdsRootTag: state.sourceCommentsIdsRootTag,
637
- sourcePeopleRootTag: state.sourcePeopleRootTag,
638
- peopleAuthors: state.sourcePeopleAuthors,
639
- });
640
- const annotatedDocument = serializeCommentAnchorsIntoDocumentXml(
641
- strippedDocumentXml,
642
- ownedCommentThreads,
643
- undefined,
644
- {
645
- exportCommentIds,
646
- },
647
- );
648
- const blockingSkippedCommentIds = annotatedDocument.skippedCommentIds.filter((commentId) => {
649
- const thread = ownedCommentThreads.find((candidate) => candidate.commentId === commentId);
650
- return !thread || thread.anchor.kind !== "detached";
651
- });
652
- if (blockingSkippedCommentIds.length > 0) {
653
- throw new Error(
654
- `DOCX export is blocked because ${blockingSkippedCommentIds.length} comments no longer map to serializable ranges.`,
655
- );
656
- }
657
- const commentsPartPath =
658
- state.sourceCommentsPartPath ?? COMMENTS_PART_PATH;
659
- const commentsExtendedPartPath =
660
- state.sourceCommentsExtendedPartPath ?? COMMENTS_EXTENDED_PART_PATH;
661
- const commentsIdsPartPath =
662
- state.sourceCommentsIdsPartPath ?? COMMENTS_IDS_PART_PATH;
663
- const peoplePartPath =
664
- state.sourcePeoplePartPath ?? PEOPLE_PART_PATH;
665
- const numberingPartPath =
666
- state.sourceNumberingPartPath ?? NUMBERING_PART_PATH;
667
- const serializedNumberingXml = hasNumberingEntries(currentDocument.numbering as NumberingCatalog)
668
- ? serializeNumberingXml(currentDocument.numbering as NumberingCatalog)
669
- : undefined;
670
- const nextRelationships = withDocumentRelatedParts(
671
- serialized.relationships,
672
- [
673
- {
674
- relationshipType: NUMBERING_RELATIONSHIP_TYPE,
675
- partPath: numberingPartPath,
676
- existingRelationshipId: state.sourceNumberingRelationshipId,
677
- include:
678
- Boolean(serializedNumberingXml) ||
679
- Boolean(state.sourceNumberingPartPath),
680
- },
681
- {
682
- relationshipType: COMMENTS_RELATIONSHIP_TYPE,
683
- partPath: commentsPartPath,
684
- existingRelationshipId: state.sourceCommentsRelationshipId,
685
- include:
686
- serializedComments.serializedCommentIds.length > 0 ||
687
- Boolean(state.sourceCommentsPartPath),
688
- },
689
- {
690
- relationshipType: COMMENTS_EXTENDED_RELATIONSHIP_TYPE,
691
- partPath: commentsExtendedPartPath,
692
- existingRelationshipId: state.sourceCommentsExtendedRelationshipId,
693
- include:
694
- Boolean(serializedComments.commentsExtendedXml) ||
695
- Boolean(state.sourceCommentsExtendedPartPath),
696
- },
697
- {
698
- relationshipType: COMMENTS_IDS_RELATIONSHIP_TYPE,
699
- partPath: commentsIdsPartPath,
700
- existingRelationshipId: state.sourceCommentsIdsRelationshipId,
701
- include:
702
- Boolean(serializedComments.commentsIdsXml) ||
703
- Boolean(state.sourceCommentsIdsPartPath),
704
- },
705
- {
706
- relationshipType: PEOPLE_RELATIONSHIP_TYPE,
707
- partPath: peoplePartPath,
708
- existingRelationshipId: state.sourcePeopleRelationshipId,
709
- include:
710
- Boolean(serializedComments.peopleXml) ||
711
- Boolean(state.sourcePeoplePartPath),
712
- },
713
- ],
714
- );
715
-
716
- const exportedSubParts = currentDocument.subParts as SubPartsCatalog | undefined;
717
- const subPartOwnedPaths: string[] = [];
718
- if (exportedSubParts) {
719
- for (const header of exportedSubParts.headers) {
720
- subPartOwnedPaths.push(header.partPath);
721
- }
722
- for (const footer of exportedSubParts.footers) {
723
- subPartOwnedPaths.push(footer.partPath);
724
- }
725
- if (exportedSubParts.footnoteCollection) {
726
- if (state.sourceSubPartPaths.footnotesPartPath) {
727
- subPartOwnedPaths.push(state.sourceSubPartPaths.footnotesPartPath);
728
- }
729
- if (state.sourceSubPartPaths.endnotesPartPath) {
730
- subPartOwnedPaths.push(state.sourceSubPartPaths.endnotesPartPath);
731
- }
732
- }
733
- if (exportedSubParts.theme && state.sourceSubPartPaths.themePartPath) {
734
- subPartOwnedPaths.push(state.sourceSubPartPaths.themePartPath);
735
- }
736
- }
737
-
738
- const exportSession = createExportSession(state.sourcePackage, [
739
- MAIN_DOCUMENT_PATH,
740
- numberingPartPath,
741
- commentsPartPath,
742
- commentsExtendedPartPath,
743
- commentsIdsPartPath,
744
- peoplePartPath,
745
- ...subPartOwnedPaths,
746
- ]);
747
-
748
- exportSession.replaceOwnedPart({
749
- path: MAIN_DOCUMENT_PATH,
750
- bytes: new TextEncoder().encode(annotatedDocument.documentXml),
751
- contentType: MAIN_DOCUMENT_CONTENT_TYPE,
752
- relationships: nextRelationships,
753
- });
754
-
755
- if (serializedNumberingXml || state.sourceNumberingPartPath) {
756
- exportSession.replaceOwnedPart({
757
- path: numberingPartPath,
758
- bytes: new TextEncoder().encode(
759
- serializedNumberingXml ??
760
- `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>\n<w:numbering xmlns:w="http://schemas.openxmlformats.org/wordprocessingml/2006/main"></w:numbering>`,
761
- ),
762
- contentType:
763
- state.sourcePackage.parts.get(numberingPartPath)?.contentType ??
764
- WORD_NUMBERING_CONTENT_TYPE,
765
- });
766
- }
767
-
768
- if (serializedComments.serializedCommentIds.length > 0 || state.sourceCommentsPartPath) {
769
- exportSession.replaceOwnedPart({
770
- path: commentsPartPath,
771
- bytes: new TextEncoder().encode(serializedComments.commentsXml),
772
- contentType:
773
- state.sourcePackage.parts.get(commentsPartPath)?.contentType ?? COMMENTS_CONTENT_TYPE,
774
- });
775
- }
776
-
777
- if (serializedComments.commentsExtendedXml || state.sourceCommentsExtendedPartPath) {
778
- exportSession.replaceOwnedPart({
779
- path: commentsExtendedPartPath,
780
- bytes: new TextEncoder().encode(
781
- serializedComments.commentsExtendedXml ?? `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>\n<w15:commentsEx xmlns:w15="http://schemas.microsoft.com/office/word/2012/wordml"></w15:commentsEx>`,
782
- ),
783
- contentType:
784
- state.sourcePackage.parts.get(commentsExtendedPartPath)?.contentType ??
785
- COMMENTS_EXTENDED_CONTENT_TYPE,
786
- });
787
- }
788
-
789
- if (serializedComments.commentsIdsXml || state.sourceCommentsIdsPartPath) {
790
- exportSession.replaceOwnedPart({
791
- path: commentsIdsPartPath,
792
- bytes: new TextEncoder().encode(
793
- serializedComments.commentsIdsXml ?? `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>\n<w16cid:commentsIds xmlns:w16cid="http://schemas.microsoft.com/office/word/2016/wordml/cid"></w16cid:commentsIds>`,
794
- ),
795
- contentType:
796
- state.sourcePackage.parts.get(commentsIdsPartPath)?.contentType ??
797
- COMMENTS_IDS_CONTENT_TYPE,
798
- });
799
- }
800
-
801
- if (serializedComments.peopleXml || state.sourcePeoplePartPath) {
802
- exportSession.replaceOwnedPart({
803
- path: peoplePartPath,
804
- bytes: new TextEncoder().encode(
805
- serializedComments.peopleXml ?? `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>\n<w15:people xmlns:w15="http://schemas.microsoft.com/office/word/2012/wordml"></w15:people>`,
806
- ),
807
- contentType:
808
- state.sourcePackage.parts.get(peoplePartPath)?.contentType ??
809
- PEOPLE_CONTENT_TYPE,
810
- });
811
- }
812
-
813
- if (exportedSubParts) {
814
- for (const header of exportedSubParts.headers) {
815
- exportSession.replaceOwnedPart({
816
- path: header.partPath,
817
- bytes: new TextEncoder().encode(serializeHeaderXml(header)),
818
- contentType:
819
- state.sourcePackage.parts.get(header.partPath)?.contentType ?? WORD_HEADER_CONTENT_TYPE,
820
- });
821
- }
822
- for (const footer of exportedSubParts.footers) {
823
- exportSession.replaceOwnedPart({
824
- path: footer.partPath,
825
- bytes: new TextEncoder().encode(serializeFooterXml(footer)),
826
- contentType:
827
- state.sourcePackage.parts.get(footer.partPath)?.contentType ?? WORD_FOOTER_CONTENT_TYPE,
828
- });
829
- }
830
- if (exportedSubParts.footnoteCollection) {
831
- if (state.sourceSubPartPaths.footnotesPartPath) {
832
- exportSession.replaceOwnedPart({
833
- path: state.sourceSubPartPaths.footnotesPartPath,
834
- bytes: new TextEncoder().encode(serializeFootnotesXml(exportedSubParts.footnoteCollection)),
835
- contentType:
836
- state.sourcePackage.parts.get(state.sourceSubPartPaths.footnotesPartPath)?.contentType ??
837
- WORD_FOOTNOTES_CONTENT_TYPE,
838
- });
839
- }
840
- if (state.sourceSubPartPaths.endnotesPartPath) {
841
- exportSession.replaceOwnedPart({
842
- path: state.sourceSubPartPaths.endnotesPartPath,
843
- bytes: new TextEncoder().encode(serializeEndnotesXml(exportedSubParts.footnoteCollection)),
844
- contentType:
845
- state.sourcePackage.parts.get(state.sourceSubPartPaths.endnotesPartPath)?.contentType ??
846
- WORD_ENDNOTES_CONTENT_TYPE,
847
- });
848
- }
849
- }
850
- if (exportedSubParts.theme && state.sourceSubPartPaths.themePartPath) {
851
- const sourceThemePart = state.sourcePackage.parts.get(state.sourceSubPartPaths.themePartPath);
852
- if (sourceThemePart) {
853
- exportSession.replaceOwnedPart({
854
- path: state.sourceSubPartPaths.themePartPath,
855
- bytes: sourceThemePart.bytes,
856
- contentType: sourceThemePart.contentType,
857
- relationships: sourceThemePart.relationships,
858
- compression: sourceThemePart.compression,
859
- });
860
- }
861
- }
862
- }
863
-
864
- return {
865
- bytes: exportSession.serialize(),
866
- mimeType: DOCX_MIME_TYPE,
867
- fileName: options?.fileName ?? `${snapshot.documentId}.docx`,
868
- };
869
- }
870
-
871
- function createImportedCanonicalDocument(input: {
872
- documentId: string;
873
- timestamp: string;
874
- numbering: CanonicalDocumentEnvelope["numbering"];
875
- media: CanonicalDocumentEnvelope["media"];
876
- content: CanonicalDocumentEnvelope["content"];
877
- subParts?: SubPartsCatalog;
878
- preservation: CanonicalDocumentEnvelope["preservation"];
879
- diagnostics: CanonicalDocumentEnvelope["diagnostics"];
880
- review: CanonicalDocumentEnvelope["review"];
881
- }): CanonicalDocumentEnvelope {
882
- return {
883
- schemaVersion: "cds/1.0.0",
884
- docId: createCanonicalDocumentId(input.documentId),
885
- createdAt: input.timestamp,
886
- updatedAt: input.timestamp,
887
- metadata: {
888
- customProperties: {},
889
- },
890
- styles: {
891
- paragraphs: {},
892
- characters: {},
893
- tables: {},
894
- },
895
- numbering: input.numbering,
896
- media: input.media,
897
- content: input.content,
898
- review: input.review,
899
- preservation: input.preservation,
900
- diagnostics: input.diagnostics,
901
- ...(input.subParts !== undefined ? { subParts: input.subParts } : {}),
902
- };
903
- }
904
-
905
- function createImportedSnapshot(input: {
906
- documentId: string;
907
- editorBuild: string;
908
- timestamp: string;
909
- document: CanonicalDocumentEnvelope;
910
- compatibility: PersistedEditorSnapshot["compatibility"];
911
- }): PersistedEditorSnapshot {
912
- return {
913
- snapshotVersion: "persisted-editor-snapshot/1",
914
- schemaVersion: input.document.schemaVersion,
915
- documentId: input.documentId,
916
- docId: input.document.docId,
917
- createdAt: input.document.createdAt,
918
- updatedAt: input.document.updatedAt,
919
- savedAt: input.timestamp,
920
- editorBuild: input.editorBuild,
921
- canonicalDocument: input.document,
922
- compatibility: input.compatibility,
923
- warningLog: input.compatibility.warnings,
924
- };
925
- }
926
-
927
- function toPublicAnchorProjection(
928
- anchor: InternalEditorAnchorProjection,
929
- ): PublicEditorAnchorProjection {
930
- switch (anchor.kind) {
931
- case "range":
932
- return {
933
- kind: "range",
934
- from: anchor.range.from,
935
- to: anchor.range.to,
936
- assoc: anchor.assoc,
937
- };
938
- case "node":
939
- return {
940
- kind: "node",
941
- at: anchor.at,
942
- assoc: anchor.assoc,
943
- };
944
- case "detached":
945
- return {
946
- kind: "detached",
947
- lastKnownRange: anchor.lastKnownRange,
948
- reason: anchor.reason,
949
- };
950
- }
951
- }
952
-
953
- function toPublicCompatibilityFeatureEntry(entry: InternalCompatibilityFeatureEntry) {
954
- return {
955
- ...entry,
956
- affectedAnchor: entry.affectedAnchor
957
- ? toPublicAnchorProjection(entry.affectedAnchor)
958
- : undefined,
959
- };
960
- }
961
-
962
- function toPublicWarning(warning: InternalEditorWarning): PublicEditorWarning {
963
- return {
964
- ...warning,
965
- affectedAnchor: warning.affectedAnchor
966
- ? toPublicAnchorProjection(warning.affectedAnchor)
967
- : undefined,
968
- };
969
- }
970
-
971
- function toPublicError(error: InternalEditorError): EditorError {
972
- return { ...error };
973
- }
974
-
975
- function toPublicCompatibilityReport(
976
- report: InternalCompatibilityReport,
977
- ): PublicCompatibilityReport {
978
- return {
979
- reportVersion: report.reportVersion,
980
- generatedAt: report.generatedAt,
981
- blockExport: report.blockExport,
982
- featureEntries: report.featureEntries.map((entry) =>
983
- toPublicCompatibilityFeatureEntry(entry),
984
- ),
985
- warnings: report.warnings.map((warning) => toPublicWarning(warning)),
986
- errors: report.errors.map((error) => toPublicError(error)),
987
- };
988
- }
989
-
990
- function createDiagnosticsSession(
991
- options: LoadDocxEditorSessionOptions,
992
- diagnostics: ImportDiagnosticsResult,
993
- ): LoadedDocxEditorSession {
994
- const timestamp = new Date().toISOString();
995
- const runtime = createReadOnlyDiagnosticsRuntime({
996
- documentId: options.documentId,
997
- sourceLabel: options.sourceLabel,
998
- editorBuild: options.editorBuild,
999
- generatedAt: timestamp,
1000
- diagnostics,
1001
- });
1002
- const initialSnapshot = runtime.getPersistedSnapshot();
1003
-
1004
- return {
1005
- initialSnapshot,
1006
- fatalError: diagnostics.fatalError,
1007
- readOnly: true,
1008
- exportDocx: async (_snapshot, exportOptions) => runtime.exportDocx(exportOptions),
1009
- };
1010
- }
1011
-
1012
- function createImportDiagnosticsFromError(error: unknown): ImportDiagnosticsResult {
1013
- if (isPackageImportError(error)) {
1014
- return createPackageImportDiagnostics({
1015
- issue: classifyCorruptPackageError(error),
1016
- });
1017
- }
1018
-
1019
- return createValidationImportDiagnostics({
1020
- message:
1021
- error instanceof Error
1022
- ? error.message
1023
- : "DOCX import failed during validation.",
1024
- });
1025
- }
1026
-
1027
- function normalizeImportedRevisionRecords(
1028
- parsed: ParsedRevisionsResult,
1029
- content: CanonicalDocumentEnvelope["content"],
1030
- opaqueFragments: Record<string, OpaqueFragmentRecord>,
1031
- ): ParsedRevisionsResult {
1032
- const opaqueRanges = Object.values(opaqueFragments).map((fragment) => fragment.lastKnownRange);
1033
- const paragraphRanges = collectCanonicalParagraphRanges(content);
1034
- if (opaqueRanges.length === 0) {
1035
- return {
1036
- ...parsed,
1037
- revisions: parsed.revisions.map((revision) => {
1038
- if (revision.anchor.kind !== "range" || revision.metadata.preserveOnlyReason) {
1039
- return revision;
1040
- }
1041
-
1042
- const preserveOnlyReason = getStructuralPreserveOnlyReason(
1043
- revision,
1044
- paragraphRanges,
1045
- );
1046
- if (!preserveOnlyReason) {
1047
- return revision;
1048
- }
1049
-
1050
- return {
1051
- ...revision,
1052
- metadata: {
1053
- ...revision.metadata,
1054
- preserveOnlyReason,
1055
- },
1056
- };
1057
- }),
1058
- };
1059
- }
1060
-
1061
- return {
1062
- ...parsed,
1063
- revisions: parsed.revisions.map((revision) => {
1064
- if (revision.anchor.kind !== "range" || revision.metadata.preserveOnlyReason) {
1065
- return revision;
1066
- }
1067
-
1068
- const preserveOnlyReason =
1069
- getStructuralPreserveOnlyReason(revision, paragraphRanges) ??
1070
- (opaqueRanges.some((range) => rangesIntersect(range, revision.anchor.range))
1071
- ? "Imported revision overlaps preserve-only OOXML and remains preserve-only."
1072
- : undefined);
1073
-
1074
- if (!preserveOnlyReason) {
1075
- return revision;
1076
- }
1077
-
1078
- return {
1079
- ...revision,
1080
- metadata: {
1081
- ...revision.metadata,
1082
- preserveOnlyReason,
1083
- },
1084
- };
1085
- }),
1086
- };
1087
- }
1088
-
1089
- function normalizeImportedCommentThreads(
1090
- parsed: ParsedCommentsResult,
1091
- opaqueFragments: Record<string, OpaqueFragmentRecord>,
1092
- revisions: readonly ReviewRevisionRecord[],
1093
- ): NormalizedImportedCommentsResult {
1094
- const opaqueRanges = Object.values(opaqueFragments).map((fragment) => fragment.lastKnownRange);
1095
- const preserveOnlyRevisionRanges = revisions
1096
- .filter(
1097
- (revision) =>
1098
- revision.anchor.kind === "range" &&
1099
- typeof revision.metadata.preserveOnlyReason === "string" &&
1100
- revision.metadata.preserveOnlyReason.length > 0,
1101
- )
1102
- .map((revision) => revision.anchor.range);
1103
- const preserveOnlyCommentIds = new Set(parsed.diagnostics.map((diagnostic) => diagnostic.commentId));
1104
- const additionalDiagnostics: CommentImportDiagnostic[] = [];
1105
- const normalizedThreads = parsed.threads.map((thread) => {
1106
- if (thread.anchor.kind !== "range") {
1107
- preserveOnlyCommentIds.add(thread.commentId);
1108
- return thread;
1109
- }
1110
-
1111
- const opaqueOverlap = opaqueRanges.some((range) => rangesIntersect(range, thread.anchor.range));
1112
- if (opaqueOverlap) {
1113
- preserveOnlyCommentIds.add(thread.commentId);
1114
- additionalDiagnostics.push({
1115
- commentId: thread.commentId,
1116
- code: "opaque_anchor_preserve_only",
1117
- message:
1118
- "Comment anchor intersects preserve-only OOXML and remains preserve-only on export.",
1119
- featureClass: "preserve-only",
1120
- });
1121
- return {
1122
- ...thread,
1123
- anchor: createDetachedAnchor(thread.anchor.range, "importAmbiguity"),
1124
- status: "detached",
1125
- };
1126
- }
1127
-
1128
- const preserveOnlyRevisionOverlap = preserveOnlyRevisionRanges.some((range) =>
1129
- rangesIntersect(range, thread.anchor.range),
1130
- );
1131
- if (preserveOnlyRevisionOverlap) {
1132
- preserveOnlyCommentIds.add(thread.commentId);
1133
- additionalDiagnostics.push({
1134
- commentId: thread.commentId,
1135
- code: "preserve_only_revision_overlap",
1136
- message:
1137
- "Comment anchor overlaps preserve-only review markup and remains preserve-only on export.",
1138
- featureClass: "preserve-only",
1139
- });
1140
- return {
1141
- ...thread,
1142
- anchor: createDetachedAnchor(thread.anchor.range, "importAmbiguity"),
1143
- status: "detached",
1144
- };
1145
- }
1146
-
1147
- return thread;
1148
- });
1149
-
1150
- return {
1151
- ...parsed,
1152
- threads: normalizedThreads,
1153
- diagnostics: [...parsed.diagnostics, ...additionalDiagnostics],
1154
- definitions: parsed.definitions,
1155
- preservedDefinitions: parsed.definitions.filter((definition) =>
1156
- preserveOnlyCommentIds.has(
1157
- resolveDefinitionRootCommentId(definition, parsed.definitions),
1158
- ),
1159
- ),
1160
- };
1161
- }
1162
-
1163
- function resolveDefinitionRootCommentId(
1164
- definition: ImportedCommentDefinition,
1165
- definitions: readonly ImportedCommentDefinition[],
1166
- ): string {
1167
- if (!definition.parentParaId) {
1168
- return definition.commentId;
1169
- }
1170
-
1171
- const definitionsByParaId = new Map(
1172
- definitions
1173
- .filter((candidate) => typeof candidate.paraId === "string")
1174
- .map((candidate) => [candidate.paraId!, candidate]),
1175
- );
1176
- const visited = new Set<string>();
1177
- let current: ImportedCommentDefinition | undefined = definition;
1178
-
1179
- while (current?.parentParaId) {
1180
- if (visited.has(current.parentParaId)) {
1181
- break;
1182
- }
1183
- visited.add(current.parentParaId);
1184
- const parent = definitionsByParaId.get(current.parentParaId);
1185
- if (!parent) {
1186
- break;
1187
- }
1188
- current = parent;
1189
- }
1190
-
1191
- return current?.commentId ?? definition.commentId;
1192
- }
1193
-
1194
- function getStructuralPreserveOnlyReason(
1195
- revision: ReviewRevisionRecord,
1196
- paragraphRanges: ReadonlyArray<{ start: number; end: number }>,
1197
- ): string | undefined {
1198
- const form = revision.metadata.importedRevisionForm;
1199
- if (!form || revision.anchor.kind !== "range") {
1200
- return undefined;
1201
- }
1202
-
1203
- if (
1204
- (form === "run-insertion" || form === "run-deletion") &&
1205
- revision.anchor.range.from === revision.anchor.range.to
1206
- ) {
1207
- return "Imported zero-width run revision remains preserve-only.";
1208
- }
1209
-
1210
- if (form === "paragraph-insertion" || form === "paragraph-deletion") {
1211
- const paragraphBoundary = paragraphRanges.find(
1212
- (boundary) =>
1213
- boundary.end === revision.anchor.range.from ||
1214
- (revision.anchor.range.from >= boundary.start &&
1215
- revision.anchor.range.from <= boundary.end),
1216
- );
1217
- return paragraphBoundary
1218
- ? undefined
1219
- : "Imported revision spans paragraph-level structure and remains preserve-only.";
1220
- }
1221
-
1222
- const paragraphBoundary = paragraphRanges.find(
1223
- (boundary) =>
1224
- revision.anchor.range.from >= boundary.start &&
1225
- revision.anchor.range.to <= boundary.end,
1226
- );
1227
- return paragraphBoundary
1228
- ? undefined
1229
- : "Imported revision spans structural boundaries and remains preserve-only.";
1230
- }
1231
-
1232
- function collectCanonicalParagraphRanges(
1233
- content: CanonicalDocumentEnvelope["content"],
1234
- ): Array<{ start: number; end: number }> {
1235
- const ranges: Array<{ start: number; end: number }> = [];
1236
- let cursor = 0;
1237
- let previousWasParagraph = false;
1238
-
1239
- for (const block of content.children) {
1240
- if (block.type === "paragraph") {
1241
- if (previousWasParagraph) {
1242
- cursor += 1;
1243
- }
1244
- const start = cursor;
1245
- cursor += measureCanonicalParagraph(block);
1246
- ranges.push({ start, end: cursor });
1247
- previousWasParagraph = true;
1248
- continue;
1249
- }
1250
-
1251
- cursor += 1;
1252
- previousWasParagraph = false;
1253
- }
1254
-
1255
- return ranges;
1256
- }
1257
-
1258
- function measureCanonicalParagraph(paragraph: CanonicalDocumentEnvelope["content"]["children"][number] & { type: "paragraph" }): number {
1259
- return paragraph.children.reduce((size, child) => {
1260
- switch (child.type) {
1261
- case "text":
1262
- return size + child.text.length;
1263
- case "tab":
1264
- case "hard_break":
1265
- case "image":
1266
- case "opaque_inline":
1267
- return size + 1;
1268
- case "hyperlink":
1269
- return (
1270
- size +
1271
- child.children.reduce((childSize, entry) => {
1272
- switch (entry.type) {
1273
- case "text":
1274
- return childSize + entry.text.length;
1275
- case "tab":
1276
- case "hard_break":
1277
- return childSize + 1;
1278
- }
1279
- }, 0)
1280
- );
1281
- }
1282
- }, 0);
1283
- }
1284
-
1285
- function rangesIntersect(
1286
- left: { from: number; to: number },
1287
- right: { from: number; to: number },
1288
- ): boolean {
1289
- return left.from < right.to && right.from < left.to;
1290
- }
1291
-
1292
- function resolveCommentsPartPath(
1293
- sourcePackage: OpcPackage,
1294
- relationships: readonly OpcRelationship[],
1295
- ): string | undefined {
1296
- return resolveDocumentRelatedPartPath(
1297
- sourcePackage,
1298
- relationships,
1299
- COMMENTS_RELATIONSHIP_TYPE,
1300
- COMMENTS_PART_PATH,
1301
- );
1302
- }
1303
-
1304
- function resolveDocumentRelatedPartPath(
1305
- sourcePackage: OpcPackage,
1306
- relationships: readonly OpcRelationship[],
1307
- relationshipType: string,
1308
- fallbackPartPath: string,
1309
- ): string | undefined {
1310
- const relationship = relationships.find(
1311
- (candidate) =>
1312
- candidate.type === relationshipType &&
1313
- candidate.targetMode === "internal",
1314
- );
1315
- if (!relationship) {
1316
- return sourcePackage.parts.has(fallbackPartPath) ? fallbackPartPath : undefined;
1317
- }
1318
-
1319
- const targetPath = resolveRelationshipTarget(MAIN_DOCUMENT_PATH, relationship);
1320
- return sourcePackage.parts.has(targetPath) ? targetPath : undefined;
1321
- }
1322
-
1323
- function toRuntimeCommentRecords(
1324
- threads: readonly CommentThread[],
1325
- ): Record<string, CommentThreadRecord> {
1326
- return Object.fromEntries(
1327
- threads.map((thread) => {
1328
- return [
1329
- thread.commentId,
1330
- {
1331
- commentId: thread.commentId,
1332
- body: thread.entries.map((entry) => entry.body).join("\n"),
1333
- anchor: thread.anchor,
1334
- createdBy: thread.createdBy,
1335
- authorId: thread.createdBy,
1336
- createdAt: thread.createdAt,
1337
- entries: thread.entries.map((entry) => ({
1338
- entryId: entry.entryId,
1339
- authorId: entry.authorId,
1340
- body: entry.body,
1341
- createdAt: entry.createdAt,
1342
- metadata: entry.metadata
1343
- ? {
1344
- ooxmlCommentId: entry.metadata.ooxmlCommentId,
1345
- paraId: entry.metadata.paraId,
1346
- parentParaId: entry.metadata.parentParaId,
1347
- durableId: entry.metadata.durableId,
1348
- initials: entry.metadata.initials,
1349
- }
1350
- : undefined,
1351
- })),
1352
- status: thread.status,
1353
- resolution: thread.resolution
1354
- ? {
1355
- resolvedAt: thread.resolution.resolvedAt,
1356
- resolvedBy: thread.resolution.resolvedBy,
1357
- }
1358
- : undefined,
1359
- resolvedAt: thread.resolution?.resolvedAt,
1360
- warningIds: [...thread.warningIds],
1361
- isResolved: thread.status === "resolved",
1362
- metadata: thread.metadata
1363
- ? {
1364
- source: thread.metadata.source,
1365
- rootOoxmlCommentId: thread.metadata.rootOoxmlCommentId,
1366
- rootParaId: thread.metadata.rootParaId,
1367
- }
1368
- : undefined,
1369
- } satisfies CommentThreadRecord,
1370
- ];
1371
- }),
1372
- );
1373
- }
1374
-
1375
- function toRuntimeRevisionRecords(
1376
- revisions: readonly ReviewRevisionRecord[],
1377
- ): Record<string, RuntimeRevisionRecord> {
1378
- return Object.fromEntries(
1379
- revisions.map((revision) => [
1380
- revision.revisionId,
1381
- {
1382
- changeId: revision.revisionId,
1383
- kind: revision.kind,
1384
- anchor: revision.anchor,
1385
- authorId: revision.authorId,
1386
- createdAt: revision.createdAt,
1387
- warningIds: [...revision.warningIds],
1388
- metadata: {
1389
- source: revision.metadata.source,
1390
- preserveOnlyReason: revision.metadata.preserveOnlyReason,
1391
- importedRevisionForm: revision.metadata.importedRevisionForm,
1392
- originalRevisionType: revision.metadata.originalRevisionType,
1393
- ooxmlRevisionId: revision.metadata.ooxmlRevisionId,
1394
- },
1395
- status: revision.status === "active" ? "open" : revision.status,
1396
- } satisfies RuntimeRevisionRecord,
1397
- ]),
1398
- );
1399
- }
1400
-
1401
- function toReviewRevisionRecords(
1402
- revisions: CanonicalDocumentEnvelope["review"]["revisions"],
1403
- ): ReviewRevisionRecord[] {
1404
- return Object.values(revisions).map((revision) => ({
1405
- revisionId: revision.changeId,
1406
- kind: revision.kind,
1407
- anchor: revision.anchor,
1408
- authorId: revision.authorId ?? "unknown",
1409
- createdAt: revision.createdAt,
1410
- warningIds: [...(revision.warningIds ?? [])],
1411
- metadata: {
1412
- source: revision.metadata?.source ?? "runtime",
1413
- preserveOnlyReason: revision.metadata?.preserveOnlyReason,
1414
- importedRevisionForm: revision.metadata?.importedRevisionForm,
1415
- originalRevisionType: revision.metadata?.originalRevisionType,
1416
- ooxmlRevisionId: revision.metadata?.ooxmlRevisionId,
1417
- },
1418
- status: revision.status === "open" ? "active" : revision.status,
1419
- }));
1420
- }
1421
-
1422
- export function stripCommentMarkup(
1423
- documentXml: string,
1424
- ownedCommentIds: readonly string[],
1425
- ): string {
1426
- if (ownedCommentIds.length === 0) {
1427
- return documentXml;
1428
- }
1429
-
1430
- let output = documentXml;
1431
- for (const commentId of ownedCommentIds) {
1432
- const escapedCommentId = escapeRegExp(commentId);
1433
- const attributePattern = `(?:w:id|id)=["']${escapedCommentId}["']`;
1434
- output = output
1435
- .replace(
1436
- new RegExp(`<w:commentRangeStart\\b[^>]*${attributePattern}[^>]*/>`, "gu"),
1437
- "",
1438
- )
1439
- .replace(
1440
- new RegExp(`<w:commentRangeEnd\\b[^>]*${attributePattern}[^>]*/>`, "gu"),
1441
- "",
1442
- )
1443
- .replace(
1444
- new RegExp(
1445
- `<w:r\\b[^>]*>(?:(?!<w:t\\b|</w:r>)[\\s\\S])*?<w:commentReference\\b[^>]*${attributePattern}[^>]*/>(?:(?!<w:t\\b|</w:r>)[\\s\\S])*?</w:r>`,
1446
- "gu",
1447
- ),
1448
- "",
1449
- );
1450
- }
1451
-
1452
- return output;
1453
- }
1454
-
1455
- function escapeRegExp(value: string): string {
1456
- return value.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
1457
- }
1458
-
1459
- function withDocumentRelatedParts(
1460
- relationships: readonly OpcRelationship[],
1461
- relatedParts: ReadonlyArray<{
1462
- relationshipType: string;
1463
- partPath: string;
1464
- existingRelationshipId?: string;
1465
- include: boolean;
1466
- }>,
1467
- ): OpcRelationship[] {
1468
- const filteredTypes = new Set(relatedParts.map((part) => part.relationshipType));
1469
- const nextRelationships = relationships
1470
- .filter((relationship) => !filteredTypes.has(relationship.type))
1471
- .map(cloneRelationship);
1472
-
1473
- for (const part of relatedParts) {
1474
- if (!part.include) {
1475
- continue;
1476
- }
1477
-
1478
- nextRelationships.push({
1479
- id: part.existingRelationshipId ?? createCommentsRelationshipId(nextRelationships),
1480
- type: part.relationshipType,
1481
- target: toDocumentRelativeTarget(part.partPath),
1482
- targetMode: "internal",
1483
- });
1484
- }
1485
-
1486
- return nextRelationships;
1487
- }
1488
-
1489
- function collectPreservedPackageParts(
1490
- sourcePackage: OpcPackage,
1491
- ownedPartPaths: ReadonlyArray<string | undefined>,
1492
- ): Record<string, PreservedPackagePart> {
1493
- const normalizedOwnedPaths = new Set(
1494
- ownedPartPaths
1495
- .filter((value): value is string => typeof value === "string")
1496
- .map((path) => normalizePartPath(path)),
1497
- );
1498
- return Object.fromEntries(
1499
- [...sourcePackage.parts.values()]
1500
- .filter((part) =>
1501
- shouldPreservePackagePart(part.path, part.surfaceKind, normalizedOwnedPaths),
1502
- )
1503
- .map((part) => [
1504
- part.path,
1505
- {
1506
- packagePartName: part.path,
1507
- contentType: part.contentType ?? "application/octet-stream",
1508
- relationshipIds: findRelationshipIdsTargetingPart(sourcePackage, part.path),
1509
- } satisfies PreservedPackagePart,
1510
- ]),
1511
- );
1512
- }
1513
-
1514
- function collectInlineMediaParts(
1515
- sourcePackage: OpcPackage,
1516
- ): ReadonlyMap<string, { path: string; contentType: string }> {
1517
- return new Map(
1518
- [...sourcePackage.parts.values()]
1519
- .filter(
1520
- (part) =>
1521
- part.path.startsWith("/word/media/") && typeof part.contentType === "string",
1522
- )
1523
- .map((part) => [
1524
- part.path,
1525
- {
1526
- path: part.path,
1527
- contentType: part.contentType ?? "application/octet-stream",
1528
- },
1529
- ]),
1530
- );
1531
- }
1532
-
1533
- function createEmptyNumberingCatalog(): NumberingCatalog {
1534
- return {
1535
- abstractDefinitions: {},
1536
- instances: {},
1537
- };
1538
- }
1539
-
1540
- function hasNumberingEntries(catalog: NumberingCatalog): boolean {
1541
- return (
1542
- Object.keys(catalog.abstractDefinitions ?? {}).length > 0 ||
1543
- Object.keys(catalog.instances ?? {}).length > 0
1544
- );
1545
- }
1546
-
1547
- function collectBrokenInternalRelationshipIssues(
1548
- sourcePackage: OpcPackage,
1549
- ): ReturnType<typeof createBrokenRelationshipIssue>[] {
1550
- const brokenTargets = new Map<string, ReturnType<typeof createBrokenRelationshipIssue>>();
1551
-
1552
- for (const relationship of sourcePackage.manifest.packageRelationships) {
1553
- if (relationship.targetMode !== "internal") {
1554
- continue;
1555
- }
1556
-
1557
- const target = resolveRelationshipTarget(null, relationship);
1558
- if (!sourcePackage.parts.has(target)) {
1559
- brokenTargets.set(
1560
- `package:${relationship.id}:${target}`,
1561
- createBrokenRelationshipIssue({
1562
- relationshipSourcePath: null,
1563
- relationshipId: relationship.id,
1564
- targetPartPath: target,
1565
- }),
1566
- );
1567
- }
1568
- }
1569
-
1570
- for (const part of sourcePackage.parts.values()) {
1571
- for (const relationship of part.relationships) {
1572
- if (relationship.targetMode !== "internal") {
1573
- continue;
1574
- }
1575
-
1576
- const target = resolveRelationshipTarget(part.path, relationship);
1577
- if (!sourcePackage.parts.has(target)) {
1578
- brokenTargets.set(
1579
- `${part.path}:${relationship.id}:${target}`,
1580
- createBrokenRelationshipIssue({
1581
- relationshipSourcePath: part.path,
1582
- relationshipId: relationship.id,
1583
- targetPartPath: target,
1584
- }),
1585
- );
1586
- }
1587
- }
1588
- }
1589
-
1590
- return [...brokenTargets.values()].sort((left, right) =>
1591
- `${left.relationshipSourcePath ?? ""}:${left.relationshipId}:${left.targetPartPath}`.localeCompare(
1592
- `${right.relationshipSourcePath ?? ""}:${right.relationshipId}:${right.targetPartPath}`,
1593
- ),
1594
- );
1595
- }
1596
-
1597
- function summarizeBrokenRelationshipIssues(
1598
- issues: readonly ReturnType<typeof createBrokenRelationshipIssue>[],
1599
- ): string {
1600
- return `DOCX package has unresolved internal relationships: ${issues
1601
- .map((issue) => issue.targetPartPath ?? issue.message)
1602
- .slice(0, 3)
1603
- .join(", ")}${issues.length > 3 ? ", ..." : ""}.`;
1604
- }
1605
-
1606
- function shouldPreservePackagePart(
1607
- partPath: string,
1608
- surfaceKind: OpcPackage["parts"] extends Map<string, infer T>
1609
- ? T extends { surfaceKind: infer U }
1610
- ? U
1611
- : never
1612
- : never,
1613
- ownedPartPaths: ReadonlySet<string>,
1614
- ): boolean {
1615
- if (surfaceKind !== "content") {
1616
- return false;
1617
- }
1618
-
1619
- if (
1620
- partPath === MAIN_DOCUMENT_PATH ||
1621
- ownedPartPaths.has(partPath) ||
1622
- partPath.startsWith("/word/media/") ||
1623
- CORE_NON_PRESERVED_PART_PATHS.has(partPath)
1624
- ) {
1625
- return false;
1626
- }
1627
-
1628
- return true;
1629
- }
1630
-
1631
- function findRelationshipIdsTargetingPart(
1632
- sourcePackage: OpcPackage,
1633
- targetPartPath: string,
1634
- ): string[] {
1635
- const ids = new Set<string>();
1636
-
1637
- for (const relationship of sourcePackage.manifest.packageRelationships) {
1638
- if (
1639
- relationship.targetMode === "internal" &&
1640
- resolveRelationshipTarget(null, relationship) === targetPartPath
1641
- ) {
1642
- ids.add(relationship.id);
1643
- }
1644
- }
1645
-
1646
- for (const part of sourcePackage.parts.values()) {
1647
- for (const relationship of part.relationships) {
1648
- if (
1649
- relationship.targetMode === "internal" &&
1650
- resolveRelationshipTarget(part.path, relationship) === targetPartPath
1651
- ) {
1652
- ids.add(relationship.id);
1653
- }
1654
- }
1655
- }
1656
-
1657
- return [...ids].sort();
1658
- }
1659
-
1660
- function extractDocumentRootAttributes(documentXml: string): Record<string, string> {
1661
- const match = documentXml.match(
1662
- /<(?:[A-Za-z_][A-Za-z0-9:._-]*:)?document\b([^>]*)>/u,
1663
- );
1664
- if (!match) {
1665
- return {};
1666
- }
1667
-
1668
- const attributes: Record<string, string> = {};
1669
- const pattern = /([A-Za-z_][A-Za-z0-9:._-]*)\s*=\s*("([^"]*)"|'([^']*)')/gu;
1670
- for (const attributeMatch of (match[1] ?? "").matchAll(pattern)) {
1671
- const name = attributeMatch[1];
1672
- const value = attributeMatch[3] ?? attributeMatch[4] ?? "";
1673
- if (!name) {
1674
- continue;
1675
- }
1676
- attributes[name] = value;
1677
- }
1678
-
1679
- return attributes;
1680
- }
1681
-
1682
- function isPackageImportError(error: unknown): boolean {
1683
- if (!(error instanceof Error)) {
1684
- return false;
1685
- }
1686
-
1687
- const normalized = error.message.toLowerCase();
1688
- return (
1689
- normalized.includes("zip") ||
1690
- normalized.includes("opc package") ||
1691
- normalized.includes("compression") ||
1692
- normalized.includes("relationship") ||
1693
- normalized.includes("/[content_types].xml") ||
1694
- normalized.includes("/word/document.xml") ||
1695
- normalized.includes("xml")
1696
- );
1697
- }
1698
-
1699
- const CORE_NON_PRESERVED_PART_PATHS = new Set([
1700
- "/docProps/app.xml",
1701
- "/docProps/core.xml",
1702
- "/docProps/custom.xml",
1703
- "/word/fontTable.xml",
1704
- "/word/numbering.xml",
1705
- "/word/settings.xml",
1706
- "/word/styles.xml",
1707
- "/word/stylesWithEffects.xml",
1708
- "/word/webSettings.xml",
1709
- ]);
1710
-
1711
- function createCommentsRelationshipId(
1712
- relationships: readonly OpcRelationship[],
1713
- ): string {
1714
- let nextIndex = 1;
1715
- while (relationships.some((relationship) => relationship.id === `rIdComments${nextIndex}`)) {
1716
- nextIndex += 1;
1717
- }
1718
-
1719
- return `rIdComments${nextIndex}`;
1720
- }
1721
-
1722
- function toDocumentRelativeTarget(partPath: string): string {
1723
- const normalized = normalizePartPath(partPath);
1724
- return normalized.startsWith("/word/") ? normalized.slice("/word/".length) : normalized.slice(1);
1725
- }
1726
-
1727
- function cloneRelationship(relationship: OpcRelationship): OpcRelationship {
1728
- return { ...relationship };
1729
- }
1730
-
1731
- function decodeUtf8(bytes: Uint8Array | undefined): string {
1732
- if (!bytes) {
1733
- return "";
1734
- }
1735
-
1736
- return Buffer.from(bytes.buffer, bytes.byteOffset, bytes.byteLength).toString("utf8");
1737
- }
1738
-
1739
- function toUint8Array(bytes: Uint8Array | ArrayBuffer): Uint8Array {
1740
- return bytes instanceof Uint8Array ? bytes : new Uint8Array(bytes);
1741
- }
1742
-
1743
- function serializeCanonicalDocumentForExport(document: CanonicalDocumentEnvelope): string {
1744
- return createCanonicalDocumentSignature(document);
1745
- }
1746
-
1747
- function canReuseSourceBytesForCurrentDocument(
1748
- state: ImportedDocxState,
1749
- document: CanonicalDocumentEnvelope,
1750
- ): boolean {
1751
- const commentThreads = Object.values(document.review.comments);
1752
- const hasLiveComments = commentThreads.some((thread) => thread.anchor.kind !== "detached");
1753
- if (!hasLiveComments) {
1754
- return true;
1755
- }
1756
-
1757
- return Boolean(
1758
- state.sourceCommentsPartPath &&
1759
- state.sourceCommentsExtendedPartPath &&
1760
- state.sourceCommentsIdsPartPath &&
1761
- state.sourcePeoplePartPath,
1762
- );
1763
- }