diff --git a/docs/archive/marketing-voice-v2-proposal.md b/docs/archive/marketing-voice-v2-proposal.md
new file mode 100644
index 0000000..e653ff6
--- /dev/null
+++ b/docs/archive/marketing-voice-v2-proposal.md
@@ -0,0 +1,422 @@
+# Marketing Voice v2 — Distillation Proposal
+
+Proposed copy changes across the site, measured against `marketing-voice.md`. Nothing applied yet — review, redline, then I'll commit the survivors.
+
+The aim: cut hedges, kill restated headings, collapse repetition across pages, and replace abstract phrasing with concrete nouns. Where two sentences carry one idea, one sentence wins.
+
+---
+
+## 1. Cross-site naming
+
+### "1:1" still appears in 6+ places
+
+We already changed the FAQ heading on the dog-walking page. The phrase still lives in navigation, footer, mega menu, decision blocks, FAQs, and SEO metadata. It's calendar-speak — customers say "solo" or "one-on-one".
+
+| Location | Current | Proposed |
+|---|---|---|
+| `homepage.ts` mobileLinks | `1:1 Walks` | `Solo Walks` |
+| `homepage.ts` megaMenuServices | `1:1 Walks` / `Personalised solo walks` | `Solo Walks` / `One dog. One walker.` |
+| `homepage.ts` services[1] | `1:1 Walks` | `Solo Walks` |
+| `homepage.ts` testimonials.service | `1:1 Walk` | `Solo Walk` |
+| `homepage.ts` booking.serviceOptions | `1:1 Walks` | `Solo Walks` |
+| `homepage.ts` footer.navigationLinks | `1:1 Walks` | `Solo Walks` |
+| `pack-walks.ts` paragraphs[4] | `our 1:1 walks are the better fit` | `our solo walks are the better fit` |
+| `dog-walking.ts` hero.eyebrow | `1:1 Walks` | `Solo Walks` |
+| `dog-walking.ts` decision.title | `Is a 1:1 walk right for your dog?` | `Is a solo walk right for your dog?` |
+| `dog-walking.ts` FAQ | `1:1 walks start from...` | `Solo walks start from...` |
+
+Open question: keep `1:1` in SEO `
` / `description` if it carries search volume, or sweep that too? I'd keep it on `llms.txt` and meta titles where the keyword has weight, drop it in body copy.
+
+---
+
+## 2. Homepage (`homepage.ts`)
+
+### Hero subtitle — already tight, leave it.
+
+✅ Current: "Reliable dog walking across Auckland Central. Happier dogs. Quieter evenings." Keep.
+
+### Intro
+
+Current: `"Professional dog walking services across Auckland."`
+
+Proposal: **`"Dog walking across Auckland Central."`**
+Why: "Professional services" is filler — readers already assume it. The location is the real specificity; drop "Auckland", say "Auckland Central" since that's where we actually are.
+
+### Founder story body
+
+Current:
+> "Most companies sell walks. We sell a calmer evening at home."
+> "Same walker. Small groups. Real attention. Your dog learns to trust one face at the door — not a rotating roster."
+> "You know who has your dog. Your dog knows who is collecting them. And you come home to a tired, happy one. Ready to"
+
+Proposal — line 3 is doing two jobs and stumbles into the CTA:
+> "Most companies sell walks. We sell a calmer evening at home."
+> "Same walker. Small groups. Real attention. Your dog learns one face at the door, not a rotating roster."
+> "You know who has your dog. They know who's at the door. You come home to a tired, happy one. Ready to"
+
+Reasons: kill "trust one face" (abstract + redundant with later "tired, happy one"), trim "to trust" (we just said we sell trust). Contraction on "they know who's" makes it sound spoken.
+
+### How it works — phase benefits
+
+The "benefit" line under each phase is doing the job of the title. Right now both compete.
+
+| Phase | Current title + benefit | Proposal |
+|---|---|---|
+| Meet | `No pressure, just clarity` + `A proper Meet & Greet at home` | Drop the benefit subtitle. Title carries enough. |
+| Settle | `A smoother start for nervous dogs` + `Your dog settles in. No rushing.` | Drop benefit subtitle. |
+| Thrive | `The outcome you actually want` + `Then the routine does the work` | Drop benefit subtitle — it's meta-commentary. |
+
+Alternative if benefit subtitles are structural: rewrite as deliverables, not vibes. `Free, 30 mins, your home` / `Two walks before regulars` / `Photo updates, every walk`.
+
+### Values
+
+Current titles read OK. Bodies have a few hedges to fix:
+
+- `"4 to 8 dogs. Always. Calm, structured walks with real attention for every dog."` → **`"4 to 8 dogs. Always. Calm walks, real attention."`** (drop "for every dog" — implied)
+- `"Pet first aid certified. Careful screening. Proactive handling. Not extras — the baseline."` → keep, but "proactive handling" is fluff. → **`"Pet first aid certified. Careful screening. Not extras — the baseline."`**
+- `"You should not have to chase your dog walker. Consistent pickup. Clear communication. Nothing to manage."` → **`"You shouldn't have to chase a dog walker. Same pickup time. Clear updates. Nothing to manage."`** (contraction, "consistent pickup" → "same pickup time")
+
+### Booking subtitles
+
+Current generalSubtitle: `"A few details. We reply properly within 24 hours."`
+
+"Properly" is a hedge that sounds defensive. Drop it.
+→ **`"A few details. We reply within 24 hours."`**
+
+### Locations & Hours intro
+
+Current: `"We cover most of Auckland Central's suburbs:"`
+
+Proposal: **`"We cover Auckland Central:"`**
+"Most of" is a hedge that undercuts the long list of suburbs immediately below.
+
+---
+
+## 3. Pack Walks (`pack-walks.ts`)
+
+### Hero — 5 paragraphs is too many
+
+The hero currently delivers: who it's for, how it works, walk length, coverage, who it isn't for. Five paragraphs. Most heroes earn their keep in 2-3. The "who it isn't for" line belongs in the decision block (and it's already there).
+
+Proposal — collapse to three paragraphs:
+
+> "Tiny Gang is built for small and medium dogs who like the right kind of company. 4 to 8 dogs, matched on size and energy."
+>
+> "Free Meet & Greet at home, then two assessment walks. Regular slot only once we know the fit is right."
+>
+> "60 to 75 minutes on the ground. We rotate Western Springs, Fowlds Park, Cornwall Park, Grey Lynn Park, and Oakley Creek — picked on the day for weather and group."
+
+Coverage line moves down to the Highlight or Pricing block where it does work. The "if your dog isn't a fit" line lives in the decision block — no need to preview it here.
+
+### Pricing intro
+
+Current: `"Right amount of exercise. Right amount of social time. Same walker every week."`
+
+Proposal: **`"Right exercise. Right social time. Same walker every week."`** Cut the repeated "amount of". Reads faster.
+
+### Pricing plan features
+
+The features hedge with "Best for…" which sounds product-y. Rewrite as straight facts:
+
+| Plan | Current 4th feature | Proposal |
+|---|---|---|
+| 1 Walk Per Week | `Best for dogs starting out` | `Good for dogs starting out` (or drop) |
+| 2-3 Walks Per Week | `Best fit for busy owners` | `Most popular routine` (matches `popular: true` flag) |
+| 4-5 Walks Per Week | `Best for high-energy social dogs` | `For high-energy social dogs` |
+| Casual Pack Walk | `Higher rate than weekly routines` | Drop — price already shows it |
+
+### Benefits intro
+
+Current: `"Small groups. Compatible dogs. No chaos. That is why it works."`
+
+Proposal: **`"Small groups. Compatible dogs. No chaos. That's why it works."`** Contraction. (Same pattern repeats elsewhere — see § 6.)
+
+### Benefit titles — abstract vs concrete
+
+- `"No overwhelming dynamics"` → **`"No bigger dogs to dodge"`** (concrete)
+- `"A routine you can rely on"` → **`"A weekly routine that sticks"`** (matches highlight block phrasing already used on the same page)
+- `"Real individual attention"` → **`"Eyes on every dog"`**
+- `"Safety, built in"` → **`"Safer than a one-size pack"`**
+
+### Booking dogIntro
+
+Current: `"Tell us about your dog. Where you are. Anything we should know. We will come back about whether Tiny Gang is the right fit."`
+
+Proposal: **`"Tell us about your dog. Where you are. Anything we should know. We'll come back about the fit."`** Contraction, drop "whether Tiny Gang is the right" (implied from page).
+
+---
+
+## 4. Solo Walks (`dog-walking.ts`)
+
+### Hero — five paragraphs again, same pattern
+
+Current paragraphs 1-5 cover: who it's for, the dog types, walk length, coverage, honesty-at-Meet-&-Greet. Same compression opportunity.
+
+Proposal — three paragraphs:
+
+> "Built for dogs who do better one-on-one. Their pace. Their walk. Same walker every time."
+>
+> "Reactive on the lead. Recovering from surgery. A senior who needs it slower. An anxious rescue still finding their feet. These are the dogs solo walks are for."
+>
+> "30 minutes for seniors or lower energy. 45 for most. 60 for dogs who want a longer outing. Door-to-door, photo update after every walk."
+
+The "honesty at Meet & Greet" sentence is already said on every other page — it's a brand-wide promise, not a hero claim.
+
+### Highlight points
+
+- `"For larger or more sensitive dogs"` body: `"When your dog needs more space, more clarity, or more attention — this gives us room to do it properly."` → **`"For dogs who need more space, clarity, or attention. We have room to give it."`** (drop "When your dog needs" → just state who it's for)
+
+### Pricing intro
+
+Current: `"Shaped around your dog, not a group schedule. For dogs who need extra attention, a steadier pace, or a more personal routine."`
+
+Proposal: **`"Shaped around your dog. For dogs who need extra attention, a steadier pace, or more personal time."`** Cuts "not a group schedule" (over-explained by context), trims "more personal routine" → "more personal time" (concrete).
+
+### Pricing plan features
+
+Same pattern as pack-walks. Drop hedging "Best for…" labels or rewrite as facts.
+
+| Plan | Current 4th feature | Proposal |
+|---|---|---|
+| 30 Min | `Good fit for lower-energy dogs` | `For seniors and lower-energy dogs` |
+| 45 Min | `Best fit for many routines` | `Most popular length` |
+| 60 Min | `Best for dogs needing a fuller outing` | `For dogs who want a longer outing` |
+
+### Pricing scarcityNote
+
+Current: `"A limited number of 1:1 slots are available each week."`
+
+Proposal: **`"Solo slots are limited each week."`** Active, shorter, no passive "are available".
+
+### Benefits intro
+
+Current: `"More space. Steadier handling. A pace that fits. The whole week feels easier."`
+
+Already strong. Keep.
+
+### Benefit titles
+
+- `"Full attention. No competition."` — keep. Strong.
+- `"The walk matches their pace"` → **`"A walk at their pace"`** (more direct)
+- `"Room to relax"` body has 21 words and ends in "without group pressure" — third time the page implies group pressure. Compress: **`"Without group pressure, anxious dogs move through the world more easily."`**
+- `"A routine built around you both"` body: `"1:1 gives us flexibility to build a routine that works for your dog and your week."` → **`"We shape the routine around your dog and your week."`** (drop "1:1 gives us flexibility" — meta-framing, redundant)
+
+### Booking dogIntro
+
+Current: `"Tell us about your dog. Where you are. Anything we should know. We will come back about the right fit."`
+
+Proposal: **`"Tell us about your dog. Where you are. Anything we should know. We'll come back about the fit."`** Match pack-walks pattern; contraction.
+
+---
+
+## 5. Puppy Visits (`puppy-visits.ts`)
+
+### Hero subtitle
+
+Current: `"While you're at work, your puppy is fed, played with, and looked after. At home."`
+
+This is the v1 voice-doc's chosen example. Keep.
+
+### Hero paragraphs
+
+Currently four. Paragraph 3 is the strongest (the growth-plates / vet rationale) and is buried. Reorder + tighten:
+
+Proposal:
+> "A visit means a toilet break, fresh water, a feed if scheduled, play, and calm settling time before we leave. Photo update lands in your phone."
+>
+> "Short visits beat long walks while your puppy is growing. Vets recommend low-impact exercise until growth plates settle — usually 12 to 18 months. Visits give them company and stimulation without the joint stress."
+>
+> "Visits are also where Goodwalk usually starts. We know your puppy early, so the move to solo walks or Tiny Gang later is smooth."
+
+Coverage line ("Across Auckland Central — Mt Eden, Ponsonby...") goes to the chip / FAQ. Three paragraphs, clearer order: what happens → why it's right for puppies → where it leads.
+
+### Highlight title
+
+Current: `"Calm routines now. A smoother Tiny Gang later."`
+
+Already nice. Keep.
+
+### Decision footnote
+
+Current: `"Free Meet & Greet first. Always."`
+
+Already nice. Keep.
+
+### Pricing intro
+
+Current: `"Built around your puppy. Real support now. Foundations for later, if Tiny Gang is the right fit."`
+
+Proposal: **`"Built around your puppy. Real support now. Foundations for whatever comes next."`** "If Tiny Gang is the right fit" hedges and over-explains. The "whatever comes next" implies the same thing without conditional language.
+
+### Plan features — same hedge pattern
+
+| Plan | Current 4th feature | Proposal |
+|---|---|---|
+| 20 Min | `Good for shorter midday support` | `For shorter midday support` |
+| 45 Min | `Best fit for many puppies` | `Most popular visit length` |
+| 60 Min | `Best for pups needing more time` | `For pups who need more time` |
+
+### scarcityNote
+
+Current: `"Puppy Visit spaces are limited so we can keep care consistent."`
+
+Proposal: **`"Puppy Visit spaces are limited."`** The reason is obvious and over-explained.
+
+### Benefits intro
+
+Current: `"The puppy stage moves fast. Daytime visits give your puppy support now — and build the routines that make later life easier."`
+
+Proposal: **`"The puppy stage moves fast. Daytime visits help now, and build routines that make later life easier."`** Drop "give your puppy support" (abstract), use "help now" (concrete verb).
+
+### Benefit body fixes
+
+- `"Foundations for Tiny Gang later"` body: `"For puppies who may join Tiny Gang one day, early visits build the confidence and routines that make the next step smooth."` — TWO hedges in one sentence ("may", "one day"). → **`"For puppies who'll join Tiny Gang later, early visits build the confidence and routines that make the next step smooth."`**
+- `"Support for busy owners too"` body has `"during a demanding stage"` — vague. → **`"Real help when puppies are learning fast. Guidance from someone who's been through this stage with dozens of dogs."`**
+
+### FAQ "How long is each visit?"
+
+Current answer says "30 minutes — the sweet spot" but the pricing plan starts at **20 minutes**. Inconsistency — fix the FAQ, not the price.
+
+Proposal: **`"20 minutes for shorter midday support. 45 minutes for most puppies. 60 minutes if your pup needs more time."`** Matches the pricing plans exactly.
+
+### FAQ "Can Puppy Visits lead into Tiny Gang…"
+
+Current: `"Exactly what they are designed for. When your puppy is old enough and the right temperament fit, we already know them well. The next step is smooth, not new."`
+
+Proposal: **`"Exactly what they're for. By the time your puppy is old enough, we already know them. The next step is smooth, not new."`** Contraction, drop "and the right temperament fit" (implied), drop "designed for" (passive-corporate).
+
+### Booking dogIntro
+
+Current: `"Tell us about your puppy. Where you are. Their routine. Anything we should know — and we will plan the right visit."`
+
+Proposal: **`"Tell us about your puppy. Where you are. Their routine. We'll plan the right visit."`** Drop "Anything we should know — and we will" (redundant, hedging).
+
+---
+
+## 6. About page (`about.ts`)
+
+### "Who we are" section
+
+Current:
+> "Alessandra started Goodwalk because she could not find a walker she trusted. So she became one."
+> "She walks every dog herself. Posts photos to Instagram so you can see your dog's day. Knows some of the Tiny Gang from ten weeks old."
+> "Thirty-plus five-star Google reviews say the same thing: the dogs adore her, and their owners finally stop worrying."
+
+Line 1 is gold. Keep. Line 2 is fine. Line 3 has "say the same thing" which is filler.
+
+Proposal line 3: **`"Thirty-plus five-star Google reviews: the dogs adore her, and their owners stop worrying."`** Drop "say the same thing" and "finally" (mild hedge).
+
+### "How we do things"
+
+Current:
+> "Calm handling. Positive reinforcement. A walker who already knows your dog. Same principles, every walk."
+> "Small packs because attention matters. Free pickup and drop-off because your day should not work around us. First aid certified. Public liability insured. That part is not negotiable."
+
+Line 2 is doing six things at once. Split:
+
+Proposal:
+> "Calm handling. Positive reinforcement. A walker who already knows your dog. Same principles, every walk."
+> "Small packs because attention matters. Free pickup and drop-off because your day shouldn't work around ours."
+> "First aid certified. Public liability insured. Not negotiable."
+
+Three paragraphs, three jobs. Contraction on "shouldn't".
+
+### "Meet the founder" — line 3 (Maya)
+
+Current: `"Maya is the reason small dogs sit at the centre of everything. A Cavalier King Charles cross Shih Tzu. Opinionated. Dramatic in the rain. Completely impossible to ignore on a walk — and the best argument we have for building a service around small dogs, not one that just makes room for them."`
+
+Strong texture, but ends with a 27-word sentence past the voice budget (max ~24). Split:
+
+Proposal:
+> "Maya is the reason small dogs sit at the centre of everything. A Cavalier King Charles cross Shih Tzu. Opinionated. Dramatic in the rain. Impossible to ignore on a walk."
+> "She's the best argument we have for a service built around small dogs — not one that just makes room for them."
+
+### FAQ "Why do you specialise in small dogs?"
+
+Current: `"Small dogs need different pace, different group dynamics, different handling. Goodwalk was built around that — not adapted from a generic dog-walking model."`
+
+Proposal: **`"Small dogs need a different pace, different group dynamics, different handling. We were built around that, not adapted from a generic model."`** Active voice ("We were built" instead of "Goodwalk was built"), trim "dog-walking" (implied).
+
+### FAQ "What suburbs do you cover?"
+
+Current has 16 suburb names listed inline. The map / chips already show them. Compress:
+
+Proposal: **`"Most of Auckland Central — Ponsonby, Grey Lynn, Mt Eden, Kingsland, Herne Bay, Remuera and surrounds. If you're nearby and unsure, just ask."`** Cuts the list to the highest-recognition six. The exhaustive list lives on the homepage Locations block and the coverage map.
+
+---
+
+## 7. Locations (`locations.ts`) — sweeping pattern
+
+Every location intro is structured the same way: descriptive lead → "well-suited / natural home / ideal place for…" → Goodwalk services available → free pickup line. Three of these in a row, the pattern shows.
+
+### Park descriptions — universal cleanup
+
+The voice doc says "Replace abstract nouns with concrete verbs". Park blurbs lean on adjective-stacks:
+
+- `"offers wide open paths, panoramic views across Auckland, and a mix of gentle and steeper terrain"` → **`"Wide open paths, panoramic views, gentle and steep terrain."`** (drop "offers")
+- `"Popular with local dog walkers and a staple route for the Tiny Gang"` → keep.
+- `"A well-used neighbourhood park… with open grass areas and shade trees"` → **`"Neighbourhood park with open grass and shade."`**
+
+Recommend a pass on all 30+ park descriptions: cut every "offers", "provides", "with X and Y" sentence opener, and every "well-suited / ideal place / natural home".
+
+### Intro pattern — propose a template
+
+Right now Mt Eden's intro is 71 words. Most location intros are 50-80 words. Voice doc says "Body sentences: max ~24". Propose a 3-sentence template:
+
+> "[Suburb] [one specific thing — geography, vibe, dog density]. [Goodwalk fact — who from here is in the Tiny Gang / what we run here]. Free pickup and drop-off."
+
+Worked example, Mt Eden:
+
+> "Mt Eden's volcanic cone, leafy streets, and mix of reserves and quiet paths make it a daily-outing favourite. We run pack walks and solo walks here weekly, with several Tiny Gang regulars based in the suburb. Free pickup and drop-off included."
+
+53 words → still long, but every clause does work. Apply the template to all 17 suburbs in a follow-up pass.
+
+---
+
+## 8. Repeated lines across pages
+
+The phrase **"Free pickup and drop-off across Auckland Central"** (or close variants) appears 14+ times: in hero chips, paragraph 4 of every service hero, FAQ answers, the coverage map, the locations page, and the about page. It's a real selling point — but said this often it stops landing.
+
+Proposal — vary the wording by surface:
+
+- **Chip** (compact): `"Free pickup & drop-off"`
+- **Hero paragraph**: usually deletable since the chip is right there. If kept: `"Pickup and drop-off included, across Auckland Central."`
+- **FAQ**: keep specific — that's where someone is actually checking.
+- **Footer / coverage block**: `"Door-to-door across Auckland Central."`
+
+Save the full sentence for places where the suburb list matters.
+
+---
+
+## 9. Summary of recurring fixes
+
+If we agree on the principles below, I can sweep these patterns site-wide without you reviewing each one:
+
+1. **Drop "Best for / Best fit"** in pricing features. State who it's for, or drop.
+2. **Contractions** ("we'll", "shouldn't", "they're") where the surrounding tone is conversational.
+3. **"1:1" → "Solo"** in body copy, navigation, decision blocks. Keep "1:1" in SEO meta/title where keyword volume might matter.
+4. **Cut "properly", "actually", "genuinely", "really"** unless the sentence dies without them.
+5. **Cut "we will come back about"** → "we'll reply about" / "we'll come back" (less corporate).
+6. **Park descriptions**: rewrite the "offers / provides / well-suited / a mix of" sentences as concrete noun phrases.
+7. **Service hero paragraphs**: target 3 paragraphs, not 5. Coverage and disqualifier lines move down the page.
+8. **Drop reasons that are already obvious from context** — "so we can keep care consistent", "Higher rate than weekly routines", "in your home" after "in-home".
+
+---
+
+## What this changes for the reader
+
+- Faster scan on every service page (3 hero paragraphs instead of 5).
+- Consistent terminology between nav, body, and CTAs (no "1:1" / "Solo" / "one-on-one" mix).
+- Pricing tables stop sounding like a SaaS comparison grid.
+- Location pages stop reading like council brochures.
+
+## What it does NOT change
+
+- The brand voice from `marketing-voice.md` — this proposal is a stricter application of that voice, not a rewrite of it.
+- SEO `` / `meta description` / `llms.txt` keywords — those remain under a separate review (the "1:1" tradeoff lives there).
+- Customer testimonials — never edited.
+- Service names ("Tiny Gang", "Meet & Greet") — kept verbatim.
+
+---
+
+**Next step**: redline this file. Strike the changes you don't want, mark anything that needs different phrasing, and I'll apply the rest in one sweep. Or pick a single section to start with (homepage? service heroes? locations?) so we can validate the voice before going site-wide.
diff --git a/docs/archive/v3-design-audit.md b/docs/archive/v3-design-audit.md
new file mode 100644
index 0000000..a73bb80
--- /dev/null
+++ b/docs/archive/v3-design-audit.md
@@ -0,0 +1,252 @@
+# Design Critique — Goodwalk
+
+Scope: read `variables.css` end-to-end, `ValuesSection.svelte` (open in IDE), `buttons.css`, and surveyed `HeroSection.svelte`. Citing concrete drift, not generalities.
+
+---
+
+## Issue list
+
+### 1. Token system has decayed into noise — the design system is the problem
+
+**Inconsistency.** `variables.css` defines **7 near-identical grays** (`--gray #59606d`, `--text-muted #4c5056`, `--text-muted-strong #4a4f55`, `--text-subtle #5f6369`, `--text-soft #666`, `--text-softest #6a6d72`, `--text-heading-soft #34363a`) and **6 near-identical off-whites** (`--surface-panel-soft`, `--surface-panel-muted`, `--surface-panel-warm`, `--surface-panel-cream`, `--surface-light`, `--off-white`). They differ by 1–3 hex points.
+
+**Why it hurts.** A design system with this many close-but-different values is functionally no system at all. Every component author picks a slightly different one. This is the engine behind every other inconsistency in the codebase.
+
+**Recommendation.** Collapse to: 3 grays (heading / body / muted), 2 off-whites (page / panel), 2 brand greens (primary / hover). Run a codemod replacing the deprecated tokens. Lock the file with a comment header forbidding additions without review.
+
+**Good looks like.** Linear's full text scale is 4 grays. Stripe's surface scale is 3.
+
+**Severity: Critical.**
+
+---
+
+### 2. No radius scale exists — `variables.css` has zero `--radius-*` tokens
+
+**Inconsistency.** Components hand-pick: `40px` (button), `28px` (photo card), `22px` (photo card mobile), `18px` (bento + caption), `16px` (bento mobile + caption mobile), `11px` (point icon), `999px` (eyebrow). 7 different radii on one page.
+
+**Why it hurts.** The eye notices radius mismatches more than almost any other inconsistency — it's the silhouette of every element. A 28px card next to an 18px card next to a 22px card reads as "ported from different templates."
+
+**Recommendation.** Add tokens: `--radius-sm 8`, `--radius-md 14`, `--radius-lg 20`, `--radius-pill 999`. Pick **one** card radius (recommend 20). Buttons stay pill. Eyebrow stays pill. Everything else uses the scale.
+
+**Severity: Critical.**
+
+---
+
+### 3. `ValuesSection.svelte` bypasses the token system almost entirely
+
+**Inconsistency.** In ~480 lines of CSS in this single component:
+- Hardcoded colors not in tokens: `#000` (×3), `#0d1a0d` (×2), `#102010`, `#5a605f`, `#3f4348`, `#fff` (×3), `#ede4d2`, `#fcfbf6`.
+- Hardcoded shadows: `0 18px 34px rgba(17,20,24,0.08)` (line 227), `0 12px 24px rgba(...)` (line 262), `0 14px 34px rgba(...)` (line 295), `0 6px 16px rgba(33,48,33,0.18)` (line 505) — when `--shadow-card`, `--shadow-panel-strong`, `--shadow-badge` already exist for these.
+- Spacing values not on the 4px scale: `padding: 0 50px`, `padding: 38px 36px`, `padding: 32px 30px`, `padding: 13px 0`, `margin-top: 32px / 36px / 52px / 26px`.
+- One shadow token (`--shadow-panel-elevated`, line 479) — out of ~6 shadows. The other 5 are inline.
+
+**Why it hurts.** This is design system drift in action. The tokens exist; they're being ignored. Every other component will do the same and the system becomes a museum.
+
+**Recommendation.** Replace every literal in this file with a token. If a token doesn't exist, add it and use it everywhere it applies. Then add a stylelint rule blocking raw hex and raw shadow tuples in component CSS.
+
+**Severity: Critical.**
+
+---
+
+### 4. Two ``s in the same section at totally different scales
+
+**Inconsistency.** `.values-contrast-cell h3` is `clamp(20px, 1.9vw, 25px)`. `.values-points-title` (also ``) is `clamp(24px, 2.4vw, 32px)`. They sit in the same section, 200px apart vertically.
+
+**Why it hurts.** Visual hierarchy breaks. The reader's brain expects the second H3 to be peer-level with the first. Instead it's larger than the first, then `.values-point h3` (also h3) drops back to 17px. Three H3s, three scales.
+
+**Recommendation.** Adopt a real type scale token set: `--text-display`, `--text-h2`, `--text-h3`, `--text-h4`, `--text-body`, `--text-small`. Map every heading to a token. The "values points" introduction is functionally a sub-section header — use the same scale as the contrast cell titles, or promote it to h2-tier visually with proper hierarchy.
+
+**Severity: Major.**
+
+---
+
+### 5. Photo caption layout breaks when `detail` is empty
+
+**Inconsistency.** `.values-photo-caption` uses `justify-content: space-between` with two spans inside. Three of five photos have `detail: ''`. The result: an empty right-side gap and a left-pinned name in a half-empty pill. On mobile, the caption switches to `display: grid; justify-content: start` (line 569) which fixes it — but the desktop is broken.
+
+**Why it hurts.** Looks accidental. A premium product never ships pills that are half-empty for half their content.
+
+**Recommendation.** Either (a) make `detail` required, or (b) when detail is empty, render only the name and center it. Conditional class in Svelte: `class:values-photo-caption-solo={!photo.detail}`.
+
+**Severity: Major.**
+
+---
+
+### 6. `.btn-yellow` uses pure black text
+
+**Inconsistency.** `buttons.css:65` — `.btn-yellow { color: var(--text-strong); }` where `--text-strong: #000`. Every other text surface in the app uses `--gw-green #213021` or `--text-heading #1f2421`.
+
+**Why it hurts.** Pure black on the brand yellow is the loudest color combo on the page. It pulls the eye like a warning sign and competes with the actual content. It's also the only place pure black appears outside `.values-inner .section-heading`.
+
+**Recommendation.** Use `var(--gw-green)` on `.btn-yellow`. Contrast is still well above AA (≈8:1 on `#ffd100`) and the button immediately feels intentional and on-brand.
+
+**Severity: Major.**
+
+---
+
+### 7. No `:focus-visible` states defined for `.btn`
+
+**Inconsistency.** `buttons.css` defines `:hover` and `:active` only. No `:focus-visible`. Keyboard users get the browser default outline, which on a custom pill button with `border-radius: 40px` looks broken.
+
+**Why it hurts.** Accessibility failure + perceived quality drop for power users. Tab through the page once and the button outlines clip the radius.
+
+**Recommendation.**
+```css
+.btn:focus-visible {
+ outline: 2px solid var(--gw-green);
+ outline-offset: 3px;
+}
+.btn-green:focus-visible { outline-color: var(--yellow); }
+```
+
+**Severity: Major.**
+
+---
+
+### 8. `.values-inner` uses hardcoded `padding: 0 50px`
+
+**Inconsistency.** Line 187. The codebase already defines `--space-container-x: clamp(24px, 4vw, 48px)`. The 50px here doesn't match any other section's container padding and breaks the visual gutter rhythm down the page.
+
+**Why it hurts.** When sections scroll past, the left/right gutters jitter by a few pixels between sections. Reads as cheap on a calibrated monitor.
+
+**Recommendation.** Replace with `padding: 0 var(--space-container-x)`. Audit every section component and do the same — almost certainly this is repeated elsewhere.
+
+**Severity: Major.**
+
+---
+
+### 9. Mobile eyebrow font-size drops to 11px
+
+**Inconsistency.** Line 549 — `.values-eyebrow { font-size: 11px; }` on mobile. Combined with `letter-spacing: 0.08em` and uppercase, this is functionally close to unreadable on a real phone.
+
+**Why it hurts.** Mobile readability and accessibility. Apple HIG and Material both recommend 12px minimum for body-adjacent text; uppercase tracked text should be 12–13px+ for legibility.
+
+**Recommendation.** Mobile eyebrow: 12px, tracking ≤ 0.06em. Or drop uppercase on mobile entirely.
+
+**Severity: Major.**
+
+---
+
+### 10. Featured photo card loses its prominence on mobile
+
+**Inconsistency.** Lines 558–560 — `.values-photo-card-featured { grid-row: auto; min-height: 178px; }` on mobile. The whole point of the "featured" card is its larger size; on mobile it becomes a peer to the rest, and the layout decision evaporates.
+
+**Why it hurts.** The featured card is the strongest brand image. On mobile (where the majority of traffic comes from for local services), it's neutered.
+
+**Recommendation.** On mobile, make the featured card span both columns of the first row (`grid-column: 1 / -1`) with a larger min-height (~240px). Visual lead image, not a peer.
+
+**Severity: Major.**
+
+---
+
+### 11. Reveal animation is slow
+
+**Inconsistency.** Lines 645–647 — `opacity 0.55s, transform 0.7s` with a `translateY(24px)` reveal. The cubic-bezier is fine but the durations are dated.
+
+**Why it hurts.** Premium products (Linear, Apple, Vercel) settle reveal animations at 0.25–0.4s. 0.7s feels like a 2015 marketing site.
+
+**Recommendation.** `opacity 0.3s ease, transform 0.45s cubic-bezier(0.2, 0.8, 0.2, 1)`. Reduce `--reveal-distance` to 16px.
+
+**Severity: Minor.**
+
+---
+
+### 12. The "with/without" contrast cell is the page's strongest moment but ends in a quiet brand voice mismatch
+
+**Inconsistency.** The "good" cell ends with: *"That is what people are really buying: peace of mind, routine, and a dog who feels cared for."* — yellow text on green, font-size 13px. The footer is the punchline but it's the smallest type in the cell.
+
+**Why it hurts.** Hierarchy inversion. The most quotable line is treated like a footnote.
+
+**Recommendation.** Promote the footer to 15–16px, maintain the yellow color and family. Add a small top divider only on the negative cell so the positive cell's footer reads as continuous emphasis.
+
+**Severity: Minor.**
+
+---
+
+### 13. `.values-bento` uses a 1px gap + ink-tinted background trick for hairlines
+
+**Inconsistency.** Line 290 — `gap: 1px; background: rgba(17, 20, 24, 0.1);` to simulate dividers. This is fine technique but the resulting line color (`rgba(ink, 0.1)`) is darker than the actual borders elsewhere in the system (`--border-soft 0.05`, `--border-muted 0.08`).
+
+**Why it hurts.** Bento dividers read heavier than every other border on the page. The card looks "boxier" than its neighbors.
+
+**Recommendation.** Drop the gap-background trick to `rgba(var(--ink-rgb), 0.06)` or a real `--border-soft-strong` token. Match the rest of the system.
+
+**Severity: Polish.**
+
+---
+
+### 14. `.values-contrast-num` ("01", "02") in `rgba(17, 20, 24, 0.22)` is barely visible
+
+**Inconsistency.** Decorative numbering at ~3:1 contrast.
+
+**Why it hurts.** If it's decorative, that's fine. But it's currently big enough to look like it's *supposed* to be read. Either commit to it being a design element (larger, more confident) or make it functional (numbered as a real step indicator).
+
+**Recommendation.** Either bump to 32px+ with the same low opacity (treats it as a visual mark, like editorial step numbering) or remove entirely. Half-measures are the worst outcome.
+
+**Severity: Polish.**
+
+---
+
+## Global Design System Drift
+
+1. **Tokens written but not consumed.** `--shadow-card`, `--shadow-panel-strong`, `--surface-panel-warm`, `--text-muted`, `--border-soft` exist, but `ValuesSection.svelte` re-implements all of them inline. Likely repeated in every section component.
+2. **Spacing scale (`--space-1` … `--space-14`) is ignored.** Every padding/margin in `ValuesSection` is a literal pixel value off the 4px grid (`50px`, `38px`, `36px`, `30px`, `26px`, `22px`, `18px`, `13px`, `9px`).
+3. **No radius scale at all.** Every component invents its own.
+4. **No type scale.** Headings use `clamp()` with different formulas everywhere.
+5. **Seven grays, six off-whites, three greens** in the token file — the surface area of the system exceeds what any human can use consistently.
+6. **Hardcoded shadows.** The shadow tokens are a *complete* shadow system that nobody is calling.
+7. **Hover states use `translateY(-2px) scale(1.012)` everywhere on `.btn`** — but card hovers in `ValuesSection` use `transform: scale(1.06)`. No shared motion language.
+
+---
+
+## Fast Wins
+
+1. **Find/replace every raw hex color in `ValuesSection.svelte`** with a token. ~30 minutes. Immediate consistency lift.
+2. **Replace `.values-inner { padding: 0 50px }` with `padding: 0 var(--space-container-x)`** and do the same in every section component. ~20 minutes. Fixes the section-to-section gutter jitter.
+3. **Add `--radius-sm/md/lg/pill` tokens and codemod components to use them.** Pick 20px as the universal card radius. ~1 hour.
+4. **Switch `.btn-yellow` text to `var(--gw-green)`.** 5 minutes. Big perceived-quality lift — the page stops shouting.
+5. **Add `:focus-visible` to `.btn`.** 5 minutes. Accessibility + polish.
+6. **Fix the photo caption layout when `detail` is empty.** 10 minutes. Removes the only broken-looking element on the page.
+7. **Bump mobile featured photo to `grid-column: 1 / -1`.** 2 minutes. Mobile hero moment.
+
+---
+
+## Premiumisation Opportunities
+
+1. **Collapse the gray palette to 3 values** and the off-white palette to 2. Premium brands feel calm because their surface palette is small.
+2. **Speed up all reveal/hover transitions to 0.25–0.4s.** Slower motion now reads as a slow site.
+3. **Replace pure-black text (`#000`) with `var(--gw-green)` or `var(--text-heading)` everywhere.** True black on warm off-white feels harsh; the dark green keeps it editorial.
+4. **Tighten letter-spacing on display headings** to `-0.03em`. Mid-2020s premium look. Bonus: add `text-wrap: balance` to all h2s.
+5. **Replace the bento dividers** with `--border-soft` and add a barely-visible inner highlight (`inset 0 1px 0 rgba(255,255,255,0.5)`) for the elevated feel Linear uses on dark surfaces.
+6. **Promote the "punchline" copy** (the contrast-footer lines) to body-lead size. Stop hiding the best writing in 13px.
+7. **Inline social proof under the contrast cells.** "Joined by 200+ Auckland owners" or a row of three small testimonial avatars with rating. The section currently asserts emotional benefits without proof — easy E-E-A-T win.
+
+---
+
+## Mobile Audit
+
+- **Eyebrow text is 11px** with uppercase + 0.08em tracking — sub-legible. Bump to 12px, drop tracking to 0.06em.
+- **Featured photo loses its purpose** — collapses to peer-card. Fix with `grid-column: 1 / -1; min-height: 240px`.
+- **Photo caption switches to vertical stack** (good) but `.values-photo-detail` runs through `-webkit-line-clamp: 2` even when detail is empty — leaves a weird invisible vertical reserve. Conditionally hide the element when empty.
+- **Container padding** uses `--space-container-x-mobile` (24px). Other components on the site may use the desktop 50px hardcode plus naive scaling — verify all sections collapse to a consistent 24px gutter.
+- **Tap targets** — the photo cards are large (~178px tall) and tappable hover scales aren't useful on touch. Hover scale lifts to `1.06` on tap-through devices look glitchy on iOS Safari. Wrap in `@media (hover: hover)` (already done — good) but verify the photo `:hover` rule is also gated (line 242 — good, it is).
+- **No visible scroll affordance** between the contrast cells and the values-points header — 30px gap on mobile (line 617). Reads as cramped.
+- **`.btn` padding `13px 28px`** on mobile yields ~44px tall tap targets only if line-height holds. Hero CTA stack untested in this audit but worth verifying tap area ≥48px on the primary booking CTA.
+
+---
+
+## Final Verdict
+
+**What makes it feel inconsistent.** A real design system exists in `variables.css` but the components don't use it. Components ship with raw hex, raw pixels, raw shadow tuples, and inconsistent radii. The token file has too many near-duplicate values, which is what enabled the drift in the first place.
+
+**What is preventing it from feeling premium.** Loud black-on-yellow CTA, slow reveal animations, seven different gray tones in the same viewport, radius drift between cards, missing focus states, and one or two broken elements (empty captions, mobile featured card collapse) that read as accidental rather than intentional.
+
+**Top 5 highest-ROI changes:**
+
+1. **Collapse the token palette** (3 grays, 2 off-whites, add `--radius-*` scale) — this is the precondition for everything else holding.
+2. **Codemod `ValuesSection.svelte` (and peer components) to consume tokens exclusively** — kills 80% of the visual inconsistency in one pass.
+3. **Switch `.btn-yellow` text to `--gw-green` and add `:focus-visible`** — instant perceived-quality lift on the most-clicked element.
+4. **Speed up reveal/hover transitions to 0.25–0.4s** — the single biggest "feels modern" lever.
+5. **Fix the photo caption (empty-detail case) and the mobile featured card** — two small fixes that remove the only "homemade" moments on the section.
+
+These are systems-level fixes, not redesigns. The CLAUDE.md mandate to preserve the WordPress visual design is respected — none of this changes layout, color brand, or hierarchy. It tightens the existing system into something that holds together.
diff --git a/docs/comparison-page-plan.md b/docs/comparison-page-plan.md
new file mode 100644
index 0000000..611fee6
--- /dev/null
+++ b/docs/comparison-page-plan.md
@@ -0,0 +1,237 @@
+# Dog Walker vs Doggy Daycare vs Pet Sitter vs Dog Boarding — Comparison Page
+
+**Page type:** Service-type comparison (category, not named competitors)
+**Target URL:** `/dog-walker-vs-daycare-vs-pet-sitter` (or `/blog/dog-walker-vs-daycare-vs-pet-sitter-auckland`)
+**Word count target:** 1,800–2,200 words
+**Last updated:** 2026-05-17
+**Author:** Alessandra (Goodwalk founder)
+
+---
+
+## Why this page exists (strategy note — not for publication)
+
+A service-type comparison captures research-stage Auckland owners who are still deciding *what kind of care* their dog needs before they decide *who to book*. It is high-intent, not transactional, but it qualifies the reader and feeds them into the booking flow once they self-identify as needing a dog walker.
+
+This page is safer than a named-competitor comparison because every claim is about service categories, not specific businesses, so there is no fairness or defamation risk. It also has broader search volume — "dog walker vs daycare" is searched far more than any "Goodwalk vs X" phrase ever will be.
+
+---
+
+## Primary keyword
+
+`dog walker vs doggy daycare`
+
+## Secondary keywords
+
+- `doggy daycare vs dog walker`
+- `dog walker vs pet sitter`
+- `dog boarding vs daycare`
+- `do I need a dog walker or daycare`
+- `is a dog walker better than daycare`
+- `best dog care option while at work auckland`
+
+## Long-tail / question keywords (for FAQ / H3s)
+
+- "How often should my dog go to daycare vs a walker"
+- "Is daycare too much for an anxious dog"
+- "Pet sitter or dog walker for a puppy"
+- "Cheapest reliable dog care while at work Auckland"
+
+## Title tag (under 60 chars)
+
+`Dog Walker vs Daycare vs Pet Sitter: Which One Fits Your Dog?`
+
+## Meta description (under 155 chars)
+
+`Auckland dog owners compare dog walking, daycare, pet sitting and boarding side by side — costs, energy fit, socialisation, and which works for working owners.`
+
+## H1
+
+`Dog Walker vs Doggy Daycare vs Pet Sitter vs Dog Boarding — Which One Actually Fits Your Dog?`
+
+---
+
+## Page structure
+
+### 1. Hero / Above-the-fold summary (120–180 words)
+
+- One sentence that names all four options.
+- Two-sentence honest verdict: "For most working Auckland owners with a healthy adult dog, a midday dog walker is the calmest and most cost-effective option. Daycare and boarding suit specific cases — high-energy dogs, long trips, multi-day absences — and pet sitters fill the in-home-care gap."
+- Primary CTA above fold: **"See Goodwalk's pack walk and puppy visit options"** → links to `/our-services` or booking section.
+- Trust line: "Written by Alessandra, founder of Goodwalk, walking dogs across Auckland Central."
+
+### 2. Quick decision matrix (the headline feature)
+
+A scannable table that answers the question in 10 seconds. This is the section that earns featured snippets and AI Overview citations.
+
+| What you need | Best fit | Why |
+|---|---|---|
+| Midday break while you're at work, calm dog | **Dog walker** | One-on-one or small group, short absence from home, no overstimulation |
+| Highly social, high-energy dog who hates being alone | **Doggy daycare** | All-day stimulation and dog-on-dog play |
+| You're away overnight or longer | **Dog boarding** or **in-home pet sitter** | Overnight care and feeding |
+| You're away 1–3 nights and want your dog at home | **Pet sitter (in-home)** | Familiar environment, lower stress |
+| Anxious, reactive, senior or recovering dog | **Dog walker (solo or tiny group)** | Predictable, low-stimulation routine |
+| Puppy mid-day toilet break + socialisation | **Puppy visit / short walk** | Daycare often too much before 6 months |
+
+### 3. Side-by-side feature comparison (the matrix the skill calls for)
+
+Word count target for this section + surrounding prose: ~400 words.
+
+| Factor | Dog walker | Doggy daycare | Pet sitter (in-home) | Dog boarding |
+|---|:---:|:---:|:---:|:---:|
+| Duration per session | 30–60 min | 4–10 hrs | Multiple visits / overnight at your home | Overnight at sitter's home or facility |
+| Dog stays in own home | ✅ | ❌ | ✅ | ❌ |
+| Overnight care | ❌ | ❌ | ✅ | ✅ |
+| Dog-on-dog socialisation | ⚠️ Small group only | ✅ High | ❌ Minimal | ⚠️ Varies |
+| Suitable for anxious / reactive dogs | ✅ Often | ❌ Usually not | ✅ Yes | ⚠️ Depends on setup |
+| Suitable for puppies under 6 months | ✅ Short visits | ⚠️ Some daycares accept, many don't | ✅ Yes | ⚠️ Varies |
+| Typical Auckland price (per session/day) | ~$30–$50 | ~$45–$70 | ~$60–$90/night | ~$55–$95/night |
+| Booking flexibility | High (recurring or ad-hoc) | Often needs membership | Medium | Low (peak periods book out) |
+| Stimulates without exhausting | ✅ | ⚠️ Can over-stimulate | ⚠️ Depends | ⚠️ Varies |
+
+> **Pricing disclaimer:** Indicative Auckland market ranges as of May 2026. Always confirm current rates directly with each provider.
+
+### 4. Detailed sections on each service type (~300 words each)
+
+#### 4a. What a dog walker actually does
+
+- One-on-one or tiny-group walks (Goodwalk runs "Tiny Gang" pack walks — max 4 dogs).
+- Typical Auckland Central session: pickup → walk → drop-off, 60 minutes including travel.
+- Best fit profile: dog who lives in an apartment or small section, owner working 8–6, needs one structured outdoor break a day.
+- Honest limitation: doesn't solve all-day separation if your dog has severe separation anxiety — pair with daycare or sitter on long days.
+- **Internal link:** to `/services/dog-walking` and `/services/pack-walks`.
+
+#### 4b. What doggy daycare actually does
+
+- All-day group play in a facility.
+- Best fit: high-drive, highly social adult dogs who genuinely enjoy busy environments.
+- Honest limitation: not every dog enjoys daycare even if they "love other dogs" on walks — sustained group play is a different intensity. Many trainers caution against daycare for under-6-month puppies due to over-stimulation and inconsistent play partners.
+- Cost: usually the most expensive recurring weekday option in Auckland.
+- No Goodwalk service in this category — acknowledge that openly. This builds trust.
+
+#### 4c. What a pet sitter does
+
+- Visits your home (or stays overnight) to feed, toilet, play, and check on your dog.
+- Best fit: short trips (1–4 nights), dogs who don't travel well, multi-pet households.
+- Honest limitation: less physical exercise than a walker provides; usually pair-able with a dog walking service for longer absences.
+- **Internal link:** to `/services/puppy-visits` (closest Goodwalk offering — short check-in visits).
+
+#### 4d. What dog boarding actually does
+
+- Your dog stays overnight at a boarder's home or kennel facility.
+- Best fit: longer absences (1+ weeks), owners without an in-home sitter option.
+- Honest limitation: change of environment is stressful for some dogs; book ahead — Auckland boarders fill up over school holidays and summer.
+
+### 5. Cost comparison over a typical month (~150 words)
+
+Example scenario: working owner, healthy 3-year-old labrador, gone 9–5 weekdays.
+
+| Option | Frequency | Monthly cost (indicative) |
+|---|---|---|
+| Dog walker, 3x/week | 12 sessions | ~$360–$600 |
+| Daycare, 3x/week | 12 days | ~$540–$840 |
+| Daycare 1x + dog walker 2x | mixed | ~$300–$600 |
+| Dog walker 5x/week | 20 sessions | ~$600–$1,000 |
+
+The "daycare 1x + walker 2x" hybrid is what many Auckland owners settle on — one social day and two calmer walking days per week.
+
+### 6. Decision flowchart (short, ~100 words + visual)
+
+A simple text flow that AI search engines can lift verbatim:
+
+> 1. Are you away for more than 24 hours? → **Pet sitter or boarder.**
+> 2. Is your dog highly social and high-energy? → **Daycare 1–2 days/week + walker on quiet days.**
+> 3. Is your dog anxious, reactive, senior or under 6 months? → **Dog walker, solo or tiny group.**
+> 4. Standard adult dog, normal energy, working hours? → **Dog walker, 2–5x/week.**
+
+### 7. FAQ section (5–7 questions, each 50–90 words)
+
+Designed for FAQPage schema and AI Overview citations.
+
+1. **Is a dog walker enough if I work 9–5?**
+2. **Is daycare too stimulating for puppies?**
+3. **Can I combine a dog walker with daycare?**
+4. **What's cheapest: walker, daycare, or sitter?**
+5. **Which option is best for an anxious or reactive dog?**
+6. **Do I need a dog walker every day?**
+7. **What's the difference between a pet sitter and a dog walker?**
+
+### 8. Closing CTA + author bio (~120 words)
+
+- One-paragraph honest summary: "If you've read this far and your dog is a normal-energy adult or a puppy who lives in Auckland Central, a midday dog walker is almost always the right starting point. Daycare and boarding are real options for specific cases — but the default answer for most owners is a walker."
+- CTA block: **"Book a Tiny Gang pack walk"** + secondary "See pricing".
+- Author bio: 2–3 lines on Alessandra, with a photo. Credibility signal for E-E-A-T.
+
+---
+
+## Conversion / CTA placement
+
+- **Above fold:** soft CTA — "See our walks" (not a booking ask yet, since they're still researching).
+- **End of section 4a (dog walker explainer):** strong CTA — "Book a Goodwalk walk".
+- **End of decision flowchart:** medium CTA — "See pricing and walk options".
+- **Footer of page:** final CTA + "Questions? Text Aless on [number]".
+
+Avoid any CTA inside sections 4b/4c/4d (daycare, sitter, boarding) — that's where the page earns trust by *not* trying to convert.
+
+---
+
+## Internal linking plan
+
+**Outbound from this page:**
+- `/our-services` (services overview)
+- `/services/dog-walking` (slug-driven service page)
+- `/services/pack-walks`
+- `/services/puppy-visits`
+- `/our-pricing`
+- `/contract` (for owners ready to onboard)
+
+**Inbound to this page (add links from):**
+- Homepage FAQ section
+- `/our-services` page footer ("Not sure which service fits? Read our comparison")
+- Each individual service landing page sidebar/below-the-fold
+- Blog posts that mention service trade-offs
+
+**Cross-links to build next:**
+- "Pack walks vs solo walks — which is right for your dog?" (Goodwalk-internal comparison)
+- "When should I start dog walks with my puppy?" (FAQ-style)
+- "Auckland Central dog walking suburbs we cover" (already in `/locations`)
+
+---
+
+## E-E-A-T signals to include on the page
+
+- Author byline with Alessandra's photo and 2-line bio
+- "Last updated: [date]" near the top
+- Methodology sentence: "Pricing ranges reflect publicly listed Auckland rates surveyed May 2026 and are indicative only."
+- At least one customer quote (from existing testimonials) about *why they chose a walker over daycare*
+- Link to Auckland Council dog registration page (existing trust signal already on site)
+
+---
+
+## Fairness checklist (skill requirement)
+
+- [x] No named-competitor claims — service-category comparison only
+- [x] Pricing flagged "as of May 2026, indicative"
+- [x] Acknowledges where Goodwalk doesn't compete (daycare, boarding)
+- [x] Recommends combining services where genuinely best for the dog
+- [x] No exaggerated claims about walker superiority — recommendation is conditional on dog profile
+
+---
+
+## Content gaps vs typical Auckland search results
+
+Most existing Auckland-region comparison content is either:
+1. **Generic global content** with US/UK pricing and no local relevance.
+2. **Daycare-operator pages** that always conclude "daycare is the answer."
+3. **Aggregator listings** with no genuine comparison.
+
+Goodwalk's advantage: written by a working Auckland walker with first-hand knowledge of which dogs thrive in which service. The honest "we don't do daycare and here's when daycare is actually the right call" framing is the differentiator.
+
+---
+
+## Recommendations (next steps after this page ships)
+
+1. **Add FAQPage schema** (see `comparison-schema.json`) so the FAQ block is eligible for rich results.
+2. **Build the two cross-link pages** ("pack walks vs solo walks", "when to start puppy walks") to strengthen the cluster.
+3. **Add a comparison shortcut from the homepage hero** — a small "Not sure which service fits?" link → this page.
+4. **Quarterly review reminder** — update pricing ranges and any factual claims every 3 months.
+5. **Consider a downloadable one-pager** of the decision flowchart for email-list signups (lead magnet).
diff --git a/docs/deployment.md b/docs/deployment.md
new file mode 100644
index 0000000..cac9354
--- /dev/null
+++ b/docs/deployment.md
@@ -0,0 +1,341 @@
+# Deployment
+
+## Hosts served by this stack
+
+The Goodwalk Svelte stack serves three subdomains from the same SvelteKit app
+container, routed by Host header at nginx:
+
+| Hostname | Purpose |
+|-----------------------------|--------------------------------------------|
+| `goodwalk.co.nz` / `www.…` | Public marketing site |
+| `clients.goodwalk.co.nz` | New-client onboarding + contract portal |
+| `cp.goodwalk.co.nz` | Owner admin dashboard (Aless only) |
+
+The shared nginx container reads TLS material from its host bind mount at
+`/docker/certbot/conf`, exposed inside the container as `/etc/letsencrypt`.
+The deploy scripts now auto-bootstrap any missing certificates referenced by
+the shared nginx config, including `clients.goodwalk.co.nz`,
+`cp.goodwalk.co.nz`, and the legacy HTTPS redirect aliases
+`onboarding.goodwalk.co.nz` and `admin.goodwalk.co.nz`. They do that by
+temporarily loading an HTTP-only nginx config and running `certbot/certbot`
+against the mounted ACME webroot. The only prerequisite is that each
+hostname's DNS A record already points at the droplet and port `80` is
+reachable.
+
+The dashboard's data (`client_profiles`, `allowed_emails`, `drafts`) lives in
+the shared postgres database alongside the marketing site content, in a single
+`admin_kv` table created by `docker/postgres/init/002-admin-kv.sql`. The
+mail-api connects with the same `DATABASE_URL` the SvelteKit app uses.
+
+### Seeding admin_kv from the old JSON files
+
+Existing installs have admin data in `client_profiles.json`,
+`allowed_emails.json`, and `drafts.json` on the mail-api Docker volume. To copy
+that data into postgres on the next deploy, run:
+
+```powershell
+./deploy.ps1 -SeedAdminData
+```
+
+That sets `ADMIN_DATA_SEED_FROM_JSON=force` for the mail-api container, which
+overwrites `admin_kv` from the JSON files on the next boot. Subsequent deploys
+default back to `auto` (seed only when `admin_kv` is empty), so they are no-ops
+for the seed. Use `-SeedAdminData` again if you ever need to force a re-seed.
+
+## Server layout confirmed
+
+The production server currently runs multiple separate Docker Compose projects:
+
+- Main public site WordPress stack:
+ - project: `goodwalkconz`
+ - path: `/docker/wordpress/goodwalk.co.nz`
+- Legacy onboarding WordPress stack:
+ - project: `onboardinggoodwalkconz`
+ - path: `/docker/wordpress/onboarding.goodwalk.co.nz`
+- Shared nginx:
+ - project: `nginx`
+ - path: `/docker/nginx`
+- Shared mysql:
+ - project: `mysql`
+ - path: `/docker/mysql`
+
+The deployment scripts in this repo are set up to deploy the new Svelte site as a
+separate stack at:
+
+- remote path: `/docker/goodwalk-svelte`
+- compose file: `docker-compose.prod.yml`
+- docker project: `goodwalk-svelte`
+
+This leaves the onboarding site, shared nginx, shared mysql, and other unrelated
+containers untouched.
+
+## Files involved
+
+- [deploy.ps1](deploy.ps1)
+ - Windows entrypoint for packaging the repo, uploading it, and running the
+ remote deployment helper over SSH.
+- [scripts/deploy.ps1](scripts/deploy.ps1)
+ - Deprecated compatibility wrapper that forwards to the repo-root
+ `deploy.ps1`. Keep using the root script directly.
+- [scripts/deploy-remote.sh](scripts/deploy-remote.sh)
+ - Server-side helper that updates only the `goodwalk-svelte` compose project.
+- [scripts/deploy-from-git.sh](scripts/deploy-from-git.sh)
+ - Standalone server-side entrypoint that pulls from Git, then runs the same
+ compose/nginx deployment steps on the server.
+- [docker-compose.prod.yml](docker-compose.prod.yml)
+ - Production compose file for the new Svelte app, mail API, and Postgres.
+- `scripts/export-homepage-content.mjs`
+ - Local helper that exports the current `src/lib/content/homepage.ts` into a
+ deployable JSON payload before each deployment.
+- `scripts/sync-homepage-content.mjs`
+ - Runtime helper that upserts the exported homepage content into PostgreSQL
+ after deploys that affect the app/database.
+- [ssh-config](ssh-config)
+ - Repo-local SSH config used by the deployment script.
+- [nginx/goodwalk.co.nz.svelte.conf.example](nginx/goodwalk.co.nz.svelte.conf.example)
+ - Example shared-nginx config for routing the main public site to the new
+ Svelte app and mail API, including the `clients` and `cp` subdomains.
+
+## First-time server preparation
+
+1. Fill in [ssh-config](ssh-config) with the real host details.
+
+2. Create the deployment directory on the server:
+
+```bash
+mkdir -p /docker/goodwalk-svelte
+```
+
+3. The first deployment will auto-create the production env file on the server at:
+
+```bash
+/docker/goodwalk-svelte/.env
+```
+
+It is created from [deploy.env.template](deploy.env.template). Current template contents:
+
+```env
+APP_VERSION=4.0.1
+ENABLE_GENERAL_ENQUIRIES=false
+PUBLIC_ENABLE_MOBILE_CTA_BUTTON=false
+PUBLIC_ENABLE_ENHANCED_CONTENT_IMAGES=false
+TZ=Pacific/Auckland
+
+POSTGRES_DB=goodwalk
+POSTGRES_USER=goodwalk
+POSTGRES_PASSWORD=gw_Pg_7Jm9!Qx4#Ld2@Vr8
+POSTGRES_PASSWORD_URLENCODED=gw_Pg_7Jm9%21Qx4%23Ld2%40Vr8
+
+RESEND_API_KEY=replace-me
+OWNER_EMAIL=replace-me
+SECONDARY_CP_EMAIL=
+SECONDARY_CP_EMAILS=
+FROM_EMAIL=GoodWalk
+REPLY_TO=aless@goodwalk.co.nz
+MAIL_API_DATA_DIR=/app/data
+
+FORM_MIN_SECONDS=4
+FORM_MAX_SECONDS=7200
+RATE_LIMIT_WINDOW_SECONDS=900
+RATE_LIMIT_MAX_PER_IP=5
+RATE_LIMIT_MAX_PER_EMAIL=3
+RATE_LIMIT_MIN_INTERVAL_SECONDS=20
+EMAIL_SEND_TIMEOUT_SECONDS=20
+```
+
+After the first deploy, edit `/docker/goodwalk-svelte/.env` on the server and replace:
+
+- `RESEND_API_KEY=replace-me`
+- `OWNER_EMAIL=replace-me`
+
+Optional CP dashboard admins:
+
+- `SECONDARY_CP_EMAIL=person@example.com`
+- `SECONDARY_CP_EMAILS=bob@smith.com;bobsmith2@smith.com`
+
+`OWNER_EMAIL` always keeps CP access. The secondary values are optional and may be
+semicolon-, comma-, or whitespace-separated.
+
+Frontend flags:
+
+- `PUBLIC_ENABLE_MOBILE_CTA_BUTTON=false` keeps the sticky mobile booking CTA hidden.
+- Set `PUBLIC_ENABLE_MOBILE_CTA_BUTTON=true` to show it again.
+- `PUBLIC_ENABLE_ENHANCED_CONTENT_IMAGES=false` skips eager `@sveltejs/enhanced-img` processing for content images during production builds. Turn it on only if you intentionally want non-WebP images from `src/lib/images` to go through the imagetools pipeline.
+
+4. Confirm the shared Docker network already exists:
+
+```bash
+docker network ls | grep webnet
+```
+
+Your server already uses `webnet`, so this should already be present.
+
+5. Confirm the shared nginx compose mounts still point at the host certbot
+ paths expected by the deploy scripts:
+
+```yaml
+- /docker/certbot/conf:/etc/letsencrypt:ro
+- /docker/certbot/www:/var/www/certbot:ro
+```
+
+The scripts inspect the running `nginx` container to derive those host paths
+before checking or issuing certificates.
+
+## First deploy
+
+From Windows PowerShell in the repo root:
+
+```powershell
+powershell -ExecutionPolicy Bypass -File .\deploy.ps1
+```
+
+This is the single supported deployment entrypoint. If you see
+`scripts/deploy.ps1`, that file now just forwards to the root script so the
+deployment logic only lives in one place.
+
+Or skip the confirmation prompt:
+
+```powershell
+powershell -ExecutionPolicy Bypass -File .\deploy.ps1 -Force
+```
+
+To rebuild and restart only one service, for example the mail API:
+
+```powershell
+powershell -ExecutionPolicy Bypass -File .\deploy.ps1 -Force -Service mail-api
+```
+
+## Remote Git deploy
+
+If you want the production server to pull straight from Gitea instead of
+receiving an uploaded tarball from Windows, use
+[scripts/deploy-from-git.sh](scripts/deploy-from-git.sh) on the server.
+
+Recommended credential setup for a private HTTPS repo:
+
+```bash
+umask 077
+cat > ~/.netrc <<'EOF'
+machine g.sublogue.com
+login YOUR_GITEA_USERNAME
+password YOUR_READ_ONLY_TOKEN
+EOF
+chmod 600 ~/.netrc
+```
+
+Install the script on the server and make it executable:
+
+```bash
+install -m 0755 scripts/deploy-from-git.sh /usr/local/bin/goodwalk-deploy
+```
+
+The remote host must have `git` and `docker`. A host-level `node` install is
+optional; if it is missing, the script will export homepage content using a
+temporary `node:22-alpine` container instead.
+
+Run a full deploy from the repo:
+
+```bash
+/usr/local/bin/goodwalk-deploy \
+ --repo-url https://g.sublogue.com/admin/gw-svelte.git \
+ --branch main \
+ --deploy-path /docker/goodwalk-svelte \
+ --compose-file docker-compose.prod.yml \
+ --project-name goodwalk-svelte \
+ --nginx-source nginx/goodwalk.co.nz.svelte.conf.example \
+ --nginx-target /docker/nginx/conf.d/goodwalk.co.nz.conf \
+ --nginx-compose-file /docker/nginx/docker-compose.yml \
+ --nginx-project-name nginx \
+ --maintenance-host-dir /docker/nginx/maintenance \
+ --maintenance-flag /docker/nginx/conf.d/maintenance.flag
+```
+
+Deploy a specific commit or tag:
+
+```bash
+/usr/local/bin/goodwalk-deploy \
+ --repo-url https://g.sublogue.com/admin/gw-svelte.git \
+ --branch main \
+ --ref \
+ --deploy-path /docker/goodwalk-svelte \
+ --compose-file docker-compose.prod.yml \
+ --project-name goodwalk-svelte \
+ --nginx-source nginx/goodwalk.co.nz.svelte.conf.example \
+ --nginx-target /docker/nginx/conf.d/goodwalk.co.nz.conf \
+ --nginx-compose-file /docker/nginx/docker-compose.yml \
+ --nginx-project-name nginx \
+ --maintenance-host-dir /docker/nginx/maintenance \
+ --maintenance-flag /docker/nginx/conf.d/maintenance.flag
+```
+
+## Homepage content sync
+
+Local development can feel fresher than production because production reads the
+homepage/shared content from PostgreSQL whenever `DATABASE_URL` is set.
+
+The deployment flow now handles that automatically:
+
+1. `deploy.ps1` exports the current `src/lib/content/homepage.ts` into
+ `deploy-data/homepage-content.json`.
+2. The deploy archive uploads that JSON payload with the app source.
+3. After the Goodwalk stack is updated, the remote helper runs a content sync
+ inside the app container.
+4. That sync upserts the `homepage` row in `site_content`.
+
+This means future deploys will carry your latest file-based homepage/navigation/
+shared content changes into production PostgreSQL automatically.
+
+## Mail auth persistence
+
+The mail API stores auth state in `DATA_DIR`, including:
+
+- `allowed_emails.json`
+- `client_profiles.json`
+- `drafts.json`
+
+Both compose files now mount a named Docker volume at `MAIL_API_DATA_DIR`
+(default `/app/data`) so previously registered client emails and saved drafts
+survive container rebuilds and redeploys.
+
+## Cutover nginx
+
+After the new Svelte stack is up and healthy, update the shared nginx config on
+the server for the main site.
+
+Current live file:
+
+```bash
+/docker/nginx/conf.d/goodwalk.co.nz.conf
+```
+
+Use the repo example as the new target config:
+
+```bash
+nginx/goodwalk.co.nz.svelte.conf.example
+```
+
+Important:
+- `deploy.ps1` now copies the repo nginx config to `/docker/nginx/conf.d/goodwalk.co.nz.conf` and reloads the shared nginx container as part of deployment.
+- The repo nginx config uses Docker's internal resolver so future app/mail container rebuilds will not leave nginx pinned to stale upstream IPs.
+- The same nginx config now also routes `clients.goodwalk.co.nz` to the Svelte app and `/api/onboarding-submit` to the shared mail API.
+- The owner dashboard is now served on `cp.goodwalk.co.nz`.
+- `onboarding.goodwalk.co.nz` and `admin.goodwalk.co.nz` should be kept only as redirect aliases once their DNS and TLS are in place.
+- The deploy script will attempt to issue any missing certificates for `clients.goodwalk.co.nz`, `cp.goodwalk.co.nz`, and `onboarding.goodwalk.co.nz` before the final nginx reload.
+
+Manual nginx commands, if you ever need them:
+
+```bash
+docker compose -p nginx -f /docker/nginx/docker-compose.yml exec nginx nginx -t
+docker compose -p nginx -f /docker/nginx/docker-compose.yml exec nginx nginx -s reload
+```
+
+## Important notes
+
+- Do not deploy the top-level `docker-compose.yml` to this server for production.
+ It includes its own nginx service and does not match the shared nginx setup on
+ the host.
+- The deployment scripts do not stop or remove the onboarding WordPress stack.
+- The deployment scripts do not touch the shared mysql compose project.
+- The deployment scripts preserve the remote `.env` file.
+- The site check in `deploy.ps1` targets `https://www.goodwalk.co.nz/api/health`.
+ Before nginx cutover, use `-SkipSiteCheck` or expect that check to fail.
diff --git a/docs/design-audit.md b/docs/design-audit.md
new file mode 100644
index 0000000..a512d00
--- /dev/null
+++ b/docs/design-audit.md
@@ -0,0 +1,455 @@
+# Design Audit — Goodwalk (Codebase-wide)
+
+Re-audited beyond ValuesSection. Read `sections.css` (1130 lines), `typography.css`, `responsive.css`, `variables.css`, and surveyed each major surface (Hero, Intro, PageHeader, Footer, FAQ, Instagram, Mobile Menu, Testimonial card). Findings below cite actual files/lines.
+
+---
+
+## Critical issues
+
+### 1. Three different "eyebrow" components exist, sharing no DNA
+**Problem.** The same UX element — small uppercase intro label above a heading — is rendered three incompatible ways:
+- `.eyebrow` (typography.css:100): font-body, 12px, 700, `0.09em`, green text, no background.
+- `.hero-kicker` (sections.css:69): font-body, 12px, 700, `0.08em`, yellow text in a *yellow pill with yellow border*.
+- `.intro-kicker` (sections.css:408): font-body, 12px, 600 (weight differs), `0.18em` (tracking differs — over 2× the others), white-58% text with a *yellow rule prefix*.
+
+Plus `.values-eyebrow`, `.booking-eyebrow`, `.footer-col-label`, `.intro-meta` are all variants.
+
+**Root cause.** No eyebrow primitive in the system. Each section author re-implemented from scratch.
+**Perception.** Reader senses different sections were designed at different times. Brand voice fragments.
+**Fix.** One `` component. Three variants only. Migrate all six existing implementations.
+**Class: System debt.**
+
+---
+
+### 2. Six different primary heading scales
+**Problem.**
+| Selector | Scale | Weight | Tracking |
+|---|---|---|---|
+| `.section-heading` | clamp(30, 4.6vw, 44) | 800 | -0.035em |
+| `.hero-text h1` | clamp(34, 4vw, 56) | 800 | -0.045em |
+| `.intro-headline` | clamp(30, 4.4vw, 54) | **500** | -0.02em |
+| `.ph-title` | clamp(34, 4vw, 56) | 800 | -0.04em |
+| `.info-block h2` | clamp(28, 2.4vw, 32) | 700 | -0.02em |
+| `#instagram h2` | clamp(30, 3vw, 36) | 700 | -0.02em |
+
+Six clamp formulas. Five tracking values. Weights ranging 500→800. The base `h2 { font-weight: 700 }` (typography.css:58) conflicts with `.section-heading { font-weight: 800 }` (typography.css:79) — actual rendered weight depends on whether the heading author remembered to apply the class.
+
+**Root cause.** No `--text-display`, `--text-h1`, `--text-h2` tokens. Every author authors a fresh clamp.
+**Perception.** Headings jump in size and weight as you scroll. The 500-weight `.intro-headline` reads as a different brand from the 800-weight `.hero h1` directly above it.
+**Fix.** Define type tokens: `--text-display` (52/800/-0.04), `--text-h1` (44/800/-0.035), `--text-h2` (32/700/-0.02), `--text-h3` (22/700/-0.02), `--text-body-lead` (17/500/0), `--text-body` (16/400/0). Map every heading. Delete the per-section clamps.
+**Class: Design debt + System debt.**
+
+---
+
+### 3. The Instagram section is a chromatic anomaly
+**Problem.** Section background sequence on the homepage:
+`#hero` (green) → `#intro` (green) → `#promise/#values` (off-white) → `#services` (white) → `#testimonials` (white) → `#info` (white) → `#newlead` (white) → `#instagram` (**solid #ffd100 full-bleed**) → `footer` (green).
+
+The Instagram block is the only place brand yellow is used as a *page section*. It's also where pure black-on-yellow body text appears at `rgba(0, 0, 0, 0.6)` (sections.css:738).
+
+**Root cause.** Brand yellow was treated as both an accent (CTAs, highlights) AND a surface (this section). Premium products pick one.
+**Perception.** Reads as advertising, not editorial. Breaks the calm green/off-white rhythm. The user scrolls into a marketing banner mid-page.
+**Fix.** Either (a) demote to a green section with yellow accents inside, or (b) shrink to a narrow card-on-off-white block. Reserve `var(--yellow)` for accents only.
+**Class: Design debt + Conversion debt** (it sits right before footer — possibly the last impression).
+
+---
+
+### 4. Hero animation system runs on a different clock than the rest of the page
+**Problem.** Hero entry: image 1.6s, text rise 0.85s, underline draw 0.9s @ 900ms delay, star pop 0.45s starting at 820ms. Total animation completes ~1.7s after load.
+
+Rest of page: reveal-block now 0.3s opacity / 0.45s transform.
+Mobile menu: 180–220ms.
+Button hover: 0.16–0.22s.
+
+**Root cause.** Hero was authored before the reveal/motion system existed. Never reconciled.
+**Perception.** The hero feels heavy and ceremonial; the rest of the page feels snappy. Two products bolted together.
+**Fix.** Cap all hero animations at 0.5s. Reduce delays — first text element by 80ms, each subsequent +60ms. Drop the underline-draw delay from 900ms to 350ms. Keep the elegance, lose the lethargy.
+**Class: Polish debt.**
+
+---
+
+### 5. `--space-container-x-tablet` breaks the gutter rhythm
+**Problem.** variables.css:145 — `--space-container-x-tablet: 30px`. Desktop value is `clamp(24px, 4vw, 48px)`. At 1024px width (the tablet breakpoint), 4vw = 41px. The tablet override drops gutters to 30px, then back up to 41–48px on desktop. A user resizing a window or rotating an iPad sees the gutters *narrow* then *widen*.
+
+**Root cause.** A patch fix applied at the tablet breakpoint that doesn't share the desktop formula.
+**Perception.** Visible engineering. Anyone who notices the layout shift on resize loses trust.
+**Fix.** Either delete `--space-container-x-tablet` and let the clamp handle the breakpoint, or make tablet = `clamp(20px, 3vw, 32px)` so the curve is continuous.
+**Class: System debt.**
+
+---
+
+### 6. Card system doesn't exist — each section invents its own
+**Problem.** Same UX primitive (an elevated content card), shipped four ways:
+| Component | Radius | Padding | Background | Shadow |
+|---|---|---|---|---|
+| `.testimonial-card` | 28px | 36px 32px | gradient panel→panel-soft | inset + shadow-md |
+| `.faq details` | 16px | 18px 22px | surface-page | shadow-lg on open only |
+| `.values-photo-card` | 28px → md mobile | n/a | beige | inset + card |
+| `.booking-field-card` | 24px | 24px 22px | (varies) | (varies) |
+
+Plus `.values-point` (no radius, pure surface), `.mobile-menu-links a` (16px radius), `.ph-media` (28px), `.intro-google` (pill).
+**Root cause.** No `` primitive. Every author re-decides.
+**Perception.** Adjacent sections feel like they were copy-pasted from different Behance projects.
+**Fix.** One Card system: `--radius-lg` (20px) for cards, `--space-7` (32px) padding, `--shadow-card` standard. Variants: `card`, `card-elevated`, `card-quiet`. Migrate all four.
+**Class: System debt + Design debt.**
+
+---
+
+### 7. Footer typography is a four-size jumble
+**Problem.** Within the footer alone: 14px (`.footer-brand p`, `.footer-nav a`), 13px (`.footer-bottom`, `.footer-contact-link`), 12px (`.footer-back-top`, `.footer-social-invite`), 10px (`.footer-col-label`), plus 14px (`footer h4`). Five sizes in a quiet auxiliary surface.
+
+Add opacities: `.footer-brand p` 0.72, `.footer-nav a` 0.72, `.footer-contact-link` 0.7, `.footer-bottom` 0.6, `.footer-back-top` 0.5, `.footer-col-label` 0.5.
+
+**Root cause.** Each footer column was sized independently to hit visual targets, not to scale.
+**Perception.** The footer feels busy in a quiet way — many small elements clamoring for slightly different attention.
+**Fix.** Two sizes: 14px (links/copy) + 12px (label/legal). Two opacities: 0.85 (active text) + 0.55 (labels). Delete the rest.
+**Class: Polish debt + UX debt.**
+
+---
+
+### 8. `.section-heading` color is pure `#000`, used on warm off-white backgrounds
+**Problem.** typography.css:82 — `.section-heading { color: var(--text-strong); }` where `--text-strong: #000`. Promise, Values, Testimonials, Services, Info all render their primary heading in pure black on a warm `--off-white #f8f7f2`. Same pattern as the `.btn-yellow` issue I already fixed: pure black against a warm surface reads as harsh and clinical.
+
+**Root cause.** `--text-strong` is a "darkest-possible" token used reflexively for headings instead of a heading-specific token.
+**Perception.** Headings shout. The rest of the type — body in `var(--text)` (#2e3031), muted in green-cast grays — is warm. Headings are not.
+**Fix.** `.section-heading { color: var(--text-heading); }` (which is `#1f2421`, near-black with a green cast). Or `var(--gw-green)` for full editorial treatment. The dark-green-on-warm-cream pairing is the brand's most premium register.
+**Class: Design debt.**
+
+---
+
+### 9. Two Google trust-mark components live as duplicates
+**Problem.** `.hero-trust-mark` (sections.css:153) and `.intro-google-mark` (sections.css:491). Same Google "G" circle, same white background, same shadow recipe with slightly different parameters (`0 2px 8px rgba(0,0,0,0.25)` vs `0 4px 12px rgba(0,0,0,0.25)`). Different sizes (28px vs 36px). Different parent chip layouts (`.hero-trust-chip` vs `.intro-google`).
+
+**Root cause.** Copy-paste authoring. The second one was built without abstracting the first.
+**Perception.** A user who scrolls hero → intro sees the same trust signal twice in slightly different sizes. Either feels redundant or sloppy depending on attention level.
+**Fix.** One `` component. Two clean sizes (24, 36). Single shadow token.
+**Class: System debt + Conversion debt** (trust signals matter; redundant ones dilute).
+
+---
+
+### 10. Letter-spacing on headings has four values
+**Problem.** `-0.02em` (h2/h3, intro-headline, info-block h2), `-0.035em` (section-heading), `-0.04em` (ph-title), `-0.045em` (hero h1). Four bespoke tracking values.
+
+**Root cause.** Each heading was tuned visually without a tracking scale.
+**Perception.** Headings at similar sizes (`.intro-headline` 54px at `-0.02em` vs `.hero h1` 56px at `-0.045em`) feel like they're set in *different fonts*, when they're actually both Unbounded.
+**Fix.** Two tracking values: `-0.035em` for display (36px+), `-0.02em` for everything else. Lock in the type tokens.
+**Class: Design debt.**
+
+---
+
+### 11. Hero text relies on hand-numbered nth-child animation delays
+**Problem.** sections.css:272–278 — `.hero-text > :nth-child(1) { animation-delay: 160ms; }` through `:nth-child(7)`. Seven hardcoded children. If the hero content shape changes (e.g., remove the kicker, add a phone CTA), the choreography breaks silently.
+
+**Root cause.** Animation was authored against current markup, not against generic children.
+**Perception.** Probably invisible to most users *until* it breaks — at which point an element pops in without animation while others slide.
+**Fix.** Use CSS counter or a Svelte action that applies `--reveal-delay` per child. Or accept staggered delays via `:nth-child(n)` formula: `animation-delay: calc(160ms + (var(--i, 0)) * 100ms)`.
+**Class: System debt + Polish debt.**
+
+---
+
+### 12. Mobile hero CTA shrinks to 12px font, 9px horizontal padding under 480px
+**Problem.** responsive.css:767–774 — primary `.btn-yellow` becomes `padding: 13px 9px; font-size: 12px; line-height: 1.1` on iPhone SE-class widths. This is the *primary booking CTA* — the conversion target.
+
+**Root cause.** Two CTAs side-by-side in a row at very narrow widths. Author chose to squeeze both rather than stack.
+**Perception.** On a 320px screen, the booking button reads as cramped, low-confidence. CTA hierarchy collapses.
+**Fix.** Keep buttons stacked under 480px (already done at 768px — extend the rule). Primary CTA: full-width, 14px font, 14px vertical padding. Secondary link below.
+**Class: Conversion debt.**
+
+---
+
+### 13. Hardcoded shadows everywhere despite a complete shadow token set
+**Problem.** Beyond ValuesSection, the same pattern repeats:
+- `.testimonial-card:hover` — `0 8px 40px rgba(0, 0, 0, 0.08)` (sections.css:620). Not a token.
+- `.hero-trust-mark` — `0 2px 8px rgba(0, 0, 0, 0.25)` (line 161). Not a token.
+- `.intro-google-mark` — `0 4px 12px rgba(0, 0, 0, 0.25)` (line 499). Not a token.
+- `.ph-media` — `0 16px 40px rgba(var(--ink-rgb), 0.08)` (line 1027). Not a token.
+- `.intro-google` — uses inset shadow tuples directly (lines 470, 482).
+
+**Root cause.** Authors didn't know which token to grab from 18 shadow options.
+**Perception.** Elevations feel inconsistent — some cards lift more, some less, even when visually they're the same depth.
+**Fix.** Five tokens, not eighteen: `--shadow-flat`, `--shadow-card`, `--shadow-elevated`, `--shadow-floating`, `--shadow-modal`. Migrate everything.
+**Class: System debt.**
+
+---
+
+### 14. The `.intro-kicker` heading-overline pattern is design-y; nothing else on the page uses it
+**Problem.** `.intro-kicker` (sections.css:408) uses a horizontal rule + uppercase text pattern (line + word). It's stylish. But it appears nowhere else. Adjacent sections use pill eyebrows, naked uppercase, or nothing.
+
+**Root cause.** Section author drew inspiration that didn't propagate.
+**Perception.** The intro feels visually distinct in a way that disconnects it from the rest of the page — over-designed compared to its neighbors.
+**Fix.** Pick one: either commit and use the rule pattern in 2-3 more sections (rhythm); or drop it and use the standard `.eyebrow`.
+**Class: Design debt.**
+
+---
+
+### 15. FAQ details radius (16px) doesn't match testimonial card (28px), values bento (20px), or photo card (28px) — they all sit on the same page
+**Problem.** Cumulative effect of issue #6. The Info section has FAQ cards next to other cards with completely different silhouettes.
+**Perception.** Visual rhythm dissolves.
+**Fix.** All cards at `--radius-lg` (20px). Period.
+**Class: Polish debt.**
+
+---
+
+### 16. Footer container padding doesn't share the global gutter system
+**Problem.** sections.css:600 — `footer { padding: 60px 50px 32px; }`. The 50px is hardcoded. The mobile override at responsive.css:678 uses `var(--space-container-x-mobile)` — correct. But desktop is a literal.
+**Same problem** at `#instagram { padding: 60px 50px; }` (sections.css:732) and `.ph-inner { padding: 0 50px; }` (line 949).
+**Perception.** Footer/instagram/page-header gutters jitter against section gutters by a few pixels — the same issue I documented for ValuesSection, repeated across the codebase.
+**Fix.** Replace every `50px` desktop gutter with `var(--space-container-x)`.
+**Class: System debt.**
+
+---
+
+### 17. Section vertical rhythm collapses on mobile to a single value
+**Problem.** variables.css:172–178 — on mobile, all four section padding tiers (`--space-section-featured-y`, `--space-section-support-y`, `--space-section-form-y`, `--space-section-page-y`) collapse to `--space-section-mobile-y` (40px). On desktop, featured sections breathe more than supporting. On mobile, every section has identical vertical padding.
+
+**Root cause.** Pragmatic — mobile space is scarce. But the *intent* (featured sections feel more important) disappears.
+**Perception.** Mobile users get a "flat" page where every section has equal visual weight. Hero → Intro → Promise → Services all feel like peers.
+**Fix.** Two mobile tiers: `--space-section-featured-y-mobile: 56px`, `--space-section-mobile-y: 40px`. Featured sections get 16px more breathing room. Still respects mobile constraints.
+**Class: Design debt.**
+
+---
+
+### 18. `#hero::after` gradient mask has a 4-stop gradient with literal RGB
+**Problem.** sections.css:32–45 — a four-stop gradient `transparent 18% → 26% brand → 78% brand → solid brand 86%`. Plus a mobile-only override at lines 293–303 with *different* stops (`22% → 45% → 88% → 78%`). The math doesn't read as deliberate; it reads as tuned by trial and error.
+
+**Root cause.** Hero image overlay was fine-tuned to one image asset. Different images won't behave the same.
+**Perception.** Probably invisible. But if the hero photo ever changes (winter scene, different dog), the overlay won't land right.
+**Fix.** Reduce to 2 stops: `transparent 30% → solid brand 95%`. Test against multiple photos.
+**Class: Polish debt.**
+
+---
+
+### 19. Hero kicker mobile color is `rgba(white, 0.48)` on dark green
+**Problem.** responsive.css:358 — `color: rgba(var(--white-rgb), 0.48)` for the kicker on mobile. At 10px uppercase 0.13em on a dark green-tinted gradient, that's somewhere around 4–5:1 contrast against the gradient. Borderline AA.
+
+**Root cause.** Designer wanted the kicker quiet against the loud headline. But mobile users get a small device + outdoor sunlight + decreased contrast.
+**Perception.** Some users won't read the kicker. Those who do will strain.
+**Fix.** `rgba(white, 0.72)` at 11px on mobile. Still quiet, properly accessible.
+**Class: UX debt.**
+
+---
+
+### 20. Three styling paradigms cohabit
+**Problem.** The codebase uses:
+1. **Global utility classes** (`.btn`, `.eyebrow`, `.section-heading`, `.testimonial-card`, `.faq`) defined in `src/lib/styles/*.css`.
+2. **Component-scoped styles** inside Svelte `