@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.
- package/dist/chunk-32W6IVQE.js +7725 -0
- package/dist/chunk-32W6IVQE.js.map +1 -0
- package/dist/index.cjs +23722 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.cts +7 -0
- package/dist/index.d.ts +7 -0
- package/dist/index.js +16011 -0
- package/dist/index.js.map +1 -0
- package/dist/public-types-DqCURAz8.d.cts +1152 -0
- package/dist/public-types-DqCURAz8.d.ts +1152 -0
- package/dist/tailwind.cjs +8295 -0
- package/dist/tailwind.cjs.map +1 -0
- package/dist/tailwind.d.cts +323 -0
- package/dist/tailwind.d.ts +323 -0
- package/dist/tailwind.js +553 -0
- package/dist/tailwind.js.map +1 -0
- package/package.json +52 -31
- 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/src/README.md +0 -85
- package/src/api/README.md +0 -22
- package/src/api/public-types.ts +0 -525
- package/src/component-inventory.md +0 -99
- package/src/core/README.md +0 -10
- package/src/core/commands/README.md +0 -3
- package/src/core/commands/formatting-commands.ts +0 -161
- package/src/core/commands/image-commands.ts +0 -144
- package/src/core/commands/index.ts +0 -1013
- package/src/core/commands/list-commands.ts +0 -370
- package/src/core/commands/review-commands.ts +0 -108
- package/src/core/commands/text-commands.ts +0 -119
- package/src/core/schema/README.md +0 -3
- package/src/core/schema/text-schema.ts +0 -512
- package/src/core/selection/README.md +0 -3
- package/src/core/selection/mapping.ts +0 -238
- package/src/core/selection/review-anchors.ts +0 -94
- package/src/core/state/README.md +0 -3
- package/src/core/state/editor-state.ts +0 -580
- package/src/core/state/text-transaction.ts +0 -276
- package/src/formats/xlsx/io/parse-shared-strings.ts +0 -41
- package/src/formats/xlsx/io/parse-sheet.ts +0 -289
- package/src/formats/xlsx/io/parse-styles.ts +0 -57
- package/src/formats/xlsx/io/parse-workbook.ts +0 -75
- package/src/formats/xlsx/io/xlsx-session.ts +0 -306
- package/src/formats/xlsx/model/cell.ts +0 -189
- package/src/formats/xlsx/model/sheet.ts +0 -244
- package/src/formats/xlsx/model/styles.ts +0 -118
- package/src/formats/xlsx/model/workbook.ts +0 -449
- package/src/io/README.md +0 -10
- package/src/io/docx-session.ts +0 -1763
- package/src/io/export/README.md +0 -3
- package/src/io/export/export-session.ts +0 -165
- package/src/io/export/minimal-docx.ts +0 -115
- package/src/io/export/reattach-preserved-parts.ts +0 -54
- package/src/io/export/serialize-comments.ts +0 -876
- package/src/io/export/serialize-footnotes.ts +0 -217
- package/src/io/export/serialize-headers-footers.ts +0 -200
- package/src/io/export/serialize-main-document.ts +0 -982
- package/src/io/export/serialize-numbering.ts +0 -97
- package/src/io/export/serialize-revisions.ts +0 -389
- package/src/io/export/serialize-runtime-revisions.ts +0 -265
- package/src/io/export/serialize-tables.ts +0 -147
- package/src/io/export/split-review-boundaries.ts +0 -194
- package/src/io/normalize/README.md +0 -3
- package/src/io/normalize/normalize-text.ts +0 -437
- package/src/io/ooxml/README.md +0 -3
- package/src/io/ooxml/parse-comments.ts +0 -779
- package/src/io/ooxml/parse-complex-content.ts +0 -287
- package/src/io/ooxml/parse-fields.ts +0 -438
- package/src/io/ooxml/parse-footnotes.ts +0 -403
- package/src/io/ooxml/parse-headers-footers.ts +0 -483
- package/src/io/ooxml/parse-inline-media.ts +0 -431
- package/src/io/ooxml/parse-main-document.ts +0 -1846
- package/src/io/ooxml/parse-numbering.ts +0 -425
- package/src/io/ooxml/parse-revisions.ts +0 -658
- package/src/io/ooxml/parse-shapes.ts +0 -271
- package/src/io/ooxml/parse-tables.ts +0 -568
- package/src/io/ooxml/parse-theme.ts +0 -314
- package/src/io/ooxml/part-manifest.ts +0 -136
- package/src/io/ooxml/revision-boundaries.ts +0 -351
- package/src/io/opc/README.md +0 -3
- package/src/io/opc/corrupt-package.ts +0 -166
- package/src/io/opc/docx-package.ts +0 -74
- package/src/io/opc/package-reader.ts +0 -320
- package/src/io/opc/package-writer.ts +0 -273
- package/src/model/README.md +0 -3
- package/src/model/canonical-document.ts +0 -1911
- package/src/model/cds-1.0.0.ts +0 -196
- package/src/model/snapshot.ts +0 -393
- package/src/preservation/README.md +0 -3
- package/src/preservation/markup-compatibility.ts +0 -48
- package/src/preservation/opaque-fragment-store.ts +0 -89
- package/src/preservation/opaque-region.ts +0 -233
- package/src/preservation/package-preservation.ts +0 -120
- package/src/preservation/preserved-part-manifest.ts +0 -56
- package/src/preservation/relationship-retention.ts +0 -57
- package/src/preservation/store.ts +0 -185
- package/src/review/README.md +0 -16
- package/src/review/store/README.md +0 -3
- package/src/review/store/comment-anchors.ts +0 -70
- package/src/review/store/comment-remapping.ts +0 -154
- package/src/review/store/comment-store.ts +0 -331
- package/src/review/store/comment-thread.ts +0 -109
- package/src/review/store/revision-actions.ts +0 -394
- package/src/review/store/revision-store.ts +0 -303
- package/src/review/store/revision-types.ts +0 -168
- package/src/review/store/runtime-comment-store.ts +0 -43
- package/src/runtime/README.md +0 -3
- package/src/runtime/ai-action-policy.ts +0 -764
- package/src/runtime/document-runtime.ts +0 -969
- package/src/runtime/read-only-diagnostics-runtime.ts +0 -232
- package/src/runtime/review-runtime.ts +0 -44
- package/src/runtime/revision-runtime.ts +0 -107
- package/src/runtime/session-capabilities.ts +0 -138
- package/src/runtime/surface-projection.ts +0 -570
- package/src/runtime/table-commands.ts +0 -84
- package/src/runtime/table-schema.ts +0 -125
- package/src/ui/README.md +0 -30
- package/src/ui/WordReviewEditor.tsx +0 -1283
- package/src/ui/comments/README.md +0 -3
- package/src/ui/compatibility/README.md +0 -3
- package/src/ui/editor-surface/README.md +0 -3
- package/src/ui/headless/comment-decoration-model.ts +0 -124
- package/src/ui/headless/revision-decoration-model.ts +0 -128
- package/src/ui/headless/selection-helpers.ts +0 -34
- package/src/ui/headless/use-editor-keyboard.ts +0 -98
- package/src/ui/review/README.md +0 -3
- package/src/ui/shared/revision-filters.ts +0 -31
- package/src/ui/status/README.md +0 -3
- package/src/ui/theme/README.md +0 -3
- package/src/ui/toolbar/README.md +0 -3
- package/src/ui-tailwind/chrome/tw-alert-banner.tsx +0 -48
- package/src/ui-tailwind/chrome/tw-selection-toolbar.tsx +0 -44
- package/src/ui-tailwind/chrome/tw-unsaved-modal.tsx +0 -58
- package/src/ui-tailwind/chrome/use-before-unload.ts +0 -20
- package/src/ui-tailwind/editor-surface/pm-command-bridge.ts +0 -139
- package/src/ui-tailwind/editor-surface/pm-decorations.ts +0 -98
- package/src/ui-tailwind/editor-surface/pm-position-map.ts +0 -123
- package/src/ui-tailwind/editor-surface/pm-schema.ts +0 -452
- package/src/ui-tailwind/editor-surface/pm-state-from-snapshot.ts +0 -327
- package/src/ui-tailwind/editor-surface/search-plugin.ts +0 -157
- package/src/ui-tailwind/editor-surface/tw-caret.tsx +0 -12
- package/src/ui-tailwind/editor-surface/tw-editor-surface.tsx +0 -150
- package/src/ui-tailwind/editor-surface/tw-inline-token.tsx +0 -118
- package/src/ui-tailwind/editor-surface/tw-opaque-block.tsx +0 -52
- package/src/ui-tailwind/editor-surface/tw-paragraph-block.tsx +0 -151
- package/src/ui-tailwind/editor-surface/tw-prosemirror-surface.tsx +0 -215
- package/src/ui-tailwind/editor-surface/tw-segment-view.tsx +0 -111
- package/src/ui-tailwind/editor-surface/tw-table-node-view.tsx +0 -108
- package/src/ui-tailwind/index.ts +0 -61
- package/src/ui-tailwind/review/tw-comment-sidebar.tsx +0 -276
- package/src/ui-tailwind/review/tw-health-panel.tsx +0 -120
- package/src/ui-tailwind/review/tw-review-rail.tsx +0 -120
- package/src/ui-tailwind/review/tw-revision-sidebar.tsx +0 -164
- package/src/ui-tailwind/status/tw-status-bar.tsx +0 -58
- package/src/ui-tailwind/theme/editor-theme.css +0 -190
- package/src/ui-tailwind/toolbar/tw-toolbar-icon-button.tsx +0 -48
- package/src/ui-tailwind/toolbar/tw-toolbar.tsx +0 -231
- package/src/ui-tailwind/tw-review-workspace.tsx +0 -140
- package/src/validation/README.md +0 -3
- package/src/validation/compatibility-engine.ts +0 -317
- package/src/validation/compatibility-report.ts +0 -160
- package/src/validation/diagnostics.ts +0 -203
- package/src/validation/import-diagnostics.ts +0 -128
- package/src/validation/low-priority-word-surfaces.ts +0 -373
- 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,1080 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @deprecated This prototype demo client is no longer rendered.
|
|
3
|
-
* The root route (/) now uses HarnessEditorPage with the real WordReviewEditor.
|
|
4
|
-
* This file is kept for reference only.
|
|
5
|
-
*/
|
|
6
|
-
"use client";
|
|
7
|
-
|
|
8
|
-
import { startTransition, useEffect, useId, useState, type ChangeEvent } from "react";
|
|
9
|
-
|
|
10
|
-
import * as Select from "@radix-ui/react-select";
|
|
11
|
-
import * as Tabs from "@radix-ui/react-tabs";
|
|
12
|
-
import * as Toggle from "@radix-ui/react-toggle";
|
|
13
|
-
import * as ToggleGroup from "@radix-ui/react-toggle-group";
|
|
14
|
-
import * as Tooltip from "@radix-ui/react-tooltip";
|
|
15
|
-
import * as ScrollArea from "@radix-ui/react-scroll-area";
|
|
16
|
-
import {
|
|
17
|
-
AlertTriangle,
|
|
18
|
-
Bold,
|
|
19
|
-
Check,
|
|
20
|
-
ChevronDown,
|
|
21
|
-
CornerDownLeft,
|
|
22
|
-
Download,
|
|
23
|
-
FileText,
|
|
24
|
-
GitCompareArrows,
|
|
25
|
-
Image,
|
|
26
|
-
Info,
|
|
27
|
-
Italic,
|
|
28
|
-
Link2,
|
|
29
|
-
Lock,
|
|
30
|
-
MessageSquare,
|
|
31
|
-
Monitor,
|
|
32
|
-
Moon,
|
|
33
|
-
MoveRight,
|
|
34
|
-
Sun,
|
|
35
|
-
Redo2,
|
|
36
|
-
Shield,
|
|
37
|
-
ShieldAlert,
|
|
38
|
-
ShieldCheck,
|
|
39
|
-
Underline,
|
|
40
|
-
Undo2,
|
|
41
|
-
X,
|
|
42
|
-
XCircle,
|
|
43
|
-
} from "lucide-react";
|
|
44
|
-
|
|
45
|
-
import type {
|
|
46
|
-
CompatibilityFeatureClass,
|
|
47
|
-
DemoCatalogResponse,
|
|
48
|
-
DemoMarkupDisplay,
|
|
49
|
-
DemoRailTab,
|
|
50
|
-
DemoSession,
|
|
51
|
-
DemoViewMode,
|
|
52
|
-
PrototypeCalloutBlock,
|
|
53
|
-
PrototypeCommentDecoration,
|
|
54
|
-
PrototypeCommentThread,
|
|
55
|
-
PrototypeFeatureEntry,
|
|
56
|
-
PrototypeRevisionDecoration,
|
|
57
|
-
PrototypeRevisionEntry,
|
|
58
|
-
PrototypeSurfaceBlock,
|
|
59
|
-
PrototypeTableBlock,
|
|
60
|
-
PrototypeTableCell,
|
|
61
|
-
SurfaceBlockSnapshot,
|
|
62
|
-
SurfaceInlineSegment,
|
|
63
|
-
} from "../../lib/demo-fixtures";
|
|
64
|
-
|
|
65
|
-
interface DemoPrototypeClientProps {
|
|
66
|
-
catalog: DemoCatalogResponse;
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
const selectionActions = [
|
|
70
|
-
{ label: "Bold", icon: Bold },
|
|
71
|
-
{ label: "Italic", icon: Italic },
|
|
72
|
-
{ label: "Underline", icon: Underline },
|
|
73
|
-
{ label: "Comment", icon: MessageSquare },
|
|
74
|
-
{ label: "Link", icon: Link2 },
|
|
75
|
-
{ label: "Clear", icon: X },
|
|
76
|
-
];
|
|
77
|
-
|
|
78
|
-
const focusRingClass =
|
|
79
|
-
"focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-accent focus-visible:ring-offset-2 focus-visible:ring-offset-canvas";
|
|
80
|
-
|
|
81
|
-
export function DemoPrototypeClient(props: DemoPrototypeClientProps) {
|
|
82
|
-
const initialSession = props.catalog.sessions[0];
|
|
83
|
-
const uploadId = useId();
|
|
84
|
-
|
|
85
|
-
const [sessionId, setSessionId] = useState(initialSession.id);
|
|
86
|
-
const [markupDisplay, setMarkupDisplay] = useState<DemoMarkupDisplay>(
|
|
87
|
-
initialSession.defaultMarkupDisplay,
|
|
88
|
-
);
|
|
89
|
-
const [viewMode, setViewMode] = useState<DemoViewMode>(initialSession.defaultViewMode);
|
|
90
|
-
const [activeRailTab, setActiveRailTab] = useState<DemoRailTab>(initialSession.defaultRailTab);
|
|
91
|
-
const [trackChangesEnabled, setTrackChangesEnabled] = useState(
|
|
92
|
-
initialSession.defaultTrackChangesEnabled,
|
|
93
|
-
);
|
|
94
|
-
const [activeCommentId, setActiveCommentId] = useState<string | undefined>(
|
|
95
|
-
initialSession.snapshot.comments.threads[0]?.commentId,
|
|
96
|
-
);
|
|
97
|
-
const [activeRevisionId, setActiveRevisionId] = useState<string | undefined>(
|
|
98
|
-
initialSession.snapshot.trackedChanges.revisions[0]?.revisionId,
|
|
99
|
-
);
|
|
100
|
-
const [selectionLabel, setSelectionLabel] = useState<string>("");
|
|
101
|
-
const [activeCellId, setActiveCellId] = useState<string | undefined>();
|
|
102
|
-
const [commentStatusOverrides, setCommentStatusOverrides] = useState<
|
|
103
|
-
Record<string, "open" | "resolved" | "detached">
|
|
104
|
-
>({});
|
|
105
|
-
const [revisionStatusOverrides, setRevisionStatusOverrides] = useState<
|
|
106
|
-
Record<string, "active" | "accepted" | "rejected" | "detached">
|
|
107
|
-
>({});
|
|
108
|
-
const [reducedMotion, setReducedMotion] = useState(false);
|
|
109
|
-
const [darkMode, setDarkMode] = useState(false);
|
|
110
|
-
useEffect(() => {
|
|
111
|
-
document.documentElement.classList.toggle("dark", darkMode);
|
|
112
|
-
}, [darkMode]);
|
|
113
|
-
useEffect(() => {
|
|
114
|
-
if (typeof window === "undefined" || typeof window.matchMedia !== "function") {
|
|
115
|
-
return;
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
const mediaQuery = window.matchMedia("(prefers-reduced-motion: reduce)");
|
|
119
|
-
const syncPreference = () => {
|
|
120
|
-
setReducedMotion(mediaQuery.matches);
|
|
121
|
-
};
|
|
122
|
-
|
|
123
|
-
syncPreference();
|
|
124
|
-
mediaQuery.addEventListener?.("change", syncPreference);
|
|
125
|
-
|
|
126
|
-
return () => {
|
|
127
|
-
mediaQuery.removeEventListener?.("change", syncPreference);
|
|
128
|
-
};
|
|
129
|
-
}, []);
|
|
130
|
-
const [isDirty, setIsDirty] = useState(false);
|
|
131
|
-
const [localUploadName, setLocalUploadName] = useState<string>("");
|
|
132
|
-
|
|
133
|
-
const currentSession =
|
|
134
|
-
props.catalog.sessions.find((s) => s.id === sessionId) ?? initialSession;
|
|
135
|
-
const snap = currentSession.snapshot;
|
|
136
|
-
|
|
137
|
-
const currentComments = snap.comments.threads.map((thread) => ({
|
|
138
|
-
...thread,
|
|
139
|
-
status: commentStatusOverrides[thread.commentId] ?? thread.status,
|
|
140
|
-
}));
|
|
141
|
-
const currentRevisions = snap.trackedChanges.revisions.map((rev) => ({
|
|
142
|
-
...rev,
|
|
143
|
-
status: revisionStatusOverrides[rev.revisionId] ?? rev.status,
|
|
144
|
-
}));
|
|
145
|
-
const activeComment = currentComments.find((t) => t.commentId === activeCommentId);
|
|
146
|
-
const activeRevision = currentRevisions.find((r) => r.revisionId === activeRevisionId);
|
|
147
|
-
const preserveOnlyCount = snap.compatibility.featureEntries.filter(
|
|
148
|
-
(e) => e.featureClass === "preserve-only",
|
|
149
|
-
).length;
|
|
150
|
-
const warningCount = snap.compatibility.featureEntries.filter(
|
|
151
|
-
(e) => e.featureClass !== "supported-roundtrip",
|
|
152
|
-
).length;
|
|
153
|
-
const isExportBlocked = snap.compatibility.blockExport;
|
|
154
|
-
|
|
155
|
-
function switchSession(nextSessionId: string) {
|
|
156
|
-
const next = props.catalog.sessions.find((s) => s.id === nextSessionId);
|
|
157
|
-
if (!next) return;
|
|
158
|
-
startTransition(() => {
|
|
159
|
-
setSessionId(next.id);
|
|
160
|
-
setMarkupDisplay(next.defaultMarkupDisplay);
|
|
161
|
-
setViewMode(next.defaultViewMode);
|
|
162
|
-
setActiveRailTab(next.defaultRailTab);
|
|
163
|
-
setTrackChangesEnabled(next.defaultTrackChangesEnabled);
|
|
164
|
-
setActiveCommentId(next.snapshot.comments.threads[0]?.commentId);
|
|
165
|
-
setActiveRevisionId(next.snapshot.trackedChanges.revisions[0]?.revisionId);
|
|
166
|
-
setSelectionLabel("");
|
|
167
|
-
setActiveCellId(undefined);
|
|
168
|
-
setCommentStatusOverrides({});
|
|
169
|
-
setRevisionStatusOverrides({});
|
|
170
|
-
setIsDirty(false);
|
|
171
|
-
});
|
|
172
|
-
}
|
|
173
|
-
|
|
174
|
-
function activateComment(thread: PrototypeCommentThread) {
|
|
175
|
-
setActiveCommentId(thread.commentId);
|
|
176
|
-
setActiveRevisionId(undefined);
|
|
177
|
-
setActiveRailTab("comments");
|
|
178
|
-
setSelectionLabel(thread.excerpt);
|
|
179
|
-
// Scroll anchor into view (rail → canvas connection)
|
|
180
|
-
setTimeout(() => {
|
|
181
|
-
const el = document.querySelector(`[data-comment-id="${thread.commentId}"]`);
|
|
182
|
-
el?.scrollIntoView({ behavior: reducedMotion ? "auto" : "smooth", block: "center" });
|
|
183
|
-
}, 50);
|
|
184
|
-
}
|
|
185
|
-
|
|
186
|
-
function activateRevision(revision: PrototypeRevisionEntry) {
|
|
187
|
-
setActiveRevisionId(revision.revisionId);
|
|
188
|
-
setActiveCommentId(undefined);
|
|
189
|
-
setActiveRailTab("changes");
|
|
190
|
-
setSelectionLabel(revision.excerpt);
|
|
191
|
-
}
|
|
192
|
-
|
|
193
|
-
function updateCommentStatus(thread: PrototypeCommentThread, nextStatus: "open" | "resolved") {
|
|
194
|
-
setCommentStatusOverrides((c) => ({ ...c, [thread.commentId]: nextStatus }));
|
|
195
|
-
setIsDirty(true);
|
|
196
|
-
setActiveCommentId(thread.commentId);
|
|
197
|
-
setActiveRailTab("comments");
|
|
198
|
-
}
|
|
199
|
-
|
|
200
|
-
function updateRevisionStatus(revision: PrototypeRevisionEntry, nextStatus: "accepted" | "rejected") {
|
|
201
|
-
setRevisionStatusOverrides((c) => ({ ...c, [revision.revisionId]: nextStatus }));
|
|
202
|
-
setIsDirty(true);
|
|
203
|
-
setActiveRevisionId(revision.revisionId);
|
|
204
|
-
setActiveRailTab("changes");
|
|
205
|
-
}
|
|
206
|
-
|
|
207
|
-
function handleSegmentClick(segmentId: string, segmentText: string, cellId?: string) {
|
|
208
|
-
const trimmed = segmentText.trim();
|
|
209
|
-
setSelectionLabel(trimmed.length > 0 ? trimmed : "Selected range");
|
|
210
|
-
setActiveCellId(cellId);
|
|
211
|
-
|
|
212
|
-
// Check if this segment is in a comment decoration
|
|
213
|
-
const commentDec = currentSession.commentDecorations.find((d) =>
|
|
214
|
-
d.segmentIds.includes(segmentId),
|
|
215
|
-
);
|
|
216
|
-
if (commentDec) {
|
|
217
|
-
const thread = currentComments.find((t) => t.commentId === commentDec.commentId);
|
|
218
|
-
if (thread) { activateComment(thread); return; }
|
|
219
|
-
}
|
|
220
|
-
|
|
221
|
-
// Check if this segment is in a revision decoration
|
|
222
|
-
const revisionDec = currentSession.revisionDecorations.find((d) =>
|
|
223
|
-
d.segmentIds.includes(segmentId),
|
|
224
|
-
);
|
|
225
|
-
if (revisionDec) {
|
|
226
|
-
const rev = currentRevisions.find((r) => r.revisionId === revisionDec.revisionId);
|
|
227
|
-
if (rev) { activateRevision(rev); return; }
|
|
228
|
-
}
|
|
229
|
-
}
|
|
230
|
-
|
|
231
|
-
function handleCellClick(cellId: string) {
|
|
232
|
-
setActiveCellId(cellId);
|
|
233
|
-
setSelectionLabel(`Cell ${cellId}`);
|
|
234
|
-
}
|
|
235
|
-
|
|
236
|
-
function handleUploadChange(event: ChangeEvent<HTMLInputElement>) {
|
|
237
|
-
const file = event.target.files?.[0];
|
|
238
|
-
if (file) setLocalUploadName(file.name);
|
|
239
|
-
}
|
|
240
|
-
|
|
241
|
-
const saveState = snap.readOnly ? "Read-only" : isDirty ? "Unsaved" : "Ready";
|
|
242
|
-
const exportState = isExportBlocked ? "Blocked" : preserveOnlyCount > 0 ? "Warnings" : "Ready";
|
|
243
|
-
|
|
244
|
-
return (
|
|
245
|
-
<Tooltip.Provider delayDuration={400}>
|
|
246
|
-
<main
|
|
247
|
-
className={`flex h-screen flex-col bg-canvas text-primary ${reducedMotion ? "[&_*]:!transition-none [&_*]:!animation-none" : ""}`}
|
|
248
|
-
>
|
|
249
|
-
{/* Demo utility bar */}
|
|
250
|
-
<div className="flex h-9 shrink-0 items-center gap-3 border-b border-border bg-surface px-3 text-xs">
|
|
251
|
-
<span className="font-medium text-tertiary">Demo</span>
|
|
252
|
-
|
|
253
|
-
<Select.Root value={sessionId} onValueChange={switchSession}>
|
|
254
|
-
<Select.Trigger className={`inline-flex items-center gap-1 rounded-md px-2 py-1 text-xs text-secondary transition-colors hover:bg-surface-hover outline-none ${focusRingClass}`}>
|
|
255
|
-
<Select.Value />
|
|
256
|
-
<Select.Icon><ChevronDown className="h-3 w-3" /></Select.Icon>
|
|
257
|
-
</Select.Trigger>
|
|
258
|
-
<Select.Portal>
|
|
259
|
-
<Select.Content className="overflow-hidden rounded-lg bg-canvas shadow-lg ring-1 ring-border z-50" position="popper" sideOffset={4}>
|
|
260
|
-
<Select.Viewport className="p-1">
|
|
261
|
-
{props.catalog.sessions.map((session) => (
|
|
262
|
-
<Select.Item key={session.id} value={session.id} className={`flex cursor-pointer items-center gap-2 rounded-md px-2 py-1.5 text-xs text-primary outline-none data-[highlighted]:bg-surface ${focusRingClass}`}>
|
|
263
|
-
<Select.ItemText>{session.title}</Select.ItemText>
|
|
264
|
-
<span className={`shrink-0 text-[10px] font-medium rounded px-1 py-px ${
|
|
265
|
-
session.sourceKind === "fixture"
|
|
266
|
-
? "text-insert bg-insert-soft"
|
|
267
|
-
: session.sourceKind === "scenario"
|
|
268
|
-
? "text-accent bg-accent-soft"
|
|
269
|
-
: "text-secondary bg-subtle"
|
|
270
|
-
}`}>
|
|
271
|
-
{session.sourceKind}
|
|
272
|
-
</span>
|
|
273
|
-
</Select.Item>
|
|
274
|
-
))}
|
|
275
|
-
</Select.Viewport>
|
|
276
|
-
</Select.Content>
|
|
277
|
-
</Select.Portal>
|
|
278
|
-
</Select.Root>
|
|
279
|
-
|
|
280
|
-
<span className={`text-[10px] font-medium rounded px-1 py-px ${
|
|
281
|
-
currentSession.sourceKind === "fixture"
|
|
282
|
-
? "text-insert bg-insert-soft"
|
|
283
|
-
: currentSession.sourceKind === "scenario"
|
|
284
|
-
? "text-accent bg-accent-soft"
|
|
285
|
-
: "text-secondary bg-subtle"
|
|
286
|
-
}`}>
|
|
287
|
-
{currentSession.sourceKind}
|
|
288
|
-
</span>
|
|
289
|
-
|
|
290
|
-
<div className="h-3 w-px bg-border" />
|
|
291
|
-
|
|
292
|
-
<label className="flex items-center gap-1.5 text-tertiary cursor-pointer">
|
|
293
|
-
<FileText className="h-3 w-3" />
|
|
294
|
-
<span>{localUploadName || "Upload .docx"}</span>
|
|
295
|
-
<input accept=".docx" className="sr-only" id={uploadId} type="file" onChange={handleUploadChange} />
|
|
296
|
-
</label>
|
|
297
|
-
|
|
298
|
-
<div className="flex-1" />
|
|
299
|
-
|
|
300
|
-
<span className="text-tertiary/50">Internal prototype · non-production</span>
|
|
301
|
-
|
|
302
|
-
<button
|
|
303
|
-
type="button"
|
|
304
|
-
onClick={() => setDarkMode((d) => !d)}
|
|
305
|
-
className="inline-flex h-6 w-6 items-center justify-center rounded-md text-tertiary hover:bg-surface-hover transition-colors"
|
|
306
|
-
aria-label={darkMode ? "Switch to light mode" : "Switch to dark mode"}
|
|
307
|
-
>
|
|
308
|
-
{darkMode ? <Sun className="h-3.5 w-3.5" /> : <Moon className="h-3.5 w-3.5" />}
|
|
309
|
-
</button>
|
|
310
|
-
|
|
311
|
-
<label className="flex items-center gap-1.5 text-tertiary cursor-pointer select-none">
|
|
312
|
-
<input checked={reducedMotion} type="checkbox" className="h-3 w-3 accent-accent" onChange={(e) => setReducedMotion(e.target.checked)} />
|
|
313
|
-
Reduced motion
|
|
314
|
-
</label>
|
|
315
|
-
</div>
|
|
316
|
-
|
|
317
|
-
{/* Editor shell */}
|
|
318
|
-
<div className="flex flex-1 min-h-0">
|
|
319
|
-
{/* Main column */}
|
|
320
|
-
<div className="flex flex-1 flex-col min-w-0">
|
|
321
|
-
{/* Toolbar */}
|
|
322
|
-
<header className="flex h-10 shrink-0 items-center gap-1 border-b border-border px-2">
|
|
323
|
-
<div className="flex items-center gap-0.5">
|
|
324
|
-
<ToolbarIconButton icon={Undo2} label="Undo" disabled={!snap.commandState.canUndo} />
|
|
325
|
-
<ToolbarIconButton icon={Redo2} label="Redo" disabled={!snap.commandState.canRedo} />
|
|
326
|
-
<div className="mx-1 h-4 w-px bg-border" />
|
|
327
|
-
<Select.Root defaultValue="body">
|
|
328
|
-
<Select.Trigger className={`inline-flex items-center gap-1 rounded-md px-2 py-1 text-xs text-primary transition-colors hover:bg-surface outline-none ${focusRingClass}`}>
|
|
329
|
-
<Select.Value />
|
|
330
|
-
<Select.Icon><ChevronDown className="h-3 w-3 text-tertiary" /></Select.Icon>
|
|
331
|
-
</Select.Trigger>
|
|
332
|
-
<Select.Portal>
|
|
333
|
-
<Select.Content className="overflow-hidden rounded-lg bg-canvas shadow-lg ring-1 ring-border z-50" position="popper" sideOffset={4}>
|
|
334
|
-
<Select.Viewport className="p-1">
|
|
335
|
-
{["Body", "Heading 1", "Heading 2"].map((v) => (
|
|
336
|
-
<Select.Item key={v} value={v.toLowerCase().replace(" ", "")} className={`flex cursor-pointer items-center rounded-md px-2 py-1.5 text-xs outline-none data-[highlighted]:bg-surface ${focusRingClass}`}>
|
|
337
|
-
<Select.ItemText>{v}</Select.ItemText>
|
|
338
|
-
</Select.Item>
|
|
339
|
-
))}
|
|
340
|
-
</Select.Viewport>
|
|
341
|
-
</Select.Content>
|
|
342
|
-
</Select.Portal>
|
|
343
|
-
</Select.Root>
|
|
344
|
-
<div className="mx-1 h-4 w-px bg-border" />
|
|
345
|
-
<ToolbarIconButton icon={Bold} label="Bold" disabled={snap.readOnly} />
|
|
346
|
-
<ToolbarIconButton icon={Italic} label="Italic" disabled={snap.readOnly} />
|
|
347
|
-
<ToolbarIconButton icon={Underline} label="Underline" disabled={snap.readOnly} />
|
|
348
|
-
</div>
|
|
349
|
-
|
|
350
|
-
<div className="flex-1 text-center min-w-0">
|
|
351
|
-
<span className="text-sm font-medium truncate block">{snap.sourceLabel}</span>
|
|
352
|
-
</div>
|
|
353
|
-
|
|
354
|
-
<div className="flex items-center gap-0.5">
|
|
355
|
-
<ToolbarIconButton icon={MessageSquare} label="Comments" disabled={snap.readOnly || snap.comments.totalCount === 0} onClick={() => { if (currentComments[0]) activateComment(currentComments[0]); }} />
|
|
356
|
-
<Tooltip.Root>
|
|
357
|
-
<Tooltip.Trigger asChild>
|
|
358
|
-
<Toggle.Root pressed={trackChangesEnabled} onPressedChange={setTrackChangesEnabled} className={`inline-flex h-7 w-7 items-center justify-center rounded-md text-secondary transition-colors hover:bg-surface data-[state=on]:bg-accent-soft data-[state=on]:text-accent outline-none disabled:opacity-40 ${focusRingClass}`}>
|
|
359
|
-
<GitCompareArrows className="h-4 w-4" />
|
|
360
|
-
</Toggle.Root>
|
|
361
|
-
</Tooltip.Trigger>
|
|
362
|
-
<Tooltip.Portal><Tooltip.Content className="rounded-md bg-primary px-2 py-1 text-xs text-white shadow-md z-50" sideOffset={6}>Track changes</Tooltip.Content></Tooltip.Portal>
|
|
363
|
-
</Tooltip.Root>
|
|
364
|
-
<div className="mx-1 h-4 w-px bg-border" />
|
|
365
|
-
<ToggleGroup.Root type="single" value={markupDisplay} onValueChange={(v) => { if (v) setMarkupDisplay(v as DemoMarkupDisplay); }} className="flex items-center gap-0.5">
|
|
366
|
-
{(["clean", "simple", "all"] as const).map((mode) => (
|
|
367
|
-
<Tooltip.Root key={mode}>
|
|
368
|
-
<Tooltip.Trigger asChild>
|
|
369
|
-
<ToggleGroup.Item value={mode} className={`inline-flex h-7 items-center rounded-md px-2 text-xs text-secondary transition-colors hover:bg-surface data-[state=on]:bg-accent-soft data-[state=on]:text-accent outline-none ${focusRingClass}`}>
|
|
370
|
-
{mode === "all" ? "Full" : mode[0].toUpperCase() + mode.slice(1)}
|
|
371
|
-
</ToggleGroup.Item>
|
|
372
|
-
</Tooltip.Trigger>
|
|
373
|
-
<Tooltip.Portal><Tooltip.Content className="rounded-md bg-primary px-2 py-1 text-xs text-white shadow-md z-50" sideOffset={6}>{mode === "clean" ? "Hide markup" : mode === "simple" ? "Simple markup" : "Full markup"}</Tooltip.Content></Tooltip.Portal>
|
|
374
|
-
</Tooltip.Root>
|
|
375
|
-
))}
|
|
376
|
-
</ToggleGroup.Root>
|
|
377
|
-
<div className="mx-1 h-4 w-px bg-border" />
|
|
378
|
-
<ToggleGroup.Root type="single" value={viewMode} onValueChange={(v) => { if (v) setViewMode(v as DemoViewMode); }} className="flex items-center gap-0.5">
|
|
379
|
-
<Tooltip.Root>
|
|
380
|
-
<Tooltip.Trigger asChild><ToggleGroup.Item value="canvas" className={`inline-flex h-7 w-7 items-center justify-center rounded-md text-secondary transition-colors hover:bg-surface data-[state=on]:bg-accent-soft data-[state=on]:text-accent outline-none ${focusRingClass}`}><Monitor className="h-3.5 w-3.5" /></ToggleGroup.Item></Tooltip.Trigger>
|
|
381
|
-
<Tooltip.Portal><Tooltip.Content className="rounded-md bg-primary px-2 py-1 text-xs text-white shadow-md z-50" sideOffset={6}>Canvas view</Tooltip.Content></Tooltip.Portal>
|
|
382
|
-
</Tooltip.Root>
|
|
383
|
-
<Tooltip.Root>
|
|
384
|
-
<Tooltip.Trigger asChild><ToggleGroup.Item value="page" className={`inline-flex h-7 w-7 items-center justify-center rounded-md text-secondary transition-colors hover:bg-surface data-[state=on]:bg-accent-soft data-[state=on]:text-accent outline-none ${focusRingClass}`}><FileText className="h-3.5 w-3.5" /></ToggleGroup.Item></Tooltip.Trigger>
|
|
385
|
-
<Tooltip.Portal><Tooltip.Content className="rounded-md bg-primary px-2 py-1 text-xs text-white shadow-md z-50" sideOffset={6}>Page view</Tooltip.Content></Tooltip.Portal>
|
|
386
|
-
</Tooltip.Root>
|
|
387
|
-
</ToggleGroup.Root>
|
|
388
|
-
<div className="mx-1 h-4 w-px bg-border" />
|
|
389
|
-
<Tooltip.Root>
|
|
390
|
-
<Tooltip.Trigger asChild>
|
|
391
|
-
<button type="button" disabled={isExportBlocked} className={`inline-flex h-7 items-center gap-1.5 rounded-md px-2.5 text-xs font-semibold transition-colors outline-none ${focusRingClass} ${isExportBlocked ? "cursor-not-allowed text-danger opacity-50" : "text-accent hover:bg-accent-soft"}`}>
|
|
392
|
-
<Download className="h-3.5 w-3.5" />Export
|
|
393
|
-
</button>
|
|
394
|
-
</Tooltip.Trigger>
|
|
395
|
-
<Tooltip.Portal><Tooltip.Content className="rounded-md bg-primary px-2 py-1 text-xs text-white shadow-md z-50" sideOffset={6}>{isExportBlocked ? "Export blocked by unsupported content" : "Export document"}</Tooltip.Content></Tooltip.Portal>
|
|
396
|
-
</Tooltip.Root>
|
|
397
|
-
</div>
|
|
398
|
-
</header>
|
|
399
|
-
|
|
400
|
-
{/* Alert banner */}
|
|
401
|
-
{snap.fatalError ? (
|
|
402
|
-
<div className="flex items-center gap-2 px-3 py-1.5 bg-danger-soft text-danger text-xs">
|
|
403
|
-
<XCircle className="h-3.5 w-3.5 shrink-0" />
|
|
404
|
-
<span>{snap.fatalError.message}</span>
|
|
405
|
-
</div>
|
|
406
|
-
) : isExportBlocked ? (
|
|
407
|
-
<div className="flex items-center gap-2 px-3 py-1.5 bg-danger-soft text-danger text-xs">
|
|
408
|
-
<XCircle className="h-3.5 w-3.5 shrink-0" />
|
|
409
|
-
<span>Export blocked — {snap.compatibility.blockExportReasons[0] ?? "unsupported content"}</span>
|
|
410
|
-
</div>
|
|
411
|
-
) : preserveOnlyCount > 0 ? (
|
|
412
|
-
<div className="flex items-center gap-2 px-3 py-1.5 bg-warning-soft text-comment text-xs">
|
|
413
|
-
<AlertTriangle className="h-3.5 w-3.5 shrink-0" />
|
|
414
|
-
<span>{preserveOnlyCount} preserve-only feature{preserveOnlyCount !== 1 ? "s" : ""} detected</span>
|
|
415
|
-
</div>
|
|
416
|
-
) : null}
|
|
417
|
-
|
|
418
|
-
{/* Document canvas */}
|
|
419
|
-
<div className="flex-1 overflow-y-auto bg-surface">
|
|
420
|
-
<div className={`mx-auto max-w-[780px] min-h-full bg-canvas ${viewMode === "page" ? "my-8 rounded-xl ring-1 ring-border shadow-sm" : ""}`}>
|
|
421
|
-
<div className={`px-12 py-10 ${markupDisplay === "clean" ? "font-[family-name:var(--font-legal-sans)]" : "font-[family-name:var(--font-legal-serif)]"}`}>
|
|
422
|
-
<h1 className="text-2xl font-medium text-primary mb-1">{snap.sourceLabel}</h1>
|
|
423
|
-
<p className="text-xs text-tertiary mb-8">{snap.sessionId} · {snap.documentStats.storyLength} chars</p>
|
|
424
|
-
|
|
425
|
-
{selectionLabel ? (
|
|
426
|
-
<div className="mb-6 inline-flex items-center gap-1 rounded-lg bg-canvas shadow-lg ring-1 ring-border p-1">
|
|
427
|
-
{selectionActions.map((action) => (
|
|
428
|
-
<Tooltip.Root key={action.label}>
|
|
429
|
-
<Tooltip.Trigger asChild>
|
|
430
|
-
<button type="button" aria-label={action.label} className="inline-flex h-7 w-7 items-center justify-center rounded-md text-secondary hover:bg-surface hover:text-primary transition-colors">
|
|
431
|
-
<action.icon className="h-3.5 w-3.5" />
|
|
432
|
-
</button>
|
|
433
|
-
</Tooltip.Trigger>
|
|
434
|
-
<Tooltip.Portal><Tooltip.Content className="rounded-md bg-primary px-2 py-1 text-xs text-white shadow-md z-50" sideOffset={6}>{action.label}</Tooltip.Content></Tooltip.Portal>
|
|
435
|
-
</Tooltip.Root>
|
|
436
|
-
))}
|
|
437
|
-
<div className="h-4 w-px bg-border mx-0.5" />
|
|
438
|
-
<span className="text-xs text-tertiary px-2 max-w-[200px] truncate">{selectionLabel}</span>
|
|
439
|
-
</div>
|
|
440
|
-
) : null}
|
|
441
|
-
|
|
442
|
-
<div className="space-y-4">
|
|
443
|
-
{snap.surface.blocks.map((block) => (
|
|
444
|
-
<BlockView
|
|
445
|
-
key={block.blockId}
|
|
446
|
-
block={block}
|
|
447
|
-
markupDisplay={markupDisplay}
|
|
448
|
-
activeCommentId={activeComment?.commentId}
|
|
449
|
-
activeRevisionId={activeRevision?.revisionId}
|
|
450
|
-
activeCellId={activeCellId}
|
|
451
|
-
commentDecorations={currentSession.commentDecorations}
|
|
452
|
-
revisionDecorations={currentSession.revisionDecorations}
|
|
453
|
-
onSegmentClick={handleSegmentClick}
|
|
454
|
-
onCellClick={handleCellClick}
|
|
455
|
-
/>
|
|
456
|
-
))}
|
|
457
|
-
</div>
|
|
458
|
-
</div>
|
|
459
|
-
</div>
|
|
460
|
-
</div>
|
|
461
|
-
|
|
462
|
-
{/* Status bar */}
|
|
463
|
-
<footer className="flex h-7 shrink-0 items-center gap-4 border-t border-border px-3 text-xs text-tertiary">
|
|
464
|
-
<span className="flex items-center gap-1.5">
|
|
465
|
-
<span className={`inline-block h-1.5 w-1.5 rounded-full ${isExportBlocked ? "bg-danger" : isDirty ? "bg-comment" : "bg-insert"}`} />
|
|
466
|
-
{saveState}
|
|
467
|
-
</span>
|
|
468
|
-
<span className="flex items-center gap-1.5">
|
|
469
|
-
<span className={`inline-block h-1.5 w-1.5 rounded-full ${isExportBlocked ? "bg-danger" : preserveOnlyCount > 0 ? "bg-comment" : "bg-insert"}`} />
|
|
470
|
-
Export {exportState.toLowerCase()}
|
|
471
|
-
</span>
|
|
472
|
-
<span>{snap.comments.totalCount} comment{snap.comments.totalCount !== 1 ? "s" : ""} · {snap.trackedChanges.totalCount} change{snap.trackedChanges.totalCount !== 1 ? "s" : ""}</span>
|
|
473
|
-
<span className="flex-1" />
|
|
474
|
-
<span>{snap.sessionId}</span>
|
|
475
|
-
</footer>
|
|
476
|
-
</div>
|
|
477
|
-
|
|
478
|
-
{/* Review rail */}
|
|
479
|
-
<aside className="flex w-[340px] shrink-0 flex-col border-l border-border bg-canvas">
|
|
480
|
-
<Tabs.Root value={activeRailTab} onValueChange={(v) => setActiveRailTab(v as DemoRailTab)} className="flex flex-1 flex-col min-h-0">
|
|
481
|
-
<Tabs.List className="flex shrink-0 border-b border-border">
|
|
482
|
-
<Tabs.Trigger value="comments" className={`flex-1 py-2 text-xs text-tertiary transition-colors data-[state=active]:text-primary data-[state=active]:shadow-[inset_0_-2px_0_var(--color-accent)] outline-none ${focusRingClass}`}>
|
|
483
|
-
Comments <span className="text-tertiary">{snap.comments.totalCount}</span>
|
|
484
|
-
</Tabs.Trigger>
|
|
485
|
-
<Tabs.Trigger value="changes" className={`flex-1 py-2 text-xs text-tertiary transition-colors data-[state=active]:text-primary data-[state=active]:shadow-[inset_0_-2px_0_var(--color-accent)] outline-none ${focusRingClass}`}>
|
|
486
|
-
Changes <span className="text-tertiary">{snap.trackedChanges.totalCount}</span>
|
|
487
|
-
</Tabs.Trigger>
|
|
488
|
-
<Tabs.Trigger value="health" className={`flex-1 py-2 text-xs text-tertiary transition-colors data-[state=active]:text-primary data-[state=active]:shadow-[inset_0_-2px_0_var(--color-accent)] outline-none ${focusRingClass}`}>
|
|
489
|
-
Health {warningCount > 0 ? <span className="text-comment">{warningCount}</span> : <span className="text-tertiary">{snap.compatibility.featureEntries.length}</span>}
|
|
490
|
-
</Tabs.Trigger>
|
|
491
|
-
</Tabs.List>
|
|
492
|
-
|
|
493
|
-
<ScrollArea.Root className="flex-1 min-h-0">
|
|
494
|
-
<ScrollArea.Viewport className="h-full w-full">
|
|
495
|
-
<Tabs.Content value="comments" className="p-3 outline-none">
|
|
496
|
-
<p className="text-xs text-tertiary mb-3">
|
|
497
|
-
{snap.comments.openCommentIds.length} open · {snap.comments.resolvedCommentIds.length} resolved · {snap.comments.detachedCommentIds.length} detached
|
|
498
|
-
</p>
|
|
499
|
-
<div className="space-y-1">
|
|
500
|
-
{currentComments.map((thread) => (
|
|
501
|
-
<div
|
|
502
|
-
key={thread.commentId}
|
|
503
|
-
role="button"
|
|
504
|
-
tabIndex={0}
|
|
505
|
-
className={`rounded-lg p-2.5 transition-colors cursor-pointer ${focusRingClass} ${activeComment?.commentId === thread.commentId ? "bg-accent-soft" : "hover:bg-surface"}`}
|
|
506
|
-
onClick={() => activateComment(thread)}
|
|
507
|
-
onKeyDown={(event) => {
|
|
508
|
-
if (event.key === "Enter" || event.key === " ") {
|
|
509
|
-
event.preventDefault();
|
|
510
|
-
activateComment(thread);
|
|
511
|
-
}
|
|
512
|
-
}}
|
|
513
|
-
>
|
|
514
|
-
<div className="flex items-start justify-between gap-2 mb-1">
|
|
515
|
-
<span className="text-sm font-medium text-primary">{thread.createdBy}</span>
|
|
516
|
-
<StatusBadge status={thread.status} />
|
|
517
|
-
</div>
|
|
518
|
-
<p className="text-xs text-tertiary mb-1">{thread.createdAt}</p>
|
|
519
|
-
<p className="text-xs font-medium text-comment bg-comment-soft rounded px-1 py-0.5 inline-block mb-1.5">{thread.excerpt}</p>
|
|
520
|
-
<p className="text-sm text-secondary leading-relaxed">{thread.body}</p>
|
|
521
|
-
{thread.entryCount > 1 ? <p className="text-xs text-tertiary mt-1.5">{thread.entryCount - 1} repl{thread.entryCount - 1 === 1 ? "y" : "ies"}</p> : null}
|
|
522
|
-
<div className="flex gap-1.5 mt-2">
|
|
523
|
-
{thread.status === "open" ? (
|
|
524
|
-
<button type="button" className="inline-flex items-center gap-1 rounded-md px-2 py-1 text-xs text-insert hover:bg-insert-soft transition-colors" onClick={(e) => { e.stopPropagation(); updateCommentStatus(thread, "resolved"); }}>
|
|
525
|
-
<Check className="h-3 w-3" /> Resolve
|
|
526
|
-
</button>
|
|
527
|
-
) : thread.status === "resolved" ? (
|
|
528
|
-
<button type="button" className="inline-flex items-center gap-1 rounded-md px-2 py-1 text-xs text-secondary hover:bg-surface transition-colors" onClick={(e) => { e.stopPropagation(); updateCommentStatus(thread, "open"); }}>
|
|
529
|
-
Reopen
|
|
530
|
-
</button>
|
|
531
|
-
) : (
|
|
532
|
-
<span className="text-xs text-tertiary px-2 py-1">Detached</span>
|
|
533
|
-
)}
|
|
534
|
-
</div>
|
|
535
|
-
</div>
|
|
536
|
-
))}
|
|
537
|
-
</div>
|
|
538
|
-
</Tabs.Content>
|
|
539
|
-
|
|
540
|
-
<Tabs.Content value="changes" className="p-3 outline-none">
|
|
541
|
-
<p className="text-xs text-tertiary mb-3">
|
|
542
|
-
{snap.trackedChanges.pendingChangeIds.length} active · {snap.trackedChanges.acceptedChangeIds.length} accepted · {snap.trackedChanges.preserveOnlyChangeIds.length} preserve-only
|
|
543
|
-
</p>
|
|
544
|
-
<div className="space-y-1">
|
|
545
|
-
{currentRevisions.map((rev) => (
|
|
546
|
-
<div
|
|
547
|
-
key={rev.revisionId}
|
|
548
|
-
role="button"
|
|
549
|
-
tabIndex={0}
|
|
550
|
-
className={`flex rounded-lg transition-colors cursor-pointer ${focusRingClass} ${activeRevision?.revisionId === rev.revisionId ? "bg-accent-soft" : "hover:bg-surface"}`}
|
|
551
|
-
onClick={() => activateRevision(rev)}
|
|
552
|
-
onKeyDown={(event) => {
|
|
553
|
-
if (event.key === "Enter" || event.key === " ") {
|
|
554
|
-
event.preventDefault();
|
|
555
|
-
activateRevision(rev);
|
|
556
|
-
}
|
|
557
|
-
}}
|
|
558
|
-
>
|
|
559
|
-
<div className={`w-0.5 shrink-0 rounded-l-lg ${rev.kind === "insertion" ? "bg-insert" : rev.kind === "deletion" ? "bg-danger" : "bg-tertiary"}`} />
|
|
560
|
-
<div className="p-2.5 flex-1 min-w-0">
|
|
561
|
-
<div className="flex items-start justify-between gap-2 mb-1">
|
|
562
|
-
<span className="text-sm font-medium text-primary">{rev.anchorLabel}</span>
|
|
563
|
-
<RevisionBadge status={rev.status} actionability={rev.actionability} />
|
|
564
|
-
</div>
|
|
565
|
-
<p className="text-xs text-tertiary mb-1">{rev.authorId} · {rev.createdAt}</p>
|
|
566
|
-
<p className={`text-sm ${rev.kind === "insertion" ? "text-insert" : rev.kind === "deletion" ? "text-danger line-through" : "text-secondary"}`}>{rev.excerpt}</p>
|
|
567
|
-
<p className="text-xs text-secondary mt-1">{rev.detail}</p>
|
|
568
|
-
<div className="flex gap-1.5 mt-2">
|
|
569
|
-
<button type="button" disabled={!rev.canAccept || rev.status === "accepted"} className="inline-flex items-center gap-1 rounded-md px-2 py-1 text-xs text-insert hover:bg-insert-soft transition-colors disabled:opacity-30 disabled:cursor-not-allowed" onClick={(e) => { e.stopPropagation(); updateRevisionStatus(rev, "accepted"); }}>
|
|
570
|
-
<Check className="h-3 w-3" /> Accept
|
|
571
|
-
</button>
|
|
572
|
-
<button type="button" disabled={!rev.canReject || rev.status === "rejected"} className="inline-flex items-center gap-1 rounded-md px-2 py-1 text-xs text-danger hover:bg-delete-soft transition-colors disabled:opacity-30 disabled:cursor-not-allowed" onClick={(e) => { e.stopPropagation(); updateRevisionStatus(rev, "rejected"); }}>
|
|
573
|
-
<X className="h-3 w-3" /> Reject
|
|
574
|
-
</button>
|
|
575
|
-
</div>
|
|
576
|
-
</div>
|
|
577
|
-
</div>
|
|
578
|
-
))}
|
|
579
|
-
</div>
|
|
580
|
-
</Tabs.Content>
|
|
581
|
-
|
|
582
|
-
<Tabs.Content value="health" className="p-3 outline-none">
|
|
583
|
-
<p className="text-xs text-tertiary mb-3">
|
|
584
|
-
{snap.compatibility.featureEntries.filter((e) => e.featureClass === "supported-roundtrip").length} supported · {snap.compatibility.featureEntries.filter((e) => e.featureClass === "preserve-only").length} preserve-only · {snap.compatibility.featureEntries.filter((e) => e.featureClass === "unsupported-fatal").length} blocked{snap.warnings.length > 0 ? ` · ${snap.warnings.length} warning${snap.warnings.length !== 1 ? "s" : ""}` : ""}
|
|
585
|
-
</p>
|
|
586
|
-
<div className="space-y-1">
|
|
587
|
-
{snap.compatibility.featureEntries.map((entry) => (
|
|
588
|
-
<div key={entry.featureEntryId} className="flex rounded-lg transition-colors hover:bg-surface">
|
|
589
|
-
{entry.featureClass !== "supported-roundtrip" ? (
|
|
590
|
-
<div className={`w-0.5 shrink-0 rounded-l-lg ${entry.featureClass === "unsupported-fatal" ? "bg-danger" : "bg-comment"}`} />
|
|
591
|
-
) : null}
|
|
592
|
-
<div className="flex items-start gap-2 mb-1 p-2.5 flex-1">
|
|
593
|
-
<HealthIcon featureClass={entry.featureClass} />
|
|
594
|
-
<div className="flex-1 min-w-0">
|
|
595
|
-
<div className="flex items-start justify-between gap-2">
|
|
596
|
-
<span className="text-sm font-medium text-primary">{entry.message}</span>
|
|
597
|
-
<FeatureClassBadge featureClass={entry.featureClass} />
|
|
598
|
-
</div>
|
|
599
|
-
<p className="text-sm text-secondary mt-0.5">{(entry as PrototypeFeatureEntry).summary}</p>
|
|
600
|
-
</div>
|
|
601
|
-
</div>
|
|
602
|
-
</div>
|
|
603
|
-
))}
|
|
604
|
-
{snap.warnings.map((warning) => (
|
|
605
|
-
<div key={warning.warningId} className="flex rounded-lg transition-colors hover:bg-surface">
|
|
606
|
-
<div className={`w-0.5 shrink-0 rounded-l-lg ${warning.severity === "warning" ? "bg-comment" : "bg-accent"}`} />
|
|
607
|
-
<div className="flex items-start gap-2 p-2.5 flex-1">
|
|
608
|
-
{warning.severity === "warning" ? (
|
|
609
|
-
<AlertTriangle className="h-4 w-4 text-comment shrink-0 mt-0.5" />
|
|
610
|
-
) : (
|
|
611
|
-
<Info className="h-4 w-4 text-accent shrink-0 mt-0.5" />
|
|
612
|
-
)}
|
|
613
|
-
<div className="flex-1 min-w-0">
|
|
614
|
-
<div className="flex items-start justify-between gap-2">
|
|
615
|
-
<span className="text-sm font-medium text-primary">{warning.message}</span>
|
|
616
|
-
<span className={`inline-flex items-center rounded px-1.5 py-0.5 text-[10px] font-medium ${warning.severity === "warning" ? "text-comment bg-warning-soft" : "text-accent bg-accent-soft"}`}>
|
|
617
|
-
{warning.code.replace(/_/g, " ")}
|
|
618
|
-
</span>
|
|
619
|
-
</div>
|
|
620
|
-
<p className="text-xs text-tertiary mt-0.5">{warning.source}</p>
|
|
621
|
-
</div>
|
|
622
|
-
</div>
|
|
623
|
-
</div>
|
|
624
|
-
))}
|
|
625
|
-
</div>
|
|
626
|
-
</Tabs.Content>
|
|
627
|
-
</ScrollArea.Viewport>
|
|
628
|
-
<ScrollArea.Scrollbar orientation="vertical" className="flex w-1.5 touch-none select-none p-0.5">
|
|
629
|
-
<ScrollArea.Thumb className="relative flex-1 rounded-full bg-black/[0.12]" />
|
|
630
|
-
</ScrollArea.Scrollbar>
|
|
631
|
-
</ScrollArea.Root>
|
|
632
|
-
</Tabs.Root>
|
|
633
|
-
</aside>
|
|
634
|
-
</div>
|
|
635
|
-
</main>
|
|
636
|
-
</Tooltip.Provider>
|
|
637
|
-
);
|
|
638
|
-
}
|
|
639
|
-
|
|
640
|
-
/* ─── Small components ─── */
|
|
641
|
-
|
|
642
|
-
function ToolbarIconButton(props: { icon: React.ComponentType<{ className?: string }>; label: string; disabled?: boolean; onClick?: () => void }) {
|
|
643
|
-
return (
|
|
644
|
-
<Tooltip.Root>
|
|
645
|
-
<Tooltip.Trigger asChild>
|
|
646
|
-
<button type="button" disabled={props.disabled} onClick={props.onClick} className={`inline-flex h-7 w-7 items-center justify-center rounded-md text-secondary transition-colors hover:bg-surface hover:text-primary outline-none disabled:opacity-30 disabled:cursor-not-allowed ${focusRingClass}`}>
|
|
647
|
-
<props.icon className="h-4 w-4" />
|
|
648
|
-
</button>
|
|
649
|
-
</Tooltip.Trigger>
|
|
650
|
-
<Tooltip.Portal><Tooltip.Content className="rounded-md bg-primary px-2 py-1 text-xs text-white shadow-md z-50" sideOffset={6}>{props.label}</Tooltip.Content></Tooltip.Portal>
|
|
651
|
-
</Tooltip.Root>
|
|
652
|
-
);
|
|
653
|
-
}
|
|
654
|
-
|
|
655
|
-
function StatusBadge(props: { status: string }) {
|
|
656
|
-
const styles: Record<string, string> = {
|
|
657
|
-
open: "text-accent bg-accent-soft",
|
|
658
|
-
resolved: "text-insert bg-insert-soft",
|
|
659
|
-
detached: "text-comment bg-warning-soft",
|
|
660
|
-
};
|
|
661
|
-
return <span className={`inline-flex items-center rounded px-1.5 py-0.5 text-[10px] font-medium ${styles[props.status] ?? "text-secondary bg-subtle"}`}>{props.status}</span>;
|
|
662
|
-
}
|
|
663
|
-
|
|
664
|
-
function RevisionBadge(props: { status: string; actionability: string }) {
|
|
665
|
-
if (props.actionability === "preserve-only") return <span className="inline-flex items-center rounded px-1.5 py-0.5 text-[10px] font-medium text-comment bg-warning-soft">preserve-only</span>;
|
|
666
|
-
const styles: Record<string, string> = {
|
|
667
|
-
active: "text-secondary bg-subtle",
|
|
668
|
-
accepted: "text-insert bg-insert-soft",
|
|
669
|
-
rejected: "text-danger bg-delete-soft",
|
|
670
|
-
detached: "text-comment bg-warning-soft",
|
|
671
|
-
};
|
|
672
|
-
return <span className={`inline-flex items-center rounded px-1.5 py-0.5 text-[10px] font-medium ${styles[props.status] ?? "text-secondary bg-subtle"}`}>{props.status}</span>;
|
|
673
|
-
}
|
|
674
|
-
|
|
675
|
-
function FeatureClassBadge(props: { featureClass: CompatibilityFeatureClass }) {
|
|
676
|
-
const styles: Record<CompatibilityFeatureClass, string> = {
|
|
677
|
-
"supported-roundtrip": "text-insert bg-insert-soft",
|
|
678
|
-
"preserve-only": "text-comment bg-warning-soft",
|
|
679
|
-
"unsupported-fatal": "text-danger bg-delete-soft",
|
|
680
|
-
};
|
|
681
|
-
const labels: Record<CompatibilityFeatureClass, string> = {
|
|
682
|
-
"supported-roundtrip": "supported",
|
|
683
|
-
"preserve-only": "preserve-only",
|
|
684
|
-
"unsupported-fatal": "blocked",
|
|
685
|
-
};
|
|
686
|
-
return <span className={`inline-flex items-center rounded px-1.5 py-0.5 text-[10px] font-medium ${styles[props.featureClass]}`}>{labels[props.featureClass]}</span>;
|
|
687
|
-
}
|
|
688
|
-
|
|
689
|
-
function HealthIcon(props: { featureClass: CompatibilityFeatureClass }) {
|
|
690
|
-
switch (props.featureClass) {
|
|
691
|
-
case "supported-roundtrip": return <ShieldCheck className="h-4 w-4 text-insert shrink-0 mt-0.5" />;
|
|
692
|
-
case "preserve-only": return <Shield className="h-4 w-4 text-comment shrink-0 mt-0.5" />;
|
|
693
|
-
case "unsupported-fatal": return <ShieldAlert className="h-4 w-4 text-danger shrink-0 mt-0.5" />;
|
|
694
|
-
}
|
|
695
|
-
}
|
|
696
|
-
|
|
697
|
-
/* ─── Document surface ─── */
|
|
698
|
-
|
|
699
|
-
function BlockView(props: {
|
|
700
|
-
block: PrototypeSurfaceBlock;
|
|
701
|
-
markupDisplay: DemoMarkupDisplay;
|
|
702
|
-
activeCommentId?: string;
|
|
703
|
-
activeRevisionId?: string;
|
|
704
|
-
activeCellId?: string;
|
|
705
|
-
commentDecorations: PrototypeCommentDecoration[];
|
|
706
|
-
revisionDecorations: PrototypeRevisionDecoration[];
|
|
707
|
-
onSegmentClick: (segmentId: string, text: string, cellId?: string) => void;
|
|
708
|
-
onCellClick?: (cellId: string) => void;
|
|
709
|
-
}) {
|
|
710
|
-
const { block } = props;
|
|
711
|
-
|
|
712
|
-
// Prototype-only: table (mirrors proposed TableNode > TableRowNode > TableCellNode)
|
|
713
|
-
if (block.kind === "table") {
|
|
714
|
-
const tb = block as PrototypeTableBlock;
|
|
715
|
-
return (
|
|
716
|
-
<div className="group relative">
|
|
717
|
-
<BlockHint label="Table" badge="prototype" />
|
|
718
|
-
{tb.caption ? <p className="text-xs text-tertiary mb-1.5">{tb.caption}</p> : null}
|
|
719
|
-
<div className="overflow-x-auto rounded-md ring-1 ring-border">
|
|
720
|
-
<table className="w-full text-sm border-collapse">
|
|
721
|
-
<tbody>
|
|
722
|
-
{tb.rows.map((row) => (
|
|
723
|
-
<tr key={row.rowId}>
|
|
724
|
-
{row.cells.map((cell) => {
|
|
725
|
-
const CellTag = row.isHeader ? "th" : "td";
|
|
726
|
-
const isCellActive = !row.isHeader && props.activeCellId === cell.cellId;
|
|
727
|
-
return (
|
|
728
|
-
<CellTag
|
|
729
|
-
key={cell.cellId}
|
|
730
|
-
colSpan={cell.colSpan > 1 ? cell.colSpan : undefined}
|
|
731
|
-
rowSpan={cell.rowSpan > 1 ? cell.rowSpan : undefined}
|
|
732
|
-
onClick={!row.isHeader ? () => props.onCellClick?.(cell.cellId) : undefined}
|
|
733
|
-
onKeyDown={
|
|
734
|
-
!row.isHeader
|
|
735
|
-
? (event) => {
|
|
736
|
-
if (event.key === "Enter" || event.key === " ") {
|
|
737
|
-
event.preventDefault();
|
|
738
|
-
props.onCellClick?.(cell.cellId);
|
|
739
|
-
}
|
|
740
|
-
}
|
|
741
|
-
: undefined
|
|
742
|
-
}
|
|
743
|
-
role={!row.isHeader ? "button" : undefined}
|
|
744
|
-
tabIndex={!row.isHeader ? 0 : undefined}
|
|
745
|
-
className={`text-left p-2.5 border-b border-border text-sm transition-colors ${!row.isHeader ? `cursor-pointer ${focusRingClass}` : "cursor-default"} ${
|
|
746
|
-
row.isHeader
|
|
747
|
-
? "font-medium text-secondary bg-surface text-xs"
|
|
748
|
-
: cell.preserveOnly
|
|
749
|
-
? "text-secondary bg-warning-soft/30"
|
|
750
|
-
: isCellActive
|
|
751
|
-
? "ring-1 ring-accent/30 bg-accent-soft/30"
|
|
752
|
-
: "text-primary hover:bg-surface/50"
|
|
753
|
-
}`}
|
|
754
|
-
>
|
|
755
|
-
{cell.preserveOnly ? (
|
|
756
|
-
<span className="flex items-center gap-1.5">
|
|
757
|
-
<Lock className="h-3 w-3 text-tertiary shrink-0" />
|
|
758
|
-
<span>
|
|
759
|
-
{cell.segments.map((seg) => (
|
|
760
|
-
<SegmentView
|
|
761
|
-
key={seg.segmentId}
|
|
762
|
-
segment={seg}
|
|
763
|
-
markupDisplay={props.markupDisplay}
|
|
764
|
-
activeCommentId={props.activeCommentId}
|
|
765
|
-
activeRevisionId={props.activeRevisionId}
|
|
766
|
-
commentDecorations={props.commentDecorations}
|
|
767
|
-
revisionDecorations={props.revisionDecorations}
|
|
768
|
-
cellId={cell.cellId}
|
|
769
|
-
onClick={props.onSegmentClick}
|
|
770
|
-
/>
|
|
771
|
-
))}
|
|
772
|
-
</span>
|
|
773
|
-
</span>
|
|
774
|
-
) : (
|
|
775
|
-
cell.segments.map((seg) => (
|
|
776
|
-
<SegmentView
|
|
777
|
-
key={seg.segmentId}
|
|
778
|
-
segment={seg}
|
|
779
|
-
markupDisplay={props.markupDisplay}
|
|
780
|
-
activeCommentId={props.activeCommentId}
|
|
781
|
-
activeRevisionId={props.activeRevisionId}
|
|
782
|
-
commentDecorations={props.commentDecorations}
|
|
783
|
-
revisionDecorations={props.revisionDecorations}
|
|
784
|
-
cellId={cell.cellId}
|
|
785
|
-
onClick={props.onSegmentClick}
|
|
786
|
-
/>
|
|
787
|
-
))
|
|
788
|
-
)}
|
|
789
|
-
</CellTag>
|
|
790
|
-
);
|
|
791
|
-
})}
|
|
792
|
-
</tr>
|
|
793
|
-
))}
|
|
794
|
-
</tbody>
|
|
795
|
-
</table>
|
|
796
|
-
</div>
|
|
797
|
-
{tb.note ? <p className="text-xs text-tertiary mt-1.5">{tb.note}</p> : null}
|
|
798
|
-
</div>
|
|
799
|
-
);
|
|
800
|
-
}
|
|
801
|
-
|
|
802
|
-
// Prototype-only: callout
|
|
803
|
-
if (block.kind === "callout") {
|
|
804
|
-
const cb = block as PrototypeCalloutBlock;
|
|
805
|
-
const toneColors: Record<string, string> = {
|
|
806
|
-
warning: "border-l-warning/30 bg-warning-soft/40",
|
|
807
|
-
danger: "border-l-danger/30 bg-danger-soft/40",
|
|
808
|
-
success: "border-l-success/30 bg-success-soft/40",
|
|
809
|
-
accent: "border-l-accent/30 bg-accent-soft/40",
|
|
810
|
-
};
|
|
811
|
-
return (
|
|
812
|
-
<div className="group relative">
|
|
813
|
-
<BlockHint label="Note" badge="prototype" />
|
|
814
|
-
<div className={`border-l-2 pl-4 py-2 rounded-r ${toneColors[cb.tone] ?? "border-l-black/10"}`}>
|
|
815
|
-
<p className="text-xs font-medium text-secondary mb-0.5">{cb.title}</p>
|
|
816
|
-
<p className="text-sm text-secondary leading-relaxed">{cb.body}</p>
|
|
817
|
-
</div>
|
|
818
|
-
</div>
|
|
819
|
-
);
|
|
820
|
-
}
|
|
821
|
-
|
|
822
|
-
// Real: opaque_block
|
|
823
|
-
if (block.kind === "opaque_block") {
|
|
824
|
-
return (
|
|
825
|
-
<div className="group relative">
|
|
826
|
-
<BlockHint label="Locked" dots={["var(--color-comment)"]} />
|
|
827
|
-
<div className="border-l-2 border-dashed border-warning/30 pl-4 py-2 rounded-r bg-warning-soft/20">
|
|
828
|
-
<div className="flex items-center gap-1.5 text-xs text-tertiary mb-1">
|
|
829
|
-
<Lock className="h-3 w-3" />
|
|
830
|
-
<span>{block.label}</span>
|
|
831
|
-
<FeatureClassBadge featureClass="preserve-only" />
|
|
832
|
-
</div>
|
|
833
|
-
<p className="text-sm text-secondary">{block.detail}</p>
|
|
834
|
-
</div>
|
|
835
|
-
</div>
|
|
836
|
-
);
|
|
837
|
-
}
|
|
838
|
-
|
|
839
|
-
// Real: paragraph (may be heading via styleId)
|
|
840
|
-
const paraBlock = block as Extract<SurfaceBlockSnapshot, { kind: "paragraph" }>;
|
|
841
|
-
const headingLevel = inferHeadingLevel(paraBlock.styleId);
|
|
842
|
-
const markers = segmentMarkers(paraBlock.segments, props.commentDecorations, props.revisionDecorations);
|
|
843
|
-
const mode = props.markupDisplay;
|
|
844
|
-
|
|
845
|
-
// Margin indicator logic per mode
|
|
846
|
-
const hasComment = markers.includes("comment");
|
|
847
|
-
const hasChange = markers.includes("change");
|
|
848
|
-
const activeCommentInBlock = hasComment && props.activeCommentId && props.commentDecorations.some(
|
|
849
|
-
(d) => d.commentId === props.activeCommentId && d.segmentIds.some((id) => paraBlock.segments.some((s) => s.segmentId === id)),
|
|
850
|
-
);
|
|
851
|
-
|
|
852
|
-
// Clean: tiny dot for comments only | Simple: colored left bar | Full: none (inline styling suffices)
|
|
853
|
-
let marginClass = "";
|
|
854
|
-
if (mode === "simple" && (hasComment || hasChange)) {
|
|
855
|
-
if (hasComment && hasChange) marginClass = "border-l-2 border-l-comment/50 pl-3 ";
|
|
856
|
-
else if (hasComment) marginClass = "border-l-2 border-l-comment/40 pl-3 ";
|
|
857
|
-
else marginClass = "border-l-2 border-l-insert/40 pl-3 ";
|
|
858
|
-
}
|
|
859
|
-
if (activeCommentInBlock && mode !== "clean") {
|
|
860
|
-
marginClass += "border-l-comment! ";
|
|
861
|
-
}
|
|
862
|
-
|
|
863
|
-
const segmentElements = paraBlock.segments.map((seg) => (
|
|
864
|
-
<SegmentView key={seg.segmentId} segment={seg} markupDisplay={mode} activeCommentId={props.activeCommentId} activeRevisionId={props.activeRevisionId} commentDecorations={props.commentDecorations} revisionDecorations={props.revisionDecorations} onClick={props.onSegmentClick} />
|
|
865
|
-
));
|
|
866
|
-
|
|
867
|
-
return (
|
|
868
|
-
<div className="group relative">
|
|
869
|
-
<BlockHint
|
|
870
|
-
label={headingLevel ? `H${headingLevel}` : "P"}
|
|
871
|
-
dots={
|
|
872
|
-
mode === "clean" && hasComment
|
|
873
|
-
? ["var(--color-comment)"]
|
|
874
|
-
: mode !== "simple"
|
|
875
|
-
? markers.map(markerDotColor)
|
|
876
|
-
: undefined
|
|
877
|
-
}
|
|
878
|
-
/>
|
|
879
|
-
{headingLevel ? (
|
|
880
|
-
<div className={`${marginClass} transition-colors`}>
|
|
881
|
-
<HeadingElement level={headingLevel}>{segmentElements}</HeadingElement>
|
|
882
|
-
</div>
|
|
883
|
-
) : (
|
|
884
|
-
<p className={`text-base leading-[1.75] text-primary ${marginClass} transition-colors`}>
|
|
885
|
-
{segmentElements}
|
|
886
|
-
</p>
|
|
887
|
-
)}
|
|
888
|
-
</div>
|
|
889
|
-
);
|
|
890
|
-
}
|
|
891
|
-
|
|
892
|
-
function BlockHint(props: { label: string; dots?: string[]; badge?: string }) {
|
|
893
|
-
return (
|
|
894
|
-
<div className="absolute -left-10 top-1 flex flex-col items-end gap-0 select-none w-8 text-right">
|
|
895
|
-
<div className="flex items-center gap-0.5">
|
|
896
|
-
<span className="text-[10px] text-tertiary/60 font-medium">{props.label}</span>
|
|
897
|
-
{props.dots?.map((color, i) => <span key={i} className="inline-block h-1 w-1 rounded-full" style={{ backgroundColor: color }} />)}
|
|
898
|
-
</div>
|
|
899
|
-
{props.badge ? <span className="text-[7px] text-tertiary/30 leading-none">{props.badge}</span> : null}
|
|
900
|
-
</div>
|
|
901
|
-
);
|
|
902
|
-
}
|
|
903
|
-
|
|
904
|
-
function HeadingElement(props: { level: 1 | 2 | 3; children: React.ReactNode }) {
|
|
905
|
-
switch (props.level) {
|
|
906
|
-
case 1: return <h2 className="text-2xl font-medium text-primary leading-tight">{props.children}</h2>;
|
|
907
|
-
case 2: return <h3 className="text-xl font-medium text-primary leading-snug">{props.children}</h3>;
|
|
908
|
-
case 3: return <h4 className="text-lg font-medium text-primary leading-snug">{props.children}</h4>;
|
|
909
|
-
}
|
|
910
|
-
}
|
|
911
|
-
|
|
912
|
-
function SegmentView(props: {
|
|
913
|
-
segment: SurfaceInlineSegment;
|
|
914
|
-
markupDisplay: DemoMarkupDisplay;
|
|
915
|
-
activeCommentId?: string;
|
|
916
|
-
activeRevisionId?: string;
|
|
917
|
-
commentDecorations: PrototypeCommentDecoration[];
|
|
918
|
-
revisionDecorations: PrototypeRevisionDecoration[];
|
|
919
|
-
cellId?: string;
|
|
920
|
-
onClick: (segmentId: string, text: string, cellId?: string) => void;
|
|
921
|
-
}) {
|
|
922
|
-
const { segment } = props;
|
|
923
|
-
|
|
924
|
-
// Non-text segment types: render as inline tokens
|
|
925
|
-
if (segment.kind === "tab") {
|
|
926
|
-
return (
|
|
927
|
-
<span className="inline-flex items-center mx-0.5 text-tertiary/50" title="Tab character">
|
|
928
|
-
<MoveRight className="h-3 w-3" />
|
|
929
|
-
</span>
|
|
930
|
-
);
|
|
931
|
-
}
|
|
932
|
-
|
|
933
|
-
if (segment.kind === "hard_break") {
|
|
934
|
-
return (
|
|
935
|
-
<>
|
|
936
|
-
<span className="inline-flex items-center mx-0.5 text-tertiary/40" title="Line break">
|
|
937
|
-
<CornerDownLeft className="h-3 w-3" />
|
|
938
|
-
</span>
|
|
939
|
-
<br />
|
|
940
|
-
</>
|
|
941
|
-
);
|
|
942
|
-
}
|
|
943
|
-
|
|
944
|
-
if (segment.kind === "image") {
|
|
945
|
-
return (
|
|
946
|
-
<span
|
|
947
|
-
className={`inline-flex items-center gap-1 mx-0.5 px-1.5 py-0.5 rounded text-xs ${
|
|
948
|
-
segment.state === "missing"
|
|
949
|
-
? "text-danger bg-delete-soft"
|
|
950
|
-
: "text-secondary bg-surface"
|
|
951
|
-
}`}
|
|
952
|
-
title={segment.detail ?? segment.altText ?? "Inline image"}
|
|
953
|
-
>
|
|
954
|
-
<Image className="h-3 w-3" />
|
|
955
|
-
{segment.altText ?? (segment.state === "missing" ? "Missing image" : "Image")}
|
|
956
|
-
</span>
|
|
957
|
-
);
|
|
958
|
-
}
|
|
959
|
-
|
|
960
|
-
if (segment.kind === "opaque_inline") {
|
|
961
|
-
return (
|
|
962
|
-
<span
|
|
963
|
-
className="inline-flex items-center gap-1 mx-0.5 px-1.5 py-0.5 rounded text-xs text-comment bg-warning-soft"
|
|
964
|
-
title={segment.detail}
|
|
965
|
-
>
|
|
966
|
-
<Lock className="h-3 w-3" />
|
|
967
|
-
{segment.label}
|
|
968
|
-
</span>
|
|
969
|
-
);
|
|
970
|
-
}
|
|
971
|
-
|
|
972
|
-
// Only text segments reach here
|
|
973
|
-
if (segment.kind !== "text") return null;
|
|
974
|
-
|
|
975
|
-
// Determine decorations
|
|
976
|
-
const commentDec = props.commentDecorations.find((d) => d.segmentIds.includes(segment.segmentId));
|
|
977
|
-
const revisionDec = props.revisionDecorations.find((d) => d.segmentIds.includes(segment.segmentId));
|
|
978
|
-
const mode = props.markupDisplay;
|
|
979
|
-
|
|
980
|
-
// Hide deletions in clean mode (document reads as final text)
|
|
981
|
-
if (revisionDec?.kind === "deletion" && mode === "clean") return null;
|
|
982
|
-
|
|
983
|
-
const isActiveComment = commentDec && commentDec.commentId === props.activeCommentId;
|
|
984
|
-
const isActiveRevision = revisionDec && revisionDec.revisionId === props.activeRevisionId;
|
|
985
|
-
|
|
986
|
-
let className = "cursor-pointer transition-colors rounded-[3px] ";
|
|
987
|
-
|
|
988
|
-
// Comment styling — varies by mode
|
|
989
|
-
if (commentDec) {
|
|
990
|
-
if (mode === "clean") {
|
|
991
|
-
// Clean: no inline highlight. Active comment gets a very subtle bg only.
|
|
992
|
-
if (isActiveComment) className += "bg-comment-soft ";
|
|
993
|
-
} else if (mode === "simple") {
|
|
994
|
-
// Simple: subtle underline, stronger when active
|
|
995
|
-
className += isActiveComment
|
|
996
|
-
? "underline decoration-comment decoration-2 underline-offset-4 "
|
|
997
|
-
: "underline decoration-comment/30 decoration-2 underline-offset-4 ";
|
|
998
|
-
} else {
|
|
999
|
-
// Full: background wash
|
|
1000
|
-
className += isActiveComment ? "bg-comment-strong " : "bg-comment-soft ";
|
|
1001
|
-
}
|
|
1002
|
-
}
|
|
1003
|
-
|
|
1004
|
-
// Revision styling — varies by mode
|
|
1005
|
-
if (revisionDec) {
|
|
1006
|
-
if (mode === "clean") {
|
|
1007
|
-
// Clean: insertions render as normal text (no color/underline)
|
|
1008
|
-
} else if (mode === "simple") {
|
|
1009
|
-
// Simple: subtle cues
|
|
1010
|
-
if (revisionDec.kind === "insertion") {
|
|
1011
|
-
className += "underline decoration-insert/40 decoration-1 underline-offset-2 ";
|
|
1012
|
-
} else if (revisionDec.kind === "deletion") {
|
|
1013
|
-
className += "text-secondary line-through decoration-1 ";
|
|
1014
|
-
}
|
|
1015
|
-
} else {
|
|
1016
|
-
// Full: strong inline styling
|
|
1017
|
-
if (revisionDec.kind === "insertion") {
|
|
1018
|
-
className += "text-insert bg-insert-soft ";
|
|
1019
|
-
} else if (revisionDec.kind === "deletion") {
|
|
1020
|
-
className += "text-danger line-through decoration-1 bg-delete-soft ";
|
|
1021
|
-
}
|
|
1022
|
-
}
|
|
1023
|
-
}
|
|
1024
|
-
|
|
1025
|
-
// Active revision highlight (all modes except clean)
|
|
1026
|
-
if (isActiveRevision && mode !== "clean") {
|
|
1027
|
-
className += "ring-1 ring-accent/30 ";
|
|
1028
|
-
}
|
|
1029
|
-
|
|
1030
|
-
if (segment.hyperlinkHref) {
|
|
1031
|
-
className += "text-accent underline decoration-1 underline-offset-2 ";
|
|
1032
|
-
}
|
|
1033
|
-
|
|
1034
|
-
return (
|
|
1035
|
-
<span
|
|
1036
|
-
role="button"
|
|
1037
|
-
tabIndex={0}
|
|
1038
|
-
className={`${className}${focusRingClass} `}
|
|
1039
|
-
data-comment-id={commentDec?.commentId}
|
|
1040
|
-
onClick={() => props.onClick(segment.segmentId, segment.text, props.cellId)}
|
|
1041
|
-
onKeyDown={(e) => {
|
|
1042
|
-
if (e.key === "Enter" || e.key === " ") {
|
|
1043
|
-
e.preventDefault();
|
|
1044
|
-
props.onClick(segment.segmentId, segment.text, props.cellId);
|
|
1045
|
-
}
|
|
1046
|
-
}}
|
|
1047
|
-
>
|
|
1048
|
-
{segment.text}
|
|
1049
|
-
</span>
|
|
1050
|
-
);
|
|
1051
|
-
}
|
|
1052
|
-
|
|
1053
|
-
/* ─── Helpers ─── */
|
|
1054
|
-
|
|
1055
|
-
function inferHeadingLevel(styleId?: string): 1 | 2 | 3 | null {
|
|
1056
|
-
if (!styleId) return null;
|
|
1057
|
-
const lower = styleId.toLowerCase();
|
|
1058
|
-
if (lower === "heading1") return 1;
|
|
1059
|
-
if (lower === "heading2") return 2;
|
|
1060
|
-
if (lower === "heading3") return 3;
|
|
1061
|
-
return null;
|
|
1062
|
-
}
|
|
1063
|
-
|
|
1064
|
-
function segmentMarkers(
|
|
1065
|
-
segments: SurfaceInlineSegment[],
|
|
1066
|
-
commentDecs: PrototypeCommentDecoration[],
|
|
1067
|
-
revisionDecs: PrototypeRevisionDecoration[],
|
|
1068
|
-
): string[] {
|
|
1069
|
-
const markers: string[] = [];
|
|
1070
|
-
const segIds = new Set(segments.map((s) => s.segmentId));
|
|
1071
|
-
if (commentDecs.some((d) => d.segmentIds.some((id) => segIds.has(id)))) markers.push("comment");
|
|
1072
|
-
if (revisionDecs.some((d) => d.segmentIds.some((id) => segIds.has(id)))) markers.push("change");
|
|
1073
|
-
return markers;
|
|
1074
|
-
}
|
|
1075
|
-
|
|
1076
|
-
function markerDotColor(marker: string): string {
|
|
1077
|
-
if (marker === "comment") return "var(--color-comment)";
|
|
1078
|
-
if (marker === "change") return "var(--color-insert)";
|
|
1079
|
-
return "var(--color-tertiary)";
|
|
1080
|
-
}
|