@beyondwork/docx-react-component 1.0.0 → 1.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +44 -104
- package/package.json +66 -15
- package/src/api/public-types.ts +1 -1
- package/src/compare/diff-engine.ts +530 -0
- package/src/compare/export-redlines.ts +162 -0
- package/src/compare/snapshot.ts +37 -0
- package/src/core/commands/index.ts +1 -1
- package/src/core/state/editor-state.ts +2 -2
- package/src/index.ts +45 -0
- package/src/legal/bookmarks.ts +196 -0
- package/src/legal/cross-references.ts +356 -0
- package/src/legal/defined-terms.ts +203 -0
- package/src/runtime/document-runtime.ts +3 -5
- package/src/runtime/table-commands.ts +4 -1
- package/src/runtime/table-schema.ts +17 -2
- package/src/runtime/virtualized-rendering.ts +258 -0
- package/src/ui/WordReviewEditor.tsx +256 -35
- package/src/ui-tailwind/editor-surface/tw-editor-surface.tsx +2 -2
- package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +16 -2
- package/.codex/config.toml +0 -5
- package/.corepack/v1/pnpm/10.30.3/.corepack +0 -1
- package/.corepack/v1/pnpm/10.30.3/LICENSE +0 -22
- package/.corepack/v1/pnpm/10.30.3/README.md +0 -240
- package/.corepack/v1/pnpm/10.30.3/dist/node-gyp-bin/node-gyp +0 -6
- package/.corepack/v1/pnpm/10.30.3/dist/node-gyp-bin/node-gyp.cmd +0 -5
- package/.corepack/v1/pnpm/10.30.3/dist/pnpm.cjs +0 -195400
- package/.corepack/v1/pnpm/10.30.3/dist/pnpmrc +0 -2
- package/.corepack/v1/pnpm/10.30.3/dist/reflink.darwin-arm64-2HJ4WGO6.node +0 -0
- package/.corepack/v1/pnpm/10.30.3/dist/reflink.darwin-x64-3G3H6IW4.node +0 -0
- package/.corepack/v1/pnpm/10.30.3/dist/reflink.win32-arm64-msvc-Q6BARPPB.node +0 -0
- package/.corepack/v1/pnpm/10.30.3/dist/reflink.win32-x64-msvc-J2TZHRQI.node +0 -0
- package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.bash +0 -31
- package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.fish +0 -22
- package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.ps1 +0 -193
- package/.corepack/v1/pnpm/10.30.3/dist/templates/completion.zsh +0 -27
- package/.corepack/v1/pnpm/10.30.3/dist/vendor/fastlist-0.3.0-x64.exe +0 -0
- package/.corepack/v1/pnpm/10.30.3/dist/vendor/fastlist-0.3.0-x86.exe +0 -0
- package/.corepack/v1/pnpm/10.30.3/dist/worker.js +0 -10119
- package/.corepack/v1/pnpm/10.30.3/package.json +0 -192
- package/.cursor/mcp.json +0 -7
- package/.github/workflows/ci.yml +0 -35
- package/.mcp.json +0 -7
- package/.openclaw/workspace-state.json +0 -4
- package/.pnpmrc.json +0 -1
- package/.wave-launch.sh +0 -7
- package/.workspace-marker +0 -1
- package/AGENTS.md +0 -78
- package/CHANGELOG.md +0 -177
- package/DESIGN.md +0 -929
- package/HEARTBEAT.md +0 -7
- package/IDENTITY.md +0 -23
- package/SOUL.md +0 -36
- package/TOOLS.md +0 -40
- package/USER.md +0 -17
- package/docs/README.md +0 -107
- package/docs/agents/wave-cont-eval-role.md +0 -36
- package/docs/agents/wave-cont-qa-role.md +0 -52
- package/docs/agents/wave-deploy-verifier-role.md +0 -34
- package/docs/agents/wave-design-role.md +0 -47
- package/docs/agents/wave-documentation-role.md +0 -34
- package/docs/agents/wave-infra-role.md +0 -34
- package/docs/agents/wave-integration-role.md +0 -37
- package/docs/agents/wave-launcher-role.md +0 -41
- package/docs/agents/wave-orchestrator-role.md +0 -52
- package/docs/agents/wave-planner-role.md +0 -39
- package/docs/agents/wave-security-role.md +0 -40
- package/docs/architecture/docx/README.md +0 -10
- package/docs/architecture/future/README.md +0 -8
- package/docs/architecture/ooxml-upgrade-analysis.md +0 -134
- package/docs/architecture/platform/shared-openxml-editor-platform.md +0 -153
- package/docs/architecture/xlsx/canonical-workbook-model-and-commands.md +0 -187
- package/docs/architecture/xlsx/spreadsheet-editor-frontend-architecture.md +0 -150
- package/docs/comment-redline-overview.md +0 -350
- package/docs/concepts/context7-vs-skills.md +0 -118
- package/docs/concepts/operating-modes.md +0 -91
- package/docs/concepts/runtime-agnostic-orchestration.md +0 -111
- package/docs/concepts/what-is-a-wave.md +0 -217
- package/docs/context7/bundles.json +0 -222
- package/docs/context7/planner-agent/README.md +0 -28
- package/docs/context7/planner-agent/manifest.json +0 -83
- package/docs/context7/planner-agent/papers/cooperbench-why-coding-agents-cannot-be-your-teammates-yet.md +0 -3283
- package/docs/context7/planner-agent/papers/dova-deliberation-first-multi-agent-orchestration-for-autonomous-research-automation.md +0 -1699
- package/docs/context7/planner-agent/papers/dpbench-large-language-models-struggle-with-simultaneous-coordination.md +0 -2251
- package/docs/context7/planner-agent/papers/incremental-planning-to-control-a-blackboard-based-problem-solver.md +0 -1729
- package/docs/context7/planner-agent/papers/silo-bench-a-scalable-environment-for-evaluating-distributed-coordination-in-multi-agent-llm-systems.md +0 -3747
- package/docs/context7/planner-agent/papers/todoevolve-learning-to-architect-agent-planning-systems.md +0 -1675
- package/docs/context7/planner-agent/papers/verified-multi-agent-orchestration-a-plan-execute-verify-replan-framework-for-complex-query-resolution.md +0 -1173
- package/docs/context7/planner-agent/papers/why-do-multi-agent-llm-systems-fail.md +0 -5211
- package/docs/context7/planner-agent/topics/planning-and-orchestration.md +0 -24
- package/docs/evals/arm-templates/README.md +0 -13
- package/docs/evals/arm-templates/full-wave.json +0 -15
- package/docs/evals/arm-templates/single-agent.json +0 -15
- package/docs/evals/benchmark-catalog.json +0 -670
- package/docs/evals/cases/README.md +0 -47
- package/docs/evals/cases/wave-blackboard-inbox-targeting.json +0 -73
- package/docs/evals/cases/wave-contradiction-conflict.json +0 -104
- package/docs/evals/cases/wave-expert-routing-preservation.json +0 -69
- package/docs/evals/cases/wave-hidden-profile-private-evidence.json +0 -81
- package/docs/evals/cases/wave-premature-closure-guard.json +0 -71
- package/docs/evals/cases/wave-silo-cross-agent-state.json +0 -77
- package/docs/evals/cases/wave-simultaneous-lockstep.json +0 -92
- package/docs/evals/external-benchmarks.json +0 -85
- package/docs/evals/external-command-config.sample.json +0 -9
- package/docs/evals/external-command-config.swe-bench-pro.json +0 -8
- package/docs/evals/pilots/README.md +0 -47
- package/docs/evals/pilots/swe-bench-pro-public-full-wave-review-10.json +0 -64
- package/docs/evals/pilots/swe-bench-pro-public-pilot.json +0 -111
- package/docs/evals/wave-benchmark-program.md +0 -302
- package/docs/guides/planner.md +0 -220
- package/docs/guides/recommendations-0.8.9.md +0 -133
- package/docs/guides/signal-wrappers.md +0 -165
- package/docs/guides/terminal-surfaces.md +0 -96
- package/docs/image copy.png +0 -0
- package/docs/image.png +0 -0
- package/docs/images/image.png +0 -0
- package/docs/legal-feedback-architecture.md +0 -498
- package/docs/plans/component-cutover-matrix.json +0 -1072
- package/docs/plans/component-cutover-matrix.md +0 -307
- package/docs/plans/context7-wave-orchestrator.md +0 -155
- package/docs/plans/current-state.md +0 -198
- package/docs/plans/docx/README.md +0 -9
- package/docs/plans/examples/wave-benchmark-improvement.md +0 -108
- package/docs/plans/examples/wave-example-live-proof.md +0 -435
- package/docs/plans/master-plan.md +0 -224
- package/docs/plans/migration.md +0 -538
- package/docs/plans/operations/README.md +0 -7
- package/docs/plans/operations/wave-10-word-certification.md +0 -87
- package/docs/plans/operations/wave-8-railway-staging.md +0 -153
- package/docs/plans/operations/wave-9-manual-certification.md +0 -73
- package/docs/plans/platform/README.md +0 -9
- package/docs/plans/reference/legal-checklist-coverage.md +0 -258
- package/docs/plans/wave-orchestrator.md +0 -423
- package/docs/plans/waves/README.md +0 -75
- package/docs/plans/waves/completed/wave-0.md +0 -195
- package/docs/plans/waves/completed/wave-1.md +0 -379
- package/docs/plans/waves/completed/wave-10.md +0 -670
- package/docs/plans/waves/completed/wave-11.md +0 -335
- package/docs/plans/waves/completed/wave-12.md +0 -417
- package/docs/plans/waves/completed/wave-13.md +0 -316
- package/docs/plans/waves/completed/wave-14.md +0 -319
- package/docs/plans/waves/completed/wave-15.md +0 -321
- package/docs/plans/waves/completed/wave-16.md +0 -316
- package/docs/plans/waves/completed/wave-17.md +0 -331
- package/docs/plans/waves/completed/wave-18.md +0 -328
- package/docs/plans/waves/completed/wave-2.md +0 -438
- package/docs/plans/waves/completed/wave-3.md +0 -435
- package/docs/plans/waves/completed/wave-4.md +0 -430
- package/docs/plans/waves/completed/wave-5.md +0 -430
- package/docs/plans/waves/completed/wave-6.md +0 -430
- package/docs/plans/waves/completed/wave-7.md +0 -526
- package/docs/plans/waves/completed/wave-8.md +0 -596
- package/docs/plans/waves/completed/wave-9.md +0 -552
- package/docs/plans/waves/deferred/README.md +0 -14
- package/docs/plans/waves/deferred/encrypted-intake-contracts.md +0 -282
- package/docs/plans/waves/deferred/legal-feedback-wave-expansion.md +0 -308
- package/docs/plans/waves/deferred/wave-encrypted-intake.md +0 -451
- package/docs/plans/waves/design/README.md +0 -5
- package/docs/plans/waves/design/wave-1-a1.md +0 -309
- package/docs/plans/waves/reviews/README.md +0 -5
- package/docs/plans/waves/reviews/wave-0-cont-qa.md +0 -151
- package/docs/plans/waves/reviews/wave-1-cont-qa.md +0 -46
- package/docs/plans/waves/reviews/wave-10-accessibility-and-design.md +0 -51
- package/docs/plans/waves/reviews/wave-10-cont-qa.md +0 -24
- package/docs/plans/waves/reviews/wave-10-dashboard-proof.md +0 -46
- package/docs/plans/waves/reviews/wave-10-performance-signoff.md +0 -55
- package/docs/plans/waves/reviews/wave-10-regression-proof.md +0 -23
- package/docs/plans/waves/reviews/wave-10-release-audit.md +0 -31
- package/docs/plans/waves/reviews/wave-10-service-proof.md +0 -83
- package/docs/plans/waves/reviews/wave-10-word-certification.md +0 -31
- package/docs/plans/waves/reviews/wave-18-ai-contract-closure.md +0 -277
- package/docs/plans/waves/reviews/wave-18-cont-qa.md +0 -255
- package/docs/plans/waves/reviews/wave-18-parity-proof.md +0 -271
- package/docs/plans/waves/reviews/wave-19-cont-qa.md +0 -59
- package/docs/plans/waves/reviews/wave-2-cont-qa.md +0 -72
- package/docs/plans/waves/reviews/wave-20-cont-qa.md +0 -60
- package/docs/plans/waves/reviews/wave-25-cont-qa.md +0 -48
- package/docs/plans/waves/reviews/wave-28-cont-qa.md +0 -46
- package/docs/plans/waves/reviews/wave-29-cont-qa.md +0 -53
- package/docs/plans/waves/reviews/wave-3-cont-qa.md +0 -53
- package/docs/plans/waves/reviews/wave-3-core-proof.md +0 -77
- package/docs/plans/waves/reviews/wave-3-validator-proof.md +0 -73
- package/docs/plans/waves/reviews/wave-32-cont-qa.md +0 -43
- package/docs/plans/waves/reviews/wave-33-cont-qa.md +0 -526
- package/docs/plans/waves/reviews/wave-34-cont-qa.md +0 -100
- package/docs/plans/waves/reviews/wave-35-cont-qa.md +0 -145
- package/docs/plans/waves/reviews/wave-4-cont-qa.md +0 -47
- package/docs/plans/waves/reviews/wave-4-structure-proof.md +0 -69
- package/docs/plans/waves/reviews/wave-5-comment-proof.md +0 -158
- package/docs/plans/waves/reviews/wave-5-cont-qa.md +0 -68
- package/docs/plans/waves/reviews/wave-6-cont-qa.md +0 -416
- package/docs/plans/waves/reviews/wave-6-redline-proof.md +0 -130
- package/docs/plans/waves/reviews/wave-7-cont-qa.md +0 -82
- package/docs/plans/waves/reviews/wave-7-ooxml-compliance.md +0 -85
- package/docs/plans/waves/reviews/wave-7-preservation-proof.md +0 -119
- package/docs/plans/waves/reviews/wave-7-trust-ux.md +0 -87
- package/docs/plans/waves/reviews/wave-8-accessibility-and-design.md +0 -128
- package/docs/plans/waves/reviews/wave-8-cont-qa.md +0 -92
- package/docs/plans/waves/reviews/wave-8-live-proof.md +0 -140
- package/docs/plans/waves/reviews/wave-8-security.md +0 -47
- package/docs/plans/waves/reviews/wave-9-editor-embedding.md +0 -39
- package/docs/plans/waves/reviews/wave-9-fixture-runner.md +0 -56
- package/docs/plans/waves/reviews/wave-9-live-proof.md +0 -105
- package/docs/plans/waves/reviews/wave-9-usability-and-performance.md +0 -152
- package/docs/plans/waves/specs/README.md +0 -5
- package/docs/plans/waves/specs/wave-1-component-boundaries.md +0 -322
- package/docs/plans/waves/specs/wave-1-ooxml-contracts.md +0 -323
- package/docs/plans/waves/specs/wave-1-review-and-ui-contracts.md +0 -339
- package/docs/plans/waves/specs/wave-1-runtime-contracts.md +0 -509
- package/docs/plans/waves/wave-19.md +0 -341
- package/docs/plans/waves/wave-20.md +0 -308
- package/docs/plans/waves/wave-21.md +0 -289
- package/docs/plans/waves/wave-22.md +0 -221
- package/docs/plans/waves/wave-23.md +0 -295
- package/docs/plans/waves/wave-24.md +0 -286
- package/docs/plans/waves/wave-25.md +0 -313
- package/docs/plans/waves/wave-26.md +0 -300
- package/docs/plans/waves/wave-27.md +0 -299
- package/docs/plans/waves/wave-28.md +0 -368
- package/docs/plans/waves/wave-29.md +0 -303
- package/docs/plans/waves/wave-30.md +0 -307
- package/docs/plans/waves/wave-31.md +0 -231
- package/docs/plans/waves/wave-32.md +0 -152
- package/docs/plans/waves/wave-33.md +0 -147
- package/docs/plans/waves/wave-34.md +0 -148
- package/docs/plans/waves/wave-35.md +0 -141
- package/docs/plans/waves/wave-36.md +0 -146
- package/docs/plans/xlsx/README.md +0 -14
- package/docs/plans/xlsx/xlsx-fixture-corpus-and-certification-plan.md +0 -126
- package/docs/reference/cli-reference.md +0 -600
- package/docs/reference/coordination-and-closure.md +0 -487
- package/docs/reference/deep-research-report (15).md +0 -25
- package/docs/reference/docx/README.md +0 -10
- package/docs/reference/legal-checklist.md +0 -445
- package/docs/reference/live-proof-waves.md +0 -199
- package/docs/reference/ooxml-compliance.md +0 -129
- package/docs/reference/ooxml-feature-parity-matrix.md +0 -172
- package/docs/reference/platform/shared-ooxml-platform-guidance.md +0 -77
- package/docs/reference/prototype-agent-prompt-legal-fidelity.md +0 -155
- package/docs/reference/public-api.md +0 -456
- package/docs/reference/repository-guidance.md +0 -58
- package/docs/reference/runtime-config/README.md +0 -182
- package/docs/reference/runtime-config/claude.md +0 -110
- package/docs/reference/runtime-config/codex.md +0 -82
- package/docs/reference/runtime-config/opencode.md +0 -93
- package/docs/reference/sample-waves.md +0 -105
- package/docs/reference/skills.md +0 -237
- package/docs/reference/templates/AGENTS.md +0 -78
- package/docs/reference/templates/HEARTBEAT.md +0 -7
- package/docs/reference/templates/IDENTITY.md +0 -23
- package/docs/reference/templates/SOUL.md +0 -36
- package/docs/reference/templates/TOOLS.md +0 -40
- package/docs/reference/templates/USER.md +0 -17
- package/docs/reference/wave-control.md +0 -184
- package/docs/reference/wave-planning-lessons.md +0 -167
- package/docs/reference/word-review-editor-frontend-architecture.md +0 -479
- package/docs/reference/word-review-editor-ux-guide.md +0 -253
- package/docs/reference/xlsx/xlsx-ooxml-compliance.md +0 -137
- package/docs/research/agent-context-sources.md +0 -178
- package/docs/research/coordination-failure-review.md +0 -290
- package/docs/research/docx-react-component/Canonical Document Schema Specification for a React-based Word-compatible Editor.md +0 -2317
- package/docs/research/docx-react-component/Feature Compatibility Matrix for a React Word Compatible Legal Editor v1.md +0 -219
- package/docs/research/docx-react-component/React Component Architecture and Front-End Structure Specification for a Word-Compatible Legal Review Editor.md +0 -1112
- package/docs/research/docx-react-component/document_compatibility_and_testing_spec.md +0 -751
- package/docs/research/xlsx/raw/README.md +0 -13
- package/docs/roadmap.md +0 -174
- package/docs/superpowers/plans/2026-03-28-harness-control-bar.md +0 -677
- package/docs/superpowers/specs/2026-03-28-harness-control-bar-design.md +0 -274
- package/docs/xlsx-react/README.md +0 -38
- package/docs/xlsx-react/agent-llm-interaction-layer-docx-xlsx.md +0 -621
- package/docs/xlsx-react/canonical-workbook-model-and-commands.md +0 -948
- package/docs/xlsx-react/shared-openxml-editor-platform-docx-xlsx.md +0 -228
- package/docs/xlsx-react/spreadsheet-editor-component-architecture.md +0 -809
- package/docs/xlsx-react/spreadsheet-editor-frontend-architecture.md +0 -537
- package/docs/xlsx-react/spreadsheet-editor-ux-guide.md +0 -520
- package/docs/xlsx-react/xlsx-editor-research-pack.md +0 -871
- package/docs/xlsx-react/xlsx-fixture-corpus-and-certification-plan.md +0 -436
- package/docs/xlsx-react/xlsx-ooxml-compliance.md +0 -320
- package/examples/README.md +0 -16
- package/memory/MEMORY.md +0 -24
- package/pnpm-workspace.yaml +0 -4
- package/scripts/check-no-authored-js.sh +0 -13
- package/scripts/context7-api-check.sh +0 -65
- package/scripts/context7-export-env.sh +0 -42
- package/scripts/run-context7-mcp.sh +0 -8
- package/scripts/run-workspace-tests.sh +0 -15
- package/scripts/start-wave-10-local.sh +0 -189
- package/scripts/wave-agent-attach.sh +0 -47
- package/scripts/wave-auto-answer.sh +0 -118
- package/scripts/wave-dashboard-attach.sh +0 -13
- package/scripts/wave-launch.sh +0 -273
- package/scripts/wave-overnight-supervisor.sh +0 -145
- package/scripts/wave-status.sh +0 -379
- package/scripts/wave-watch.sh +0 -231
- package/services/README.md +0 -17
- package/services/openxml-validator/Dockerfile +0 -29
- package/services/openxml-validator/OpenXmlValidator.Api.csproj +0 -12
- package/services/openxml-validator/Program.cs +0 -436
- package/services/openxml-validator/README.md +0 -152
- package/services/openxml-validator/railway.json +0 -16
- package/services/react-word-editor/.tmp-a4/src/api/public-types.ts +0 -318
- package/services/react-word-editor/.tmp-a4/src/ui/WordReviewEditor.tsx +0 -1302
- package/services/react-word-editor/.tmp-a4/src/ui/editor-surface/editor-surface.tsx +0 -546
- package/services/react-word-editor/.tmp-a4/test/ui/word-review-editor.test.tsx +0 -146
- package/services/react-word-editor/.tmp-a4-build/src/api/public-types.js +0 -2
- package/services/react-word-editor/.tmp-a4-build/src/ui/WordReviewEditor.js +0 -818
- package/services/react-word-editor/.tmp-a4-build/src/ui/editor-surface/editor-surface.js +0 -229
- package/services/react-word-editor/.tmp-a4-build/test/ui/word-review-editor.test.js +0 -121
- package/services/react-word-editor/.tmp-wave-4-a3-tsconfig.json +0 -21
- package/services/react-word-editor/.tmp-wave-4-a3-tsconfig.tsbuildinfo +0 -1
- package/services/react-word-editor/Dockerfile +0 -26
- package/services/react-word-editor/README.md +0 -254
- package/services/react-word-editor/app/api/certification/route.ts +0 -79
- package/services/react-word-editor/app/api/demo-sessions/route.ts +0 -109
- package/services/react-word-editor/app/api/deploy-health/route.ts +0 -23
- package/services/react-word-editor/app/api/exports/[exportId]/route.ts +0 -34
- package/services/react-word-editor/app/api/exports/route.ts +0 -81
- package/services/react-word-editor/app/api/fixtures/[fixtureId]/run/route.ts +0 -100
- package/services/react-word-editor/app/api/health/route.ts +0 -70
- package/services/react-word-editor/app/api/runs/[runId]/route.ts +0 -36
- package/services/react-word-editor/app/api/scenarios/[scenarioId]/run/route.ts +0 -85
- package/services/react-word-editor/app/api/sessions/[sessionId]/route.ts +0 -199
- package/services/react-word-editor/app/api/sessions/[sessionId]/source/route.ts +0 -45
- package/services/react-word-editor/app/api/uploads/route.ts +0 -70
- package/services/react-word-editor/app/api/validate/route.ts +0 -310
- package/services/react-word-editor/app/certification/[runId]/page.tsx +0 -14
- package/services/react-word-editor/app/certification/page.tsx +0 -32
- package/services/react-word-editor/app/dashboard/page.tsx +0 -7
- package/services/react-word-editor/app/demo/page.tsx +0 -30
- package/services/react-word-editor/app/demo/prototype-client.tsx +0 -1080
- package/services/react-word-editor/app/editor/[sessionId]/page.tsx +0 -33
- package/services/react-word-editor/app/fixtures/page.tsx +0 -7
- package/services/react-word-editor/app/globals.css +0 -121
- package/services/react-word-editor/app/layout.tsx +0 -32
- package/services/react-word-editor/app/page.tsx +0 -30
- package/services/react-word-editor/app/runs/[runId]/page.tsx +0 -34
- package/services/react-word-editor/app/wave-10-word-review/page.tsx +0 -7
- package/services/react-word-editor/components/harness-control-bar.tsx +0 -289
- package/services/react-word-editor/components/harness-editor-session-client.tsx +0 -1214
- package/services/react-word-editor/components/harness-workspace-page.tsx +0 -715
- package/services/react-word-editor/components/reduced-motion-toggle.tsx +0 -79
- package/services/react-word-editor/components/workspace-certification-panel.tsx +0 -307
- package/services/react-word-editor/lib/certification-bundle.ts +0 -796
- package/services/react-word-editor/lib/certification-store.ts +0 -661
- package/services/react-word-editor/lib/demo-fixtures.test.mjs +0 -195
- package/services/react-word-editor/lib/demo-fixtures.ts +0 -1519
- package/services/react-word-editor/lib/editor-session-summary.test.mjs +0 -68
- package/services/react-word-editor/lib/editor-session-summary.ts +0 -14
- package/services/react-word-editor/lib/editor-session.ts +0 -228
- package/services/react-word-editor/lib/exports-route.test.mjs +0 -32
- package/services/react-word-editor/lib/harness-client.ts +0 -347
- package/services/react-word-editor/lib/harness-config.json +0 -30
- package/services/react-word-editor/lib/harness-config.test.mjs +0 -31
- package/services/react-word-editor/lib/harness-config.ts +0 -21
- package/services/react-word-editor/lib/harness-editor-datastore.test.mjs +0 -220
- package/services/react-word-editor/lib/harness-editor-datastore.ts +0 -161
- package/services/react-word-editor/lib/private-mode.test.mjs +0 -42
- package/services/react-word-editor/lib/private-mode.ts +0 -61
- package/services/react-word-editor/lib/regression-report.test.mjs +0 -352
- package/services/react-word-editor/lib/regression-report.ts +0 -896
- package/services/react-word-editor/lib/run-artifacts.ts +0 -934
- package/services/react-word-editor/lib/run-history.ts +0 -755
- package/services/react-word-editor/lib/scenario-artifacts.test.mjs +0 -41
- package/services/react-word-editor/lib/scenario-artifacts.ts +0 -44
- package/services/react-word-editor/lib/storage.ts +0 -953
- package/services/react-word-editor/lib/validator-client.test.mjs +0 -54
- package/services/react-word-editor/lib/validator-client.ts +0 -95
- package/services/react-word-editor/lib/workspace-navigation.ts +0 -79
- package/services/react-word-editor/middleware.ts +0 -35
- package/services/react-word-editor/next-env.d.ts +0 -6
- package/services/react-word-editor/next.config.mjs +0 -15
- package/services/react-word-editor/package.json +0 -38
- package/services/react-word-editor/postcss.config.mjs +0 -8
- package/services/react-word-editor/railway.json +0 -21
- package/services/react-word-editor/scripts/wave-10-certification.mjs +0 -101
- package/services/react-word-editor/scripts/wave-9-live-usability-pilot.mjs +0 -911
- package/services/react-word-editor/tsconfig.json +0 -39
- package/services/react-word-editor/tsconfig.tsbuildinfo +0 -1
- package/skills/README.md +0 -48
- package/skills/domain-docx-compatibility/SKILL.md +0 -44
- package/skills/domain-docx-compatibility/skill.json +0 -19
- package/skills/domain-editor-architecture/SKILL.md +0 -49
- package/skills/domain-editor-architecture/skill.json +0 -19
- package/skills/domain-legal-review/SKILL.md +0 -39
- package/skills/domain-legal-review/skill.json +0 -19
- package/skills/provider-aws/SKILL.md +0 -117
- package/skills/provider-aws/adapters/claude.md +0 -1
- package/skills/provider-aws/adapters/codex.md +0 -1
- package/skills/provider-aws/references/service-verification.md +0 -39
- package/skills/provider-aws/skill.json +0 -54
- package/skills/provider-custom-deploy/SKILL.md +0 -64
- package/skills/provider-custom-deploy/skill.json +0 -50
- package/skills/provider-docker-compose/SKILL.md +0 -96
- package/skills/provider-docker-compose/adapters/local.md +0 -1
- package/skills/provider-docker-compose/skill.json +0 -53
- package/skills/provider-github-release/SKILL.md +0 -121
- package/skills/provider-github-release/adapters/claude.md +0 -1
- package/skills/provider-github-release/adapters/codex.md +0 -1
- package/skills/provider-github-release/skill.json +0 -55
- package/skills/provider-kubernetes/SKILL.md +0 -143
- package/skills/provider-kubernetes/adapters/claude.md +0 -1
- package/skills/provider-kubernetes/adapters/codex.md +0 -1
- package/skills/provider-kubernetes/references/kubectl-patterns.md +0 -58
- package/skills/provider-kubernetes/skill.json +0 -52
- package/skills/provider-railway/SKILL.md +0 -123
- package/skills/provider-railway/adapters/claude.md +0 -1
- package/skills/provider-railway/adapters/codex.md +0 -1
- package/skills/provider-railway/adapters/local.md +0 -1
- package/skills/provider-railway/adapters/opencode.md +0 -1
- package/skills/provider-railway/references/verification-commands.md +0 -39
- package/skills/provider-railway/skill.json +0 -71
- package/skills/provider-ssh-manual/SKILL.md +0 -97
- package/skills/provider-ssh-manual/skill.json +0 -54
- package/skills/repo-coding-rules/SKILL.md +0 -55
- package/skills/repo-coding-rules/skill.json +0 -34
- package/skills/role-cont-eval/SKILL.md +0 -91
- package/skills/role-cont-eval/adapters/codex.md +0 -1
- package/skills/role-cont-eval/skill.json +0 -36
- package/skills/role-cont-qa/SKILL.md +0 -100
- package/skills/role-cont-qa/adapters/claude.md +0 -1
- package/skills/role-cont-qa/skill.json +0 -36
- package/skills/role-deploy/SKILL.md +0 -97
- package/skills/role-deploy/skill.json +0 -36
- package/skills/role-design/SKILL.md +0 -50
- package/skills/role-design/skill.json +0 -36
- package/skills/role-documentation/SKILL.md +0 -76
- package/skills/role-documentation/skill.json +0 -36
- package/skills/role-implementation/SKILL.md +0 -45
- package/skills/role-implementation/skill.json +0 -36
- package/skills/role-infra/SKILL.md +0 -81
- package/skills/role-infra/skill.json +0 -36
- package/skills/role-integration/SKILL.md +0 -91
- package/skills/role-integration/skill.json +0 -36
- package/skills/role-planner/SKILL.md +0 -39
- package/skills/role-planner/skill.json +0 -21
- package/skills/role-research/SKILL.md +0 -65
- package/skills/role-research/skill.json +0 -36
- package/skills/role-security/SKILL.md +0 -60
- package/skills/role-security/skill.json +0 -36
- package/skills/runtime-claude/SKILL.md +0 -66
- package/skills/runtime-claude/skill.json +0 -36
- package/skills/runtime-codex/SKILL.md +0 -58
- package/skills/runtime-codex/skill.json +0 -36
- package/skills/runtime-local/SKILL.md +0 -46
- package/skills/runtime-local/skill.json +0 -36
- package/skills/runtime-opencode/SKILL.md +0 -58
- package/skills/runtime-opencode/skill.json +0 -36
- package/skills/signal-hygiene/SKILL.md +0 -51
- package/skills/signal-hygiene/skill.json +0 -20
- package/skills/tui-design/SKILL.md +0 -77
- package/skills/tui-design/references/tui-design.md +0 -259
- package/skills/tui-design/skill.json +0 -36
- package/skills/wave-core/SKILL.md +0 -141
- package/skills/wave-core/references/marker-syntax.md +0 -70
- package/skills/wave-core/skill.json +0 -35
- package/test/README.md +0 -16
- package/test/core/formatting-commands.test.ts +0 -285
- package/test/core/image-commands.test.ts +0 -298
- package/test/core/mapping.test.ts +0 -186
- package/test/core/text-commands.test.ts +0 -176
- package/test/fixtures/docx/F01-basic-contract.docx +0 -0
- package/test/fixtures/docx/F01-basic-contract.md +0 -33
- package/test/fixtures/docx/F02-headings-styles.docx +0 -0
- package/test/fixtures/docx/F02-headings-styles.md +0 -33
- package/test/fixtures/docx/F03-legal-outline-numbering.docx +0 -0
- package/test/fixtures/docx/F03-legal-outline-numbering.md +0 -34
- package/test/fixtures/docx/F04-restart-numbering-schedules.docx +0 -0
- package/test/fixtures/docx/F04-restart-numbering-schedules.md +0 -33
- package/test/fixtures/docx/F05-table-heavy-agreement.docx +0 -0
- package/test/fixtures/docx/F05-table-heavy-agreement.md +0 -34
- package/test/fixtures/docx/F06-merged-cells-signature-table.docx +0 -0
- package/test/fixtures/docx/F06-merged-cells-signature-table.md +0 -34
- package/test/fixtures/docx/F07-inline-images-exhibit.docx +0 -0
- package/test/fixtures/docx/F07-inline-images-exhibit.md +0 -34
- package/test/fixtures/docx/F08-hyperlinks.docx +0 -0
- package/test/fixtures/docx/F08-hyperlinks.md +0 -33
- package/test/fixtures/docx/F09-comments-single-paragraph.docx +0 -0
- package/test/fixtures/docx/F09-comments-single-paragraph.md +0 -33
- package/test/fixtures/docx/F10-threaded-comments-resolve.docx +0 -0
- package/test/fixtures/docx/F10-threaded-comments-resolve.md +0 -33
- package/test/fixtures/docx/F11-redlines-basic.docx +0 -0
- package/test/fixtures/docx/F11-redlines-basic.md +0 -33
- package/test/fixtures/docx/F12-redlines-paragraph-joins-splits.docx +0 -0
- package/test/fixtures/docx/F12-redlines-paragraph-joins-splits.md +0 -33
- package/test/fixtures/docx/F13-comments-on-deleted-text.docx +0 -0
- package/test/fixtures/docx/F13-comments-on-deleted-text.md +0 -33
- package/test/fixtures/docx/F14-revisions-in-tables-and-lists.docx +0 -0
- package/test/fixtures/docx/F14-revisions-in-tables-and-lists.md +0 -33
- package/test/fixtures/docx/F15-sections-headers-footers.docx +0 -0
- package/test/fixtures/docx/F15-sections-headers-footers.md +0 -33
- package/test/fixtures/docx/F16-footnotes-endnotes.docx +0 -0
- package/test/fixtures/docx/F16-footnotes-endnotes.md +0 -33
- package/test/fixtures/docx/F17-fields-and-toc.docx +0 -0
- package/test/fixtures/docx/F17-fields-and-toc.md +0 -33
- package/test/fixtures/docx/F18-content-controls-template.docx +0 -0
- package/test/fixtures/docx/F18-content-controls-template.md +0 -33
- package/test/fixtures/docx/F19-custom-xml-doc-assembly.docx +0 -0
- package/test/fixtures/docx/F19-custom-xml-doc-assembly.md +0 -35
- package/test/fixtures/docx/F20-unknown-ooxml-and-alternatecontent.docx +0 -0
- package/test/fixtures/docx/F20-unknown-ooxml-and-alternatecontent.md +0 -33
- package/test/fixtures/docx/F21-malformed-broken-docx.docx +0 -0
- package/test/fixtures/docx/F21-malformed-broken-docx.md +0 -33
- package/test/fixtures/docx/README.md +0 -74
- package/test/fixtures/docx/certification-manifest.json +0 -104
- package/test/fixtures/docx/fixtures.manifest.json +0 -196
- package/test/fixtures/encrypted-docx/README.md +0 -27
- package/test/fixtures/encrypted-docx/certification-manifest.json +0 -9
- package/test/fixtures/encrypted-docx/fixtures.manifest.json +0 -47
- package/test/fixtures/scenarios/docx/README.md +0 -25
- package/test/fixtures/scenarios/docx/S01-sow-template.docx +0 -0
- package/test/fixtures/scenarios/docx/S01-sow-template.md +0 -30
- package/test/fixtures/scenarios/docx/S02-bw-partner-user-licence-agreement-redlines.docx +0 -0
- package/test/fixtures/scenarios/docx/S02-bw-partner-user-licence-agreement-redlines.md +0 -32
- package/test/fixtures/scenarios/docx/scenario-manifest.json +0 -53
- package/test/formats/xlsx/io/xlsx-import.test.ts +0 -766
- package/test/formats/xlsx/model/workbook.test.ts +0 -669
- package/test/helpers/dom-setup.ts +0 -124
- package/test/io/comment-roundtrip.test.ts +0 -272
- package/test/io/complex-content-roundtrip.test.ts +0 -632
- package/test/io/docx-compatibility-regression.test.ts +0 -199
- package/test/io/docx-session.test.ts +0 -1495
- package/test/io/footnotes-roundtrip.test.ts +0 -318
- package/test/io/headers-footers-roundtrip.test.ts +0 -547
- package/test/io/numbering-roundtrip.test.ts +0 -234
- package/test/io/package-reader.test.ts +0 -199
- package/test/io/paragraph-properties-roundtrip.test.ts +0 -129
- package/test/io/preserved-package-roundtrip.test.ts +0 -365
- package/test/io/property-completeness.test.ts +0 -292
- package/test/io/revision-roundtrip.test.ts +0 -347
- package/test/io/structural-blocks.test.ts +0 -202
- package/test/io/table-media-roundtrip.test.ts +0 -448
- package/test/io/table-properties-roundtrip.test.ts +0 -569
- package/test/io/table-roundtrip.test.ts +0 -302
- package/test/io/text-roundtrip.test.ts +0 -344
- package/test/model/canonical-document.test.ts +0 -285
- package/test/preservation/opaque-fragment-store.test.ts +0 -121
- package/test/preservation/package-preservation.test.ts +0 -395
- package/test/preservation/store.test.ts +0 -84
- package/test/review/comment-remapping.test.ts +0 -220
- package/test/review/comment-store.test.ts +0 -180
- package/test/review/move-revisions.test.ts +0 -143
- package/test/review/property-change-revisions.test.ts +0 -225
- package/test/review/revision-actions.test.ts +0 -330
- package/test/review/revision-store.test.ts +0 -193
- package/test/runtime/session-capabilities.test.ts +0 -260
- package/test/runtime/table-commands.test.ts +0 -356
- package/test/runtime/table-schema.test.ts +0 -221
- package/test/runtime/tracked-changes-toggle.test.ts +0 -107
- package/test/ui/comment-review-surface.test.tsx +0 -114
- package/test/ui/reduced-motion-toggle.test.tsx +0 -137
- package/test/ui/word-review-editor.imported-scenarios.test.tsx +0 -169
- package/test/ui/word-review-editor.interaction.test.tsx +0 -1198
- package/test/ui/word-review-editor.test.js +0 -188
- package/test/ui/word-review-editor.test.tsx +0 -280
- package/test/ui-tailwind/search-plugin.test.ts +0 -286
- package/test/validation/compatibility-engine.test.ts +0 -336
- package/test/validation/compatibility-report.test.ts +0 -189
- package/test/validation/low-priority-word-surfaces.test.ts +0 -282
- package/test/validation/malformed-doc.test.ts +0 -113
- package/test-results/.last-run.json +0 -4
- package/wave.config.json +0 -406
|
@@ -1,669 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Tests for the canonical workbook model.
|
|
3
|
-
*
|
|
4
|
-
* Covers:
|
|
5
|
-
* - workbook creation and metadata
|
|
6
|
-
* - sheet registry (add, remove, rename, move, get by name)
|
|
7
|
-
* - sparse cell storage (set, get, delete, extent)
|
|
8
|
-
* - all cell value kinds (blank, text, number, boolean, error, formula)
|
|
9
|
-
* - shared string table
|
|
10
|
-
* - merge range operations
|
|
11
|
-
* - style references
|
|
12
|
-
*/
|
|
13
|
-
|
|
14
|
-
import assert from "node:assert/strict";
|
|
15
|
-
import { mkdirSync, writeFileSync } from "node:fs";
|
|
16
|
-
import test from "node:test";
|
|
17
|
-
|
|
18
|
-
import {
|
|
19
|
-
makeBlankCell,
|
|
20
|
-
makeBooleanCell,
|
|
21
|
-
makeErrorCell,
|
|
22
|
-
makeFormulaCell,
|
|
23
|
-
makeNumberCell,
|
|
24
|
-
makeTextCell,
|
|
25
|
-
cellKey,
|
|
26
|
-
parseCellKey,
|
|
27
|
-
} from "../../../../src/formats/xlsx/model/cell.ts";
|
|
28
|
-
import type { CellValue } from "../../../../src/formats/xlsx/model/cell.ts";
|
|
29
|
-
|
|
30
|
-
import {
|
|
31
|
-
addMerge,
|
|
32
|
-
createSheet,
|
|
33
|
-
findMergesContaining,
|
|
34
|
-
getCell,
|
|
35
|
-
getSheetExtent,
|
|
36
|
-
iterateCellAddresses,
|
|
37
|
-
removeMerge,
|
|
38
|
-
setCell,
|
|
39
|
-
} from "../../../../src/formats/xlsx/model/sheet.ts";
|
|
40
|
-
|
|
41
|
-
import {
|
|
42
|
-
createEmptyStyleRegistry,
|
|
43
|
-
getXfRecord,
|
|
44
|
-
isDateFormatHeuristic,
|
|
45
|
-
} from "../../../../src/formats/xlsx/model/styles.ts";
|
|
46
|
-
import type { XfRecord } from "../../../../src/formats/xlsx/model/styles.ts";
|
|
47
|
-
|
|
48
|
-
import {
|
|
49
|
-
WORKBOOK_SCHEMA_VERSION,
|
|
50
|
-
addSheet,
|
|
51
|
-
appendSharedString,
|
|
52
|
-
createWorkbook,
|
|
53
|
-
getSheet,
|
|
54
|
-
getSheetByName,
|
|
55
|
-
listSheets,
|
|
56
|
-
moveSheet,
|
|
57
|
-
removeSheet,
|
|
58
|
-
renameSheet,
|
|
59
|
-
resolveSharedString,
|
|
60
|
-
workbookToJson,
|
|
61
|
-
workbookFromJson,
|
|
62
|
-
} from "../../../../src/formats/xlsx/model/workbook.ts";
|
|
63
|
-
|
|
64
|
-
// ---------------------------------------------------------------------------
|
|
65
|
-
// Proof bundle accumulator
|
|
66
|
-
// ---------------------------------------------------------------------------
|
|
67
|
-
|
|
68
|
-
const PROOF_DIR = ".tmp/wave-25-xlsx-proof";
|
|
69
|
-
const PROOF_PATH = `${PROOF_DIR}/model-tests.json`;
|
|
70
|
-
|
|
71
|
-
interface TestResult {
|
|
72
|
-
id: string;
|
|
73
|
-
description: string;
|
|
74
|
-
passed: boolean;
|
|
75
|
-
detail?: string;
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
const results: TestResult[] = [];
|
|
79
|
-
|
|
80
|
-
function record(id: string, description: string, passed: boolean, detail?: string): void {
|
|
81
|
-
results.push({ id, description, passed, detail });
|
|
82
|
-
}
|
|
83
|
-
|
|
84
|
-
// ---------------------------------------------------------------------------
|
|
85
|
-
// Cell value helpers
|
|
86
|
-
// ---------------------------------------------------------------------------
|
|
87
|
-
|
|
88
|
-
test("cell.ts: cellKey and parseCellKey are inverses", () => {
|
|
89
|
-
const key = cellKey(3, 7);
|
|
90
|
-
assert.equal(key, "3:7");
|
|
91
|
-
const addr = parseCellKey(key);
|
|
92
|
-
assert.deepEqual(addr, { row: 3, col: 7 });
|
|
93
|
-
record("cell-key-roundtrip", "cellKey / parseCellKey roundtrip", true);
|
|
94
|
-
});
|
|
95
|
-
|
|
96
|
-
test("cell.ts: parseCellKey throws on invalid input", () => {
|
|
97
|
-
assert.throws(() => parseCellKey("invalid"), /Invalid cell key/);
|
|
98
|
-
record("cell-key-invalid", "parseCellKey throws on bad input", true);
|
|
99
|
-
});
|
|
100
|
-
|
|
101
|
-
test("cell.ts: makeBlankCell produces correct kind", () => {
|
|
102
|
-
const cell = makeBlankCell();
|
|
103
|
-
assert.equal(cell.kind, "blank");
|
|
104
|
-
assert.equal(cell.styleRef, undefined);
|
|
105
|
-
record("blank-cell", "makeBlankCell produces blank kind", true);
|
|
106
|
-
});
|
|
107
|
-
|
|
108
|
-
test("cell.ts: makeBlankCell with styleRef", () => {
|
|
109
|
-
const cell = makeBlankCell({ xfIndex: 2 });
|
|
110
|
-
assert.equal(cell.kind, "blank");
|
|
111
|
-
assert.deepEqual(cell.styleRef, { xfIndex: 2 });
|
|
112
|
-
record("blank-cell-style", "makeBlankCell with styleRef", true);
|
|
113
|
-
});
|
|
114
|
-
|
|
115
|
-
test("cell.ts: makeTextCell produces correct value", () => {
|
|
116
|
-
const cell = makeTextCell("Hello", false);
|
|
117
|
-
assert.equal(cell.kind, "text");
|
|
118
|
-
assert.equal(cell.value, "Hello");
|
|
119
|
-
assert.equal(cell.fromSharedString, false);
|
|
120
|
-
record("text-cell", "makeTextCell produces text kind", true);
|
|
121
|
-
});
|
|
122
|
-
|
|
123
|
-
test("cell.ts: makeTextCell with fromSharedString flag", () => {
|
|
124
|
-
const cell = makeTextCell("World", true, { xfIndex: 1 });
|
|
125
|
-
assert.equal(cell.fromSharedString, true);
|
|
126
|
-
assert.deepEqual(cell.styleRef, { xfIndex: 1 });
|
|
127
|
-
record("text-cell-shared", "makeTextCell fromSharedString flag", true);
|
|
128
|
-
});
|
|
129
|
-
|
|
130
|
-
test("cell.ts: makeNumberCell produces correct value", () => {
|
|
131
|
-
const cell = makeNumberCell(42.5);
|
|
132
|
-
assert.equal(cell.kind, "number");
|
|
133
|
-
assert.equal(cell.value, 42.5);
|
|
134
|
-
record("number-cell", "makeNumberCell value", true);
|
|
135
|
-
});
|
|
136
|
-
|
|
137
|
-
test("cell.ts: makeBooleanCell produces correct value", () => {
|
|
138
|
-
const t = makeBooleanCell(true);
|
|
139
|
-
const f = makeBooleanCell(false);
|
|
140
|
-
assert.equal(t.kind, "boolean");
|
|
141
|
-
assert.equal(t.value, true);
|
|
142
|
-
assert.equal(f.value, false);
|
|
143
|
-
record("boolean-cell", "makeBooleanCell true/false", true);
|
|
144
|
-
});
|
|
145
|
-
|
|
146
|
-
test("cell.ts: makeErrorCell produces correct error code", () => {
|
|
147
|
-
const cell = makeErrorCell("#DIV/0!");
|
|
148
|
-
assert.equal(cell.kind, "error");
|
|
149
|
-
assert.equal(cell.errorCode, "#DIV/0!");
|
|
150
|
-
record("error-cell", "makeErrorCell error code", true);
|
|
151
|
-
});
|
|
152
|
-
|
|
153
|
-
test("cell.ts: makeFormulaCell with cached value", () => {
|
|
154
|
-
const cell = makeFormulaCell("SUM(A1:A10)", { type: "number", value: 123 });
|
|
155
|
-
assert.equal(cell.kind, "formula");
|
|
156
|
-
assert.equal(cell.formula, "SUM(A1:A10)");
|
|
157
|
-
assert.deepEqual(cell.cachedValue, { type: "number", value: 123 });
|
|
158
|
-
record("formula-cell-cached", "makeFormulaCell with cached value", true);
|
|
159
|
-
});
|
|
160
|
-
|
|
161
|
-
test("cell.ts: makeFormulaCell without cached value", () => {
|
|
162
|
-
const cell = makeFormulaCell("A1+B1");
|
|
163
|
-
assert.equal(cell.cachedValue, undefined);
|
|
164
|
-
record("formula-cell-no-cache", "makeFormulaCell without cached value", true);
|
|
165
|
-
});
|
|
166
|
-
|
|
167
|
-
// ---------------------------------------------------------------------------
|
|
168
|
-
// Sheet operations
|
|
169
|
-
// ---------------------------------------------------------------------------
|
|
170
|
-
|
|
171
|
-
test("sheet.ts: createSheet creates empty sheet", () => {
|
|
172
|
-
const sheet = createSheet("s1", "Sheet1", 0);
|
|
173
|
-
assert.equal(sheet.sheetId, "s1");
|
|
174
|
-
assert.equal(sheet.name, "Sheet1");
|
|
175
|
-
assert.equal(sheet.orderIndex, 0);
|
|
176
|
-
assert.equal(sheet.hidden, false);
|
|
177
|
-
assert.equal(sheet.cells.size, 0);
|
|
178
|
-
assert.equal(sheet.rowProps.size, 0);
|
|
179
|
-
assert.equal(sheet.colProps.size, 0);
|
|
180
|
-
assert.deepEqual(sheet.merges, []);
|
|
181
|
-
record("sheet-create", "createSheet produces empty sheet", true);
|
|
182
|
-
});
|
|
183
|
-
|
|
184
|
-
test("sheet.ts: setCell and getCell round-trip", () => {
|
|
185
|
-
const sheet = createSheet("s1", "Sheet1", 0);
|
|
186
|
-
const cell = makeNumberCell(99);
|
|
187
|
-
setCell(sheet, 0, 0, cell);
|
|
188
|
-
const retrieved = getCell(sheet, 0, 0);
|
|
189
|
-
assert.deepEqual(retrieved, cell);
|
|
190
|
-
record("cell-set-get", "setCell / getCell round-trip", true);
|
|
191
|
-
});
|
|
192
|
-
|
|
193
|
-
test("sheet.ts: getCell returns undefined for absent cell", () => {
|
|
194
|
-
const sheet = createSheet("s1", "Sheet1", 0);
|
|
195
|
-
assert.equal(getCell(sheet, 5, 5), undefined);
|
|
196
|
-
record("cell-get-absent", "getCell returns undefined for absent cell", true);
|
|
197
|
-
});
|
|
198
|
-
|
|
199
|
-
test("sheet.ts: setCell with undefined removes cell", () => {
|
|
200
|
-
const sheet = createSheet("s1", "Sheet1", 0);
|
|
201
|
-
setCell(sheet, 2, 3, makeTextCell("x", false));
|
|
202
|
-
setCell(sheet, 2, 3, undefined);
|
|
203
|
-
assert.equal(getCell(sheet, 2, 3), undefined);
|
|
204
|
-
assert.equal(sheet.cells.size, 0);
|
|
205
|
-
record("cell-delete", "setCell with undefined removes cell", true);
|
|
206
|
-
});
|
|
207
|
-
|
|
208
|
-
test("sheet.ts: getSheetExtent returns null for empty sheet", () => {
|
|
209
|
-
const sheet = createSheet("s1", "Sheet1", 0);
|
|
210
|
-
assert.equal(getSheetExtent(sheet), null);
|
|
211
|
-
record("extent-empty", "getSheetExtent null for empty sheet", true);
|
|
212
|
-
});
|
|
213
|
-
|
|
214
|
-
test("sheet.ts: getSheetExtent computes correct bounds", () => {
|
|
215
|
-
const sheet = createSheet("s1", "Sheet1", 0);
|
|
216
|
-
setCell(sheet, 0, 0, makeTextCell("A1", false));
|
|
217
|
-
setCell(sheet, 3, 5, makeTextCell("F4", false));
|
|
218
|
-
setCell(sheet, 1, 2, makeNumberCell(7));
|
|
219
|
-
const extent = getSheetExtent(sheet);
|
|
220
|
-
assert.deepEqual(extent, { minRow: 0, maxRow: 3, minCol: 0, maxCol: 5 });
|
|
221
|
-
record("extent-bounds", "getSheetExtent computes correct bounds", true);
|
|
222
|
-
});
|
|
223
|
-
|
|
224
|
-
test("sheet.ts: iterateCellAddresses yields all occupied cells", () => {
|
|
225
|
-
const sheet = createSheet("s1", "Sheet1", 0);
|
|
226
|
-
setCell(sheet, 0, 0, makeTextCell("a", false));
|
|
227
|
-
setCell(sheet, 1, 1, makeNumberCell(1));
|
|
228
|
-
setCell(sheet, 2, 2, makeBooleanCell(true));
|
|
229
|
-
const addresses = [...iterateCellAddresses(sheet)];
|
|
230
|
-
assert.equal(addresses.length, 3);
|
|
231
|
-
record("iterate-addresses", "iterateCellAddresses yields correct count", true);
|
|
232
|
-
});
|
|
233
|
-
|
|
234
|
-
// ---------------------------------------------------------------------------
|
|
235
|
-
// Merge operations
|
|
236
|
-
// ---------------------------------------------------------------------------
|
|
237
|
-
|
|
238
|
-
test("sheet.ts: addMerge stores a merge range", () => {
|
|
239
|
-
const sheet = createSheet("s1", "Sheet1", 0);
|
|
240
|
-
addMerge(sheet, { startRow: 0, startCol: 0, endRow: 2, endCol: 3 });
|
|
241
|
-
assert.equal(sheet.merges.length, 1);
|
|
242
|
-
assert.deepEqual(sheet.merges[0], { startRow: 0, startCol: 0, endRow: 2, endCol: 3 });
|
|
243
|
-
record("merge-add", "addMerge stores merge range", true);
|
|
244
|
-
});
|
|
245
|
-
|
|
246
|
-
test("sheet.ts: removeMerge removes exact range", () => {
|
|
247
|
-
const sheet = createSheet("s1", "Sheet1", 0);
|
|
248
|
-
const merge = { startRow: 1, startCol: 1, endRow: 3, endCol: 3 };
|
|
249
|
-
addMerge(sheet, merge);
|
|
250
|
-
addMerge(sheet, { startRow: 5, startCol: 0, endRow: 6, endCol: 1 });
|
|
251
|
-
const removed = removeMerge(sheet, merge);
|
|
252
|
-
assert.equal(removed, true);
|
|
253
|
-
assert.equal(sheet.merges.length, 1);
|
|
254
|
-
record("merge-remove", "removeMerge removes exact merge", true);
|
|
255
|
-
});
|
|
256
|
-
|
|
257
|
-
test("sheet.ts: removeMerge returns false when range not found", () => {
|
|
258
|
-
const sheet = createSheet("s1", "Sheet1", 0);
|
|
259
|
-
const removed = removeMerge(sheet, { startRow: 0, startCol: 0, endRow: 1, endCol: 1 });
|
|
260
|
-
assert.equal(removed, false);
|
|
261
|
-
record("merge-remove-absent", "removeMerge returns false for absent merge", true);
|
|
262
|
-
});
|
|
263
|
-
|
|
264
|
-
test("sheet.ts: findMergesContaining returns overlapping merges", () => {
|
|
265
|
-
const sheet = createSheet("s1", "Sheet1", 0);
|
|
266
|
-
addMerge(sheet, { startRow: 0, startCol: 0, endRow: 2, endCol: 2 });
|
|
267
|
-
addMerge(sheet, { startRow: 5, startCol: 5, endRow: 7, endCol: 7 });
|
|
268
|
-
assert.equal(findMergesContaining(sheet, 1, 1).length, 1);
|
|
269
|
-
assert.equal(findMergesContaining(sheet, 6, 6).length, 1);
|
|
270
|
-
assert.equal(findMergesContaining(sheet, 3, 3).length, 0);
|
|
271
|
-
record("merge-find", "findMergesContaining returns correct merges", true);
|
|
272
|
-
});
|
|
273
|
-
|
|
274
|
-
test("sheet.ts: cell at top-left of merge is included in merge", () => {
|
|
275
|
-
const sheet = createSheet("s1", "Sheet1", 0);
|
|
276
|
-
addMerge(sheet, { startRow: 2, startCol: 3, endRow: 4, endCol: 5 });
|
|
277
|
-
assert.equal(findMergesContaining(sheet, 2, 3).length, 1);
|
|
278
|
-
assert.equal(findMergesContaining(sheet, 4, 5).length, 1);
|
|
279
|
-
assert.equal(findMergesContaining(sheet, 5, 5).length, 0);
|
|
280
|
-
record("merge-boundary", "merge boundary cells correctly included/excluded", true);
|
|
281
|
-
});
|
|
282
|
-
|
|
283
|
-
// ---------------------------------------------------------------------------
|
|
284
|
-
// Workbook registry
|
|
285
|
-
// ---------------------------------------------------------------------------
|
|
286
|
-
|
|
287
|
-
test("workbook.ts: createWorkbook produces correct initial state", () => {
|
|
288
|
-
const wb = createWorkbook();
|
|
289
|
-
assert.equal(wb.schemaVersion, WORKBOOK_SCHEMA_VERSION);
|
|
290
|
-
assert.equal(wb.metadata.date1904, false);
|
|
291
|
-
assert.equal(wb.sheets.size, 0);
|
|
292
|
-
assert.deepEqual(wb.sheetOrder, []);
|
|
293
|
-
assert.deepEqual(wb.sharedStrings.strings, []);
|
|
294
|
-
assert.deepEqual(wb.namedItems, []);
|
|
295
|
-
record("workbook-create", "createWorkbook initial state", true);
|
|
296
|
-
});
|
|
297
|
-
|
|
298
|
-
test("workbook.ts: createWorkbook accepts metadata overrides", () => {
|
|
299
|
-
const wb = createWorkbook({ title: "Budget 2026", date1904: true });
|
|
300
|
-
assert.equal(wb.metadata.title, "Budget 2026");
|
|
301
|
-
assert.equal(wb.metadata.date1904, true);
|
|
302
|
-
record("workbook-metadata", "createWorkbook metadata override", true);
|
|
303
|
-
});
|
|
304
|
-
|
|
305
|
-
test("workbook.ts: addSheet adds sheet and registers in order", () => {
|
|
306
|
-
const wb = createWorkbook();
|
|
307
|
-
addSheet(wb, "s1", "Sheet1");
|
|
308
|
-
addSheet(wb, "s2", "Sheet2");
|
|
309
|
-
assert.equal(wb.sheets.size, 2);
|
|
310
|
-
assert.deepEqual(wb.sheetOrder, ["s1", "s2"]);
|
|
311
|
-
record("sheet-add", "addSheet registers sheets in order", true);
|
|
312
|
-
});
|
|
313
|
-
|
|
314
|
-
test("workbook.ts: addSheet assigns correct orderIndex", () => {
|
|
315
|
-
const wb = createWorkbook();
|
|
316
|
-
addSheet(wb, "s1", "Sheet1");
|
|
317
|
-
addSheet(wb, "s2", "Sheet2");
|
|
318
|
-
assert.equal(wb.sheets.get("s1")?.orderIndex, 0);
|
|
319
|
-
assert.equal(wb.sheets.get("s2")?.orderIndex, 1);
|
|
320
|
-
record("sheet-order-index", "addSheet assigns correct orderIndex", true);
|
|
321
|
-
});
|
|
322
|
-
|
|
323
|
-
test("workbook.ts: addSheet throws on duplicate id", () => {
|
|
324
|
-
const wb = createWorkbook();
|
|
325
|
-
addSheet(wb, "s1", "Sheet1");
|
|
326
|
-
assert.throws(() => addSheet(wb, "s1", "Sheet1-dup"), /already exists/);
|
|
327
|
-
record("sheet-dup-id", "addSheet throws on duplicate id", true);
|
|
328
|
-
});
|
|
329
|
-
|
|
330
|
-
test("workbook.ts: addSheet at specific index inserts correctly", () => {
|
|
331
|
-
const wb = createWorkbook();
|
|
332
|
-
addSheet(wb, "s1", "Sheet1");
|
|
333
|
-
addSheet(wb, "s2", "Sheet2");
|
|
334
|
-
addSheet(wb, "s3", "Sheet3", 1); // insert before Sheet2
|
|
335
|
-
assert.deepEqual(wb.sheetOrder, ["s1", "s3", "s2"]);
|
|
336
|
-
assert.equal(wb.sheets.get("s3")?.orderIndex, 1);
|
|
337
|
-
assert.equal(wb.sheets.get("s2")?.orderIndex, 2);
|
|
338
|
-
record("sheet-insert", "addSheet at index inserts correctly", true);
|
|
339
|
-
});
|
|
340
|
-
|
|
341
|
-
test("workbook.ts: removeSheet removes sheet from registry", () => {
|
|
342
|
-
const wb = createWorkbook();
|
|
343
|
-
addSheet(wb, "s1", "Sheet1");
|
|
344
|
-
addSheet(wb, "s2", "Sheet2");
|
|
345
|
-
const removed = removeSheet(wb, "s1");
|
|
346
|
-
assert.equal(removed?.sheetId, "s1");
|
|
347
|
-
assert.equal(wb.sheets.size, 1);
|
|
348
|
-
assert.deepEqual(wb.sheetOrder, ["s2"]);
|
|
349
|
-
assert.equal(wb.sheets.get("s2")?.orderIndex, 0);
|
|
350
|
-
record("sheet-remove", "removeSheet removes sheet and reindexes", true);
|
|
351
|
-
});
|
|
352
|
-
|
|
353
|
-
test("workbook.ts: removeSheet returns undefined for unknown id", () => {
|
|
354
|
-
const wb = createWorkbook();
|
|
355
|
-
const removed = removeSheet(wb, "nonexistent");
|
|
356
|
-
assert.equal(removed, undefined);
|
|
357
|
-
record("sheet-remove-absent", "removeSheet returns undefined for unknown id", true);
|
|
358
|
-
});
|
|
359
|
-
|
|
360
|
-
test("workbook.ts: getSheet returns correct sheet", () => {
|
|
361
|
-
const wb = createWorkbook();
|
|
362
|
-
addSheet(wb, "s1", "Sheet1");
|
|
363
|
-
const sheet = getSheet(wb, "s1");
|
|
364
|
-
assert.equal(sheet?.sheetId, "s1");
|
|
365
|
-
record("sheet-get", "getSheet returns correct sheet", true);
|
|
366
|
-
});
|
|
367
|
-
|
|
368
|
-
test("workbook.ts: getSheetByName finds sheet by display name", () => {
|
|
369
|
-
const wb = createWorkbook();
|
|
370
|
-
addSheet(wb, "s1", "Budget");
|
|
371
|
-
addSheet(wb, "s2", "Summary");
|
|
372
|
-
assert.equal(getSheetByName(wb, "Budget")?.sheetId, "s1");
|
|
373
|
-
assert.equal(getSheetByName(wb, "Summary")?.sheetId, "s2");
|
|
374
|
-
assert.equal(getSheetByName(wb, "Missing"), undefined);
|
|
375
|
-
record("sheet-get-by-name", "getSheetByName finds sheet", true);
|
|
376
|
-
});
|
|
377
|
-
|
|
378
|
-
test("workbook.ts: listSheets returns sheets in tab order", () => {
|
|
379
|
-
const wb = createWorkbook();
|
|
380
|
-
addSheet(wb, "s1", "First");
|
|
381
|
-
addSheet(wb, "s2", "Second");
|
|
382
|
-
addSheet(wb, "s3", "Third");
|
|
383
|
-
const sheets = listSheets(wb);
|
|
384
|
-
assert.equal(sheets.length, 3);
|
|
385
|
-
assert.equal(sheets[0].name, "First");
|
|
386
|
-
assert.equal(sheets[1].name, "Second");
|
|
387
|
-
assert.equal(sheets[2].name, "Third");
|
|
388
|
-
record("sheet-list", "listSheets returns sheets in tab order", true);
|
|
389
|
-
});
|
|
390
|
-
|
|
391
|
-
test("workbook.ts: renameSheet updates display name", () => {
|
|
392
|
-
const wb = createWorkbook();
|
|
393
|
-
addSheet(wb, "s1", "OldName");
|
|
394
|
-
renameSheet(wb, "s1", "NewName");
|
|
395
|
-
assert.equal(wb.sheets.get("s1")?.name, "NewName");
|
|
396
|
-
record("sheet-rename", "renameSheet updates display name", true);
|
|
397
|
-
});
|
|
398
|
-
|
|
399
|
-
test("workbook.ts: renameSheet throws for unknown id", () => {
|
|
400
|
-
const wb = createWorkbook();
|
|
401
|
-
assert.throws(() => renameSheet(wb, "nonexistent", "X"), /not found/);
|
|
402
|
-
record("sheet-rename-absent", "renameSheet throws for unknown id", true);
|
|
403
|
-
});
|
|
404
|
-
|
|
405
|
-
test("workbook.ts: moveSheet reorders sheets", () => {
|
|
406
|
-
const wb = createWorkbook();
|
|
407
|
-
addSheet(wb, "s1", "First");
|
|
408
|
-
addSheet(wb, "s2", "Second");
|
|
409
|
-
addSheet(wb, "s3", "Third");
|
|
410
|
-
moveSheet(wb, "s1", 2);
|
|
411
|
-
assert.deepEqual(wb.sheetOrder, ["s2", "s3", "s1"]);
|
|
412
|
-
assert.equal(wb.sheets.get("s1")?.orderIndex, 2);
|
|
413
|
-
assert.equal(wb.sheets.get("s2")?.orderIndex, 0);
|
|
414
|
-
record("sheet-move", "moveSheet reorders sheets correctly", true);
|
|
415
|
-
});
|
|
416
|
-
|
|
417
|
-
test("workbook.ts: moveSheet no-op when already at target", () => {
|
|
418
|
-
const wb = createWorkbook();
|
|
419
|
-
addSheet(wb, "s1", "First");
|
|
420
|
-
addSheet(wb, "s2", "Second");
|
|
421
|
-
moveSheet(wb, "s1", 0); // already at 0
|
|
422
|
-
assert.deepEqual(wb.sheetOrder, ["s1", "s2"]);
|
|
423
|
-
record("sheet-move-noop", "moveSheet is no-op when already at target", true);
|
|
424
|
-
});
|
|
425
|
-
|
|
426
|
-
// ---------------------------------------------------------------------------
|
|
427
|
-
// Shared string table
|
|
428
|
-
// ---------------------------------------------------------------------------
|
|
429
|
-
|
|
430
|
-
test("workbook.ts: appendSharedString adds strings in order", () => {
|
|
431
|
-
const wb = createWorkbook();
|
|
432
|
-
const idx0 = appendSharedString(wb, "Alpha");
|
|
433
|
-
const idx1 = appendSharedString(wb, "Beta");
|
|
434
|
-
assert.equal(idx0, 0);
|
|
435
|
-
assert.equal(idx1, 1);
|
|
436
|
-
assert.deepEqual(wb.sharedStrings.strings, ["Alpha", "Beta"]);
|
|
437
|
-
record("shared-strings-append", "appendSharedString appends in order", true);
|
|
438
|
-
});
|
|
439
|
-
|
|
440
|
-
test("workbook.ts: resolveSharedString returns correct string", () => {
|
|
441
|
-
const wb = createWorkbook();
|
|
442
|
-
appendSharedString(wb, "Hello");
|
|
443
|
-
appendSharedString(wb, "World");
|
|
444
|
-
assert.equal(resolveSharedString(wb, 0), "Hello");
|
|
445
|
-
assert.equal(resolveSharedString(wb, 1), "World");
|
|
446
|
-
assert.equal(resolveSharedString(wb, 99), undefined);
|
|
447
|
-
record("shared-strings-resolve", "resolveSharedString returns correct string", true);
|
|
448
|
-
});
|
|
449
|
-
|
|
450
|
-
// ---------------------------------------------------------------------------
|
|
451
|
-
// Style registry
|
|
452
|
-
// ---------------------------------------------------------------------------
|
|
453
|
-
|
|
454
|
-
test("styles.ts: createEmptyStyleRegistry is empty", () => {
|
|
455
|
-
const reg = createEmptyStyleRegistry();
|
|
456
|
-
assert.equal(reg.cellFormats.length, 0);
|
|
457
|
-
assert.equal(reg.numFormats.size, 0);
|
|
458
|
-
assert.equal(reg.cellFormatCount, 0);
|
|
459
|
-
record("styles-empty", "createEmptyStyleRegistry is empty", true);
|
|
460
|
-
});
|
|
461
|
-
|
|
462
|
-
test("styles.ts: getXfRecord returns undefined for missing index", () => {
|
|
463
|
-
const reg = createEmptyStyleRegistry();
|
|
464
|
-
assert.equal(getXfRecord(reg, 0), undefined);
|
|
465
|
-
record("styles-xf-missing", "getXfRecord returns undefined for missing index", true);
|
|
466
|
-
});
|
|
467
|
-
|
|
468
|
-
test("styles.ts: getXfRecord returns stored xf record", () => {
|
|
469
|
-
const reg = createEmptyStyleRegistry();
|
|
470
|
-
const xf: XfRecord = {
|
|
471
|
-
xfIndex: 0,
|
|
472
|
-
fontId: 0,
|
|
473
|
-
fillId: 0,
|
|
474
|
-
borderId: 0,
|
|
475
|
-
numFmtId: 0,
|
|
476
|
-
rawAttributes: { applyFont: "1" },
|
|
477
|
-
};
|
|
478
|
-
reg.cellFormats.push(xf);
|
|
479
|
-
reg.cellFormatCount = 1;
|
|
480
|
-
assert.deepEqual(getXfRecord(reg, 0), xf);
|
|
481
|
-
record("styles-xf-stored", "getXfRecord returns stored xf record", true);
|
|
482
|
-
});
|
|
483
|
-
|
|
484
|
-
test("styles.ts: isDateFormatHeuristic recognizes built-in date numFmtId", () => {
|
|
485
|
-
const reg = createEmptyStyleRegistry();
|
|
486
|
-
const xf: XfRecord = {
|
|
487
|
-
xfIndex: 0,
|
|
488
|
-
fontId: 0,
|
|
489
|
-
fillId: 0,
|
|
490
|
-
borderId: 0,
|
|
491
|
-
numFmtId: 14, // built-in short date
|
|
492
|
-
rawAttributes: {},
|
|
493
|
-
};
|
|
494
|
-
reg.cellFormats.push(xf);
|
|
495
|
-
assert.equal(isDateFormatHeuristic(reg, 0), true);
|
|
496
|
-
record("styles-date-builtin", "isDateFormatHeuristic recognizes built-in date", true);
|
|
497
|
-
});
|
|
498
|
-
|
|
499
|
-
test("styles.ts: isDateFormatHeuristic returns false for general numFmtId 0", () => {
|
|
500
|
-
const reg = createEmptyStyleRegistry();
|
|
501
|
-
const xf: XfRecord = {
|
|
502
|
-
xfIndex: 0,
|
|
503
|
-
fontId: 0,
|
|
504
|
-
fillId: 0,
|
|
505
|
-
borderId: 0,
|
|
506
|
-
numFmtId: 0,
|
|
507
|
-
rawAttributes: {},
|
|
508
|
-
};
|
|
509
|
-
reg.cellFormats.push(xf);
|
|
510
|
-
assert.equal(isDateFormatHeuristic(reg, 0), false);
|
|
511
|
-
record("styles-date-general", "isDateFormatHeuristic false for numFmtId 0", true);
|
|
512
|
-
});
|
|
513
|
-
|
|
514
|
-
// ---------------------------------------------------------------------------
|
|
515
|
-
// Integration scenario: populate a sheet with multiple cell types
|
|
516
|
-
// ---------------------------------------------------------------------------
|
|
517
|
-
|
|
518
|
-
test("integration: build a small workbook with multiple sheets and cell types", () => {
|
|
519
|
-
const wb = createWorkbook({ title: "Test Workbook" });
|
|
520
|
-
|
|
521
|
-
// Add two sheets
|
|
522
|
-
const s1 = addSheet(wb, "sheet1", "Data");
|
|
523
|
-
const s2 = addSheet(wb, "sheet2", "Summary");
|
|
524
|
-
|
|
525
|
-
// Populate sheet 1 with various cell types
|
|
526
|
-
setCell(s1, 0, 0, makeTextCell("Name", false));
|
|
527
|
-
setCell(s1, 0, 1, makeTextCell("Value", false));
|
|
528
|
-
setCell(s1, 1, 0, makeTextCell("Alpha", false));
|
|
529
|
-
setCell(s1, 1, 1, makeNumberCell(100));
|
|
530
|
-
setCell(s1, 2, 0, makeTextCell("Beta", false));
|
|
531
|
-
setCell(s1, 2, 1, makeFormulaCell("B2*2", { type: "number", value: 200 }));
|
|
532
|
-
setCell(s1, 3, 0, makeBooleanCell(true));
|
|
533
|
-
setCell(s1, 3, 1, makeErrorCell("#N/A"));
|
|
534
|
-
|
|
535
|
-
// Add a merge in sheet 1
|
|
536
|
-
addMerge(s1, { startRow: 4, startCol: 0, endRow: 4, endCol: 3 });
|
|
537
|
-
|
|
538
|
-
// Populate summary sheet with a shared-string reference
|
|
539
|
-
appendSharedString(wb, "Total");
|
|
540
|
-
setCell(s2, 0, 0, makeTextCell("Total", true, { xfIndex: 0 }));
|
|
541
|
-
setCell(s2, 0, 1, makeFormulaCell("Data!B2+Data!B3", { type: "number", value: 300 }));
|
|
542
|
-
|
|
543
|
-
// Assertions
|
|
544
|
-
assert.equal(wb.sheets.size, 2);
|
|
545
|
-
assert.equal(s1.cells.size, 8);
|
|
546
|
-
assert.equal(s1.merges.length, 1);
|
|
547
|
-
assert.equal(getCell(s1, 1, 1)?.kind, "number");
|
|
548
|
-
assert.equal((getCell(s1, 2, 1) as { kind: string; formula: string })?.formula, "B2*2");
|
|
549
|
-
assert.equal(getCell(s1, 3, 0)?.kind, "boolean");
|
|
550
|
-
assert.equal(getCell(s1, 3, 1)?.kind, "error");
|
|
551
|
-
assert.equal(getCell(s2, 0, 0)?.kind, "text");
|
|
552
|
-
assert.equal(resolveSharedString(wb, 0), "Total");
|
|
553
|
-
|
|
554
|
-
const extent = getSheetExtent(s1);
|
|
555
|
-
assert.equal(extent?.maxRow, 3);
|
|
556
|
-
assert.equal(extent?.maxCol, 1);
|
|
557
|
-
|
|
558
|
-
record("integration-scenario", "full workbook integration scenario", true);
|
|
559
|
-
});
|
|
560
|
-
|
|
561
|
-
// ---------------------------------------------------------------------------
|
|
562
|
-
// Formula reference tokens (architecture requirement for safe remapping)
|
|
563
|
-
// ---------------------------------------------------------------------------
|
|
564
|
-
|
|
565
|
-
test("cell.ts: makeFormulaCell stores referenceTokens", () => {
|
|
566
|
-
const cell = makeFormulaCell("SUM(A1:A10)+B2", undefined, undefined, ["A1:A10", "B2"]);
|
|
567
|
-
assert.equal(cell.kind, "formula");
|
|
568
|
-
assert.deepEqual(cell.referenceTokens, ["A1:A10", "B2"]);
|
|
569
|
-
record("formula-ref-tokens", "makeFormulaCell stores referenceTokens", true);
|
|
570
|
-
});
|
|
571
|
-
|
|
572
|
-
test("cell.ts: makeFormulaCell defaults referenceTokens to empty array", () => {
|
|
573
|
-
const cell = makeFormulaCell("TODAY()");
|
|
574
|
-
assert.deepEqual(cell.referenceTokens, []);
|
|
575
|
-
record("formula-ref-tokens-default", "makeFormulaCell defaults referenceTokens to []", true);
|
|
576
|
-
});
|
|
577
|
-
|
|
578
|
-
test("cell.ts: makeFormulaCell with cached value preserves referenceTokens", () => {
|
|
579
|
-
const cell = makeFormulaCell(
|
|
580
|
-
"A1+B1",
|
|
581
|
-
{ type: "number", value: 42 },
|
|
582
|
-
{ xfIndex: 0 },
|
|
583
|
-
["A1", "B1"],
|
|
584
|
-
);
|
|
585
|
-
assert.deepEqual(cell.referenceTokens, ["A1", "B1"]);
|
|
586
|
-
assert.deepEqual(cell.cachedValue, { type: "number", value: 42 });
|
|
587
|
-
assert.deepEqual(cell.styleRef, { xfIndex: 0 });
|
|
588
|
-
record("formula-ref-tokens-with-cached", "referenceTokens coexists with cachedValue and styleRef", true);
|
|
589
|
-
});
|
|
590
|
-
|
|
591
|
-
// ---------------------------------------------------------------------------
|
|
592
|
-
// Workbook serialization round-trip (Map → JSON → Map)
|
|
593
|
-
// ---------------------------------------------------------------------------
|
|
594
|
-
|
|
595
|
-
test("workbook.ts: workbookToJson produces JSON-serializable output", () => {
|
|
596
|
-
const wb = createWorkbook({ title: "Serialize Test" });
|
|
597
|
-
const s = addSheet(wb, "s1", "Sheet1");
|
|
598
|
-
setCell(s, 0, 0, makeTextCell("Hello", false));
|
|
599
|
-
setCell(s, 1, 0, makeNumberCell(99));
|
|
600
|
-
addMerge(s, { startRow: 0, startCol: 0, endRow: 0, endCol: 2 });
|
|
601
|
-
appendSharedString(wb, "shared");
|
|
602
|
-
|
|
603
|
-
const plain = workbookToJson(wb);
|
|
604
|
-
// Should round-trip through JSON without throwing
|
|
605
|
-
const json = JSON.stringify(plain);
|
|
606
|
-
assert.ok(json.length > 0, "JSON output is non-empty");
|
|
607
|
-
assert.ok(!json.includes("[object Map]"), "No un-serialized Map in output");
|
|
608
|
-
record("workbook-to-json", "workbookToJson produces JSON-serializable plain object", true);
|
|
609
|
-
});
|
|
610
|
-
|
|
611
|
-
test("workbook.ts: workbookFromJson restores workbook faithfully", () => {
|
|
612
|
-
const wb = createWorkbook({ title: "Round Trip" });
|
|
613
|
-
const s = addSheet(wb, "s1", "Data");
|
|
614
|
-
setCell(s, 0, 0, makeTextCell("Alpha", true));
|
|
615
|
-
setCell(s, 1, 1, makeNumberCell(3.14));
|
|
616
|
-
addMerge(s, { startRow: 2, startCol: 0, endRow: 2, endCol: 1 });
|
|
617
|
-
appendSharedString(wb, "Alpha");
|
|
618
|
-
|
|
619
|
-
const serialized = workbookToJson(wb);
|
|
620
|
-
const restored = workbookFromJson(serialized);
|
|
621
|
-
|
|
622
|
-
assert.equal(restored.schemaVersion, WORKBOOK_SCHEMA_VERSION);
|
|
623
|
-
assert.equal(restored.metadata.title, "Round Trip");
|
|
624
|
-
assert.equal(restored.sheets.size, 1);
|
|
625
|
-
assert.deepEqual(restored.sheetOrder, ["s1"]);
|
|
626
|
-
assert.equal(restored.sharedStrings.strings[0], "Alpha");
|
|
627
|
-
|
|
628
|
-
const rs = restored.sheets.get("s1");
|
|
629
|
-
assert.ok(rs, "sheet s1 present after restore");
|
|
630
|
-
assert.equal(rs!.cells.size, 2);
|
|
631
|
-
assert.equal(rs!.merges.length, 1);
|
|
632
|
-
const cell = getCell(rs!, 0, 0);
|
|
633
|
-
assert.equal(cell?.kind, "text");
|
|
634
|
-
assert.equal((cell as { value: string }).value, "Alpha");
|
|
635
|
-
record("workbook-round-trip", "workbookToJson/workbookFromJson round-trip is faithful", true);
|
|
636
|
-
});
|
|
637
|
-
|
|
638
|
-
test("workbook.ts: workbookFromJson throws on schema version mismatch", () => {
|
|
639
|
-
const wb = createWorkbook();
|
|
640
|
-
const plain = workbookToJson(wb);
|
|
641
|
-
// Corrupt the schema version
|
|
642
|
-
const corrupted = { ...plain, schemaVersion: "workbook/0" as typeof WORKBOOK_SCHEMA_VERSION };
|
|
643
|
-
assert.throws(() => workbookFromJson(corrupted), /Unsupported workbook schema version/);
|
|
644
|
-
record("workbook-schema-mismatch", "workbookFromJson throws on schema version mismatch", true);
|
|
645
|
-
});
|
|
646
|
-
|
|
647
|
-
// ---------------------------------------------------------------------------
|
|
648
|
-
// Write proof bundle (deferred to process exit so all test records are captured)
|
|
649
|
-
// ---------------------------------------------------------------------------
|
|
650
|
-
|
|
651
|
-
process.on("exit", () => {
|
|
652
|
-
const passed = results.filter((r) => r.passed).length;
|
|
653
|
-
const failed = results.filter((r) => !r.passed).length;
|
|
654
|
-
|
|
655
|
-
const proofBundle = {
|
|
656
|
-
reportVersion: "wave-25-model/1",
|
|
657
|
-
component: "xlsx-canonical-workbook-model",
|
|
658
|
-
generatedAt: new Date().toISOString(),
|
|
659
|
-
summary: {
|
|
660
|
-
total: results.length,
|
|
661
|
-
passed,
|
|
662
|
-
failed,
|
|
663
|
-
},
|
|
664
|
-
tests: results,
|
|
665
|
-
};
|
|
666
|
-
|
|
667
|
-
mkdirSync(PROOF_DIR, { recursive: true });
|
|
668
|
-
writeFileSync(PROOF_PATH, JSON.stringify(proofBundle, null, 2), "utf8");
|
|
669
|
-
});
|