@hiai-gg/hiai-opencode 0.2.1 → 0.2.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.env.example +4 -0
- package/AGENTS.md +34 -38
- package/ARCHITECTURE.md +4 -3
- package/LICENSE.md +14 -0
- package/README.md +52 -21
- package/config/hiai-opencode.schema.json +11 -13
- package/dist/agents/{bob.d.ts → bob/claude.d.ts} +6 -2
- package/dist/agents/bob/core.d.ts +6 -0
- package/dist/agents/bob/gpt.d.ts +11 -0
- package/dist/agents/bob/index.d.ts +3 -0
- package/dist/agents/coder/core.d.ts +4 -0
- package/dist/agents/coder/gpt.d.ts +1 -4
- package/dist/agents/coder/index.d.ts +1 -0
- package/dist/agents/manager/agent.d.ts +1 -1
- package/dist/agents/manager/default-prompt-sections.d.ts +3 -3
- package/dist/agents/manager/guard-integration.d.ts +1 -0
- package/dist/agents/prompt-library/index.d.ts +0 -1
- package/dist/agents/prompt-library/shared-execution.d.ts +9 -0
- package/dist/agents/strategist/behavioral-summary.d.ts +1 -1
- package/dist/agents/strategist/identity-constraints.d.ts +1 -1
- package/dist/agents/strategist/plan-generation.d.ts +1 -1
- package/dist/agents/types.d.ts +2 -1
- package/dist/config/defaults.d.ts +1 -0
- package/dist/config/platform-schema.d.ts +26 -26
- package/dist/config/schema/agent-names.d.ts +6 -6
- package/dist/config/schema/agent-overrides.d.ts +0 -128
- package/dist/config/schema/hiai-opencode-config.d.ts +0 -128
- package/dist/config/types.d.ts +2 -2
- package/dist/features/background-agent/manager-notifier.d.ts +46 -0
- package/dist/features/background-agent/manager-types.d.ts +40 -0
- package/dist/features/background-agent/manager.d.ts +3 -19
- package/dist/features/background-agent/polling-manager.d.ts +51 -0
- package/dist/features/boulder-state/storage.d.ts +1 -0
- package/dist/features/builtin-commands/templates/loop.d.ts +2 -0
- package/dist/features/builtin-commands/templates/start-work.d.ts +1 -1
- package/dist/features/builtin-skills/skills/interview-me.d.ts +2 -0
- package/dist/features/builtin-skills/skills/planning-and-task-breakdown.d.ts +2 -0
- package/dist/hooks/reasoning-content-cache/hook.d.ts +11 -0
- package/dist/hooks/reasoning-content-cache/index.d.ts +1 -0
- package/dist/hooks/session-recovery/checkpoint.d.ts +48 -0
- package/dist/hooks/session-recovery/enhanced-hook.d.ts +30 -0
- package/dist/hooks/session-recovery/state-backup.d.ts +76 -0
- package/dist/hooks/shared/compaction-in-progress.d.ts +4 -0
- package/dist/hooks/start-work/git-operations.d.ts +47 -0
- package/dist/hooks/token-budget.d.ts +30 -0
- package/dist/index.js +1185 -1078
- package/dist/mcp/rate-limiter.d.ts +68 -0
- package/dist/plugin/chat-message.d.ts +8 -0
- package/dist/plugin/command-execute-before.d.ts +1 -1
- package/dist/plugin/event-handlers/message-updated.d.ts +2 -0
- package/dist/plugin/event-handlers/session-error.d.ts +2 -0
- package/dist/plugin/event-handlers/session-status.d.ts +2 -0
- package/dist/plugin/event-handlers/types.d.ts +62 -0
- package/dist/plugin/event-handlers/utils.d.ts +11 -0
- package/dist/plugin/event.d.ts +1 -1
- package/dist/shared/data-path.d.ts +1 -1
- package/dist/shared/errors.d.ts +70 -0
- package/dist/shared/extract-session-id.d.ts +8 -0
- package/dist/shared/git-worktree/get-git-state-summary.d.ts +14 -0
- package/dist/shared/index.d.ts +67 -68
- package/dist/shared/internal-initiator-marker.d.ts +1 -1
- package/dist/shared/logger.d.ts +5 -1
- package/dist/shared/reasoning-content-cache.d.ts +68 -0
- package/dist/shared/safe-create-hook.d.ts +4 -4
- package/dist/tools/call-hiai-agent/constants.d.ts +2 -2
- package/dist/tools/delegate-task/sub-agent.d.ts +1 -1
- package/dist/tools/look-at/constants.d.ts +1 -1
- package/docs/architecture/bob-manager-architecture.md +244 -0
- package/docs/hiai-opencode/adr/ADR-001-agent-identity-section-injection.md +66 -0
- package/docs/hiai-opencode/adr/ADR-002-anti-loop-guard-priority.md +63 -0
- package/docs/hiai-opencode/adr/ADR-003-compaction-mechanism.md +71 -0
- package/docs/hiai-opencode/adr/ADR-004-session-recovery.md +76 -0
- package/docs/hiai-opencode/api.md +305 -0
- package/docs/hiai-opencode/hooks-architecture.md +225 -0
- package/docs/hiai-opencode/migration.md +209 -0
- package/docs/skill-discovery.md +288 -0
- package/package.json +1 -1
- package/skills/agent-browser/SKILL.md +193 -0
- package/skills/apple-hig/SKILL.md +43 -0
- package/skills/article-magazine/SKILL.md +46 -0
- package/skills/article-magazine/example.html +81 -0
- package/skills/article-magazine/example.md +38 -0
- package/skills/canvas-design/SKILL.md +45 -0
- package/skills/design-templates/audio-jingle/SKILL.md +132 -0
- package/skills/design-templates/audio-jingle/example.html +128 -0
- package/skills/design-templates/blog-post/SKILL.md +80 -0
- package/skills/design-templates/blog-post/example.html +80 -0
- package/skills/design-templates/clinical-case-report/SKILL.md +209 -0
- package/skills/design-templates/clinical-case-report/example.html +698 -0
- package/skills/design-templates/clinical-case-report/examples/example-stemi.html +698 -0
- package/skills/design-templates/clinical-case-report/references/case-formats.md +94 -0
- package/skills/design-templates/clinical-case-report/references/checklist.md +41 -0
- package/skills/design-templates/critique/SKILL.md +258 -0
- package/skills/design-templates/critique/example.html +671 -0
- package/skills/design-templates/dashboard/SKILL.md +76 -0
- package/skills/design-templates/dashboard/example.html +118 -0
- package/skills/design-templates/dating-web/SKILL.md +92 -0
- package/skills/design-templates/dating-web/example.html +265 -0
- package/skills/design-templates/dcf-valuation/SKILL.md +140 -0
- package/skills/design-templates/dcf-valuation/references/sector-wacc.md +42 -0
- package/skills/design-templates/digital-eguide/SKILL.md +94 -0
- package/skills/design-templates/digital-eguide/example.html +204 -0
- package/skills/design-templates/docs-page/SKILL.md +80 -0
- package/skills/design-templates/docs-page/example.html +122 -0
- package/skills/design-templates/email-marketing/SKILL.md +84 -0
- package/skills/design-templates/email-marketing/example.html +159 -0
- package/skills/design-templates/eng-runbook/SKILL.md +51 -0
- package/skills/design-templates/eng-runbook/example.html +250 -0
- package/skills/design-templates/finance-report/SKILL.md +61 -0
- package/skills/design-templates/finance-report/example.html +242 -0
- package/skills/design-templates/flowai-live-dashboard-template/SKILL.md +87 -0
- package/skills/design-templates/flowai-live-dashboard-template/assets/template.html +387 -0
- package/skills/design-templates/flowai-live-dashboard-template/example.html +13 -0
- package/skills/design-templates/flowai-live-dashboard-template/references/checklist.md +35 -0
- package/skills/design-templates/gamified-app/SKILL.md +108 -0
- package/skills/design-templates/gamified-app/example.html +292 -0
- package/skills/design-templates/github-dashboard/SKILL.md +130 -0
- package/skills/design-templates/github-dashboard/example.html +473 -0
- package/skills/design-templates/github-dashboard/references/README.md +10 -0
- package/skills/design-templates/github-dashboard/references/artifact-example.json +15 -0
- package/skills/design-templates/github-dashboard/references/example-data.json +138 -0
- package/skills/design-templates/github-dashboard/references/provenance-example.json +92 -0
- package/skills/design-templates/github-dashboard/references/template.html +473 -0
- package/skills/design-templates/guizang-ppt/LICENSE +21 -0
- package/skills/design-templates/guizang-ppt/README.en.md +119 -0
- package/skills/design-templates/guizang-ppt/README.md +120 -0
- package/skills/design-templates/guizang-ppt/README.pt-BR.md +121 -0
- package/skills/design-templates/guizang-ppt/SKILL.md +313 -0
- package/skills/design-templates/guizang-ppt/assets/example-slides.html +318 -0
- package/skills/design-templates/guizang-ppt/assets/template.html +647 -0
- package/skills/design-templates/guizang-ppt/references/checklist.md +265 -0
- package/skills/design-templates/guizang-ppt/references/components.md +363 -0
- package/skills/design-templates/guizang-ppt/references/layouts.md +630 -0
- package/skills/design-templates/guizang-ppt/references/styles.md +195 -0
- package/skills/design-templates/guizang-ppt/references/themes.md +122 -0
- package/skills/design-templates/hr-onboarding/SKILL.md +52 -0
- package/skills/design-templates/hr-onboarding/example.html +219 -0
- package/skills/design-templates/html-ppt/.clawscan-allow +12 -0
- package/skills/design-templates/html-ppt/LICENSE +21 -0
- package/skills/design-templates/html-ppt/README.md +234 -0
- package/skills/design-templates/html-ppt/README.pt-BR.md +239 -0
- package/skills/design-templates/html-ppt/README.zh-CN.md +238 -0
- package/skills/design-templates/html-ppt/SKILL.md +250 -0
- package/skills/design-templates/html-ppt/assets/animations/animations.css +138 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/_util.js +63 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/chain-react.js +41 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/confetti-cannon.js +49 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/constellation.js +44 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/counter-explosion.js +58 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/data-stream.js +45 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/firework.js +51 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/galaxy-swirl.js +33 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/gradient-blob.js +39 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/knowledge-graph.js +69 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/letter-explode.js +50 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/magnetic-field.js +40 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/matrix-rain.js +33 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/neural-net.js +75 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/orbit-ring.js +38 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/particle-burst.js +42 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/shockwave.js +39 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/sparkle-trail.js +62 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/starfield.js +30 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/typewriter-multi.js +51 -0
- package/skills/design-templates/html-ppt/assets/animations/fx/word-cascade.js +47 -0
- package/skills/design-templates/html-ppt/assets/animations/fx-runtime.js +99 -0
- package/skills/design-templates/html-ppt/assets/base.css +150 -0
- package/skills/design-templates/html-ppt/assets/fonts.css +9 -0
- package/skills/design-templates/html-ppt/assets/runtime.js +960 -0
- package/skills/design-templates/html-ppt/assets/themes/academic-paper.css +23 -0
- package/skills/design-templates/html-ppt/assets/themes/arctic-cool.css +14 -0
- package/skills/design-templates/html-ppt/assets/themes/aurora.css +20 -0
- package/skills/design-templates/html-ppt/assets/themes/bauhaus.css +16 -0
- package/skills/design-templates/html-ppt/assets/themes/blueprint.css +19 -0
- package/skills/design-templates/html-ppt/assets/themes/catppuccin-latte.css +14 -0
- package/skills/design-templates/html-ppt/assets/themes/catppuccin-mocha.css +14 -0
- package/skills/design-templates/html-ppt/assets/themes/corporate-clean.css +19 -0
- package/skills/design-templates/html-ppt/assets/themes/cyberpunk-neon.css +23 -0
- package/skills/design-templates/html-ppt/assets/themes/dracula.css +14 -0
- package/skills/design-templates/html-ppt/assets/themes/editorial-serif.css +18 -0
- package/skills/design-templates/html-ppt/assets/themes/engineering-whiteprint.css +26 -0
- package/skills/design-templates/html-ppt/assets/themes/glassmorphism.css +21 -0
- package/skills/design-templates/html-ppt/assets/themes/gruvbox-dark.css +14 -0
- package/skills/design-templates/html-ppt/assets/themes/japanese-minimal.css +21 -0
- package/skills/design-templates/html-ppt/assets/themes/magazine-bold.css +21 -0
- package/skills/design-templates/html-ppt/assets/themes/memphis-pop.css +20 -0
- package/skills/design-templates/html-ppt/assets/themes/midcentury.css +19 -0
- package/skills/design-templates/html-ppt/assets/themes/minimal-white.css +16 -0
- package/skills/design-templates/html-ppt/assets/themes/neo-brutalism.css +17 -0
- package/skills/design-templates/html-ppt/assets/themes/news-broadcast.css +20 -0
- package/skills/design-templates/html-ppt/assets/themes/nord.css +14 -0
- package/skills/design-templates/html-ppt/assets/themes/pitch-deck-vc.css +21 -0
- package/skills/design-templates/html-ppt/assets/themes/rainbow-gradient.css +16 -0
- package/skills/design-templates/html-ppt/assets/themes/retro-tv.css +22 -0
- package/skills/design-templates/html-ppt/assets/themes/rose-pine.css +14 -0
- package/skills/design-templates/html-ppt/assets/themes/sharp-mono.css +17 -0
- package/skills/design-templates/html-ppt/assets/themes/soft-pastel.css +14 -0
- package/skills/design-templates/html-ppt/assets/themes/solarized-light.css +14 -0
- package/skills/design-templates/html-ppt/assets/themes/sunset-warm.css +14 -0
- package/skills/design-templates/html-ppt/assets/themes/swiss-grid.css +17 -0
- package/skills/design-templates/html-ppt/assets/themes/terminal-green.css +18 -0
- package/skills/design-templates/html-ppt/assets/themes/tokyo-night.css +14 -0
- package/skills/design-templates/html-ppt/assets/themes/vaporwave.css +21 -0
- package/skills/design-templates/html-ppt/assets/themes/xiaohongshu-white.css +16 -0
- package/skills/design-templates/html-ppt/assets/themes/y2k-chrome.css +20 -0
- package/skills/design-templates/html-ppt/docs/readme/_theme-cell.html +56 -0
- package/skills/design-templates/html-ppt/docs/readme/animations.png +0 -0
- package/skills/design-templates/html-ppt/docs/readme/hero.gif +0 -0
- package/skills/design-templates/html-ppt/docs/readme/layouts-live.gif +0 -0
- package/skills/design-templates/html-ppt/docs/readme/layouts.png +0 -0
- package/skills/design-templates/html-ppt/docs/readme/montage-animations.html +61 -0
- package/skills/design-templates/html-ppt/docs/readme/montage-layouts.html +72 -0
- package/skills/design-templates/html-ppt/docs/readme/montage-templates.html +72 -0
- package/skills/design-templates/html-ppt/docs/readme/montage-themes.html +38 -0
- package/skills/design-templates/html-ppt/docs/readme/presenter-mode.png +0 -0
- package/skills/design-templates/html-ppt/docs/readme/templates.png +0 -0
- package/skills/design-templates/html-ppt/docs/readme/themes.png +0 -0
- package/skills/design-templates/html-ppt/examples/demo-deck/index.html +161 -0
- package/skills/design-templates/html-ppt/references/animations.md +147 -0
- package/skills/design-templates/html-ppt/references/authoring-guide.md +141 -0
- package/skills/design-templates/html-ppt/references/full-decks.md +98 -0
- package/skills/design-templates/html-ppt/references/layouts.md +103 -0
- package/skills/design-templates/html-ppt/references/presenter-mode.md +240 -0
- package/skills/design-templates/html-ppt/references/themes.md +107 -0
- package/skills/design-templates/html-ppt/scripts/new-deck.sh +46 -0
- package/skills/design-templates/html-ppt/scripts/render.sh +71 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_01.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_02.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_03.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_04.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_05.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_06.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_07.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_08.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_09.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_10.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_11.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_12.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_13.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_14.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_15.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_16.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_17.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_18.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_19.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/animation-showcase/animation-showcase_20.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_01.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_02.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_03.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_04.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_05.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_06.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_07.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_08.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_09.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_10.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_11.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_12.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_13.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_14.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_15.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_16.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_17.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_18.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_19.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_20.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_21.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_22.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_23.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_24.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_25.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_26.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_27.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_28.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_29.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_30.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_31.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_32.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_33.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_34.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_35.png +0 -0
- package/skills/design-templates/html-ppt/scripts/verify-output/theme-showcase/theme-showcase_36.png +0 -0
- package/skills/design-templates/html-ppt/templates/animation-showcase.html +172 -0
- package/skills/design-templates/html-ppt/templates/deck.html +69 -0
- package/skills/design-templates/html-ppt/templates/full-decks/course-module/README.md +8 -0
- package/skills/design-templates/html-ppt/templates/full-decks/course-module/index.html +189 -0
- package/skills/design-templates/html-ppt/templates/full-decks/course-module/style.css +46 -0
- package/skills/design-templates/html-ppt/templates/full-decks/dir-key-nav-minimal/README.md +11 -0
- package/skills/design-templates/html-ppt/templates/full-decks/dir-key-nav-minimal/index.html +138 -0
- package/skills/design-templates/html-ppt/templates/full-decks/dir-key-nav-minimal/style.css +60 -0
- package/skills/design-templates/html-ppt/templates/full-decks/graphify-dark-graph/README.md +11 -0
- package/skills/design-templates/html-ppt/templates/full-decks/graphify-dark-graph/index.html +180 -0
- package/skills/design-templates/html-ppt/templates/full-decks/graphify-dark-graph/style.css +54 -0
- package/skills/design-templates/html-ppt/templates/full-decks/hermes-cyber-terminal/README.md +11 -0
- package/skills/design-templates/html-ppt/templates/full-decks/hermes-cyber-terminal/index.html +199 -0
- package/skills/design-templates/html-ppt/templates/full-decks/hermes-cyber-terminal/style.css +55 -0
- package/skills/design-templates/html-ppt/templates/full-decks/knowledge-arch-blueprint/README.md +11 -0
- package/skills/design-templates/html-ppt/templates/full-decks/knowledge-arch-blueprint/index.html +190 -0
- package/skills/design-templates/html-ppt/templates/full-decks/knowledge-arch-blueprint/style.css +49 -0
- package/skills/design-templates/html-ppt/templates/full-decks/obsidian-claude-gradient/README.md +11 -0
- package/skills/design-templates/html-ppt/templates/full-decks/obsidian-claude-gradient/index.html +144 -0
- package/skills/design-templates/html-ppt/templates/full-decks/obsidian-claude-gradient/style.css +59 -0
- package/skills/design-templates/html-ppt/templates/full-decks/pitch-deck/README.md +9 -0
- package/skills/design-templates/html-ppt/templates/full-decks/pitch-deck/index.html +148 -0
- package/skills/design-templates/html-ppt/templates/full-decks/pitch-deck/style.css +40 -0
- package/skills/design-templates/html-ppt/templates/full-decks/presenter-mode-reveal/README.md +102 -0
- package/skills/design-templates/html-ppt/templates/full-decks/presenter-mode-reveal/index.html +187 -0
- package/skills/design-templates/html-ppt/templates/full-decks/presenter-mode-reveal/style.css +216 -0
- package/skills/design-templates/html-ppt/templates/full-decks/product-launch/README.md +8 -0
- package/skills/design-templates/html-ppt/templates/full-decks/product-launch/index.html +121 -0
- package/skills/design-templates/html-ppt/templates/full-decks/product-launch/style.css +39 -0
- package/skills/design-templates/html-ppt/templates/full-decks/tech-sharing/README.md +8 -0
- package/skills/design-templates/html-ppt/templates/full-decks/tech-sharing/index.html +156 -0
- package/skills/design-templates/html-ppt/templates/full-decks/tech-sharing/style.css +49 -0
- package/skills/design-templates/html-ppt/templates/full-decks/testing-safety-alert/README.md +11 -0
- package/skills/design-templates/html-ppt/templates/full-decks/testing-safety-alert/index.html +183 -0
- package/skills/design-templates/html-ppt/templates/full-decks/testing-safety-alert/style.css +62 -0
- package/skills/design-templates/html-ppt/templates/full-decks/weekly-report/README.md +8 -0
- package/skills/design-templates/html-ppt/templates/full-decks/weekly-report/index.html +127 -0
- package/skills/design-templates/html-ppt/templates/full-decks/weekly-report/style.css +55 -0
- package/skills/design-templates/html-ppt/templates/full-decks/xhs-pastel-card/README.md +11 -0
- package/skills/design-templates/html-ppt/templates/full-decks/xhs-pastel-card/index.html +147 -0
- package/skills/design-templates/html-ppt/templates/full-decks/xhs-pastel-card/style.css +66 -0
- package/skills/design-templates/html-ppt/templates/full-decks/xhs-post/README.md +9 -0
- package/skills/design-templates/html-ppt/templates/full-decks/xhs-post/index.html +133 -0
- package/skills/design-templates/html-ppt/templates/full-decks/xhs-post/style.css +47 -0
- package/skills/design-templates/html-ppt/templates/full-decks/xhs-white-editorial/README.md +11 -0
- package/skills/design-templates/html-ppt/templates/full-decks/xhs-white-editorial/index.html +187 -0
- package/skills/design-templates/html-ppt/templates/full-decks/xhs-white-editorial/style.css +63 -0
- package/skills/design-templates/html-ppt/templates/full-decks-index.html +82 -0
- package/skills/design-templates/html-ppt/templates/layout-showcase.html +47 -0
- package/skills/design-templates/html-ppt/templates/single-page/arch-diagram.html +46 -0
- package/skills/design-templates/html-ppt/templates/single-page/big-quote.html +18 -0
- package/skills/design-templates/html-ppt/templates/single-page/bullets.html +19 -0
- package/skills/design-templates/html-ppt/templates/single-page/chart-bar.html +30 -0
- package/skills/design-templates/html-ppt/templates/single-page/chart-line.html +35 -0
- package/skills/design-templates/html-ppt/templates/single-page/chart-pie.html +36 -0
- package/skills/design-templates/html-ppt/templates/single-page/chart-radar.html +31 -0
- package/skills/design-templates/html-ppt/templates/single-page/code.html +33 -0
- package/skills/design-templates/html-ppt/templates/single-page/comparison.html +47 -0
- package/skills/design-templates/html-ppt/templates/single-page/cover.html +32 -0
- package/skills/design-templates/html-ppt/templates/single-page/cta.html +27 -0
- package/skills/design-templates/html-ppt/templates/single-page/diff.html +35 -0
- package/skills/design-templates/html-ppt/templates/single-page/flow-diagram.html +33 -0
- package/skills/design-templates/html-ppt/templates/single-page/gantt.html +29 -0
- package/skills/design-templates/html-ppt/templates/single-page/image-grid.html +34 -0
- package/skills/design-templates/html-ppt/templates/single-page/image-hero.html +33 -0
- package/skills/design-templates/html-ppt/templates/single-page/kpi-grid.html +19 -0
- package/skills/design-templates/html-ppt/templates/single-page/mindmap.html +38 -0
- package/skills/design-templates/html-ppt/templates/single-page/process-steps.html +27 -0
- package/skills/design-templates/html-ppt/templates/single-page/pros-cons.html +31 -0
- package/skills/design-templates/html-ppt/templates/single-page/roadmap.html +46 -0
- package/skills/design-templates/html-ppt/templates/single-page/section-divider.html +17 -0
- package/skills/design-templates/html-ppt/templates/single-page/stat-highlight.html +17 -0
- package/skills/design-templates/html-ppt/templates/single-page/table.html +33 -0
- package/skills/design-templates/html-ppt/templates/single-page/terminal.html +35 -0
- package/skills/design-templates/html-ppt/templates/single-page/thanks.html +21 -0
- package/skills/design-templates/html-ppt/templates/single-page/three-column.html +18 -0
- package/skills/design-templates/html-ppt/templates/single-page/timeline.html +32 -0
- package/skills/design-templates/html-ppt/templates/single-page/toc.html +26 -0
- package/skills/design-templates/html-ppt/templates/single-page/todo-checklist.html +33 -0
- package/skills/design-templates/html-ppt/templates/single-page/two-column.html +39 -0
- package/skills/design-templates/html-ppt/templates/theme-showcase.html +151 -0
- package/skills/design-templates/html-ppt-course-module/SKILL.md +78 -0
- package/skills/design-templates/html-ppt-course-module/example.html +542 -0
- package/skills/design-templates/html-ppt-dir-key-nav-minimal/SKILL.md +77 -0
- package/skills/design-templates/html-ppt-dir-key-nav-minimal/example.html +366 -0
- package/skills/design-templates/html-ppt-graphify-dark-graph/SKILL.md +77 -0
- package/skills/design-templates/html-ppt-graphify-dark-graph/example.html +402 -0
- package/skills/design-templates/html-ppt-hermes-cyber-terminal/SKILL.md +77 -0
- package/skills/design-templates/html-ppt-hermes-cyber-terminal/example.html +422 -0
- package/skills/design-templates/html-ppt-knowledge-arch-blueprint/SKILL.md +77 -0
- package/skills/design-templates/html-ppt-knowledge-arch-blueprint/example.html +407 -0
- package/skills/design-templates/html-ppt-obsidian-claude-gradient/SKILL.md +77 -0
- package/skills/design-templates/html-ppt-obsidian-claude-gradient/example.html +371 -0
- package/skills/design-templates/html-ppt-pitch-deck/SKILL.md +78 -0
- package/skills/design-templates/html-ppt-pitch-deck/example.html +495 -0
- package/skills/design-templates/html-ppt-presenter-mode-reveal/SKILL.md +78 -0
- package/skills/design-templates/html-ppt-presenter-mode-reveal/example.html +725 -0
- package/skills/design-templates/html-ppt-product-launch/SKILL.md +77 -0
- package/skills/design-templates/html-ppt-product-launch/example.html +467 -0
- package/skills/design-templates/html-ppt-taste-brutalist/SKILL.md +70 -0
- package/skills/design-templates/html-ppt-taste-brutalist/example.html +774 -0
- package/skills/design-templates/html-ppt-taste-editorial/SKILL.md +62 -0
- package/skills/design-templates/html-ppt-taste-editorial/example.html +689 -0
- package/skills/design-templates/html-ppt-tech-sharing/SKILL.md +77 -0
- package/skills/design-templates/html-ppt-tech-sharing/example.html +512 -0
- package/skills/design-templates/html-ppt-testing-safety-alert/SKILL.md +78 -0
- package/skills/design-templates/html-ppt-testing-safety-alert/example.html +413 -0
- package/skills/design-templates/html-ppt-weekly-report/SKILL.md +77 -0
- package/skills/design-templates/html-ppt-weekly-report/example.html +489 -0
- package/skills/design-templates/html-ppt-xhs-pastel-card/SKILL.md +78 -0
- package/skills/design-templates/html-ppt-xhs-pastel-card/example.html +381 -0
- package/skills/design-templates/html-ppt-xhs-post/SKILL.md +78 -0
- package/skills/design-templates/html-ppt-xhs-post/example.html +487 -0
- package/skills/design-templates/html-ppt-xhs-white-editorial/SKILL.md +77 -0
- package/skills/design-templates/html-ppt-xhs-white-editorial/example.html +418 -0
- package/skills/design-templates/html-ppt-zhangzara-8-bit-orbit/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-8-bit-orbit/SKILL.md +93 -0
- package/skills/design-templates/html-ppt-zhangzara-8-bit-orbit/example.html +1640 -0
- package/skills/design-templates/html-ppt-zhangzara-8-bit-orbit/template.json +48 -0
- package/skills/design-templates/html-ppt-zhangzara-biennale-yellow/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-biennale-yellow/SKILL.md +93 -0
- package/skills/design-templates/html-ppt-zhangzara-biennale-yellow/example.html +833 -0
- package/skills/design-templates/html-ppt-zhangzara-biennale-yellow/template.json +49 -0
- package/skills/design-templates/html-ppt-zhangzara-block-frame/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-block-frame/SKILL.md +93 -0
- package/skills/design-templates/html-ppt-zhangzara-block-frame/example.html +1453 -0
- package/skills/design-templates/html-ppt-zhangzara-block-frame/template.json +47 -0
- package/skills/design-templates/html-ppt-zhangzara-blue-professional/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-blue-professional/SKILL.md +93 -0
- package/skills/design-templates/html-ppt-zhangzara-blue-professional/example.html +1423 -0
- package/skills/design-templates/html-ppt-zhangzara-blue-professional/template.json +44 -0
- package/skills/design-templates/html-ppt-zhangzara-bold-poster/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-bold-poster/SKILL.md +93 -0
- package/skills/design-templates/html-ppt-zhangzara-bold-poster/example.html +876 -0
- package/skills/design-templates/html-ppt-zhangzara-bold-poster/template.json +45 -0
- package/skills/design-templates/html-ppt-zhangzara-broadside/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-broadside/SKILL.md +92 -0
- package/skills/design-templates/html-ppt-zhangzara-broadside/example.html +2144 -0
- package/skills/design-templates/html-ppt-zhangzara-broadside/template.json +49 -0
- package/skills/design-templates/html-ppt-zhangzara-capsule/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-capsule/SKILL.md +92 -0
- package/skills/design-templates/html-ppt-zhangzara-capsule/example.html +1413 -0
- package/skills/design-templates/html-ppt-zhangzara-capsule/template.json +51 -0
- package/skills/design-templates/html-ppt-zhangzara-cartesian/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-cartesian/SKILL.md +92 -0
- package/skills/design-templates/html-ppt-zhangzara-cartesian/example.html +1136 -0
- package/skills/design-templates/html-ppt-zhangzara-cartesian/template.json +47 -0
- package/skills/design-templates/html-ppt-zhangzara-cobalt-grid/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-cobalt-grid/SKILL.md +93 -0
- package/skills/design-templates/html-ppt-zhangzara-cobalt-grid/example.html +1205 -0
- package/skills/design-templates/html-ppt-zhangzara-cobalt-grid/template.json +49 -0
- package/skills/design-templates/html-ppt-zhangzara-coral/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-coral/SKILL.md +92 -0
- package/skills/design-templates/html-ppt-zhangzara-coral/example.html +1487 -0
- package/skills/design-templates/html-ppt-zhangzara-coral/template.json +45 -0
- package/skills/design-templates/html-ppt-zhangzara-creative-mode/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-creative-mode/SKILL.md +99 -0
- package/skills/design-templates/html-ppt-zhangzara-creative-mode/assets/deck-stage.js +619 -0
- package/skills/design-templates/html-ppt-zhangzara-creative-mode/example.html +636 -0
- package/skills/design-templates/html-ppt-zhangzara-creative-mode/template.json +47 -0
- package/skills/design-templates/html-ppt-zhangzara-daisy-days/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-daisy-days/SKILL.md +93 -0
- package/skills/design-templates/html-ppt-zhangzara-daisy-days/example.html +469 -0
- package/skills/design-templates/html-ppt-zhangzara-daisy-days/template.json +54 -0
- package/skills/design-templates/html-ppt-zhangzara-editorial-tri-tone/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-editorial-tri-tone/SKILL.md +98 -0
- package/skills/design-templates/html-ppt-zhangzara-editorial-tri-tone/assets/deck-stage.js +619 -0
- package/skills/design-templates/html-ppt-zhangzara-editorial-tri-tone/example.html +737 -0
- package/skills/design-templates/html-ppt-zhangzara-editorial-tri-tone/template.json +44 -0
- package/skills/design-templates/html-ppt-zhangzara-grove/LICENSE +21 -0
- package/skills/design-templates/html-ppt-zhangzara-grove/SKILL.md +92 -0
- package/skills/design-templates/html-ppt-zhangzara-grove/example.html +1676 -0
- package/skills/design-templates/html-ppt-zhangzara-grove/template.json +51 -0
- package/skills/figma-code-connect-components/SKILL.md +42 -0
- package/skills/figma-create-design-system-rules/SKILL.md +42 -0
- package/skills/figma-create-new-file/SKILL.md +41 -0
- package/skills/figma-generate-design/SKILL.md +42 -0
- package/skills/figma-generate-library/SKILL.md +42 -0
- package/skills/figma-implement-design/SKILL.md +42 -0
- package/skills/figma-use/SKILL.md +42 -0
- package/skills/full-page-screenshot/SKILL.md +42 -0
- package/skills/interview-me/SKILL.md +64 -0
- package/skills/planning-and-task-breakdown/SKILL.md +52 -0
- package/skills/sora/SKILL.md +43 -0
- package/skills/theme-factory/SKILL.md +43 -0
- package/skills/web-design-guidelines/SKILL.md +42 -0
- package/dist/agents/prompt-library/orchestration.d.ts +0 -4
- package/skills/brainstorming/SKILL.md +0 -164
- package/skills/brainstorming/scripts/frame-template.html +0 -214
- package/skills/brainstorming/scripts/helper.js +0 -88
- package/skills/brainstorming/scripts/server.cjs +0 -354
- package/skills/brainstorming/scripts/start-server.sh +0 -148
- package/skills/brainstorming/scripts/stop-server.sh +0 -56
- package/skills/brainstorming/spec-document-reviewer-prompt.md +0 -49
- package/skills/brainstorming/visual-companion.md +0 -287
|
@@ -0,0 +1,1676 @@
|
|
|
1
|
+
<!doctype html>
|
|
2
|
+
<html lang="zh-CN">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="UTF-8" />
|
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
6
|
+
<title>Grove Presentation</title>
|
|
7
|
+
|
|
8
|
+
<link rel="preconnect" href="https://fonts.googleapis.com" />
|
|
9
|
+
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
|
|
10
|
+
<!--
|
|
11
|
+
Grove style uses four font families:
|
|
12
|
+
1. Playfair Display — editorial serif for headlines, italic in coral for accent emphasis
|
|
13
|
+
2. Jost — clean light grotesque for body copy (weight 300 — the "good paper" feel)
|
|
14
|
+
3. JetBrains Mono — labels, metadata, vertical sidebar text
|
|
15
|
+
4. Noto Serif SC / Noto Sans SC — Chinese fallbacks for every role
|
|
16
|
+
-->
|
|
17
|
+
<link
|
|
18
|
+
href="https://fonts.googleapis.com/css2?family=Jost:wght@200;300;400;500&family=Playfair+Display:ital,wght@0,400;0,500;1,400;1,500&family=JetBrains+Mono:wght@300;400&family=Noto+Serif+SC:wght@300;400;500&family=Noto+Sans+SC:wght@300;400;500&display=swap"
|
|
19
|
+
rel="stylesheet"
|
|
20
|
+
/>
|
|
21
|
+
|
|
22
|
+
<style>
|
|
23
|
+
/* ╔══════════════════════════════════════════════════════════════════════╗
|
|
24
|
+
║ ZONE A · TOKENS — GROVE STYLE ║
|
|
25
|
+
║ ║
|
|
26
|
+
║ Replace this block to change the visual style entirely. ║
|
|
27
|
+
║ Every color, font, and size in this file reads from these vars. ║
|
|
28
|
+
║ Never write raw hex values, font names, or px sizes outside here. ║
|
|
29
|
+
╚══════════════════════════════════════════════════════════════════════╝ */
|
|
30
|
+
:root {
|
|
31
|
+
/* ── Palette ──────────────────────────────────────────────────────── */
|
|
32
|
+
/* Deep forest green: grounded, considered, editorial — the Grove base */
|
|
33
|
+
--c-bg: #192b1b;
|
|
34
|
+
/* Slightly lighter forest green for secondary dark surfaces */
|
|
35
|
+
--c-bg-alt: #1e3221;
|
|
36
|
+
/* Warm parchment: light slide background — feels like good paper */
|
|
37
|
+
--c-bg-light: #e8e4d6;
|
|
38
|
+
/* Slightly cooler parchment for secondary light surfaces */
|
|
39
|
+
--c-bg-light-alt: #dedad0;
|
|
40
|
+
|
|
41
|
+
/* Primary text on dark: warm cream, never pure white */
|
|
42
|
+
--c-fg: #d4cfbf;
|
|
43
|
+
/* Secondary text on dark: muted cream at 60% opacity */
|
|
44
|
+
--c-fg-2: rgba(212, 207, 191, 0.6);
|
|
45
|
+
/* Tertiary / hint text on dark: 32% opacity — near-invisible */
|
|
46
|
+
--c-fg-3: rgba(212, 207, 191, 0.32);
|
|
47
|
+
/* Primary text on light: forest green near-black */
|
|
48
|
+
--c-fg-light: #192b1b;
|
|
49
|
+
/* Secondary text on light */
|
|
50
|
+
--c-fg-light-2: rgba(25, 43, 27, 0.58);
|
|
51
|
+
/* Tertiary text on light */
|
|
52
|
+
--c-fg-light-3: rgba(25, 43, 27, 0.33);
|
|
53
|
+
|
|
54
|
+
/* Accent: terracotta coral — the single warm note; used sparingly */
|
|
55
|
+
--c-accent: #c8524a;
|
|
56
|
+
/* Dividers on dark: faint cream border at 12% opacity */
|
|
57
|
+
--c-border: rgba(212, 207, 191, 0.12);
|
|
58
|
+
/* Dividers on light: faint green border at 14% opacity */
|
|
59
|
+
--c-border-light: rgba(25, 43, 27, 0.14);
|
|
60
|
+
|
|
61
|
+
/* ── Typography ──────────────────────────────────────────────────── */
|
|
62
|
+
/* Display + Heading: Playfair Display — editorial serif.
|
|
63
|
+
Italic in accent coral is the Grove signature move.
|
|
64
|
+
NEVER use weight 700 or bold on a serif in Grove. */
|
|
65
|
+
--f-display: "Playfair Display", "Noto Serif SC", Georgia, serif;
|
|
66
|
+
--f-heading: "Playfair Display", "Noto Serif SC", Georgia, serif;
|
|
67
|
+
/* Body: Jost weight 300 — light grotesque that steps back so the serif leads */
|
|
68
|
+
--f-body: "Jost", "Noto Sans SC", system-ui, sans-serif;
|
|
69
|
+
/* Mono: JetBrains Mono — labels, vertical sidebar text, metadata */
|
|
70
|
+
--f-mono: "JetBrains Mono", monospace;
|
|
71
|
+
|
|
72
|
+
/* ── Type Scale ───────────────────────────────────────────────────── */
|
|
73
|
+
/* vw units keep all sizes proportional regardless of window width */
|
|
74
|
+
--sz-display: 10vw; /* hero cover title — very large, commanding */
|
|
75
|
+
--sz-h1: 5.5vw; /* chapter and statement titles */
|
|
76
|
+
--sz-h2: 3.2vw; /* slide headlines */
|
|
77
|
+
--sz-h3: 2vw; /* sub-headlines, compare panel titles */
|
|
78
|
+
--sz-lead: 1.45vw; /* lead paragraph */
|
|
79
|
+
--sz-body: 1.05vw; /* body text, bullets */
|
|
80
|
+
--sz-caption: 0.82vw; /* captions, footnotes, chart sources */
|
|
81
|
+
--sz-label: 0.7vw; /* chrome labels, kickers, mono metadata */
|
|
82
|
+
|
|
83
|
+
/* ── Spacing ─────────────────────────────────────────────────────── */
|
|
84
|
+
/* Grove is intentionally spacious — breathing room is the aesthetic */
|
|
85
|
+
--pad-x: 8vw; /* horizontal slide padding */
|
|
86
|
+
--pad-y: 6.5vh; /* vertical slide padding */
|
|
87
|
+
--gap-lg: 4.5vh; /* between major content sections */
|
|
88
|
+
--gap-md: 2.8vh; /* between related elements */
|
|
89
|
+
--gap-sm: 1.4vh; /* between tightly coupled elements */
|
|
90
|
+
|
|
91
|
+
/* ── Motion ──────────────────────────────────────────────────────── */
|
|
92
|
+
/* Slide pan: sharp and deliberate — Grove moves with intention */
|
|
93
|
+
--ease-slide: cubic-bezier(0.77, 0, 0.175, 1);
|
|
94
|
+
--dur-slide: 0.9s;
|
|
95
|
+
/* Element entrance: spring-y ease-out for content reveals */
|
|
96
|
+
--ease-enter: cubic-bezier(0.16, 1, 0.3, 1);
|
|
97
|
+
--dur-enter: 0.7s;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
/* ╔══════════════════════════════════════════════════════════════════════╗
|
|
101
|
+
║ ZONE B · ENGINE — DO NOT MODIFY ║
|
|
102
|
+
║ ║
|
|
103
|
+
║ Layout engine, transitions, animation system, navigation. ║
|
|
104
|
+
║ Touching this breaks the mechanics. ║
|
|
105
|
+
╚══════════════════════════════════════════════════════════════════════╝ */
|
|
106
|
+
|
|
107
|
+
*,
|
|
108
|
+
*::before,
|
|
109
|
+
*::after {
|
|
110
|
+
box-sizing: border-box;
|
|
111
|
+
margin: 0;
|
|
112
|
+
padding: 0;
|
|
113
|
+
}
|
|
114
|
+
html,
|
|
115
|
+
body {
|
|
116
|
+
width: 100%;
|
|
117
|
+
height: 100%;
|
|
118
|
+
overflow: hidden;
|
|
119
|
+
background: var(--c-bg);
|
|
120
|
+
-webkit-font-smoothing: antialiased;
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
/* Deck container — all slides sit side by side in one horizontal strip */
|
|
124
|
+
#deck {
|
|
125
|
+
display: flex;
|
|
126
|
+
height: 100vh;
|
|
127
|
+
transition: transform var(--dur-slide) var(--ease-slide);
|
|
128
|
+
will-change: transform;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/* Slide base — each slide occupies exactly one full viewport */
|
|
132
|
+
.slide {
|
|
133
|
+
flex: 0 0 100vw;
|
|
134
|
+
width: 100vw;
|
|
135
|
+
height: 100vh;
|
|
136
|
+
position: relative;
|
|
137
|
+
padding: var(--pad-y) var(--pad-x);
|
|
138
|
+
display: grid;
|
|
139
|
+
grid-template-rows: auto 1fr auto;
|
|
140
|
+
overflow: hidden;
|
|
141
|
+
}
|
|
142
|
+
.slide-body {
|
|
143
|
+
min-height: 0;
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
/* Slide themes */
|
|
147
|
+
.slide.dark {
|
|
148
|
+
background: var(--c-bg);
|
|
149
|
+
color: var(--c-fg);
|
|
150
|
+
}
|
|
151
|
+
.slide.light {
|
|
152
|
+
background: var(--c-bg-light);
|
|
153
|
+
color: var(--c-fg-light);
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
/* ── Animation system ─────────────────────────────────────────────── */
|
|
157
|
+
/* Elements start invisible; become visible only when slide is active */
|
|
158
|
+
[data-anim] {
|
|
159
|
+
opacity: 0;
|
|
160
|
+
}
|
|
161
|
+
.slide.is-active [data-anim] {
|
|
162
|
+
animation-duration: var(--dur-enter);
|
|
163
|
+
animation-timing-function: var(--ease-enter);
|
|
164
|
+
animation-fill-mode: forwards;
|
|
165
|
+
}
|
|
166
|
+
.slide.is-active [data-anim="fade-up"] {
|
|
167
|
+
animation-name: kFadeUp;
|
|
168
|
+
}
|
|
169
|
+
.slide.is-active [data-anim="fade-in"] {
|
|
170
|
+
animation-name: kFadeIn;
|
|
171
|
+
}
|
|
172
|
+
.slide.is-active [data-anim="reveal-right"] {
|
|
173
|
+
animation-name: kRevealRight;
|
|
174
|
+
}
|
|
175
|
+
.slide.is-active [data-anim="reveal-left"] {
|
|
176
|
+
animation-name: kRevealLeft;
|
|
177
|
+
}
|
|
178
|
+
.slide.is-active [data-anim="scale-in"] {
|
|
179
|
+
animation-name: kScaleIn;
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
/* Staggered delays via data-delay attribute (0–6) */
|
|
183
|
+
[data-delay="0"] {
|
|
184
|
+
animation-delay: 0s;
|
|
185
|
+
}
|
|
186
|
+
[data-delay="1"] {
|
|
187
|
+
animation-delay: 0.08s;
|
|
188
|
+
}
|
|
189
|
+
[data-delay="2"] {
|
|
190
|
+
animation-delay: 0.18s;
|
|
191
|
+
}
|
|
192
|
+
[data-delay="3"] {
|
|
193
|
+
animation-delay: 0.3s;
|
|
194
|
+
}
|
|
195
|
+
[data-delay="4"] {
|
|
196
|
+
animation-delay: 0.44s;
|
|
197
|
+
}
|
|
198
|
+
[data-delay="5"] {
|
|
199
|
+
animation-delay: 0.6s;
|
|
200
|
+
}
|
|
201
|
+
[data-delay="6"] {
|
|
202
|
+
animation-delay: 0.78s;
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
@keyframes kFadeUp {
|
|
206
|
+
from {
|
|
207
|
+
opacity: 0;
|
|
208
|
+
transform: translateY(28px);
|
|
209
|
+
}
|
|
210
|
+
to {
|
|
211
|
+
opacity: 1;
|
|
212
|
+
transform: none;
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
@keyframes kFadeIn {
|
|
216
|
+
from {
|
|
217
|
+
opacity: 0;
|
|
218
|
+
}
|
|
219
|
+
to {
|
|
220
|
+
opacity: 1;
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
@keyframes kRevealRight {
|
|
224
|
+
from {
|
|
225
|
+
clip-path: inset(0 100% 0 0);
|
|
226
|
+
opacity: 1;
|
|
227
|
+
}
|
|
228
|
+
to {
|
|
229
|
+
clip-path: inset(0 0% 0 0);
|
|
230
|
+
opacity: 1;
|
|
231
|
+
}
|
|
232
|
+
}
|
|
233
|
+
@keyframes kRevealLeft {
|
|
234
|
+
from {
|
|
235
|
+
clip-path: inset(0 0 0 100%);
|
|
236
|
+
opacity: 1;
|
|
237
|
+
}
|
|
238
|
+
to {
|
|
239
|
+
clip-path: inset(0 0 0 0%);
|
|
240
|
+
opacity: 1;
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
@keyframes kScaleIn {
|
|
244
|
+
from {
|
|
245
|
+
opacity: 0;
|
|
246
|
+
transform: scale(0.94);
|
|
247
|
+
}
|
|
248
|
+
to {
|
|
249
|
+
opacity: 1;
|
|
250
|
+
transform: none;
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
/* ── Navigation UI ───────────────────────────────────────────────── */
|
|
255
|
+
#nav-dots {
|
|
256
|
+
position: fixed;
|
|
257
|
+
bottom: 24px;
|
|
258
|
+
left: 50%;
|
|
259
|
+
transform: translateX(-50%);
|
|
260
|
+
display: flex;
|
|
261
|
+
gap: 7px;
|
|
262
|
+
z-index: 100;
|
|
263
|
+
}
|
|
264
|
+
.nav-dot {
|
|
265
|
+
width: 5px;
|
|
266
|
+
height: 5px;
|
|
267
|
+
border-radius: 50%;
|
|
268
|
+
border: none;
|
|
269
|
+
background: rgba(255, 255, 255, 0.22);
|
|
270
|
+
cursor: pointer;
|
|
271
|
+
transition:
|
|
272
|
+
background 0.3s,
|
|
273
|
+
transform 0.3s;
|
|
274
|
+
padding: 0;
|
|
275
|
+
}
|
|
276
|
+
.nav-dot.is-active {
|
|
277
|
+
background: rgba(255, 255, 255, 0.8);
|
|
278
|
+
transform: scale(1.4);
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
/* Disabled: the slide-foot bar already shows "NN / TT" on every
|
|
282
|
+
slide; the fixed counter at the viewport edge was a duplicate. */
|
|
283
|
+
#slide-counter {
|
|
284
|
+
display: none;
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
/* ╔══════════════════════════════════════════════════════════════════════╗
|
|
288
|
+
║ ZONE C · TYPOGRAPHY ║
|
|
289
|
+
╚══════════════════════════════════════════════════════════════════════╝ */
|
|
290
|
+
|
|
291
|
+
/* Display: Playfair 400 — the Grove rule: never bold on serif */
|
|
292
|
+
.display {
|
|
293
|
+
font-family: var(--f-display);
|
|
294
|
+
font-size: var(--sz-display);
|
|
295
|
+
font-weight: 400;
|
|
296
|
+
line-height: 1;
|
|
297
|
+
letter-spacing: -0.01em;
|
|
298
|
+
}
|
|
299
|
+
.h1 {
|
|
300
|
+
font-family: var(--f-heading);
|
|
301
|
+
font-size: var(--sz-h1);
|
|
302
|
+
font-weight: 400;
|
|
303
|
+
line-height: 1.1;
|
|
304
|
+
}
|
|
305
|
+
.h2 {
|
|
306
|
+
font-family: var(--f-heading);
|
|
307
|
+
font-size: var(--sz-h2);
|
|
308
|
+
font-weight: 400;
|
|
309
|
+
line-height: 1.2;
|
|
310
|
+
}
|
|
311
|
+
.h3 {
|
|
312
|
+
font-family: var(--f-heading);
|
|
313
|
+
font-size: var(--sz-h3);
|
|
314
|
+
font-weight: 400;
|
|
315
|
+
line-height: 1.3;
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
/* Grove signature: italic serif in terracotta coral for em inside headings */
|
|
319
|
+
.h1 em,
|
|
320
|
+
.h2 em,
|
|
321
|
+
.h3 em {
|
|
322
|
+
font-style: italic;
|
|
323
|
+
color: var(--c-accent);
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
.lead {
|
|
327
|
+
font-family: var(--f-body);
|
|
328
|
+
font-size: var(--sz-lead);
|
|
329
|
+
font-weight: 300;
|
|
330
|
+
line-height: 1.65;
|
|
331
|
+
}
|
|
332
|
+
.body {
|
|
333
|
+
font-family: var(--f-body);
|
|
334
|
+
font-size: var(--sz-body);
|
|
335
|
+
font-weight: 300;
|
|
336
|
+
line-height: 1.75;
|
|
337
|
+
}
|
|
338
|
+
.caption {
|
|
339
|
+
font-family: var(--f-body);
|
|
340
|
+
font-size: var(--sz-caption);
|
|
341
|
+
font-weight: 300;
|
|
342
|
+
line-height: 1.55;
|
|
343
|
+
}
|
|
344
|
+
.label {
|
|
345
|
+
font-family: var(--f-mono);
|
|
346
|
+
font-size: var(--sz-label);
|
|
347
|
+
font-weight: 300;
|
|
348
|
+
letter-spacing: 0.12em;
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
/* Muted text — adapts to dark/light context automatically */
|
|
352
|
+
.muted {
|
|
353
|
+
color: var(--c-fg-2);
|
|
354
|
+
}
|
|
355
|
+
.accent {
|
|
356
|
+
color: var(--c-accent);
|
|
357
|
+
}
|
|
358
|
+
.light .muted {
|
|
359
|
+
color: var(--c-fg-light-2);
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
/* Bullet list — clean coral dash leaders, Jost 300 */
|
|
363
|
+
.bullet-list {
|
|
364
|
+
list-style: none;
|
|
365
|
+
display: flex;
|
|
366
|
+
flex-direction: column;
|
|
367
|
+
gap: var(--gap-md);
|
|
368
|
+
font-family: var(--f-body);
|
|
369
|
+
font-size: var(--sz-body);
|
|
370
|
+
font-weight: 300;
|
|
371
|
+
line-height: 1.65;
|
|
372
|
+
padding-left: 0;
|
|
373
|
+
}
|
|
374
|
+
.bullet-list li {
|
|
375
|
+
display: grid;
|
|
376
|
+
grid-template-columns: 2em 1fr;
|
|
377
|
+
gap: 0.5em;
|
|
378
|
+
}
|
|
379
|
+
/* Coral em-dash replaces bullet point */
|
|
380
|
+
.bullet-list li::before {
|
|
381
|
+
content: "—";
|
|
382
|
+
color: var(--c-accent);
|
|
383
|
+
font-family: var(--f-mono);
|
|
384
|
+
}
|
|
385
|
+
|
|
386
|
+
/* ╔══════════════════════════════════════════════════════════════════════╗
|
|
387
|
+
║ ZONE D · CHROME ║
|
|
388
|
+
╚══════════════════════════════════════════════════════════════════════╝ */
|
|
389
|
+
|
|
390
|
+
/* Chrome (top bar) and footer */
|
|
391
|
+
.slide-chrome,
|
|
392
|
+
.slide-foot {
|
|
393
|
+
display: flex;
|
|
394
|
+
justify-content: space-between;
|
|
395
|
+
align-items: center;
|
|
396
|
+
}
|
|
397
|
+
.slide-chrome {
|
|
398
|
+
padding-bottom: var(--gap-sm);
|
|
399
|
+
border-bottom: 1px solid var(--c-border);
|
|
400
|
+
margin-bottom: var(--gap-md);
|
|
401
|
+
}
|
|
402
|
+
.slide-foot {
|
|
403
|
+
padding-top: var(--gap-sm);
|
|
404
|
+
border-top: 1px solid var(--c-border);
|
|
405
|
+
margin-top: var(--gap-md);
|
|
406
|
+
}
|
|
407
|
+
.light .slide-chrome,
|
|
408
|
+
.light .slide-foot {
|
|
409
|
+
border-color: var(--c-border-light);
|
|
410
|
+
}
|
|
411
|
+
|
|
412
|
+
/* Layouts that are chromeless — no top bar or footer */
|
|
413
|
+
.slide--cover .slide-chrome,
|
|
414
|
+
.slide--cover .slide-foot,
|
|
415
|
+
.slide--chapter .slide-chrome,
|
|
416
|
+
.slide--chapter .slide-foot,
|
|
417
|
+
.slide--quote .slide-chrome,
|
|
418
|
+
.slide--quote .slide-foot,
|
|
419
|
+
.slide--end .slide-chrome,
|
|
420
|
+
.slide--end .slide-foot {
|
|
421
|
+
display: none;
|
|
422
|
+
}
|
|
423
|
+
|
|
424
|
+
/* ╔══════════════════════════════════════════════════════════════════════╗
|
|
425
|
+
║ ZONE E · GROVE COMPONENTS + LAYOUT PATTERNS ║
|
|
426
|
+
╚══════════════════════════════════════════════════════════════════════╝ */
|
|
427
|
+
|
|
428
|
+
/* ── Grove: large decorative watermark number ────────────────────────── */
|
|
429
|
+
/* Massive serif digit in the background — compositional texture, not UI.
|
|
430
|
+
Near-invisible: only 6% opacity. Positions in bottom-right corner. */
|
|
431
|
+
.grove-num {
|
|
432
|
+
font-family: var(--f-display);
|
|
433
|
+
font-size: 18vw;
|
|
434
|
+
font-weight: 400;
|
|
435
|
+
line-height: 1;
|
|
436
|
+
color: rgba(212, 207, 191, 0.06); /* near-invisible watermark on dark */
|
|
437
|
+
position: absolute;
|
|
438
|
+
right: var(--pad-x);
|
|
439
|
+
bottom: -0.15em;
|
|
440
|
+
pointer-events: none;
|
|
441
|
+
user-select: none;
|
|
442
|
+
letter-spacing: -0.03em;
|
|
443
|
+
}
|
|
444
|
+
/* Same watermark effect on light slides uses forest green tint */
|
|
445
|
+
.light .grove-num {
|
|
446
|
+
color: rgba(25, 43, 27, 0.06);
|
|
447
|
+
}
|
|
448
|
+
|
|
449
|
+
/* ── Grove: vertical sidebar label ──────────────────────────────────── */
|
|
450
|
+
/* Disabled: vertical/rotated sidebar text was added as a chapter-tab
|
|
451
|
+
decoration but read as overflow / clutter on most viewports.
|
|
452
|
+
Hidden across the deck; the slide-chrome bar and slide-foot already
|
|
453
|
+
provide section name and page number. */
|
|
454
|
+
.grove-sidebar {
|
|
455
|
+
display: none !important;
|
|
456
|
+
}
|
|
457
|
+
.light .grove-sidebar {
|
|
458
|
+
color: var(--c-fg-light-3);
|
|
459
|
+
}
|
|
460
|
+
|
|
461
|
+
/* ── Grove: stat card ────────────────────────────────────────────────── */
|
|
462
|
+
/* Spacious layout: large serif number in coral, mono label beneath, border-bottom rule */
|
|
463
|
+
.grove-stat {
|
|
464
|
+
display: flex;
|
|
465
|
+
flex-direction: column;
|
|
466
|
+
gap: var(--gap-sm);
|
|
467
|
+
padding-bottom: var(--gap-md);
|
|
468
|
+
border-bottom: 1px solid var(--c-border);
|
|
469
|
+
}
|
|
470
|
+
/* Large serif stat value in terracotta coral */
|
|
471
|
+
.grove-stat-val {
|
|
472
|
+
font-family: var(--f-display);
|
|
473
|
+
font-size: 4.5vw;
|
|
474
|
+
font-weight: 400;
|
|
475
|
+
line-height: 1;
|
|
476
|
+
color: var(--c-accent);
|
|
477
|
+
letter-spacing: -0.02em;
|
|
478
|
+
}
|
|
479
|
+
/* Italic suffix within stat values: e.g. 73<em>%</em> */
|
|
480
|
+
.grove-stat-val em {
|
|
481
|
+
font-style: italic;
|
|
482
|
+
}
|
|
483
|
+
/* Mono uppercase label beneath the number */
|
|
484
|
+
.grove-stat-label {
|
|
485
|
+
font-family: var(--f-mono);
|
|
486
|
+
font-size: var(--sz-label);
|
|
487
|
+
letter-spacing: 0.12em;
|
|
488
|
+
text-transform: uppercase;
|
|
489
|
+
color: var(--c-fg-2);
|
|
490
|
+
}
|
|
491
|
+
.light .grove-stat-label {
|
|
492
|
+
color: var(--c-fg-light-2);
|
|
493
|
+
}
|
|
494
|
+
.light .grove-stat {
|
|
495
|
+
border-color: var(--c-border-light);
|
|
496
|
+
}
|
|
497
|
+
|
|
498
|
+
/* ── Image placeholder ────────────────────────────────────────────── */
|
|
499
|
+
/* Used in place of <img> tags — marks where real images will go */
|
|
500
|
+
.img-placeholder {
|
|
501
|
+
background: var(--c-bg-alt);
|
|
502
|
+
display: flex;
|
|
503
|
+
align-items: center;
|
|
504
|
+
justify-content: center;
|
|
505
|
+
font-family: var(--f-mono);
|
|
506
|
+
font-size: var(--sz-label);
|
|
507
|
+
letter-spacing: 0.1em;
|
|
508
|
+
color: var(--c-fg-3);
|
|
509
|
+
width: 100%;
|
|
510
|
+
flex: 1;
|
|
511
|
+
min-height: 30vh;
|
|
512
|
+
}
|
|
513
|
+
.light .img-placeholder {
|
|
514
|
+
background: var(--c-border-light);
|
|
515
|
+
color: var(--c-fg-light-3);
|
|
516
|
+
}
|
|
517
|
+
|
|
518
|
+
/* ── Decorative rule lines ────────────────────────────────────────── */
|
|
519
|
+
/* Short coral horizontal rule — a compositional beat between kicker and headline */
|
|
520
|
+
.rule {
|
|
521
|
+
width: 36px;
|
|
522
|
+
height: 1px;
|
|
523
|
+
background: var(--c-accent);
|
|
524
|
+
}
|
|
525
|
+
.rule.full {
|
|
526
|
+
width: 100%;
|
|
527
|
+
background: var(--c-border);
|
|
528
|
+
}
|
|
529
|
+
.light .rule.full {
|
|
530
|
+
background: var(--c-border-light);
|
|
531
|
+
}
|
|
532
|
+
|
|
533
|
+
/* ── Kicker label (above headlines) ──────────────────────────────── */
|
|
534
|
+
.kicker {
|
|
535
|
+
font-family: var(--f-mono);
|
|
536
|
+
font-size: var(--sz-label);
|
|
537
|
+
letter-spacing: 0.14em;
|
|
538
|
+
text-transform: uppercase;
|
|
539
|
+
color: var(--c-accent);
|
|
540
|
+
}
|
|
541
|
+
|
|
542
|
+
/* ── 1. COVER ───────────────────────────────────────────────────────── */
|
|
543
|
+
/* Chromeless — flex column, fills full viewport height */
|
|
544
|
+
.slide--cover {
|
|
545
|
+
display: flex;
|
|
546
|
+
flex-direction: column;
|
|
547
|
+
justify-content: space-between;
|
|
548
|
+
}
|
|
549
|
+
|
|
550
|
+
/* ── 2. CHAPTER ─────────────────────────────────────────────────────── */
|
|
551
|
+
.slide--chapter {
|
|
552
|
+
display: flex;
|
|
553
|
+
flex-direction: column;
|
|
554
|
+
justify-content: center;
|
|
555
|
+
}
|
|
556
|
+
/* Mono kicker above chapter title */
|
|
557
|
+
.chapter-num {
|
|
558
|
+
font-family: var(--f-mono);
|
|
559
|
+
font-size: var(--sz-label);
|
|
560
|
+
letter-spacing: 0.2em;
|
|
561
|
+
text-transform: uppercase;
|
|
562
|
+
color: var(--c-accent);
|
|
563
|
+
margin-bottom: var(--gap-md);
|
|
564
|
+
}
|
|
565
|
+
/* Short coral rule between kicker and title */
|
|
566
|
+
.chapter-rule {
|
|
567
|
+
width: 36px;
|
|
568
|
+
height: 1px;
|
|
569
|
+
background: var(--c-accent);
|
|
570
|
+
margin-bottom: var(--gap-md);
|
|
571
|
+
}
|
|
572
|
+
|
|
573
|
+
/* ── 3. STATEMENT ───────────────────────────────────────────────────── */
|
|
574
|
+
.slide--statement .statement-body {
|
|
575
|
+
display: flex;
|
|
576
|
+
flex-direction: column;
|
|
577
|
+
justify-content: center;
|
|
578
|
+
gap: var(--gap-md);
|
|
579
|
+
}
|
|
580
|
+
/* Cap the statement headline so a 3-line thesis can't overflow on
|
|
581
|
+
shorter viewports. min() takes the smallest of width-based, height-
|
|
582
|
+
based, and absolute caps, ensuring the line-stack always fits. */
|
|
583
|
+
.slide--statement .h1 {
|
|
584
|
+
font-size: min(4.5vw, 7.5vh, 88px);
|
|
585
|
+
line-height: 1.15;
|
|
586
|
+
}
|
|
587
|
+
|
|
588
|
+
/* ── 4. SPLIT ───────────────────────────────────────────────────────── */
|
|
589
|
+
/* Two columns: text left, image right */
|
|
590
|
+
.slide--split .slide-body {
|
|
591
|
+
display: grid;
|
|
592
|
+
grid-template-columns: 1fr 1fr;
|
|
593
|
+
gap: calc(var(--pad-x) * 0.7);
|
|
594
|
+
align-items: center;
|
|
595
|
+
}
|
|
596
|
+
.split-text {
|
|
597
|
+
display: flex;
|
|
598
|
+
flex-direction: column;
|
|
599
|
+
gap: var(--gap-md);
|
|
600
|
+
}
|
|
601
|
+
.split-image {
|
|
602
|
+
display: flex;
|
|
603
|
+
flex-direction: column;
|
|
604
|
+
gap: var(--gap-sm);
|
|
605
|
+
height: 100%;
|
|
606
|
+
}
|
|
607
|
+
|
|
608
|
+
/* ── 5. STATS ───────────────────────────────────────────────────────── */
|
|
609
|
+
.slide--stats .slide-body {
|
|
610
|
+
display: flex;
|
|
611
|
+
flex-direction: column;
|
|
612
|
+
justify-content: center;
|
|
613
|
+
gap: var(--gap-lg);
|
|
614
|
+
}
|
|
615
|
+
/* Three grove-stat cards in a row — generous spacing between them */
|
|
616
|
+
.stats-row {
|
|
617
|
+
display: grid;
|
|
618
|
+
grid-template-columns: repeat(3, 1fr);
|
|
619
|
+
gap: calc(var(--pad-x) * 0.6);
|
|
620
|
+
}
|
|
621
|
+
|
|
622
|
+
/* ── 6. LIST ────────────────────────────────────────────────────────── */
|
|
623
|
+
/* Two-column: intro left, bullets right */
|
|
624
|
+
.slide--list .slide-body {
|
|
625
|
+
display: grid;
|
|
626
|
+
grid-template-columns: 2fr 3fr;
|
|
627
|
+
gap: calc(var(--pad-x) * 0.8);
|
|
628
|
+
align-items: center;
|
|
629
|
+
}
|
|
630
|
+
.list-head {
|
|
631
|
+
display: flex;
|
|
632
|
+
flex-direction: column;
|
|
633
|
+
gap: var(--gap-md);
|
|
634
|
+
padding-top: var(--gap-sm);
|
|
635
|
+
}
|
|
636
|
+
/* Body copy on the list slide: bumped up for legibility. The default
|
|
637
|
+
--sz-body of 1.05vw reads as ~12px on typical laptop viewports. */
|
|
638
|
+
.slide--list .body,
|
|
639
|
+
.slide--list .bullet-list {
|
|
640
|
+
font-size: max(1.4vw, 17px);
|
|
641
|
+
line-height: 1.6;
|
|
642
|
+
}
|
|
643
|
+
|
|
644
|
+
/* ── 7. QUOTE ───────────────────────────────────────────────────────── */
|
|
645
|
+
/* Chromeless — the emotional centerpiece of the deck */
|
|
646
|
+
.slide--quote {
|
|
647
|
+
display: flex;
|
|
648
|
+
flex-direction: column;
|
|
649
|
+
justify-content: center;
|
|
650
|
+
padding: calc(var(--pad-y) * 1.2) calc(var(--pad-x) * 1.1);
|
|
651
|
+
}
|
|
652
|
+
/* Large decorative opening quote mark in coral */
|
|
653
|
+
.quote-mark {
|
|
654
|
+
font-family: var(--f-display);
|
|
655
|
+
font-size: 8vw;
|
|
656
|
+
line-height: 0.6;
|
|
657
|
+
color: var(--c-accent);
|
|
658
|
+
margin-bottom: var(--gap-md);
|
|
659
|
+
font-weight: 400;
|
|
660
|
+
}
|
|
661
|
+
/* Large italic serif quote body */
|
|
662
|
+
.quote-text {
|
|
663
|
+
font-family: var(--f-display);
|
|
664
|
+
font-size: 3.2vw;
|
|
665
|
+
font-weight: 400;
|
|
666
|
+
line-height: 1.35;
|
|
667
|
+
letter-spacing: -0.01em;
|
|
668
|
+
max-width: 75%;
|
|
669
|
+
margin-bottom: var(--gap-lg);
|
|
670
|
+
font-style: italic;
|
|
671
|
+
}
|
|
672
|
+
.quote-attr {
|
|
673
|
+
display: flex;
|
|
674
|
+
flex-direction: column;
|
|
675
|
+
gap: 0.4vh;
|
|
676
|
+
}
|
|
677
|
+
|
|
678
|
+
/* ── 8. COMPARE ─────────────────────────────────────────────────────── */
|
|
679
|
+
/* Two panels divided by a vertical line */
|
|
680
|
+
.slide--compare .slide-body {
|
|
681
|
+
display: grid;
|
|
682
|
+
grid-template-columns: 1fr 1fr;
|
|
683
|
+
height: 100%;
|
|
684
|
+
}
|
|
685
|
+
.compare-panel {
|
|
686
|
+
display: flex;
|
|
687
|
+
flex-direction: column;
|
|
688
|
+
gap: var(--gap-md);
|
|
689
|
+
padding: var(--gap-md) 0;
|
|
690
|
+
}
|
|
691
|
+
.compare-panel.left {
|
|
692
|
+
padding-right: calc(var(--pad-x) * 0.55);
|
|
693
|
+
border-right: 1px solid var(--c-border);
|
|
694
|
+
}
|
|
695
|
+
.compare-panel.right {
|
|
696
|
+
padding-left: calc(var(--pad-x) * 0.55);
|
|
697
|
+
}
|
|
698
|
+
.light .compare-panel.left {
|
|
699
|
+
border-color: var(--c-border-light);
|
|
700
|
+
}
|
|
701
|
+
/* Panel label: coral on the "after" side */
|
|
702
|
+
.compare-label {
|
|
703
|
+
font-family: var(--f-mono);
|
|
704
|
+
font-size: var(--sz-label);
|
|
705
|
+
letter-spacing: 0.16em;
|
|
706
|
+
text-transform: uppercase;
|
|
707
|
+
padding-bottom: var(--gap-sm);
|
|
708
|
+
border-bottom: 1px solid var(--c-border);
|
|
709
|
+
}
|
|
710
|
+
.compare-label.after {
|
|
711
|
+
color: var(--c-accent);
|
|
712
|
+
}
|
|
713
|
+
.light .compare-label {
|
|
714
|
+
border-color: var(--c-border-light);
|
|
715
|
+
}
|
|
716
|
+
|
|
717
|
+
/* ── 9. END ─────────────────────────────────────────────────────────── */
|
|
718
|
+
/* Chromeless — closes the deck with generosity of space */
|
|
719
|
+
.slide--end {
|
|
720
|
+
display: flex;
|
|
721
|
+
flex-direction: column;
|
|
722
|
+
justify-content: center;
|
|
723
|
+
gap: var(--gap-md);
|
|
724
|
+
}
|
|
725
|
+
|
|
726
|
+
/* ── CHART ──────────────────────────────────────────────────────────── */
|
|
727
|
+
.slide--chart .slide-body {
|
|
728
|
+
display: flex;
|
|
729
|
+
flex-direction: column;
|
|
730
|
+
gap: var(--gap-md);
|
|
731
|
+
min-height: 0;
|
|
732
|
+
}
|
|
733
|
+
.chart-header {
|
|
734
|
+
display: flex;
|
|
735
|
+
justify-content: space-between;
|
|
736
|
+
align-items: baseline;
|
|
737
|
+
flex-shrink: 0;
|
|
738
|
+
}
|
|
739
|
+
.chart-wrapper {
|
|
740
|
+
flex: 1;
|
|
741
|
+
display: flex;
|
|
742
|
+
flex-direction: column;
|
|
743
|
+
justify-content: flex-end;
|
|
744
|
+
min-height: 0;
|
|
745
|
+
}
|
|
746
|
+
/* Bars aligned to a left spine line */
|
|
747
|
+
.bar-track {
|
|
748
|
+
height: 30vh;
|
|
749
|
+
display: flex;
|
|
750
|
+
align-items: flex-end;
|
|
751
|
+
gap: 4vw;
|
|
752
|
+
border-left: 1px solid var(--c-border);
|
|
753
|
+
padding-left: 0.5vw;
|
|
754
|
+
}
|
|
755
|
+
.light .bar-track {
|
|
756
|
+
border-color: var(--c-border-light);
|
|
757
|
+
}
|
|
758
|
+
.bar-col {
|
|
759
|
+
flex: 1;
|
|
760
|
+
display: flex;
|
|
761
|
+
flex-direction: column;
|
|
762
|
+
align-items: flex-start;
|
|
763
|
+
justify-content: flex-end;
|
|
764
|
+
gap: 1vh;
|
|
765
|
+
height: 100%;
|
|
766
|
+
}
|
|
767
|
+
/* Default bar: muted fg-3 color */
|
|
768
|
+
.bar-fill {
|
|
769
|
+
width: 100%;
|
|
770
|
+
background: var(--c-fg-3);
|
|
771
|
+
}
|
|
772
|
+
/* Accent bar: terracotta coral highlight for the featured value */
|
|
773
|
+
.bar-fill.accent {
|
|
774
|
+
background: var(--c-accent);
|
|
775
|
+
}
|
|
776
|
+
.light .bar-fill {
|
|
777
|
+
background: var(--c-fg-light-3);
|
|
778
|
+
}
|
|
779
|
+
.light .bar-fill.accent {
|
|
780
|
+
background: var(--c-accent);
|
|
781
|
+
}
|
|
782
|
+
.bar-x-label {
|
|
783
|
+
font-family: var(--f-mono);
|
|
784
|
+
/* Bumped from var(--sz-caption) (~10px on laptop) so the category
|
|
785
|
+
labels are actually readable. */
|
|
786
|
+
font-size: max(1.1vw, 13px);
|
|
787
|
+
letter-spacing: 0.1em;
|
|
788
|
+
/* Lifted from --c-fg-3 (32% opacity) to --c-fg-2 (60% opacity);
|
|
789
|
+
the 32% reading was nearly invisible on the dark background. */
|
|
790
|
+
color: var(--c-fg-2);
|
|
791
|
+
white-space: nowrap;
|
|
792
|
+
text-transform: uppercase;
|
|
793
|
+
}
|
|
794
|
+
.light .bar-x-label {
|
|
795
|
+
color: var(--c-fg-light-2);
|
|
796
|
+
}
|
|
797
|
+
.bar-val {
|
|
798
|
+
font-family: var(--f-body);
|
|
799
|
+
font-size: var(--sz-body);
|
|
800
|
+
font-weight: 300;
|
|
801
|
+
color: var(--c-fg-2);
|
|
802
|
+
}
|
|
803
|
+
/* Highlighted bar value: coral and slightly bolder */
|
|
804
|
+
.bar-val.hi {
|
|
805
|
+
color: var(--c-accent);
|
|
806
|
+
font-weight: 400;
|
|
807
|
+
}
|
|
808
|
+
.light .bar-val {
|
|
809
|
+
color: var(--c-fg-light-2);
|
|
810
|
+
}
|
|
811
|
+
/* Thin baseline rule below bars */
|
|
812
|
+
.chart-baseline {
|
|
813
|
+
height: 1px;
|
|
814
|
+
background: var(--c-border);
|
|
815
|
+
flex-shrink: 0;
|
|
816
|
+
margin-top: 1px;
|
|
817
|
+
}
|
|
818
|
+
.light .chart-baseline {
|
|
819
|
+
background: var(--c-border-light);
|
|
820
|
+
}
|
|
821
|
+
/* Source attribution below chart */
|
|
822
|
+
.chart-source {
|
|
823
|
+
flex-shrink: 0;
|
|
824
|
+
font-family: var(--f-mono);
|
|
825
|
+
font-size: var(--sz-caption);
|
|
826
|
+
color: var(--c-fg-3);
|
|
827
|
+
letter-spacing: 0.06em;
|
|
828
|
+
margin-top: var(--gap-sm);
|
|
829
|
+
}
|
|
830
|
+
.light .chart-source {
|
|
831
|
+
color: var(--c-fg-light-3);
|
|
832
|
+
}
|
|
833
|
+
</style>
|
|
834
|
+
</head>
|
|
835
|
+
|
|
836
|
+
<body>
|
|
837
|
+
<!-- NAV DOTS: generated by JS -->
|
|
838
|
+
<nav id="nav-dots"></nav>
|
|
839
|
+
<!-- SLIDE COUNTER: generated by JS -->
|
|
840
|
+
<div id="slide-counter"></div>
|
|
841
|
+
|
|
842
|
+
<!-- ═══════════════════════════════════════════════════════════════════════
|
|
843
|
+
DECK — all slides live here, side by side in one horizontal strip
|
|
844
|
+
═══════════════════════════════════════════════════════════════════════ -->
|
|
845
|
+
<div id="deck">
|
|
846
|
+
<!-- ═══════ SLIDE 1 · COVER dark ══════════════════════════════════════
|
|
847
|
+
Chromeless. grove-sidebar rotated along left edge.
|
|
848
|
+
grove-num watermark "01" in bottom-right corner.
|
|
849
|
+
Display title in Playfair 400 with italic coral emphasis word.
|
|
850
|
+
══════════════════════════════════════════════════════════════════ -->
|
|
851
|
+
<section class="slide dark slide--cover">
|
|
852
|
+
<!-- Vertical sidebar label: rotated text on the left edge -->
|
|
853
|
+
<div class="grove-sidebar" data-anim="fade-in" data-delay="0">
|
|
854
|
+
Strategy · Presentation
|
|
855
|
+
</div>
|
|
856
|
+
<!-- Watermark number: near-invisible serif "01" in background -->
|
|
857
|
+
<div class="grove-num">01</div>
|
|
858
|
+
|
|
859
|
+
<!-- Main content: grows to fill, centers vertically -->
|
|
860
|
+
<div
|
|
861
|
+
style="
|
|
862
|
+
padding-top: calc(var(--pad-y) * 1.5);
|
|
863
|
+
flex: 1;
|
|
864
|
+
display: flex;
|
|
865
|
+
flex-direction: column;
|
|
866
|
+
justify-content: center;
|
|
867
|
+
"
|
|
868
|
+
>
|
|
869
|
+
<!-- Studio name kicker: mono uppercase, coral -->
|
|
870
|
+
<div class="kicker" data-anim="fade-in" data-delay="1">
|
|
871
|
+
[Studio Name] · [Year]
|
|
872
|
+
</div>
|
|
873
|
+
<!-- Short coral rule revealing right -->
|
|
874
|
+
<div
|
|
875
|
+
class="rule"
|
|
876
|
+
data-anim="reveal-right"
|
|
877
|
+
data-delay="2"
|
|
878
|
+
style="margin: var(--gap-sm) 0"
|
|
879
|
+
></div>
|
|
880
|
+
<!-- h1 headline: Playfair 400, max 55% width, italic em in coral -->
|
|
881
|
+
<h1
|
|
882
|
+
class="h1"
|
|
883
|
+
style="max-width: 55%; margin-top: var(--gap-md)"
|
|
884
|
+
data-anim="fade-up"
|
|
885
|
+
data-delay="3"
|
|
886
|
+
>
|
|
887
|
+
[Presentation Title Goes <em>Here</em>]
|
|
888
|
+
</h1>
|
|
889
|
+
<!-- Lead subtitle: Jost 300, muted, generous line-height -->
|
|
890
|
+
<p
|
|
891
|
+
class="lead muted"
|
|
892
|
+
style="max-width: 40%; margin-top: var(--gap-md)"
|
|
893
|
+
data-anim="fade-up"
|
|
894
|
+
data-delay="4"
|
|
895
|
+
>
|
|
896
|
+
A [type of work] for [audience or occasion]. [Month, Year].
|
|
897
|
+
</p>
|
|
898
|
+
</div>
|
|
899
|
+
|
|
900
|
+
<!-- Bottom meta bar: thin top border, two mono labels -->
|
|
901
|
+
<div
|
|
902
|
+
style="
|
|
903
|
+
border-top: 1px solid var(--c-border);
|
|
904
|
+
padding-top: var(--gap-sm);
|
|
905
|
+
display: flex;
|
|
906
|
+
justify-content: space-between;
|
|
907
|
+
"
|
|
908
|
+
data-anim="fade-in"
|
|
909
|
+
data-delay="5"
|
|
910
|
+
>
|
|
911
|
+
<span class="label muted">[Prepared by]</span>
|
|
912
|
+
<span class="label muted">[Confidential]</span>
|
|
913
|
+
</div>
|
|
914
|
+
</section>
|
|
915
|
+
|
|
916
|
+
<!-- ═══════ SLIDE 2 · CHAPTER dark ════════════════════════════════════
|
|
917
|
+
Section 01 divider. Giant watermark number creates depth.
|
|
918
|
+
Kicker shows "01 / CONTEXT" then rule then h1 title.
|
|
919
|
+
══════════════════════════════════════════════════════════════════ -->
|
|
920
|
+
<section class="slide dark slide--chapter">
|
|
921
|
+
<!-- Watermark: massive near-invisible "01" in background -->
|
|
922
|
+
<div class="grove-num">01</div>
|
|
923
|
+
<!-- Section kicker: "01 / CONTEXT" in mono coral -->
|
|
924
|
+
<div class="chapter-num" data-anim="fade-in" data-delay="0">
|
|
925
|
+
01 / Context
|
|
926
|
+
</div>
|
|
927
|
+
<!-- Short coral rule -->
|
|
928
|
+
<div class="chapter-rule" data-anim="reveal-right" data-delay="1"></div>
|
|
929
|
+
<!-- Chapter title: Playfair 400, max 62% width -->
|
|
930
|
+
<h1
|
|
931
|
+
class="h1"
|
|
932
|
+
style="max-width: 62%; margin-bottom: var(--gap-md)"
|
|
933
|
+
data-anim="fade-up"
|
|
934
|
+
data-delay="2"
|
|
935
|
+
>
|
|
936
|
+
The landscape has shifted. Now we must decide where to <em>stand</em>.
|
|
937
|
+
</h1>
|
|
938
|
+
<!-- Supporting line: Jost 300 muted -->
|
|
939
|
+
<p
|
|
940
|
+
class="lead muted"
|
|
941
|
+
style="max-width: 48%"
|
|
942
|
+
data-anim="fade-up"
|
|
943
|
+
data-delay="3"
|
|
944
|
+
>
|
|
945
|
+
An honest assessment of where the market is, and where the opportunity
|
|
946
|
+
lies.
|
|
947
|
+
</p>
|
|
948
|
+
</section>
|
|
949
|
+
|
|
950
|
+
<!-- ═══════ SLIDE 3 · STATEMENT dark ══════════════════════════════════
|
|
951
|
+
Single bold thesis — the most important idea in the deck.
|
|
952
|
+
One sentence, max 62% width. Italic coral emphasis word.
|
|
953
|
+
grove-sidebar provides category context.
|
|
954
|
+
══════════════════════════════════════════════════════════════════ -->
|
|
955
|
+
<section class="slide dark slide--statement">
|
|
956
|
+
<!-- Vertical sidebar label -->
|
|
957
|
+
<div class="grove-sidebar">The Thesis</div>
|
|
958
|
+
<!-- Chrome bar: section name left, page number right -->
|
|
959
|
+
<div class="slide-chrome">
|
|
960
|
+
<span class="label muted" data-anim="fade-in" data-delay="0"
|
|
961
|
+
>Core Insight</span
|
|
962
|
+
>
|
|
963
|
+
<span class="label muted" data-anim="fade-in" data-delay="0">03</span>
|
|
964
|
+
</div>
|
|
965
|
+
|
|
966
|
+
<div class="slide-body">
|
|
967
|
+
<div class="statement-body">
|
|
968
|
+
<div class="kicker" data-anim="fade-in" data-delay="1">
|
|
969
|
+
The Argument
|
|
970
|
+
</div>
|
|
971
|
+
<div class="rule" data-anim="reveal-right" data-delay="2"></div>
|
|
972
|
+
<!-- Main thesis: large serif, generous white space around it -->
|
|
973
|
+
<h1
|
|
974
|
+
class="h1"
|
|
975
|
+
style="max-width: 62%"
|
|
976
|
+
data-anim="fade-up"
|
|
977
|
+
data-delay="3"
|
|
978
|
+
>
|
|
979
|
+
The brands that will lead the next decade are not the ones with
|
|
980
|
+
the best product. They are the ones with the deepest
|
|
981
|
+
<em>understanding</em>.
|
|
982
|
+
</h1>
|
|
983
|
+
</div>
|
|
984
|
+
</div>
|
|
985
|
+
|
|
986
|
+
<div class="slide-foot">
|
|
987
|
+
<span class="label muted"></span>
|
|
988
|
+
<span class="label muted">03 / 12</span>
|
|
989
|
+
</div>
|
|
990
|
+
</section>
|
|
991
|
+
|
|
992
|
+
<!-- ═══════ SLIDE 4 · SPLIT light ═════════════════════════════════════
|
|
993
|
+
Text left, image placeholder right.
|
|
994
|
+
Warm parchment background — editorial, considered.
|
|
995
|
+
grove-sidebar provides section context.
|
|
996
|
+
══════════════════════════════════════════════════════════════════ -->
|
|
997
|
+
<section class="slide light slide--split">
|
|
998
|
+
<div class="grove-sidebar">The Evidence</div>
|
|
999
|
+
<div class="slide-chrome">
|
|
1000
|
+
<span
|
|
1001
|
+
class="label"
|
|
1002
|
+
style="color: var(--c-fg-light-2)"
|
|
1003
|
+
data-anim="fade-in"
|
|
1004
|
+
data-delay="0"
|
|
1005
|
+
>Research · Insight</span
|
|
1006
|
+
>
|
|
1007
|
+
<span
|
|
1008
|
+
class="label"
|
|
1009
|
+
style="color: var(--c-fg-light-2)"
|
|
1010
|
+
data-anim="fade-in"
|
|
1011
|
+
data-delay="0"
|
|
1012
|
+
>04</span
|
|
1013
|
+
>
|
|
1014
|
+
</div>
|
|
1015
|
+
|
|
1016
|
+
<div class="slide-body">
|
|
1017
|
+
<!-- Left: text content -->
|
|
1018
|
+
<div class="split-text" data-anim="fade-up" data-delay="1">
|
|
1019
|
+
<div class="kicker">What We Found</div>
|
|
1020
|
+
<h2 class="h2">
|
|
1021
|
+
Audiences have outgrown the stories being told about <em>them</em>
|
|
1022
|
+
</h2>
|
|
1023
|
+
<p class="lead" style="color: var(--c-fg-light-2)">
|
|
1024
|
+
Three years of primary research across six markets revealed a
|
|
1025
|
+
consistent pattern: the gap between how brands communicate and how
|
|
1026
|
+
people actually live is widening.
|
|
1027
|
+
</p>
|
|
1028
|
+
<ul class="bullet-list" style="color: var(--c-fg-light)">
|
|
1029
|
+
<li>
|
|
1030
|
+
Authenticity is valued over aspiration in all categories tested
|
|
1031
|
+
</li>
|
|
1032
|
+
<li>Trust is earned through consistency, not campaigns</li>
|
|
1033
|
+
<li>
|
|
1034
|
+
Communities form around shared values, not product features
|
|
1035
|
+
</li>
|
|
1036
|
+
</ul>
|
|
1037
|
+
</div>
|
|
1038
|
+
|
|
1039
|
+
<!-- Right: image placeholder fills the column -->
|
|
1040
|
+
<div
|
|
1041
|
+
data-anim="fade-in"
|
|
1042
|
+
data-delay="3"
|
|
1043
|
+
style="
|
|
1044
|
+
display: flex;
|
|
1045
|
+
flex-direction: column;
|
|
1046
|
+
gap: var(--gap-sm);
|
|
1047
|
+
height: 100%;
|
|
1048
|
+
"
|
|
1049
|
+
>
|
|
1050
|
+
<div class="img-placeholder">[IMAGE PLACEHOLDER]</div>
|
|
1051
|
+
<p
|
|
1052
|
+
class="caption"
|
|
1053
|
+
style="
|
|
1054
|
+
color: var(--c-fg-light-3);
|
|
1055
|
+
font-family: var(--f-mono);
|
|
1056
|
+
letter-spacing: 0.08em;
|
|
1057
|
+
"
|
|
1058
|
+
>
|
|
1059
|
+
[Caption: research context or visual annotation]
|
|
1060
|
+
</p>
|
|
1061
|
+
</div>
|
|
1062
|
+
</div>
|
|
1063
|
+
|
|
1064
|
+
<div class="slide-foot">
|
|
1065
|
+
<span class="label" style="color: var(--c-fg-light-2)"></span>
|
|
1066
|
+
<span class="label" style="color: var(--c-fg-light-2)">04 / 12</span>
|
|
1067
|
+
</div>
|
|
1068
|
+
</section>
|
|
1069
|
+
|
|
1070
|
+
<!-- ═══════ SLIDE 5 · STATS dark ═══════════════════════════════════════
|
|
1071
|
+
Three grove-stat cards in a row.
|
|
1072
|
+
Large serif numbers in terracotta coral — let them breathe.
|
|
1073
|
+
grove-sidebar provides section context.
|
|
1074
|
+
══════════════════════════════════════════════════════════════════ -->
|
|
1075
|
+
<section class="slide dark slide--stats">
|
|
1076
|
+
<div class="grove-sidebar">By The Numbers</div>
|
|
1077
|
+
<div class="slide-chrome">
|
|
1078
|
+
<span class="label muted" data-anim="fade-in" data-delay="0"
|
|
1079
|
+
>Market · Metrics</span
|
|
1080
|
+
>
|
|
1081
|
+
<span class="label muted" data-anim="fade-in" data-delay="0">05</span>
|
|
1082
|
+
</div>
|
|
1083
|
+
|
|
1084
|
+
<div class="slide-body">
|
|
1085
|
+
<!-- Section headline above stats -->
|
|
1086
|
+
<h2 class="h2" data-anim="fade-up" data-delay="1">
|
|
1087
|
+
Three numbers that define the <em>opportunity</em>
|
|
1088
|
+
</h2>
|
|
1089
|
+
|
|
1090
|
+
<!-- Three stat cards — spacious, no crowding -->
|
|
1091
|
+
<div class="stats-row" data-anim="fade-up" data-delay="2">
|
|
1092
|
+
<div class="grove-stat">
|
|
1093
|
+
<!-- Serif stat value in coral: 73 + italic % suffix -->
|
|
1094
|
+
<div class="grove-stat-val">73<em>%</em></div>
|
|
1095
|
+
<div class="grove-stat-label">
|
|
1096
|
+
Of consumers distrust brand-created content
|
|
1097
|
+
</div>
|
|
1098
|
+
</div>
|
|
1099
|
+
<div class="grove-stat">
|
|
1100
|
+
<div class="grove-stat-val">4.8<em>×</em></div>
|
|
1101
|
+
<div class="grove-stat-label">
|
|
1102
|
+
Higher engagement for community-driven campaigns
|
|
1103
|
+
</div>
|
|
1104
|
+
</div>
|
|
1105
|
+
<div class="grove-stat">
|
|
1106
|
+
<div class="grove-stat-val">#1</div>
|
|
1107
|
+
<div class="grove-stat-label">
|
|
1108
|
+
Driver of purchase decisions: peer recommendation
|
|
1109
|
+
</div>
|
|
1110
|
+
</div>
|
|
1111
|
+
</div>
|
|
1112
|
+
|
|
1113
|
+
<!-- Source line below stats -->
|
|
1114
|
+
<p
|
|
1115
|
+
class="caption muted"
|
|
1116
|
+
style="font-family: var(--f-mono); letter-spacing: 0.08em"
|
|
1117
|
+
data-anim="fade-in"
|
|
1118
|
+
data-delay="3"
|
|
1119
|
+
>
|
|
1120
|
+
Source: [Primary Research] · [Year] · N=[sample size] across
|
|
1121
|
+
[geographies]
|
|
1122
|
+
</p>
|
|
1123
|
+
</div>
|
|
1124
|
+
|
|
1125
|
+
<div class="slide-foot">
|
|
1126
|
+
<span class="label muted"></span>
|
|
1127
|
+
<span class="label muted">05 / 12</span>
|
|
1128
|
+
</div>
|
|
1129
|
+
</section>
|
|
1130
|
+
|
|
1131
|
+
<!-- ═══════ SLIDE 6 · LIST light ══════════════════════════════════════
|
|
1132
|
+
Two-column layout: intro heading left, bullet list right.
|
|
1133
|
+
grove-sidebar on left edge.
|
|
1134
|
+
Light parchment — feels like a printed document.
|
|
1135
|
+
══════════════════════════════════════════════════════════════════ -->
|
|
1136
|
+
<section class="slide light slide--list">
|
|
1137
|
+
<div class="grove-sidebar">Our Approach</div>
|
|
1138
|
+
<div class="slide-chrome">
|
|
1139
|
+
<span
|
|
1140
|
+
class="label"
|
|
1141
|
+
style="color: var(--c-fg-light-2)"
|
|
1142
|
+
data-anim="fade-in"
|
|
1143
|
+
data-delay="0"
|
|
1144
|
+
>Framework</span
|
|
1145
|
+
>
|
|
1146
|
+
<span
|
|
1147
|
+
class="label"
|
|
1148
|
+
style="color: var(--c-fg-light-2)"
|
|
1149
|
+
data-anim="fade-in"
|
|
1150
|
+
data-delay="0"
|
|
1151
|
+
>06</span
|
|
1152
|
+
>
|
|
1153
|
+
</div>
|
|
1154
|
+
|
|
1155
|
+
<div class="slide-body">
|
|
1156
|
+
<!-- Left: heading and intro -->
|
|
1157
|
+
<div class="list-head" data-anim="fade-up" data-delay="1">
|
|
1158
|
+
<div class="kicker">What Changes</div>
|
|
1159
|
+
<h2 class="h2">
|
|
1160
|
+
Five principles that <em>reframe</em> how we think about brand
|
|
1161
|
+
</h2>
|
|
1162
|
+
<p class="body" style="color: var(--c-fg-light-2)">
|
|
1163
|
+
These are not tactics. They are the underlying commitments that
|
|
1164
|
+
make everything else possible.
|
|
1165
|
+
</p>
|
|
1166
|
+
</div>
|
|
1167
|
+
|
|
1168
|
+
<!-- Right: bullet list with coral dash leaders -->
|
|
1169
|
+
<ul
|
|
1170
|
+
class="bullet-list"
|
|
1171
|
+
style="color: var(--c-fg-light)"
|
|
1172
|
+
data-anim="fade-up"
|
|
1173
|
+
data-delay="2"
|
|
1174
|
+
>
|
|
1175
|
+
<li>
|
|
1176
|
+
Start with the community, not the product — earn presence before
|
|
1177
|
+
claiming it
|
|
1178
|
+
</li>
|
|
1179
|
+
<li>
|
|
1180
|
+
Replace broadcast with conversation — listen before speaking
|
|
1181
|
+
</li>
|
|
1182
|
+
<li>
|
|
1183
|
+
Make the values visible in operations, not just in messaging
|
|
1184
|
+
</li>
|
|
1185
|
+
<li>
|
|
1186
|
+
Treat long-term relationship as the primary metric, not reach
|
|
1187
|
+
</li>
|
|
1188
|
+
<li>
|
|
1189
|
+
Give audiences ownership of the narrative — participation over
|
|
1190
|
+
performance
|
|
1191
|
+
</li>
|
|
1192
|
+
</ul>
|
|
1193
|
+
</div>
|
|
1194
|
+
|
|
1195
|
+
<div class="slide-foot">
|
|
1196
|
+
<span class="label" style="color: var(--c-fg-light-2)"></span>
|
|
1197
|
+
<span class="label" style="color: var(--c-fg-light-2)">06 / 12</span>
|
|
1198
|
+
</div>
|
|
1199
|
+
</section>
|
|
1200
|
+
|
|
1201
|
+
<!-- ═══════ SLIDE 7 · QUOTE dark ═══════════════════════════════════════
|
|
1202
|
+
Long italic serif quote — emotional centerpiece of the deck.
|
|
1203
|
+
Chromeless. Large coral opening quote mark.
|
|
1204
|
+
Attribution below: coral name, muted role/year.
|
|
1205
|
+
══════════════════════════════════════════════════════════════════ -->
|
|
1206
|
+
<section class="slide dark slide--quote">
|
|
1207
|
+
<!-- Large decorative opening quote mark in terracotta coral -->
|
|
1208
|
+
<div class="quote-mark" data-anim="fade-in" data-delay="0">"</div>
|
|
1209
|
+
|
|
1210
|
+
<!-- Quote body: italic Playfair, spacious line-height -->
|
|
1211
|
+
<p class="quote-text" data-anim="fade-up" data-delay="1">
|
|
1212
|
+
The most radical thing a brand can do right now is simply tell the
|
|
1213
|
+
truth about what it is, and what it is not.
|
|
1214
|
+
</p>
|
|
1215
|
+
|
|
1216
|
+
<!-- Attribution: coral name, muted role -->
|
|
1217
|
+
<div class="quote-attr" data-anim="fade-up" data-delay="3">
|
|
1218
|
+
<span class="label accent">[Author Name]</span>
|
|
1219
|
+
<span class="label muted">[Title] · [Year]</span>
|
|
1220
|
+
</div>
|
|
1221
|
+
</section>
|
|
1222
|
+
|
|
1223
|
+
<!-- ═══════ SLIDE 8 · COMPARE light ════════════════════════════════════
|
|
1224
|
+
Before and after in two panels, divided by vertical rule.
|
|
1225
|
+
Light parchment — document-like, considered.
|
|
1226
|
+
Left panel: old model (no coral). Right panel: new model (coral label).
|
|
1227
|
+
══════════════════════════════════════════════════════════════════ -->
|
|
1228
|
+
<section class="slide light slide--compare">
|
|
1229
|
+
<div class="grove-sidebar">Before / After</div>
|
|
1230
|
+
<div class="slide-chrome">
|
|
1231
|
+
<span
|
|
1232
|
+
class="label"
|
|
1233
|
+
style="color: var(--c-fg-light-2)"
|
|
1234
|
+
data-anim="fade-in"
|
|
1235
|
+
data-delay="0"
|
|
1236
|
+
>The Shift</span
|
|
1237
|
+
>
|
|
1238
|
+
<span
|
|
1239
|
+
class="label"
|
|
1240
|
+
style="color: var(--c-fg-light-2)"
|
|
1241
|
+
data-anim="fade-in"
|
|
1242
|
+
data-delay="0"
|
|
1243
|
+
>08</span
|
|
1244
|
+
>
|
|
1245
|
+
</div>
|
|
1246
|
+
|
|
1247
|
+
<div class="slide-body">
|
|
1248
|
+
<!-- Left panel: the old way — muted label, no coral -->
|
|
1249
|
+
<div class="compare-panel left" data-anim="fade-up" data-delay="1">
|
|
1250
|
+
<div class="compare-label" style="color: var(--c-fg-light-2)">
|
|
1251
|
+
The Old Model
|
|
1252
|
+
</div>
|
|
1253
|
+
<h3 class="h3">Brand as broadcaster — pushing messages outward</h3>
|
|
1254
|
+
<p class="lead" style="color: var(--c-fg-light-2)">
|
|
1255
|
+
The organization speaks. The audience receives. Feedback is
|
|
1256
|
+
collected in annual surveys and processed into next year's
|
|
1257
|
+
messaging brief.
|
|
1258
|
+
</p>
|
|
1259
|
+
<ul
|
|
1260
|
+
class="bullet-list"
|
|
1261
|
+
style="font-size: var(--sz-body); color: var(--c-fg-light)"
|
|
1262
|
+
>
|
|
1263
|
+
<li>Campaigns replace conversations</li>
|
|
1264
|
+
<li>Reach is the primary metric</li>
|
|
1265
|
+
<li>Community is a distribution channel</li>
|
|
1266
|
+
</ul>
|
|
1267
|
+
</div>
|
|
1268
|
+
|
|
1269
|
+
<!-- Right panel: the new way — coral label, italic emphasis -->
|
|
1270
|
+
<div class="compare-panel right" data-anim="fade-up" data-delay="2">
|
|
1271
|
+
<div class="compare-label after">The New Model</div>
|
|
1272
|
+
<h3 class="h3">
|
|
1273
|
+
Brand as participant — embedded in the <em>community</em>
|
|
1274
|
+
</h3>
|
|
1275
|
+
<p class="lead" style="color: var(--c-fg-light-2)">
|
|
1276
|
+
The organization listens first and speaks in response. Feedback is
|
|
1277
|
+
constant, not a project. The community owns the story as much as
|
|
1278
|
+
the brand does.
|
|
1279
|
+
</p>
|
|
1280
|
+
<ul
|
|
1281
|
+
class="bullet-list"
|
|
1282
|
+
style="font-size: var(--sz-body); color: var(--c-fg-light)"
|
|
1283
|
+
>
|
|
1284
|
+
<li>Relationships replace campaigns</li>
|
|
1285
|
+
<li>Trust is the primary metric</li>
|
|
1286
|
+
<li>Community is the source of strategy</li>
|
|
1287
|
+
</ul>
|
|
1288
|
+
</div>
|
|
1289
|
+
</div>
|
|
1290
|
+
|
|
1291
|
+
<div class="slide-foot">
|
|
1292
|
+
<span class="label" style="color: var(--c-fg-light-2)"></span>
|
|
1293
|
+
<span class="label" style="color: var(--c-fg-light-2)">08 / 12</span>
|
|
1294
|
+
</div>
|
|
1295
|
+
</section>
|
|
1296
|
+
|
|
1297
|
+
<!-- ═══════ SLIDE 9 · CHAPTER dark ════════════════════════════════════
|
|
1298
|
+
Section 02 divider — same pattern as slide 2, watermark "02".
|
|
1299
|
+
══════════════════════════════════════════════════════════════════ -->
|
|
1300
|
+
<section class="slide dark slide--chapter">
|
|
1301
|
+
<!-- Watermark: massive near-invisible "02" -->
|
|
1302
|
+
<div class="grove-num">02</div>
|
|
1303
|
+
<div class="chapter-num" data-anim="fade-in" data-delay="0">
|
|
1304
|
+
02 / Recommendation
|
|
1305
|
+
</div>
|
|
1306
|
+
<div class="chapter-rule" data-anim="reveal-right" data-delay="1"></div>
|
|
1307
|
+
<h1
|
|
1308
|
+
class="h1"
|
|
1309
|
+
style="max-width: 58%; margin-bottom: var(--gap-md)"
|
|
1310
|
+
data-anim="fade-up"
|
|
1311
|
+
data-delay="2"
|
|
1312
|
+
>
|
|
1313
|
+
What we propose — and why we believe it will <em>work</em>
|
|
1314
|
+
</h1>
|
|
1315
|
+
<p
|
|
1316
|
+
class="lead muted"
|
|
1317
|
+
style="max-width: 45%"
|
|
1318
|
+
data-anim="fade-up"
|
|
1319
|
+
data-delay="3"
|
|
1320
|
+
>
|
|
1321
|
+
A practical framework built on the evidence, with clear priorities and
|
|
1322
|
+
measurable outcomes.
|
|
1323
|
+
</p>
|
|
1324
|
+
</section>
|
|
1325
|
+
|
|
1326
|
+
<!-- ═══════ SLIDE 10 · STATEMENT light ════════════════════════════════
|
|
1327
|
+
Strong thesis on warm parchment — a change of emotional register.
|
|
1328
|
+
Light version of the statement slide. Same structure, different texture.
|
|
1329
|
+
══════════════════════════════════════════════════════════════════ -->
|
|
1330
|
+
<section class="slide light slide--statement">
|
|
1331
|
+
<div class="grove-sidebar">The Recommendation</div>
|
|
1332
|
+
<div class="slide-chrome">
|
|
1333
|
+
<span
|
|
1334
|
+
class="label"
|
|
1335
|
+
style="color: var(--c-fg-light-2)"
|
|
1336
|
+
data-anim="fade-in"
|
|
1337
|
+
data-delay="0"
|
|
1338
|
+
>Strategic Direction</span
|
|
1339
|
+
>
|
|
1340
|
+
<span
|
|
1341
|
+
class="label"
|
|
1342
|
+
style="color: var(--c-fg-light-2)"
|
|
1343
|
+
data-anim="fade-in"
|
|
1344
|
+
data-delay="0"
|
|
1345
|
+
>10</span
|
|
1346
|
+
>
|
|
1347
|
+
</div>
|
|
1348
|
+
|
|
1349
|
+
<div class="slide-body">
|
|
1350
|
+
<div class="statement-body">
|
|
1351
|
+
<div class="kicker" data-anim="fade-in" data-delay="1">
|
|
1352
|
+
The Path Forward
|
|
1353
|
+
</div>
|
|
1354
|
+
<div class="rule" data-anim="reveal-right" data-delay="2"></div>
|
|
1355
|
+
<!-- Forest green text on parchment, coral italic emphasis -->
|
|
1356
|
+
<h1
|
|
1357
|
+
class="h1"
|
|
1358
|
+
style="max-width: 65%; color: var(--c-fg-light)"
|
|
1359
|
+
data-anim="fade-up"
|
|
1360
|
+
data-delay="3"
|
|
1361
|
+
>
|
|
1362
|
+
Stop managing perception. Start <em>deserving</em> it.
|
|
1363
|
+
</h1>
|
|
1364
|
+
<p
|
|
1365
|
+
class="lead"
|
|
1366
|
+
style="
|
|
1367
|
+
max-width: 52%;
|
|
1368
|
+
color: var(--c-fg-light-2);
|
|
1369
|
+
margin-top: var(--gap-md);
|
|
1370
|
+
"
|
|
1371
|
+
data-anim="fade-up"
|
|
1372
|
+
data-delay="4"
|
|
1373
|
+
>
|
|
1374
|
+
The organizations that win the next decade will earn trust slowly,
|
|
1375
|
+
through consistent action — not through the perfection of their
|
|
1376
|
+
messaging.
|
|
1377
|
+
</p>
|
|
1378
|
+
</div>
|
|
1379
|
+
</div>
|
|
1380
|
+
|
|
1381
|
+
<div class="slide-foot">
|
|
1382
|
+
<span class="label" style="color: var(--c-fg-light-2)"></span>
|
|
1383
|
+
<span class="label" style="color: var(--c-fg-light-2)">10 / 12</span>
|
|
1384
|
+
</div>
|
|
1385
|
+
</section>
|
|
1386
|
+
|
|
1387
|
+
<!-- ═══════ SLIDE 11 · CHART dark ═════════════════════════════════════
|
|
1388
|
+
Simple bar chart using grove design tokens.
|
|
1389
|
+
Muted bars for most values, accent coral bar for the highlight.
|
|
1390
|
+
All colors from CSS vars — no raw hex values.
|
|
1391
|
+
══════════════════════════════════════════════════════════════════ -->
|
|
1392
|
+
<section class="slide dark slide--chart">
|
|
1393
|
+
<div class="grove-sidebar">The Data</div>
|
|
1394
|
+
<div class="slide-chrome">
|
|
1395
|
+
<span class="label muted" data-anim="fade-in" data-delay="0"
|
|
1396
|
+
>Trust Index · Category Benchmarks</span
|
|
1397
|
+
>
|
|
1398
|
+
<span class="label muted" data-anim="fade-in" data-delay="0">11</span>
|
|
1399
|
+
</div>
|
|
1400
|
+
|
|
1401
|
+
<div class="slide-body">
|
|
1402
|
+
<!-- Chart header: headline left, unit annotation right -->
|
|
1403
|
+
<div class="chart-header">
|
|
1404
|
+
<h2 class="h2" data-anim="fade-up" data-delay="1">
|
|
1405
|
+
Consumer trust by <em>category</em>
|
|
1406
|
+
</h2>
|
|
1407
|
+
<span class="caption muted" data-anim="fade-in" data-delay="1"
|
|
1408
|
+
>Score out of 100 · [Year] · N=[X]</span
|
|
1409
|
+
>
|
|
1410
|
+
</div>
|
|
1411
|
+
|
|
1412
|
+
<!-- Bar chart: 5 categories, coral accent on the highest bar -->
|
|
1413
|
+
<div class="chart-wrapper" data-anim="fade-up" data-delay="2">
|
|
1414
|
+
<div class="bar-track">
|
|
1415
|
+
<div class="bar-col">
|
|
1416
|
+
<span class="bar-val">38</span>
|
|
1417
|
+
<div class="bar-fill" style="height: 12vh"></div>
|
|
1418
|
+
<span class="bar-x-label">Finance</span>
|
|
1419
|
+
</div>
|
|
1420
|
+
<div class="bar-col">
|
|
1421
|
+
<span class="bar-val">44</span>
|
|
1422
|
+
<div class="bar-fill" style="height: 15vh"></div>
|
|
1423
|
+
<span class="bar-x-label">Media</span>
|
|
1424
|
+
</div>
|
|
1425
|
+
<div class="bar-col">
|
|
1426
|
+
<span class="bar-val">56</span>
|
|
1427
|
+
<div class="bar-fill" style="height: 18vh"></div>
|
|
1428
|
+
<span class="bar-x-label">Retail</span>
|
|
1429
|
+
</div>
|
|
1430
|
+
<div class="bar-col">
|
|
1431
|
+
<span class="bar-val">62</span>
|
|
1432
|
+
<div class="bar-fill" style="height: 21vh"></div>
|
|
1433
|
+
<span class="bar-x-label">Healthcare</span>
|
|
1434
|
+
</div>
|
|
1435
|
+
<!-- Highest bar: coral accent fill, coral value label -->
|
|
1436
|
+
<div class="bar-col">
|
|
1437
|
+
<span class="bar-val hi">79</span>
|
|
1438
|
+
<div class="bar-fill accent" style="height: 27vh"></div>
|
|
1439
|
+
<span class="bar-x-label">Community</span>
|
|
1440
|
+
</div>
|
|
1441
|
+
</div>
|
|
1442
|
+
<!-- Thin baseline rule below all bars -->
|
|
1443
|
+
<div class="chart-baseline"></div>
|
|
1444
|
+
</div>
|
|
1445
|
+
|
|
1446
|
+
<!-- Source attribution -->
|
|
1447
|
+
<p class="chart-source" data-anim="fade-in" data-delay="3">
|
|
1448
|
+
Source: [Research Institute] · Consumer Trust Index · [Year]
|
|
1449
|
+
</p>
|
|
1450
|
+
</div>
|
|
1451
|
+
|
|
1452
|
+
<div class="slide-foot">
|
|
1453
|
+
<span class="label muted">[Organization] · [Year]</span>
|
|
1454
|
+
<span class="label muted">11 / 12</span>
|
|
1455
|
+
</div>
|
|
1456
|
+
</section>
|
|
1457
|
+
|
|
1458
|
+
<!-- ═══════ SLIDE 12 · END dark ════════════════════════════════════════
|
|
1459
|
+
Closing thought + contact information. Chromeless.
|
|
1460
|
+
The deck ends as it began: with generosity of space.
|
|
1461
|
+
Watermark "12" in the background grounds the composition.
|
|
1462
|
+
══════════════════════════════════════════════════════════════════ -->
|
|
1463
|
+
<section class="slide dark slide--end">
|
|
1464
|
+
<!-- Watermark number in the background -->
|
|
1465
|
+
<div class="grove-num">12</div>
|
|
1466
|
+
|
|
1467
|
+
<!-- Organization kicker -->
|
|
1468
|
+
<div class="kicker" data-anim="fade-in" data-delay="0">
|
|
1469
|
+
[Organization]
|
|
1470
|
+
</div>
|
|
1471
|
+
|
|
1472
|
+
<!-- Short coral rule -->
|
|
1473
|
+
<div class="rule" data-anim="reveal-right" data-delay="1"></div>
|
|
1474
|
+
|
|
1475
|
+
<!-- Closing headline: the last thing they read -->
|
|
1476
|
+
<h1
|
|
1477
|
+
class="h1"
|
|
1478
|
+
style="max-width: 55%"
|
|
1479
|
+
data-anim="fade-up"
|
|
1480
|
+
data-delay="2"
|
|
1481
|
+
>
|
|
1482
|
+
The work begins when the presentation <em>ends</em>.
|
|
1483
|
+
</h1>
|
|
1484
|
+
|
|
1485
|
+
<!-- Contact information: lead size, muted, spacious -->
|
|
1486
|
+
<p
|
|
1487
|
+
class="lead muted"
|
|
1488
|
+
style="max-width: 45%"
|
|
1489
|
+
data-anim="fade-up"
|
|
1490
|
+
data-delay="3"
|
|
1491
|
+
>
|
|
1492
|
+
[Author Name] · [author@organization.com] · [organization.com]
|
|
1493
|
+
</p>
|
|
1494
|
+
|
|
1495
|
+
<!-- Small follow-up line: mono, very muted -->
|
|
1496
|
+
<p
|
|
1497
|
+
class="label muted"
|
|
1498
|
+
style="margin-top: var(--gap-lg)"
|
|
1499
|
+
data-anim="fade-in"
|
|
1500
|
+
data-delay="4"
|
|
1501
|
+
>
|
|
1502
|
+
[Deck version] · [Date] · [Confidentiality note]
|
|
1503
|
+
</p>
|
|
1504
|
+
</section>
|
|
1505
|
+
</div>
|
|
1506
|
+
<!-- /deck -->
|
|
1507
|
+
|
|
1508
|
+
<script>
|
|
1509
|
+
/* ══════════════════════════════════════════════════════════════════════
|
|
1510
|
+
GROVE PRESENTATION ENGINE
|
|
1511
|
+
Self-contained navigation: keyboard, touch, mouse-wheel, nav-dots.
|
|
1512
|
+
No external dependencies.
|
|
1513
|
+
══════════════════════════════════════════════════════════════════════ */
|
|
1514
|
+
(function () {
|
|
1515
|
+
"use strict";
|
|
1516
|
+
|
|
1517
|
+
/* ── State ─────────────────────────────────────────────────────────── */
|
|
1518
|
+
const deck = document.getElementById("deck");
|
|
1519
|
+
const slides = Array.from(document.querySelectorAll(".slide"));
|
|
1520
|
+
const dotsEl = document.getElementById("nav-dots");
|
|
1521
|
+
const counter = document.getElementById("slide-counter");
|
|
1522
|
+
const total = slides.length;
|
|
1523
|
+
let current = 0;
|
|
1524
|
+
let animating = false;
|
|
1525
|
+
|
|
1526
|
+
/* ── Bootstrap: set deck width, build nav dots ─────────────────────── */
|
|
1527
|
+
function init() {
|
|
1528
|
+
// The deck is one wide strip — width equals N full viewports
|
|
1529
|
+
deck.style.width = total + "00vw";
|
|
1530
|
+
|
|
1531
|
+
// Build one dot per slide
|
|
1532
|
+
slides.forEach(function (_, i) {
|
|
1533
|
+
const btn = document.createElement("button");
|
|
1534
|
+
btn.className = "nav-dot";
|
|
1535
|
+
btn.setAttribute("aria-label", "Go to slide " + (i + 1));
|
|
1536
|
+
btn.addEventListener("click", function () {
|
|
1537
|
+
goTo(i);
|
|
1538
|
+
});
|
|
1539
|
+
dotsEl.appendChild(btn);
|
|
1540
|
+
});
|
|
1541
|
+
|
|
1542
|
+
// Activate the first slide immediately
|
|
1543
|
+
goTo(0, true); // true = skip transition on load
|
|
1544
|
+
}
|
|
1545
|
+
|
|
1546
|
+
/* ── goTo: the single source of truth for navigation ───────────────── */
|
|
1547
|
+
function goTo(index, instant) {
|
|
1548
|
+
// Clamp to valid range
|
|
1549
|
+
index = Math.max(0, Math.min(total - 1, index));
|
|
1550
|
+
if (index === current && !instant) return;
|
|
1551
|
+
|
|
1552
|
+
// Mark animating to block rapid key/swipe spam
|
|
1553
|
+
animating = true;
|
|
1554
|
+
|
|
1555
|
+
// Deactivate current slide: reset animations by toggling is-active
|
|
1556
|
+
const prev = slides[current];
|
|
1557
|
+
prev.classList.remove("is-active");
|
|
1558
|
+
// Force a reflow so removing then re-adding is-active (on same slide)
|
|
1559
|
+
// actually re-triggers the keyframe animations
|
|
1560
|
+
void prev.offsetWidth; // eslint-disable-line no-void
|
|
1561
|
+
|
|
1562
|
+
// Translate the deck
|
|
1563
|
+
if (instant) {
|
|
1564
|
+
// Remove transition temporarily for the initial load snap
|
|
1565
|
+
deck.style.transition = "none";
|
|
1566
|
+
deck.style.transform = "translateX(-" + index + "00vw)";
|
|
1567
|
+
void deck.offsetWidth; // force reflow
|
|
1568
|
+
deck.style.transition = "";
|
|
1569
|
+
} else {
|
|
1570
|
+
deck.style.transform = "translateX(-" + index + "00vw)";
|
|
1571
|
+
}
|
|
1572
|
+
|
|
1573
|
+
// Activate new slide
|
|
1574
|
+
current = index;
|
|
1575
|
+
const next = slides[current];
|
|
1576
|
+
next.classList.add("is-active");
|
|
1577
|
+
|
|
1578
|
+
// Update nav dots
|
|
1579
|
+
Array.from(dotsEl.children).forEach(function (dot, i) {
|
|
1580
|
+
dot.classList.toggle("is-active", i === current);
|
|
1581
|
+
});
|
|
1582
|
+
|
|
1583
|
+
// Update counter: "3 / 12"
|
|
1584
|
+
counter.textContent = current + 1 + " / " + total;
|
|
1585
|
+
|
|
1586
|
+
// Unlock after transition completes
|
|
1587
|
+
const delay = instant
|
|
1588
|
+
? 0
|
|
1589
|
+
: parseFloat(getComputedStyle(deck).transitionDuration) * 1000;
|
|
1590
|
+
setTimeout(function () {
|
|
1591
|
+
animating = false;
|
|
1592
|
+
}, delay);
|
|
1593
|
+
}
|
|
1594
|
+
|
|
1595
|
+
/* ── Keyboard navigation ──────────────────────────────────────────── */
|
|
1596
|
+
document.addEventListener("keydown", function (e) {
|
|
1597
|
+
if (animating) return;
|
|
1598
|
+
if (
|
|
1599
|
+
e.key === "ArrowRight" ||
|
|
1600
|
+
e.key === "ArrowDown" ||
|
|
1601
|
+
e.key === " "
|
|
1602
|
+
) {
|
|
1603
|
+
e.preventDefault();
|
|
1604
|
+
goTo(current + 1);
|
|
1605
|
+
} else if (e.key === "ArrowLeft" || e.key === "ArrowUp") {
|
|
1606
|
+
e.preventDefault();
|
|
1607
|
+
goTo(current - 1);
|
|
1608
|
+
} else if (e.key === "Home") {
|
|
1609
|
+
e.preventDefault();
|
|
1610
|
+
goTo(0);
|
|
1611
|
+
} else if (e.key === "End") {
|
|
1612
|
+
e.preventDefault();
|
|
1613
|
+
goTo(total - 1);
|
|
1614
|
+
}
|
|
1615
|
+
});
|
|
1616
|
+
|
|
1617
|
+
/* ── Mouse-wheel navigation ───────────────────────────────────────── */
|
|
1618
|
+
let wheelCooldown = false;
|
|
1619
|
+
document.addEventListener(
|
|
1620
|
+
"wheel",
|
|
1621
|
+
function (e) {
|
|
1622
|
+
e.preventDefault();
|
|
1623
|
+
if (animating || wheelCooldown) return;
|
|
1624
|
+
wheelCooldown = true;
|
|
1625
|
+
setTimeout(function () {
|
|
1626
|
+
wheelCooldown = false;
|
|
1627
|
+
}, 900);
|
|
1628
|
+
if (e.deltaY > 0 || e.deltaX > 0) {
|
|
1629
|
+
goTo(current + 1);
|
|
1630
|
+
} else {
|
|
1631
|
+
goTo(current - 1);
|
|
1632
|
+
}
|
|
1633
|
+
},
|
|
1634
|
+
{ passive: false },
|
|
1635
|
+
);
|
|
1636
|
+
|
|
1637
|
+
/* ── Touch swipe navigation ───────────────────────────────────────── */
|
|
1638
|
+
let touchStartX = null;
|
|
1639
|
+
let touchStartY = null;
|
|
1640
|
+
|
|
1641
|
+
document.addEventListener(
|
|
1642
|
+
"touchstart",
|
|
1643
|
+
function (e) {
|
|
1644
|
+
touchStartX = e.touches[0].clientX;
|
|
1645
|
+
touchStartY = e.touches[0].clientY;
|
|
1646
|
+
},
|
|
1647
|
+
{ passive: true },
|
|
1648
|
+
);
|
|
1649
|
+
|
|
1650
|
+
document.addEventListener(
|
|
1651
|
+
"touchend",
|
|
1652
|
+
function (e) {
|
|
1653
|
+
if (touchStartX === null) return;
|
|
1654
|
+
const dx = touchStartX - e.changedTouches[0].clientX;
|
|
1655
|
+
const dy = touchStartY - e.changedTouches[0].clientY;
|
|
1656
|
+
// Only respond if horizontal swipe is dominant
|
|
1657
|
+
if (Math.abs(dx) > Math.abs(dy) && Math.abs(dx) > 40) {
|
|
1658
|
+
if (animating) return;
|
|
1659
|
+
if (dx > 0) {
|
|
1660
|
+
goTo(current + 1); // swipe left: next
|
|
1661
|
+
} else {
|
|
1662
|
+
goTo(current - 1); // swipe right: prev
|
|
1663
|
+
}
|
|
1664
|
+
}
|
|
1665
|
+
touchStartX = null;
|
|
1666
|
+
touchStartY = null;
|
|
1667
|
+
},
|
|
1668
|
+
{ passive: true },
|
|
1669
|
+
);
|
|
1670
|
+
|
|
1671
|
+
/* ── Boot ─────────────────────────────────────────────────────────── */
|
|
1672
|
+
init();
|
|
1673
|
+
})();
|
|
1674
|
+
</script>
|
|
1675
|
+
</body>
|
|
1676
|
+
</html>
|