v4.0.0.2
@@ -8,7 +8,15 @@
|
||||
"Bash(node \"C:\\\\Users\\\\mattc\\\\.claude\\\\plugins\\\\cache\\\\impeccable\\\\impeccable\\\\3.1.1\\\\skills\\\\impeccable\\\\scripts\\\\load-context.mjs\")",
|
||||
"Bash(cat \"C:\\\\Users\\\\mattc\\\\.claude\\\\plugins\\\\cache\\\\impeccable\\\\impeccable\\\\3.1.1\\\\skills\\\\impeccable\\\\reference\\\\distill.md\" 2>&1 | head -100)",
|
||||
"Read(//c/Users/mattc/.claude/plugins/cache/impeccable/impeccable/3.1.1/skills/impeccable/reference/**)",
|
||||
"Bash(node *)"
|
||||
"Bash(node *)",
|
||||
"WebFetch(domain:sticksandbones.co.nz)",
|
||||
"WebFetch(domain:fetchdogwalking.co.nz)",
|
||||
"Bash(npx --no-install svelte-check --tsconfig ./tsconfig.json --threshold error)",
|
||||
"Bash(python -c \"import ast; ast.parse\\(open\\('main.py',encoding='utf-8'\\).read\\(\\)\\); ast.parse\\(open\\('mail_api/models.py',encoding='utf-8'\\).read\\(\\)\\); print\\('ok'\\)\")",
|
||||
"Bash(python *)",
|
||||
"Bash(npm test *)",
|
||||
"Bash(npx svelte-kit *)",
|
||||
"Bash(npx svelte-check *)"
|
||||
]
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,237 +0,0 @@
|
||||
# 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).
|
||||
@@ -1,341 +0,0 @@
|
||||
# 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 <bookings@goodwalk.co.nz>
|
||||
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 <commit-or-tag> \
|
||||
--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.
|
||||
@@ -1,730 +0,0 @@
|
||||
# Goodwalk Design Language (2026 Refined Edition)
|
||||
|
||||
This document is the source of truth for Goodwalk’s visual design system. It defines the visual language, emotional tone, interaction philosophy, and implementation rules used across the website and future digital products.
|
||||
|
||||
The goal is not trend-chasing. The goal is to create a calm, trustworthy, modern experience that feels premium without becoming cold, corporate, or overly “tech”.
|
||||
|
||||
---
|
||||
|
||||
# Core Principles
|
||||
|
||||
## 1. Warmth over perfection
|
||||
Goodwalk is a human service business. The experience should feel:
|
||||
- calm
|
||||
- warm
|
||||
- trustworthy
|
||||
- emotionally safe
|
||||
- premium without arrogance
|
||||
|
||||
Avoid interfaces that feel:
|
||||
- clinical
|
||||
- overly corporate
|
||||
- aggressive
|
||||
- “SaaS-like”
|
||||
- excessively polished
|
||||
|
||||
Slight softness and imperfection are intentional.
|
||||
|
||||
---
|
||||
|
||||
## 2. Refinement over redesign
|
||||
The brand identity already works.
|
||||
|
||||
Modernisation should happen through:
|
||||
- spacing
|
||||
- typography
|
||||
- image treatment
|
||||
- motion
|
||||
- surface depth
|
||||
- consistency
|
||||
- restraint
|
||||
|
||||
Do not redesign for novelty.
|
||||
|
||||
---
|
||||
|
||||
## 3. Emotion before decoration
|
||||
Visual decisions must support:
|
||||
- trust
|
||||
- clarity
|
||||
- calmness
|
||||
- conversion
|
||||
|
||||
Do not add visual effects simply because they are modern.
|
||||
|
||||
Every effect must have emotional purpose.
|
||||
|
||||
---
|
||||
|
||||
## 4. Mobile-first experience
|
||||
The site must feel premium on:
|
||||
- 375px width
|
||||
- average brightness outdoors
|
||||
- thumb-driven interaction
|
||||
- imperfect network conditions
|
||||
|
||||
Mobile is not a secondary layout.
|
||||
|
||||
---
|
||||
|
||||
## 5. Quiet luxury
|
||||
The site should feel:
|
||||
- expensive
|
||||
- intentional
|
||||
- calm
|
||||
- understated
|
||||
|
||||
Not:
|
||||
- flashy
|
||||
- loud
|
||||
- hyper-animated
|
||||
- trend-driven
|
||||
|
||||
Modern premium design in 2026 is increasingly about restraint.
|
||||
|
||||
---
|
||||
|
||||
# Colour System
|
||||
|
||||
Defined in:
|
||||
`src/lib/styles/variables.css`
|
||||
|
||||
| Token | Hex | Purpose |
|
||||
|---|---|---|
|
||||
| `--gw-green` | `#213021` | Primary brand colour |
|
||||
| `--green-mid` | `#2d4230` | Hover states, elevated surfaces |
|
||||
| `--green-soft` | `#344b38` | Optional softer elevated green |
|
||||
| `--yellow` | `#ffd100` | Primary CTA accent |
|
||||
| `--yellow-soft` | `#f2bf2f` | Premium warm accent alternative |
|
||||
| `--off-white` | `#fbfbfb` | Primary background |
|
||||
| `--surface-light` | `#f7f8f6` | Elevated light surface |
|
||||
| `--text` | `#2e3031` | Default text |
|
||||
| `--gray` | `#59606d` | Secondary copy |
|
||||
| `--beige` | `#e5d6c2` | Warm neutral accent |
|
||||
|
||||
---
|
||||
|
||||
# Colour Philosophy
|
||||
|
||||
## Greens
|
||||
The green palette represents:
|
||||
- safety
|
||||
- reliability
|
||||
- groundedness
|
||||
- nature
|
||||
- professionalism
|
||||
|
||||
Greens should feel deep and organic rather than synthetic.
|
||||
|
||||
Avoid:
|
||||
- saturated emerald tones
|
||||
- neon greens
|
||||
- cold blue-greens
|
||||
|
||||
---
|
||||
|
||||
## Yellow usage
|
||||
Yellow is an accent, not a dominant UI colour.
|
||||
|
||||
Yellow should:
|
||||
- guide attention
|
||||
- signal warmth
|
||||
- create optimism
|
||||
|
||||
Yellow should NOT:
|
||||
- overpower sections
|
||||
- become large reading surfaces
|
||||
- create visual fatigue
|
||||
|
||||
For premium applications:
|
||||
```css
|
||||
background:
|
||||
linear-gradient(
|
||||
135deg,
|
||||
#ffd54a,
|
||||
#f2bf2f
|
||||
);
|
||||
```
|
||||
|
||||
This creates a warmer, more refined feel.
|
||||
|
||||
---
|
||||
|
||||
# Surface Design
|
||||
|
||||
## Philosophy
|
||||
Modern interfaces should feel layered rather than flat.
|
||||
|
||||
Depth should come from:
|
||||
- tonal separation
|
||||
- soft gradients
|
||||
- atmospheric shadows
|
||||
- subtle edge highlights
|
||||
|
||||
NOT heavy shadows or strong borders.
|
||||
|
||||
---
|
||||
|
||||
## Surface Rules
|
||||
|
||||
### Avoid pure white
|
||||
Never use:
|
||||
```css
|
||||
#ffffff
|
||||
```
|
||||
|
||||
Prefer:
|
||||
```css
|
||||
#fbfbfb
|
||||
#f7f8f6
|
||||
#f5f6f3
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Avoid pure black
|
||||
Never use:
|
||||
```css
|
||||
#000000
|
||||
```
|
||||
|
||||
Prefer:
|
||||
```css
|
||||
#0f1115
|
||||
#16181d
|
||||
#1b1d21
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Micro gradients
|
||||
Large flat surfaces should contain extremely subtle tonal variation.
|
||||
|
||||
Example:
|
||||
```css
|
||||
background:
|
||||
linear-gradient(
|
||||
180deg,
|
||||
rgba(255,255,255,0.95),
|
||||
rgba(247,248,246,0.95)
|
||||
);
|
||||
```
|
||||
|
||||
These gradients should be nearly invisible.
|
||||
|
||||
---
|
||||
|
||||
# Typography
|
||||
|
||||
Defined in:
|
||||
`src/lib/styles/typography.css`
|
||||
|
||||
| Token | Family | Purpose |
|
||||
|---|---|---|
|
||||
| `--font-head` | Unbounded | Headlines |
|
||||
| `--font-body` | Readex Pro | Body text |
|
||||
|
||||
---
|
||||
|
||||
# Typography Philosophy
|
||||
|
||||
Typography should create:
|
||||
- confidence
|
||||
- calmness
|
||||
- readability
|
||||
- rhythm
|
||||
|
||||
Avoid:
|
||||
- excessive weight changes
|
||||
- overly dense layouts
|
||||
- tiny text
|
||||
- over-stylised headings
|
||||
|
||||
---
|
||||
|
||||
# Typography Rules
|
||||
|
||||
## Body copy
|
||||
Desktop:
|
||||
```css
|
||||
font-size: 15px;
|
||||
line-height: 1.65;
|
||||
```
|
||||
|
||||
Mobile:
|
||||
```css
|
||||
font-size: 16px;
|
||||
line-height: 1.7;
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Headings
|
||||
Hero headings:
|
||||
```css
|
||||
letter-spacing: -0.04em;
|
||||
```
|
||||
|
||||
Section headings:
|
||||
```css
|
||||
letter-spacing: -0.02em;
|
||||
```
|
||||
|
||||
Large headings should feel:
|
||||
- compact
|
||||
- intentional
|
||||
- editorial
|
||||
|
||||
---
|
||||
|
||||
## Weight restraint
|
||||
Prefer:
|
||||
- 400
|
||||
- 500
|
||||
- 700
|
||||
|
||||
Avoid excessive font-weight variety.
|
||||
|
||||
Spacing should create hierarchy more than font weight.
|
||||
|
||||
---
|
||||
|
||||
# Spacing System
|
||||
|
||||
## Philosophy
|
||||
Modern premium interfaces feel expensive because they are under-filled.
|
||||
|
||||
Whitespace is a feature.
|
||||
|
||||
Do not compress layouts simply to fit more information.
|
||||
|
||||
---
|
||||
|
||||
## Spacing Scale
|
||||
|
||||
Only use:
|
||||
- 4
|
||||
- 8
|
||||
- 12
|
||||
- 16
|
||||
- 24
|
||||
- 32
|
||||
- 48
|
||||
- 64
|
||||
- 72
|
||||
- 96
|
||||
|
||||
Avoid arbitrary spacing values.
|
||||
|
||||
---
|
||||
|
||||
# Border Radius
|
||||
|
||||
| Context | Radius |
|
||||
|---|---|
|
||||
| Standard cards | `28px` |
|
||||
| Large feature containers | `32px` |
|
||||
| Pills | `999px` |
|
||||
| Small UI surfaces | `16px` |
|
||||
|
||||
The system should feel soft and approachable.
|
||||
|
||||
Avoid:
|
||||
- sharp corners
|
||||
- overly circular “bubble UI”
|
||||
|
||||
---
|
||||
|
||||
# Shadows & Depth
|
||||
|
||||
## Philosophy
|
||||
Modern depth is atmospheric, not dramatic.
|
||||
|
||||
Shadows should feel:
|
||||
- diffused
|
||||
- soft
|
||||
- realistic
|
||||
|
||||
Avoid:
|
||||
- heavy elevation
|
||||
- dark shadows
|
||||
- obvious floating cards
|
||||
|
||||
---
|
||||
|
||||
## Standard shadows
|
||||
|
||||
### Definition shadow
|
||||
```css
|
||||
inset 0 0 0 1px rgba(17,20,24,0.045);
|
||||
```
|
||||
|
||||
### Ambient depth
|
||||
```css
|
||||
0 8px 40px rgba(0,0,0,0.06);
|
||||
```
|
||||
|
||||
### Inner edge highlight
|
||||
```css
|
||||
inset 0 1px 0 rgba(255,255,255,0.04);
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
# Texture & Grain
|
||||
|
||||
## Philosophy
|
||||
Digital flatness feels artificial.
|
||||
|
||||
Subtle texture adds:
|
||||
- realism
|
||||
- warmth
|
||||
- richness
|
||||
|
||||
Very subtle grain is encouraged on:
|
||||
- hero sections
|
||||
- dark surfaces
|
||||
- footers
|
||||
- large empty areas
|
||||
|
||||
Texture must never become visibly noisy.
|
||||
|
||||
---
|
||||
|
||||
# Motion System
|
||||
|
||||
## Philosophy
|
||||
Motion should feel calm and physical.
|
||||
|
||||
Avoid:
|
||||
- excessive movement
|
||||
- bouncing
|
||||
- playful overshoot
|
||||
- aggressive transitions
|
||||
|
||||
---
|
||||
|
||||
## Preferred transition timing
|
||||
|
||||
```css
|
||||
220ms cubic-bezier(0.22, 1, 0.36, 1)
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Hover behaviour
|
||||
|
||||
Modern hover effects should rely more on:
|
||||
- brightness shifts
|
||||
- opacity
|
||||
- slight elevation
|
||||
- subtle glow
|
||||
|
||||
Less:
|
||||
```css
|
||||
translateY(-6px)
|
||||
```
|
||||
|
||||
More:
|
||||
```css
|
||||
translateY(-2px)
|
||||
filter: brightness(1.02)
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
# Cards
|
||||
|
||||
## Philosophy
|
||||
Cards should not feel disconnected from the layout.
|
||||
|
||||
Modern premium layouts use:
|
||||
- softer separation
|
||||
- lower contrast
|
||||
- quieter surfaces
|
||||
|
||||
Avoid:
|
||||
- harsh borders
|
||||
- obvious “dashboard card” styling
|
||||
- excessive shadowing
|
||||
|
||||
---
|
||||
|
||||
# Photography Direction
|
||||
|
||||
## This is one of the most important parts of the brand.
|
||||
|
||||
Photography should feel:
|
||||
- candid
|
||||
- warm
|
||||
- emotionally genuine
|
||||
- lightly cinematic
|
||||
- calm
|
||||
- naturally lit
|
||||
|
||||
Avoid:
|
||||
- harsh HDR
|
||||
- over-sharpening
|
||||
- fake bokeh
|
||||
- obvious AI look
|
||||
- cluttered backgrounds
|
||||
|
||||
---
|
||||
|
||||
## Image treatment
|
||||
|
||||
Preferred:
|
||||
- shallow depth of field
|
||||
- warm colour grading
|
||||
- natural greens
|
||||
- soft contrast
|
||||
- realistic shadows
|
||||
|
||||
Images should feel:
|
||||
“premium lifestyle”
|
||||
not:
|
||||
“local flyer advertisement”
|
||||
|
||||
---
|
||||
|
||||
# Layout Philosophy
|
||||
|
||||
## Editorial rhythm
|
||||
Not every section should feel identical.
|
||||
|
||||
The site should alternate between:
|
||||
- dense
|
||||
- airy
|
||||
- emotional
|
||||
- informational
|
||||
|
||||
This creates pacing and reduces fatigue.
|
||||
|
||||
---
|
||||
|
||||
## Controlled asymmetry
|
||||
Small asymmetry is encouraged:
|
||||
- offset images
|
||||
- uneven crops
|
||||
- staggered alignment
|
||||
- imperfect positioning
|
||||
|
||||
This creates humanity and visual interest.
|
||||
|
||||
Avoid perfect grid rigidity everywhere.
|
||||
|
||||
---
|
||||
|
||||
# Interaction Design
|
||||
|
||||
## Buttons
|
||||
Buttons should feel:
|
||||
- tactile
|
||||
- confident
|
||||
- soft
|
||||
|
||||
Never aggressive.
|
||||
|
||||
---
|
||||
|
||||
## Hover states
|
||||
Hover should:
|
||||
- reward
|
||||
- guide
|
||||
- reassure
|
||||
|
||||
Not distract.
|
||||
|
||||
---
|
||||
|
||||
# Navigation
|
||||
|
||||
The navigation should feel:
|
||||
- lightweight
|
||||
- stable
|
||||
- unobtrusive
|
||||
|
||||
Avoid:
|
||||
- oversized sticky headers
|
||||
- excessive blur
|
||||
- flashy dropdowns
|
||||
|
||||
The navigation exists to support trust and conversion.
|
||||
|
||||
---
|
||||
|
||||
# Mobile UX Principles
|
||||
|
||||
## Thumb-first interaction
|
||||
Primary actions should remain reachable.
|
||||
|
||||
Spacing should prevent accidental taps.
|
||||
|
||||
---
|
||||
|
||||
## Reduced visual noise
|
||||
Mobile layouts should:
|
||||
- simplify aggressively
|
||||
- reduce simultaneous options
|
||||
- preserve emotional tone
|
||||
|
||||
Not simply shrink desktop layouts.
|
||||
|
||||
---
|
||||
|
||||
# Accessibility
|
||||
|
||||
Always support:
|
||||
- reduced motion
|
||||
- readable contrast
|
||||
- large tap targets
|
||||
- visible focus states
|
||||
- scalable text
|
||||
|
||||
Accessibility should feel integrated, not bolted on.
|
||||
|
||||
---
|
||||
|
||||
# Design Language Keywords
|
||||
|
||||
The Goodwalk experience should feel:
|
||||
|
||||
- calm
|
||||
- premium
|
||||
- trustworthy
|
||||
- editorial
|
||||
- grounded
|
||||
- warm
|
||||
- modern
|
||||
- understated
|
||||
- human
|
||||
- emotionally safe
|
||||
- refined
|
||||
- spacious
|
||||
|
||||
Never:
|
||||
- corporate
|
||||
- loud
|
||||
- trendy
|
||||
- hyper-minimal
|
||||
- cold
|
||||
- sterile
|
||||
- flashy
|
||||
|
||||
---
|
||||
|
||||
# What To Prioritise Next
|
||||
|
||||
Highest impact improvements:
|
||||
|
||||
1. Better photography consistency
|
||||
2. Softer surface contrast
|
||||
3. More atmospheric depth
|
||||
4. Grain/texture implementation
|
||||
5. Reduced hover movement
|
||||
6. More editorial layouts
|
||||
7. More restrained motion
|
||||
8. More premium CTA treatment
|
||||
9. Improved mobile spacing rhythm
|
||||
10. More subtle card separation
|
||||
|
||||
---
|
||||
|
||||
# Final Rule
|
||||
|
||||
If a design decision looks impressive but reduces:
|
||||
- clarity
|
||||
- warmth
|
||||
- trust
|
||||
- calmness
|
||||
|
||||
Do not ship it.
|
||||
|
||||
Goodwalk should feel premium because it feels thoughtful — not because it feels flashy.
|
||||
|
||||
---
|
||||
|
||||
# Implementation Reference
|
||||
|
||||
Technical specs for what is currently live. Update this section when the codebase changes.
|
||||
|
||||
## Colour tokens (variables.css)
|
||||
|
||||
| Token | Hex | Status |
|
||||
|---|---|---|
|
||||
| `--gw-green` | `#213021` | ✅ Live |
|
||||
| `--green-mid` | `#2d4230` | ✅ Live |
|
||||
| `--green-soft` | `#344b38` | ✅ Live |
|
||||
| `--yellow` | `#ffd100` | ✅ Live |
|
||||
| `--yellow-soft` | `#f2bf2f` | ✅ Live |
|
||||
| `--gray` | `#59606d` | ✅ Live |
|
||||
| `--beige` | `#e5d6c2` | ✅ Live |
|
||||
| `--off-white` | `#fbfbfb` | ✅ Live |
|
||||
| `--surface-light` | `#f7f8f6` | ✅ Live — use where a surface sits above `--off-white` |
|
||||
| `--text` | `#2e3031` | ✅ Live |
|
||||
|
||||
## Typography (live values)
|
||||
|
||||
| Context | Size | Weight | Tracking | Line-height |
|
||||
|---|---|---|---|---|
|
||||
| Hero h1 | `clamp(34px, 4vw, 56px)` | 800 | `-0.04em` | `1.05` |
|
||||
| Section headings | `42px` | 700 | `-0.02em` | `1.08` |
|
||||
| Body (desktop) | `15px` | 400 | — | `1.65` |
|
||||
| Body (mobile) | `16px` | 400 | — | `1.70` |
|
||||
| Buttons | `14px` | 700 | `0.01em` | `1.2` |
|
||||
| Eyebrow | `13px` | 700 | `0.08em` | — |
|
||||
|
||||
## Section padding rhythm (live values)
|
||||
|
||||
| Section | Padding |
|
||||
|---|---|
|
||||
| `#promise`, `#services` | `96px 0` |
|
||||
| `#values`, `#testimonials`, `#info` | `72px 0` |
|
||||
| `#newlead` | `80px 0` |
|
||||
| `#instagram` | `60px` |
|
||||
|
||||
## Card hover behaviour (live values)
|
||||
|
||||
```css
|
||||
/* Service and testimonial cards */
|
||||
transform: translateY(-2px);
|
||||
filter: brightness(1.02);
|
||||
box-shadow: inset 0 0 0 1px rgba(17,20,24,0.06), 0 8px 40px rgba(0,0,0,0.08);
|
||||
```
|
||||
|
||||
Reduced from `translateY(-6px)` — calmer, more restrained per design philosophy.
|
||||
|
||||
## Service icon bubble gradient (live)
|
||||
|
||||
```css
|
||||
background: linear-gradient(135deg, #ffd54a, var(--yellow-soft));
|
||||
```
|
||||
|
||||
Angled, warmer gradient — replaces the flat vertical yellow.
|
||||
|
||||
## Testimonial card surface (live)
|
||||
|
||||
```css
|
||||
background: linear-gradient(180deg, #ffffff 0%, var(--off-white) 100%);
|
||||
```
|
||||
|
||||
Micro-gradient — avoids flat pure white per surface design rules.
|
||||
|
||||
## Scroll reveals (live)
|
||||
|
||||
JS (`+layout.svelte`): `IntersectionObserver` targets `.section-heading`, `.service-card`, `.testimonial-card`, `.value-card`. Adds `data-reveal` attribute on mount, toggles `.is-visible` when the element crosses `40px` from the viewport bottom.
|
||||
|
||||
CSS (`base.css`): `opacity 0 → 1`, `translateY(20px) → none`, `0.5s`. Wrapped in `prefers-reduced-motion: no-preference`.
|
||||
|
||||
## Hero image gradient (live)
|
||||
|
||||
`#hero::after` pseudo-element: `linear-gradient(to top, var(--gw-green), transparent)`, `height: 120px`, covering the right 58% of the section. Hidden on mobile.
|
||||
|
||||
## Not yet implemented
|
||||
|
||||
| Item | Notes |
|
||||
|---|---|
|
||||
| Grain / texture | Needs a noise SVG or canvas overlay — skip until photography is consistent |
|
||||
| `--surface-light` usage | Token is defined; not yet applied to any component |
|
||||
| `--green-soft` usage | Token is defined; candidate for mega-menu icon hover state |
|
||||
| More editorial layouts | Structural work — needs a design pass per page |
|
||||
@@ -1,3 +1,4 @@
|
||||
# syntax=docker/dockerfile:1.7
|
||||
ARG APP_VERSION=4.0.1
|
||||
|
||||
FROM node:22-alpine AS builder
|
||||
@@ -5,13 +6,30 @@ ARG APP_VERSION
|
||||
|
||||
WORKDIR /app
|
||||
|
||||
COPY package.json ./
|
||||
COPY package-lock.json ./
|
||||
# 1. Dependencies — only re-runs when package*.json change.
|
||||
COPY package.json package-lock.json ./
|
||||
RUN npm ci
|
||||
|
||||
# 2. Image assets in their own layer. Enhanced-img content-hashes its inputs,
|
||||
# so when these files are unchanged the BuildKit cache mount below replays
|
||||
# the previous WebP/AVIF outputs instead of regenerating them.
|
||||
COPY src/lib/images ./src/lib/images
|
||||
COPY static ./static
|
||||
|
||||
# 3. Build config (changes rarely).
|
||||
COPY svelte.config.js vite.config.ts tsconfig.json ./
|
||||
|
||||
# 4. Everything else. Component-only edits invalidate this layer but the vite
|
||||
# transform cache below keeps the expensive image pipeline warm.
|
||||
COPY . .
|
||||
|
||||
RUN node --experimental-strip-types --import="file:///app/scripts/sveltekit-resolver.mjs" scripts/export-homepage-content.mjs
|
||||
RUN npm run build
|
||||
|
||||
# Persist vite's transform cache across builds. Enhanced-img stores its
|
||||
# generated raster variants here keyed by source-image hash, so unchanged
|
||||
# images skip the sharp/resize/encode pass entirely.
|
||||
RUN --mount=type=cache,target=/app/node_modules/.vite,sharing=locked \
|
||||
npm run build
|
||||
|
||||
FROM node:22-alpine AS runner
|
||||
ARG APP_VERSION
|
||||
|
||||
@@ -1,185 +0,0 @@
|
||||
# Marketing Principles for Goodwalk
|
||||
|
||||
A working reference for the Goodwalk site rebuild and ongoing marketing decisions. Drawn from Chris Do (The Futur) and Debbie Millman (Design Matters), applied to the goal of acquiring 10 new clients.
|
||||
|
||||
# Checklist
|
||||
* Prioritise emotional trust before visual impressiveness.
|
||||
* Reduce cognitive load on every screen and interaction.
|
||||
* Every page should answer: “Am I in the right place?”
|
||||
* Use whitespace intentionally to create calmness and confidence.
|
||||
* Interfaces should feel predictable, stable, and effortless.
|
||||
Avoid clutter, excessive animations, and visual noise.
|
||||
Design for clarity first, aesthetics second.
|
||||
Premium experiences rely on restraint, not excess.
|
||||
Typography hierarchy must immediately guide the eye.
|
||||
Use fewer colours, but apply them consistently.
|
||||
Every component should have a clear purpose.
|
||||
Remove unnecessary borders, labels, and UI chrome.
|
||||
Make primary actions visually obvious within 2 seconds.
|
||||
Ensure pages feel fast even before fully loading.
|
||||
Consistent spacing creates perceived quality and trust.
|
||||
Use authentic photography over generic stock imagery.
|
||||
Human faces increase emotional connection and trust.
|
||||
Testimonials should feel personal and believable, not corporate.
|
||||
Buttons and CTAs should sound conversational and reassuring.
|
||||
Interfaces should feel welcoming, not technical.
|
||||
Avoid overwhelming users with too many choices.
|
||||
Users should never wonder what happens next.
|
||||
Design layouts around scanning behaviour, not reading behaviour.
|
||||
Mobile layouts should feel intentionally designed, not compressed desktop pages.
|
||||
Use subtle depth, shadows, and contrast to create hierarchy.
|
||||
Premium brands often use less content, but communicate more clearly.
|
||||
Calm interfaces increase perceived professionalism.
|
||||
Align visuals, copy, and interaction style into one consistent tone.
|
||||
The homepage should communicate trust before features.
|
||||
Every visual element should reinforce simplicity and confidence.
|
||||
Reduce form friction wherever possible.
|
||||
Users should be able to understand the business in under 5 seconds.
|
||||
Make service quality visually obvious through imagery and spacing.
|
||||
Avoid sharp transitions or jarring visual elements.
|
||||
Consistency across pages matters more than visual complexity.
|
||||
Good UX feels invisible to the user.
|
||||
Use natural language instead of corporate wording.
|
||||
Remove anything that feels “template-like”.
|
||||
Create visual breathing room around important content.
|
||||
Make interactions feel human, warm, and intentional.
|
||||
Ensure hover states and animations feel subtle and refined.
|
||||
Use imagery that reflects real customers and real experiences.
|
||||
Trust is built through consistency, polish, and predictability.
|
||||
Pages should feel curated, not crowded.
|
||||
Premium experiences rely heavily on pacing and rhythm.
|
||||
Focus attention using contrast, spacing, and hierarchy.
|
||||
Design should lower anxiety and decision fatigue.
|
||||
Avoid overexplaining when visuals already communicate meaning.
|
||||
The best interfaces feel calm, simple, and inevitable.
|
||||
Every redesign decision should improve trust, clarity, or emotional comfort.
|
||||
|
||||
---
|
||||
|
||||
## Chris Do's Principles
|
||||
|
||||
### 1. Sell the transformation, not the service
|
||||
|
||||
People don't buy "dog walking" — they buy peace of mind at work, a tired happy dog, not feeling guilty.
|
||||
|
||||
The headline shouldn't be "Professional Dog Walking in Wellington." It should speak to the outcome:
|
||||
|
||||
- "Come home to a happy, exercised dog"
|
||||
- "Your dog's best part of the day, while you're at work"
|
||||
|
||||
### 2. Niche down to stand out
|
||||
|
||||
"Dog walker" competes with everyone. "Dog walker for working professionals in [suburb] with anxious or reactive dogs" competes with almost no one — and can charge more.
|
||||
|
||||
Pick a wedge.
|
||||
|
||||
### 3. Price on value, not time
|
||||
|
||||
Don't lead with "$25 per walk." Lead with packages and outcomes:
|
||||
|
||||
> **The Working Professional Plan** — 3 walks/week, GPS updates, photo reports
|
||||
|
||||
Hide the hourly rate. Make it about what they get, not what you do.
|
||||
|
||||
### 4. Show, don't tell
|
||||
|
||||
Testimonials and proof crush adjectives. "Reliable and caring" is meaningless.
|
||||
|
||||
A photo of a muddy grinning dog with a one-line quote from the owner sells:
|
||||
|
||||
> "Bowie pulls me to the door when he sees Sarah's car."
|
||||
|
||||
### 5. Free is a magnet
|
||||
|
||||
Most dog walking sites just have a contact form — that's a closed door. Open one with:
|
||||
|
||||
- A free first walk
|
||||
- A free meet-and-greet
|
||||
- A downloadable "Is your dog getting enough exercise?" checklist
|
||||
|
||||
Get people into the funnel.
|
||||
|
||||
---
|
||||
|
||||
## Debbie Millman's Principles
|
||||
|
||||
### 1. Brand is a story people tell themselves about you
|
||||
|
||||
Branding is deliberate differentiation through storytelling.
|
||||
|
||||
What's the Goodwalk story? Why do you do this? Are you the ex-vet-nurse who only walks small dogs? The runner who takes high-energy breeds on actual trail runs?
|
||||
|
||||
That story belongs on the homepage, not buried on About.
|
||||
|
||||
### 2. Consistency builds trust
|
||||
|
||||
One voice, one visual identity, everywhere:
|
||||
|
||||
- Website
|
||||
- Instagram
|
||||
- Car magnet
|
||||
- The message sent when running 5 minutes late
|
||||
|
||||
Owners are handing you keys to their house and the life of their dog. Visual and verbal consistency signals "I am organised and reliable" before you've said a word.
|
||||
|
||||
### 3. Design is a tool for clarity, not decoration
|
||||
|
||||
Debbie often quotes Massimo Vignelli — design should make the message clearer.
|
||||
|
||||
In 3 seconds, can a stranger answer:
|
||||
|
||||
- What do you do?
|
||||
- Who is it for?
|
||||
- How do I book?
|
||||
|
||||
If they have to scroll or think, you're losing them.
|
||||
|
||||
---
|
||||
|
||||
## Applied: A Plan for 10 New Clients
|
||||
|
||||
A site rewrite with these principles in mind.
|
||||
|
||||
### 1. Homepage hero
|
||||
|
||||
- Outcome-focused headline
|
||||
- One strong photo of a happy dog mid-walk
|
||||
- One button: **"Book a free meet-and-greet"**
|
||||
|
||||
### 2. Pick a niche and say it out loud
|
||||
|
||||
Even just "for [your suburb] working professionals" narrows the field and helps you rank.
|
||||
|
||||
### 3. Three packages, not an hourly rate
|
||||
|
||||
Make the middle one the obvious choice (the "decoy effect" — Chris talks about this).
|
||||
|
||||
### 4. Three testimonials with photos and dog names
|
||||
|
||||
Real names, real dogs. Not "J.S. — Customer."
|
||||
|
||||
### 5. One story section
|
||||
|
||||
Who you are, why you do this, why someone should trust you with their dog and their house key.
|
||||
|
||||
### 6. Lead magnet
|
||||
|
||||
A free PDF like "How much exercise does your dog actually need?" in exchange for an email. Then you have a list to follow up with.
|
||||
|
||||
### 7. Kill booking friction
|
||||
|
||||
One-click to a calendar or a WhatsApp link. Not a 7-field form.
|
||||
|
||||
---
|
||||
|
||||
## Quick Reference Checklist
|
||||
|
||||
- [ ] Headline sells the outcome, not the service
|
||||
- [ ] Niche is named explicitly on the homepage
|
||||
- [ ] Pricing presented as packages, not hourly
|
||||
- [ ] At least 3 testimonials with real names, dog names, and photos
|
||||
- [ ] Founder story visible on homepage
|
||||
- [ ] Lead magnet (PDF or free meet-and-greet) above the fold
|
||||
- [ ] Booking is one click — calendar link or WhatsApp
|
||||
- [ ] Visual and verbal identity consistent across site, Instagram, and comms
|
||||
- [ ] In 3 seconds: what / who / how-to-book is obvious
|
||||
@@ -1,332 +0,0 @@
|
||||
# Mobile Polish — Conversion & Comfort Audit Tracker
|
||||
|
||||
## New Rescan Items — Mobile Conversion Opportunities
|
||||
|
||||
Fresh opportunities from a second mobile-first pass over the main site.
|
||||
These are intentionally only the new items, kept separate from the
|
||||
existing audit below.
|
||||
|
||||
### High — conversion strategy and flow
|
||||
|
||||
- [ ] **Hero CTA hierarchy still prioritises browsing over booking**
|
||||
- Files: `src/lib/content/homepage.ts:37-39`, `src/lib/components/HeroSection.svelte`
|
||||
- Current: the yellow primary CTA is `Explore our services →`, while
|
||||
`Book a Meet & Greet` is secondary.
|
||||
- Why: on mobile, high-intent users should be able to choose the next
|
||||
step immediately. Making the exploratory path more visually dominant
|
||||
adds friction before the user reaches the lead form.
|
||||
- Opportunity: test flipping the hierarchy on mobile so booking
|
||||
becomes the primary CTA and service exploration becomes secondary.
|
||||
|
||||
- [ ] **Homepage social proof appears too late in the scroll**
|
||||
- File: `src/routes/+page.svelte:143-147`
|
||||
- Current order: `Services -> Values -> Testimonials -> Booking`.
|
||||
- Why: testimonials are one of the strongest conversion levers, but
|
||||
on mobile they arrive after several large sections. Users are asked
|
||||
to keep scrolling before seeing the strongest emotional proof.
|
||||
- Opportunity: move testimonials above values on the homepage, or
|
||||
surface one featured review snippet earlier in the page.
|
||||
|
||||
- [ ] **Hero still relies on the next section for trust**
|
||||
- Files: `src/lib/content/homepage.ts:43-49`, `src/lib/components/HeroSection.svelte`,
|
||||
`src/lib/components/IntroStrip.svelte`
|
||||
- Current: the hero presents the headline and CTAs, but the review
|
||||
proof sits in the intro strip below.
|
||||
- Why: on mobile, the hero needs to answer both "what is this?" and
|
||||
"can I trust them?" before the user scrolls away. Separating those
|
||||
two jobs weakens the first decision moment.
|
||||
- Opportunity: add a compact review/trust chip directly under the
|
||||
hero subtitle or near the hero CTAs on mobile.
|
||||
|
||||
- [ ] **Booking flow asks for dog details before it captures the lead**
|
||||
- File: `src/lib/components/BookingSection.svelte:298-441`
|
||||
- Current: step 1 asks for dog name, location, message, and services;
|
||||
contact details are only requested in step 2.
|
||||
- Why: this is a higher-friction sequence on mobile. Users often feel
|
||||
more comfortable giving owner details first, then expanding into pet
|
||||
specifics once they have mentally committed.
|
||||
- Opportunity: test reversing the step order so step 1 captures name,
|
||||
email, and phone first, then dog details second.
|
||||
|
||||
### Medium — mobile persuasion and CTA timing
|
||||
|
||||
- [ ] **Sticky mobile CTA appears on a fixed pixel threshold rather than page context**
|
||||
- File: `src/lib/components/MobileBookBar.svelte:26-37`
|
||||
- Current: visibility is driven by `SHOW_AFTER_PX = 480` and
|
||||
`HIDE_BELOW_PX = 120`.
|
||||
- Why: a fixed threshold will feel early on some phones and late on
|
||||
others. It also ignores whether the hero or booking section is
|
||||
actually in view.
|
||||
- Opportunity: switch to an `IntersectionObserver` tied to the hero or
|
||||
booking section so the bar appears based on user context rather than
|
||||
raw scroll position.
|
||||
|
||||
- [ ] **Testimonials section pushes users off-site before finishing the proof story**
|
||||
- File: `src/lib/components/TestimonialsSection.svelte:128-134`
|
||||
- Current: the Instagram CTA appears near the top of the testimonials
|
||||
section, before the user has fully consumed the review content.
|
||||
- Why: on mobile, sending users to Instagram this early interrupts the
|
||||
conversion journey and competes with the booking path.
|
||||
- Opportunity: demote the Instagram CTA below the carousel, or replace
|
||||
it with a tighter trust-oriented proof CTA higher up.
|
||||
|
||||
- [ ] **Mobile pricing pages lose the consultative "not sure?" nudge**
|
||||
- File: `src/lib/components/PricingPage.svelte:12-19`
|
||||
- Current: the meet-and-greet reminder prompt is gated behind
|
||||
`min-width: 769px`, so desktop gets a tailored nudge and mobile
|
||||
does not.
|
||||
- Why: mobile users are more likely to feel overwhelmed by stacked
|
||||
pricing cards, not less. Removing the consultative reassurance on
|
||||
the smallest screens is directionally backwards for conversion.
|
||||
- Opportunity: add an inline mobile prompt after the first pricing
|
||||
section that says, in effect, "Not sure which option fits? Book a
|
||||
free Meet & Greet and we’ll help you choose."
|
||||
|
||||
### Medium — stacked-page CTA noise
|
||||
|
||||
- [ ] **Stacked pricing/service cards repeat the same CTA too many times**
|
||||
- Files: `src/lib/components/PricingPage.svelte:141-159`,
|
||||
`src/lib/components/ServiceLandingPage.svelte:94-112`
|
||||
- Current: when cards collapse to one column on mobile, each card
|
||||
keeps a full "Book a Meet & Greet" button.
|
||||
- Why: the repetition turns persuasive choice architecture into visual
|
||||
noise. Instead of helping the user decide, the page starts feeling
|
||||
like a stack of repeated asks.
|
||||
- Opportunity: treat this as a shared mobile pattern across pricing
|
||||
and service pages. Keep one strong CTA per section, let the popular
|
||||
card carry the primary action, and demote the rest.
|
||||
|
||||
Findings from a focused mobile-experience review (≤768px, with extra
|
||||
attention to 375px small-phones). Desktop is considered done. Each item
|
||||
records the where, why, and the concrete change.
|
||||
|
||||
> Important context for prioritisation: a dog-walking business is a
|
||||
> jobs-to-be-done service that users research on the couch. Mobile
|
||||
> conversion lower-bound is "they Meet & Greet". So the dial-movers
|
||||
> are: thumb-reach for the booking CTA, legibility, friction-free
|
||||
> form, and trust signals visible on the first scroll.
|
||||
|
||||
---
|
||||
|
||||
## High — direct conversion impact
|
||||
|
||||
- [x] **Hero buttons stack awkwardly at ~375px**
|
||||
- File: `src/lib/styles/responsive.css` (new ≤480px block)
|
||||
- Implementation: At `@media (max-width: 480px)` the hero buttons
|
||||
flip to `flex-direction: column; gap: 12px;` and each button
|
||||
becomes full-width with padding `16px 24px`. The 768px-specific
|
||||
`padding-right: 18px` on the buttons row and the
|
||||
`margin-right: 12px` on `:last-child` are both reset so the two
|
||||
CTAs read as a clean stacked stack.
|
||||
- Why: At 375px the side-by-side primary + outline CTAs were wrapping
|
||||
unevenly and the primary's prominence collapsed.
|
||||
|
||||
- [x] **Mobile header phone button is low-contrast and small**
|
||||
- File: `src/lib/styles/responsive.css:88-100`
|
||||
- Implementation: Background bumped from `rgba(33, 48, 33, 0.06)` →
|
||||
`0.10`; padding 9px 12px → 11px 14px (and at ≤480px, 10px 12px);
|
||||
`font-weight: 600`; `min-height: 44px` to meet the touch-target
|
||||
minimum; icon size also bumped from 13px → 14px.
|
||||
- Why: The tap-to-call on the mobile header is one of the highest
|
||||
intent actions on the site. It now reads as a button rather than a
|
||||
barely-visible label.
|
||||
|
||||
- [x] **Booking form inputs trigger iOS zoom-on-focus**
|
||||
- File: `src/lib/styles/responsive.css:453-462`
|
||||
- Implementation: Input `font-size: 15px` → `16px` at ≤768px. Comment
|
||||
added explaining the iOS-Safari 16px threshold so a future edit
|
||||
doesn't accidentally drop it again.
|
||||
- Why: Below 16px Safari auto-zooms on focus; the page jolts every
|
||||
time a field is tapped. Critical at the booking conversion step.
|
||||
|
||||
- [ ] **Testimonial carousel arrows hard to reach at 375px**
|
||||
- File: `src/lib/components/TestimonialsSection.svelte` (mobile rules
|
||||
lines ~640-660)
|
||||
- Current: arrows pinned to `bottom: 24px` inside a stage with
|
||||
`padding-bottom: 116px`. Visually at the bottom of a tall card —
|
||||
requires deliberate stretching.
|
||||
- Why: The carousel feels passive. Users don't realise they can advance
|
||||
it; testimonials sell — losing that engagement matters.
|
||||
- Fix: Lift arrows on small screens:
|
||||
```css
|
||||
@media (max-width: 480px) {
|
||||
.testimonial-arrow { bottom: 80px; }
|
||||
}
|
||||
```
|
||||
|
||||
- [x] **No persistent "Book Meet & Greet" CTA on mobile after hero scrolls past**
|
||||
- Files: `src/lib/components/MobileBookBar.svelte` (new),
|
||||
`src/routes/+layout.svelte` (mount), `src/lib/styles/responsive.css`
|
||||
(`body { padding-bottom: 64px }` at ≤768px).
|
||||
- Implementation v2 — *Airbnb-style soft-container, scroll-triggered*:
|
||||
- **Soft container**: a translucent white tray (`rgba(255, 255,
|
||||
255, 0.96)` with backdrop-filter blur) sits flush at the bottom
|
||||
with a thin top hairline and a soft shadow. The brand-yellow CTA
|
||||
pill lives *inside* the tray, not as the tray itself — so the
|
||||
action stays unmistakable but the surrounding chrome is calm.
|
||||
- **Scroll-triggered**: the bar slides up + fades in only after the
|
||||
user has scrolled ~480px (≈ one mobile viewport, hero out of
|
||||
view). Slides back out below ~120px to avoid flicker near the
|
||||
top. `prefers-reduced-motion` respected — the slide is dropped,
|
||||
only the opacity fade remains.
|
||||
- **Hidden on `/contact-us` and `/booking`** (already in the form).
|
||||
- **Resets on navigation**: `afterNavigate` resets `visible = false`
|
||||
so each new page starts hidden until the user has earned it again.
|
||||
- **Accessibility**: `aria-hidden` toggles with visibility; CTA
|
||||
`tabindex` flips to `-1` while hidden so keyboard users don't
|
||||
stumble onto an off-screen control; `pointer-events: none` while
|
||||
hidden so the user can't accidentally tap it during the fade.
|
||||
- **Safe-area aware**: `env(safe-area-inset-bottom)` so iPhones
|
||||
with the home-bar gesture area get correct spacing.
|
||||
- Body bottom-padding of 64px on mobile keeps the footer from sitting
|
||||
behind the bar when it's visible at the bottom of long pages.
|
||||
- Why: Sticky mobile CTAs are a validated conversion pattern (Airbnb,
|
||||
Booking.com, etc.), but the v1 full-yellow bar was tonally wrong
|
||||
for Goodwalk — it competed with the calm brand voice and dominated
|
||||
the screen permanently. v2 keeps the conversion benefit (one-tap
|
||||
booking, always one swipe away after engagement) without yelling.
|
||||
|
||||
## Medium — legibility & polish
|
||||
|
||||
- [x] **Hero title can wrap awkwardly at 375px**
|
||||
- File: `src/lib/styles/responsive.css` (≤480px block)
|
||||
- Implementation: At ≤480px the desktop H1 drops to 32px /
|
||||
line-height 1.12 and the dedicated `.hero-heading-mobile` element
|
||||
drops to 30px / 1.12 (it was 33.5px). The two-line "Unleashing Fun
|
||||
in / Your Dog's Day!" now sits comfortably with breathing room
|
||||
above the subtitle.
|
||||
- Why: Previous 38px / 33.5px sizings were a hair too big for 375px;
|
||||
line-2 felt cramped against the subtitle.
|
||||
|
||||
- [x] **Body text 15px feels small on mobile (and on ultra-wide desktop)**
|
||||
- File: `src/lib/styles/responsive.css` (≤768px body rule)
|
||||
- Implementation: `body { font-size: 16px; }` at ≤768px, with a
|
||||
comment noting the iOS-Safari 16px zoom threshold so this rule
|
||||
isn't accidentally undone.
|
||||
- Desktop (≥769px) still inherits 15px from `base.css` for now — the
|
||||
ultra-wide bump is tracked separately at the bottom of this file
|
||||
so it can be reasoned about independently (clamp vs. breakpoint).
|
||||
- Why: 16px is the modern legibility standard on mobile, dovetails
|
||||
with the iOS zoom-on-focus rule for inputs, and reduces read
|
||||
fatigue on long pages (about, FAQ answers, legal).
|
||||
|
||||
- [ ] **FAQ summary tap target too small**
|
||||
- File: `src/lib/styles/sections.css` (`.faq summary` rules)
|
||||
- Current: just text height (~22-26px) — below the 44px minimum.
|
||||
- Fix:
|
||||
```css
|
||||
.faq summary {
|
||||
padding: 12px 0;
|
||||
min-height: 44px;
|
||||
display: flex; align-items: center;
|
||||
}
|
||||
```
|
||||
|
||||
- [ ] **Booking service-option chips wrap awkwardly at 375px**
|
||||
- File: `src/lib/styles/responsive.css:468-470`
|
||||
- Fix at ≤480px: 2-up grid with tighter gap:
|
||||
```css
|
||||
.booking-service-options { gap: 10px 12px; }
|
||||
.booking-toggle-option { flex: 1 1 calc(50% - 6px); }
|
||||
```
|
||||
|
||||
- [ ] **Footer social icons spaced too tight for thumbs**
|
||||
- File: `src/lib/styles/sections.css` (`.social-links { gap: 14px }`)
|
||||
- Current 14px gap with 40px icons → centres are 54px apart. Apple
|
||||
HIG wants 8px+ between targets after the 44px minimum.
|
||||
- Fix:
|
||||
```css
|
||||
@media (max-width: 768px) { .social-links { gap: 18px; } }
|
||||
```
|
||||
|
||||
- [ ] **Pricing cards stack to 1-col with multiple "Book" buttons in a row**
|
||||
- File: `src/lib/components/PricingPage.svelte` (and ServiceLandingPage
|
||||
plan grids)
|
||||
- Why: After stacking, the user sees Plan → Book → Plan → Book → Plan
|
||||
→ Book in vertical sequence. The repetition reads as noise rather
|
||||
than choice; the "popular" anchor disappears.
|
||||
- Fix (opinionated): on mobile, only the popular plan keeps its CTA
|
||||
button. Other plans show a smaller "Choose this plan" link instead,
|
||||
or no per-card CTA at all (a single CTA appears under the grid):
|
||||
```css
|
||||
@media (max-width: 768px) {
|
||||
.pricing-plan-card:not(.pricing-plan-popular) .pricing-plan-cta {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
```
|
||||
Then keep the existing under-grid `.service-plan-reassurance` pill
|
||||
and add a single "Book a Meet & Greet" button below it.
|
||||
|
||||
- [ ] **Mobile nav header is too tall — eats above-the-fold real estate**
|
||||
- File: `src/lib/styles/responsive.css:74`
|
||||
- Current: `nav { padding: 20px 24px; }` + 25px logo = ~65px header.
|
||||
On iPhone 13 (844px), this leaves ~380px for hero before scroll —
|
||||
less than what the Goodwalk dog-image needs to feel like a hero.
|
||||
- Fix at ≤480px:
|
||||
```css
|
||||
nav { padding: 14px 20px; }
|
||||
.logo img { height: 22px; }
|
||||
```
|
||||
|
||||
## Low — incremental polish
|
||||
|
||||
- [ ] **Hero top padding generous at 375px**
|
||||
- File: `src/lib/styles/responsive.css:179`
|
||||
- Reduce hero `padding-top` from 50px to 32px at ≤480px.
|
||||
|
||||
- [ ] **Intro trust badge feels edge-to-edge at 375px**
|
||||
- File: `src/lib/styles/responsive.css:296-301`
|
||||
- Add `padding: 18px 16px; margin: 0 12px;` at ≤480px.
|
||||
|
||||
- [ ] **Testimonial quote mark too large at 375px**
|
||||
- File: `src/lib/components/TestimonialsSection.svelte:~595`
|
||||
- Current: 44px. Reduce to 36px at ≤480px.
|
||||
|
||||
- [ ] **Booking form labels could shrink slightly at 375px**
|
||||
- File: `src/lib/styles/responsive.css:450`
|
||||
- Optional: 16px → 15px at ≤480px to give field width back to the
|
||||
input value (where it actually matters for legibility).
|
||||
|
||||
- [ ] **No scroll-to-top affordance on long pages (booking, pricing)**
|
||||
- Currently absent. Low priority but helpful when users have scrolled
|
||||
past the booking form and want to re-read service details. Could be
|
||||
folded into the same sticky-book-bar work above (one bar, both jobs).
|
||||
|
||||
## Open from elsewhere
|
||||
|
||||
- [ ] **Ultra-wide desktop body font feels small** *(noted by user)*
|
||||
- Currently `body { font-size: 15px }` ([base.css:15](src/lib/styles/base.css#L15)).
|
||||
On ≥1800px screens with the `--max-w` already widening (per the
|
||||
1800px breakpoint), 15px in long-form sections (about, FAQ answers,
|
||||
legal) becomes uncomfortable.
|
||||
- Suggested fix: bump to `clamp(15px, 0.95vw, 17px)` on `body`, OR
|
||||
introduce a `@media (min-width: 1600px) { body { font-size: 17px; } }`.
|
||||
Either keeps the desktop ≤1599px experience identical and only
|
||||
expands type when there's genuinely more reading width.
|
||||
|
||||
## Deliberately not actioning
|
||||
|
||||
- **Drop reveal animations on mobile to "save bandwidth".** They're
|
||||
IntersectionObserver-driven, cost nothing perceivable, and add brand
|
||||
polish. Removing them would make the mobile site feel cheaper for no
|
||||
measurable performance gain.
|
||||
- **Replace the testimonial carousel with a stacked list on mobile.**
|
||||
Tempting (carousels famously hide content), but the carousel is
|
||||
central to the brand's "see real dogs" pitch. Better to fix the arrow
|
||||
reachability and let the autoplay do the work.
|
||||
|
||||
---
|
||||
|
||||
## Suggested order of attack
|
||||
|
||||
If you want one batch that moves the dial: **High items 1, 2, 3, 5**
|
||||
together is roughly an hour of work — they hit the hero, the header
|
||||
tap-to-call, the booking form's biggest mobile bug (zoom-on-focus), and
|
||||
add the sticky CTA. That's the package I'd ship first. Item 4 (carousel
|
||||
arrows) is a one-line fix once you're already in `responsive.css`.
|
||||
|
||||
The Medium list is best as a second pass — body-text bump,
|
||||
header-padding reduction, FAQ tap-target, and pricing-card-CTA dedupe
|
||||
all compound into a noticeably more "intentional on mobile" feel
|
||||
without any structural change.
|
||||
@@ -1,58 +0,0 @@
|
||||
# NZ Citations — Submission Sheet (C3)
|
||||
|
||||
Use the exact NAP block below for every directory. Consistency is the whole
|
||||
point — even small variations (brackets in phone, trailing punctuation,
|
||||
`Ltd.` vs `Limited`) split your local trust signals.
|
||||
|
||||
## Canonical NAP
|
||||
|
||||
| Field | Value |
|
||||
|---|---|
|
||||
| Business name | `Goodwalk` |
|
||||
| Phone (visible) | `022 642 1011` |
|
||||
| Phone (E.164 / forms that ask for international) | `+64 22 642 1011` |
|
||||
| Email | `info@goodwalk.co.nz` |
|
||||
| Website | `https://www.goodwalk.co.nz` |
|
||||
| Service area | Auckland Central (list 17 suburbs if a field allows: Morningside, Kingsland, Ponsonby, Grey Lynn, Mt Albert, Mt Eden, Sandringham, Mt Roskill, Arch Hill, Freemans Bay, Herne Bay, Pt Chevalier, Avondale, Three Kings, Hillsborough, Eden Terrace, Balmoral) |
|
||||
| Address | Service-area business — do **not** publish a home address. If a directory mandates an address, use a postcode-only entry where possible (e.g. Auckland 1021). |
|
||||
| Hours | Mon–Fri 8:00am–4:00pm |
|
||||
| Category (primary) | Dog Walker |
|
||||
| Category (secondary, where allowed) | Pet Sitter / Pet Care Service |
|
||||
| Short description (160 chars) | Goodwalk runs Tiny Gang pack walks, 1:1 walks, and puppy visits across Auckland Central. Small-dog specialists, free pickup and drop-off. |
|
||||
| Long description (use where 500+ chars allowed) | Goodwalk is an Auckland Central dog walking service run personally by Alessandra, a small-dog specialist. We offer Tiny Gang pack walks (4–8 dogs, from $49.50), one-on-one walks (from $45), and in-home puppy visits (from $39). Free pickup and drop-off across 17 inner-west suburbs including Ponsonby, Grey Lynn, Mt Eden, Kingsland and Morningside. Every walker holds public liability insurance and a current pet first aid certificate. New clients begin with a free, no-obligation Meet & Greet. 30+ five-star Google reviews. |
|
||||
| Logo | `/static/images/goodwalk-auckland-dog-walking-logo.png` (export at 600×600 for directories) |
|
||||
| Instagram | `https://www.instagram.com/goodwalk.nz/` |
|
||||
| Google Business Profile | `https://g.page/r/CUsvrWPhkYrAEB0` |
|
||||
|
||||
## Directories to claim (in priority order)
|
||||
|
||||
| # | Directory | URL | Cost | Notes |
|
||||
|---|---|---|---|---|
|
||||
| 1 | **Google Business Profile** | already claimed | Free | Verify category is "Dog Walker"; add 8+ photos; respond to all reviews |
|
||||
| 2 | **Yellow.co.nz** | https://yellow.co.nz/add-a-business | Free tier | NZ's largest directory — non-negotiable |
|
||||
| 3 | **Finda.co.nz** | https://www.finda.co.nz/add-business | Free | Crawled by every local SEO tool |
|
||||
| 4 | **Localist.co.nz** | https://www.localist.co.nz/business/add | Free | Auckland-focused, high local relevance |
|
||||
| 5 | **Neighbourly.co.nz** | https://www.neighbourly.co.nz | Free | Suburb-level visibility — critical for pet services where locals ask "anyone know a good dog walker in Ponsonby?" |
|
||||
| 6 | **NZS.com** | https://www.nzs.com | Free | General NZ directory, broad backlink value |
|
||||
| 7 | **Facebook Business Page** | https://business.facebook.com | Free | Footer already links here — make sure the page actually exists with NAP matching |
|
||||
| 8 | **NoCowboys** | https://www.nocowboys.co.nz | Free + paid | Reputation-focused; aim to gather a few reviews here too |
|
||||
| 9 | **DogFriendly NZ** | https://dogfriendly.co.nz | Free | Industry-specific, low competition |
|
||||
| 10 | **Pet Directory NZ** | https://www.petdirectory.co.nz | Free | Pet-specific authority signal |
|
||||
|
||||
## After each citation goes live
|
||||
|
||||
1. Save the public profile URL in a spreadsheet (you'll need the list to
|
||||
update `sameAs` in the LocalBusiness JSON-LD on the homepage —
|
||||
`src/routes/+page.svelte`).
|
||||
2. Re-run `/seo local https://goodwalk.co.nz` after ~2 weeks to confirm
|
||||
crawler discovery.
|
||||
|
||||
## Anti-patterns to avoid
|
||||
|
||||
- Do **not** vary the business name ("Goodwalk Auckland", "Goodwalk Ltd",
|
||||
"Goodwalk Dog Walking") across listings. Pick `Goodwalk` and stick with it.
|
||||
- Do **not** publish a home street address anywhere. SAB = service-area only.
|
||||
- Do **not** use a tracking phone number — Google penalises NAP mismatches.
|
||||
- Do **not** auto-syndicate via paid "submit to 50 directories" services —
|
||||
they typically use slightly different NAP per source and create the exact
|
||||
inconsistency you're trying to avoid.
|
||||
@@ -1,278 +0,0 @@
|
||||
# OVERLOAD.md — Section flow audit & cuts
|
||||
|
||||
Evaluating each section of the Goodwalk homepage against one question: **does this section move a researching dog owner closer to booking, or does it just take up scroll?**
|
||||
|
||||
Scope: `src/routes/+page.svelte` (the homepage). Findings drawn from reading the actual section components and copy.
|
||||
|
||||
---
|
||||
|
||||
## The current sequence
|
||||
|
||||
```
|
||||
1. Header
|
||||
2. HeroSection ── hook + primary CTA
|
||||
3. ValuesSection ── photo gallery + before/after + values points (3 sub-blocks)
|
||||
4. ServicesSection ── service grid
|
||||
5. HowItWorksSection ── 3-4 step process
|
||||
6. TestimonialsSection ── reviews
|
||||
7. FounderStorySection ── 4 paragraphs + 2 CTAs + portrait
|
||||
8. InfoSection ── locations + hours + FAQ (2 sub-blocks)
|
||||
9. BookingSection ── lead form (THE conversion target)
|
||||
10. InstagramSection ── follow CTA
|
||||
11. Footer
|
||||
```
|
||||
|
||||
**9 content sections**, but really **13+ sub-blocks** once you unpack Values (3), FounderStory (dense), and Info (2). The booking form sits 8th in sequence — a user who's "sold" by Testimonials still has to scroll through two more sections before they can ask to book.
|
||||
|
||||
---
|
||||
|
||||
## Section-by-section verdict
|
||||
|
||||
### 1. HeroSection — **KEEP as-is**
|
||||
|
||||
The hero is doing its job: photographic hook, headline, primary "Book a walk" CTA, Google trust chip, three subtitle proof chips. This is the only section a 5-second-attention user actually sees, and the primary CTA + Google rating are in it.
|
||||
|
||||
**One nit:** the seoHeading (`<h2>` below `<h1>`) is yellow Unbounded at 18px on green. Reads to humans as a subtitle but ships as an H2 for SEO. Fine, but a careful eye spots it as "two headings of equal importance" — that's why mobile bumps it to 15px to quiet it. Acceptable trade-off.
|
||||
|
||||
**Verdict:** earns its place outright.
|
||||
|
||||
---
|
||||
|
||||
### 2. ValuesSection — **TRIM AGGRESSIVELY**
|
||||
|
||||
This is the worst offender for overload. It's three sections stacked under one `<section id="values">`:
|
||||
|
||||
| Sub-block | Content | Job |
|
||||
|---|---|---|
|
||||
| Photo gallery | 5 client dog photos with names | Visual proof, decorative |
|
||||
| Before/after contrast | Two cells: "Without the right routine" vs "With Goodwalk" + bullets each | Emotional pitch |
|
||||
| Values points | 6 icon cards | Rational claims |
|
||||
|
||||
That's three different rhetorical moves — photographic warmth → emotional contrast → rational checklist — packed into one section. The user is asked to switch modes twice within ~1500px of scrolling.
|
||||
|
||||
**Friction created:**
|
||||
- Photo gallery is the third visual moment in 1500px of page (after hero photo and any micro-imagery in the hero chips). It doesn't *add* social proof — Testimonials at section 6 will do that with quotes + faces. It's redundant.
|
||||
- Before/after is genuinely the strongest copy on the page. *That's the section that earns its place.* The yellow "With Goodwalk" cell carries the brand-emotional payload of the homepage in two short paragraphs.
|
||||
- Values points are claims the FounderStory section will later repeat ("Little groups, never a crowded van" / "The same friendly face at the door"). Duplicate.
|
||||
|
||||
**Proposal:**
|
||||
- **Cut the photo gallery from this section entirely.** Move it to a thin band *under* the hero (or kill it; Testimonials photos cover the same emotional ground).
|
||||
- **Keep the before/after contrast as the entire section.** Rename `id="values"` → `id="why-goodwalk"`. Make it shorter (drop "What we stand for" subheader and the 6-point grid).
|
||||
- **Either kill the values points OR collapse them into a 3-icon strip embedded inside the Hero or Services section** as inline reassurance.
|
||||
|
||||
**Cognitive load saved:** ~50% reduction in section height. The strongest copy gets isolated. The user reaches Services faster.
|
||||
|
||||
---
|
||||
|
||||
### 3. ServicesSection — **KEEP, MAYBE PROMOTE**
|
||||
|
||||
Services answers the question the user came to the site asking: "what do you actually offer and what does it cost?" In local-services SEO research, this is the highest-intent section on the page.
|
||||
|
||||
**Currently 4th in the scroll order.** Most local-service homepages put services 2nd or 3rd because intent-driven visitors want to confirm fit *before* reading testimonials. Consider promoting to position 2 (immediately after Hero).
|
||||
|
||||
**Verdict:** earns its place, **but order matters** — see proposed reordering below.
|
||||
|
||||
---
|
||||
|
||||
### 4. HowItWorksSection — **KEEP, possibly MERGE with Services**
|
||||
|
||||
Process clarity is a real conversion accelerator for a service that requires giving someone a key to your house. Three steps ("Book in → Meet & greet → Regular walks" or similar) reduces the perceived commitment.
|
||||
|
||||
**Watch for:** if HowItWorks duplicates the messaging in Services or FounderStory ("you'll meet Aless first"), it stops being a friction-reducer and becomes a third "trust us" pitch.
|
||||
|
||||
**Proposal:** keep as a separate section unless the steps are <3. If <3 steps, fold into a strip inside the Booking section as "what happens after you submit this form" reassurance.
|
||||
|
||||
**Verdict:** earns its place if it shows *process* (steps with icons), not values.
|
||||
|
||||
---
|
||||
|
||||
### 5. TestimonialsSection — **KEEP**
|
||||
|
||||
Social proof is the highest-converting section type in local services. Five-star reviews with names and dog photos do real work.
|
||||
|
||||
**Watch for:** card padding (36px 32px), 28px radius (just fixed to 20px). 3-up grid → 1-up on mobile is correct.
|
||||
|
||||
**One question:** how many testimonials show? If it's 6+, that's overload too. 3 is the sweet spot. 5 max.
|
||||
|
||||
**Verdict:** earns its place. May need a content trim (count of cards), not a structure change.
|
||||
|
||||
---
|
||||
|
||||
### 6. FounderStorySection — **TRIM HARD**
|
||||
|
||||
Currently:
|
||||
- Eyebrow "A note from Aless"
|
||||
- Greeting "Hi, I'm Aless."
|
||||
- Heading with main + sub
|
||||
- "What owners notice first" trust strip with 3 bullets
|
||||
- **Four paragraphs of body copy** (≈ 250 words)
|
||||
- Closing line in bold
|
||||
- Two CTAs (email-Aless + book CTA)
|
||||
- Signoff with name, tagline, portrait
|
||||
|
||||
This is the densest section on the page. It re-pitches values that Values + Services + HowItWorks have already pitched.
|
||||
|
||||
**The problem:** a user who reached this section is *already convinced or already gone*. The 250-word essay is for the still-convinced reader, but they're scrolling for the form.
|
||||
|
||||
**Proposal:**
|
||||
- **Cut to 2 paragraphs.** Para 1: "Why I started Goodwalk." Para 2: "The Tiny Gang philosophy + sign-off."
|
||||
- **Remove the "What owners notice first" trust strip.** It duplicates values content already on the page.
|
||||
- **Remove the email-Aless secondary CTA.** Founder pages with two CTAs split attention. Keep only the primary booking CTA.
|
||||
- **Keep the portrait + signoff** — that's the emotional payoff for the section's existence.
|
||||
|
||||
**Word-count target:** ≤ 120 words of body copy + signature. Reader spends 30 seconds here, not 90.
|
||||
|
||||
---
|
||||
|
||||
### 7. InfoSection — **SPLIT + DEMOTE**
|
||||
|
||||
This section is bundling two unrelated jobs:
|
||||
|
||||
| Block | Content | Real job |
|
||||
|---|---|---|
|
||||
| Block 1 | Suburb chips + nearby card + hours | "Do you cover my area?" — qualifying signal |
|
||||
| Block 2 | FAQ accordion | "What if X?" — friction reducer |
|
||||
|
||||
These should not be in the same section. They serve different reader states.
|
||||
|
||||
**Block 1 (suburbs/hours)** is a *qualifier* — it tells the user whether to even bother with the booking form. It belongs **inside or directly above the BookingSection**, where it removes the "do you walk in [my suburb]" objection right at the point of conversion.
|
||||
|
||||
**Block 2 (FAQ)** is the only place to address objections like "what about wet weather," "what's your cancellation policy." It belongs **after** the booking form OR collapsed-by-default near the bottom of the page. FAQ before the form is friction; FAQ after the form catches the not-yet-convinced.
|
||||
|
||||
**Proposal:**
|
||||
- Split InfoSection into two components.
|
||||
- Move suburb chips + hours into a slim band ABOVE the BookingSection.
|
||||
- Keep FAQ as a section, but move it BELOW the BookingSection (right before Instagram or Footer).
|
||||
|
||||
---
|
||||
|
||||
### 8. BookingSection — **KEEP and PROMOTE in flow**
|
||||
|
||||
This is *the* conversion target. Currently it sits 8th on a 9-section page. On mobile this is at least 6+ screens of scrolling. The sticky `MobileBookBar` helps, but desktop has no equivalent.
|
||||
|
||||
**Watch:** the BookingSection currently uses the `card-stepper` variant (saw in `+page.svelte`), which is a multi-step card form. Step forms convert well when the steps are short, badly when they feel like a survey. Verify the first step is *radically* lightweight (1 field max).
|
||||
|
||||
**Proposal:**
|
||||
- **Promote BookingSection to position 5 or 6** (after social proof, before founder-story-as-trust-confirmation).
|
||||
- Add suburb chip band immediately above.
|
||||
- Add the 3-step "how it works" strip immediately below.
|
||||
- Remove redundant section padding that double-spaces it from neighbors.
|
||||
|
||||
---
|
||||
|
||||
### 9. InstagramSection — **DEMOTE or KILL**
|
||||
|
||||
Currently the last content section before footer. Yellow billboard (until just reverted) with a "Follow us on Instagram" CTA.
|
||||
|
||||
**Hard question:** how many homepage visitors will Goodwalk *acquire* by getting them to follow the Instagram instead of booking? Almost none. The Instagram link is a *brand-discovery* mechanism, not a conversion path. It belongs in the footer as a social icon, not as a section.
|
||||
|
||||
**Proposal options:**
|
||||
| Option | Effort | Trade-off |
|
||||
|---|---|---|
|
||||
| **A. Kill the section.** Move Instagram link to footer social icons only. | 1 hour | Loses a real estate moment for users who want to lurk-research before booking. |
|
||||
| **B. Demote to a thin strip** above the footer — Instagram icon + grid of 4-6 recent photos as a passive link. No yellow background. | Half day | Less in-your-face; lets visual lurkers find it. |
|
||||
| **C. Keep as-is but move ABOVE the BookingSection.** | 5 min | Probably worst option: yellow billboard interrupts the path to conversion. |
|
||||
|
||||
**Recommend B.** A subtle Instagram strip — six recent posts, no copy, link out — gives the brand-discovery user what they want without competing with the booking CTA.
|
||||
|
||||
---
|
||||
|
||||
### Header & Footer — out of scope here
|
||||
|
||||
Header (sticky nav + mobile menu + book CTA) is fine. Footer carries the legal/contact/social/locations real estate well after the typography pass.
|
||||
|
||||
---
|
||||
|
||||
## Proposed new sequence
|
||||
|
||||
```
|
||||
1. Header
|
||||
2. HeroSection (hook + Google trust)
|
||||
3. ServicesSection (PROMOTED ↑ — answers intent)
|
||||
4. TestimonialsSection (PROMOTED ↑ — social proof)
|
||||
5. WhyGoodwalkSection (= old Values, before/after only) (TRIMMED — emotional payoff)
|
||||
6. HowItWorksSection (process clarity)
|
||||
7. FounderStorySection (TRIMMED to 120 words)
|
||||
8. LocationsBand (= old Info block 1) (NEW slim qualifier band)
|
||||
9. BookingSection (PROMOTED ↑ — the conversion)
|
||||
10. FaqSection (= old Info block 2) (catches not-yet-convinced)
|
||||
11. InstagramStrip (= demoted Instagram) (RECOMPOSED — passive)
|
||||
12. Footer
|
||||
```
|
||||
|
||||
**Net: 10 sections instead of 9, but ~30% less total scroll height** because Values lost 2 sub-blocks and FounderStory lost half its body copy.
|
||||
|
||||
**Story arc:**
|
||||
1. Hook
|
||||
2. Offering ("what can I get")
|
||||
3. Proof ("who else gets it")
|
||||
4. Promise ("what changes for my dog")
|
||||
5. Process ("how does it work")
|
||||
6. Trust ("who runs this")
|
||||
7. Qualification ("do you cover me")
|
||||
8. **Action** ("book")
|
||||
9. Objection handling ("what if X")
|
||||
10. Brand discovery ("see more")
|
||||
|
||||
The booking form sits 8th in the new flow vs 8th in the current flow — same numeric position, but with ~40% less scroll above it.
|
||||
|
||||
---
|
||||
|
||||
## Conversion-flow analysis
|
||||
|
||||
### Friction points in the current page
|
||||
|
||||
| # | Issue | Impact |
|
||||
|---|---|---|
|
||||
| 1 | Booking form below 7 sections of content | Desktop users without sticky CTA must scroll a long way to convert. |
|
||||
| 2 | FounderStory has 2 CTAs (email + book) | Splits attention at a high-conviction moment. |
|
||||
| 3 | InfoSection FAQ before BookingSection | FAQ before the form makes the form feel optional. FAQ after the form catches the abandoners. |
|
||||
| 4 | Values + FounderStory both run the "Tiny Gang philosophy" pitch | Diminishing returns on emotional copy. |
|
||||
| 5 | InstagramSection competes with BookingSection's CTA | Section after form siphons attention from the form. |
|
||||
| 6 | Values gallery + Testimonials = duplicate photographic social proof | One earns its keep; the other dilutes. |
|
||||
|
||||
### Conversion-flow wins from the proposal
|
||||
|
||||
| Win | Mechanism |
|
||||
|---|---|
|
||||
| Faster to "where do I book" | Services 3rd, Booking 9th (in a shorter page). |
|
||||
| Stronger social proof early | Testimonials at 4 instead of 6. Confidence cascades into the rest of the read. |
|
||||
| Pre-qualification | Suburb chips immediately above the form reduce "do you walk in my area" abandon. |
|
||||
| Single CTA per section | FounderStory loses its email link. Every emotional moment funnels to the booking form. |
|
||||
| Objection-handling at the right place | FAQ moves below the form so it catches the hesitant, not the eager. |
|
||||
|
||||
---
|
||||
|
||||
## What I'd actually ship first (highest ROI)
|
||||
|
||||
If you only do three things from this audit:
|
||||
|
||||
1. **Cut the Values photo gallery.** 30 minutes. Removes the most redundant block and saves a screen of scroll. (Just delete the `clientPhotos` constant and the `.values-photo-grid` figure block in `ValuesSection.svelte` — confirmed the rest of the section stands on its own.)
|
||||
|
||||
2. **Move suburb chips above the BookingSection.** 1 hour. Pre-qualifies the lead inline at the highest-conviction moment. Best single-shot conversion-rate move on the page.
|
||||
|
||||
3. **Trim FounderStory body copy from 4 paragraphs to 2 and remove the email-Aless CTA.** 30 minutes. Restores the section's tempo. Single CTA = stronger conversion path.
|
||||
|
||||
Everything else (Services promotion, Instagram demotion, Info split, FAQ relocation) is good follow-up work but those three deliver the bulk of the win.
|
||||
|
||||
---
|
||||
|
||||
## What NOT to cut
|
||||
|
||||
- **Hero.** Untouchable.
|
||||
- **Testimonials.** Local services live and die on five-star reviews. Trim count if needed, never remove.
|
||||
- **BookingSection.** It's the conversion target.
|
||||
- **Google trust chips in the Hero.** Smallest UI on the page; biggest conversion signal.
|
||||
|
||||
---
|
||||
|
||||
## Notes on implementation order
|
||||
|
||||
If you decide to proceed, recommended PR sequence to keep risk low:
|
||||
|
||||
1. **PR 1 — Values cut.** Photo gallery + values points removed. Section keeps the before/after contrast only. Update homepage to match new section size. *Visual: -1 screen.*
|
||||
2. **PR 2 — Founder trim.** Reduce body copy to 2 paragraphs. Remove email CTA. *Visual: -300px.*
|
||||
3. **PR 3 — Info split.** New `<LocationsBand />` + new `<FaqSection />`. Update `+page.svelte` to render them in new positions.
|
||||
4. **PR 4 — Section reorder + Instagram recompose.** Last because it touches `+page.svelte` order, has the most visual impact, and benefits from the earlier trims already being in.
|
||||
|
||||
Each PR is independently shippable and visually reviewable. None block the others. None require a redesign of any individual section — just structural surgery on what's bundled where.
|
||||
@@ -1,33 +0,0 @@
|
||||
# Product
|
||||
|
||||
## Register
|
||||
|
||||
brand
|
||||
|
||||
## Users
|
||||
|
||||
Busy Auckland dog owners, especially working professionals in the inner-west and nearby suburbs, who need dependable weekday care for dogs they treat like family. They are usually short on time, want a calmer home routine, and need to trust both the walker and the experience before booking.
|
||||
|
||||
## Product Purpose
|
||||
|
||||
Goodwalk exists to turn weekday dog care into a source of confidence rather than guilt or logistical stress. The site should quickly show what Goodwalk does, who it is for, why the service feels safer and more personal than generic dog walking, and how to book a free meet-and-greet with minimal friction.
|
||||
|
||||
## Brand Personality
|
||||
|
||||
Warm, grounded, premium. The voice should feel calm, reassuring, and human, with clear expertise but no corporate distance. The emotional goal is peace of mind: owners should feel that their dog will be known, safe, and genuinely looked after.
|
||||
|
||||
## Anti-references
|
||||
|
||||
Avoid generic SaaS landing-page patterns, loud pet-industry gimmicks, and clinical service-business layouts. This should not look hyper-animated, overly polished, corporate, cold, bargain-oriented, or like a template full of interchangeable cards and stock-style marketing language.
|
||||
|
||||
## Design Principles
|
||||
|
||||
1. Lead with trust before features.
|
||||
2. Show the emotional outcome, not just the operational service.
|
||||
3. Use restraint to signal quality: fewer, better elements with generous space.
|
||||
4. Make booking feel easy, personal, and low-pressure.
|
||||
5. Keep every page grounded in real dogs, real routines, and real care.
|
||||
|
||||
## Accessibility & Inclusion
|
||||
|
||||
Target clear, readable contrast, large tap targets, visible focus states, and strong mobile usability in outdoor and on-the-go conditions. Support reduced motion, preserve legibility at larger text sizes, and keep copy plain enough to reduce decision fatigue for anxious or time-poor users.
|
||||
@@ -1,143 +0,0 @@
|
||||
# UX Polish — Conversion Audit Tracker
|
||||
|
||||
Findings from the senior-marketing-lens audit, with completion status. Each
|
||||
item has a one-line rationale and the file/line where the change lives (or
|
||||
will live).
|
||||
|
||||
> Only commit to "We'll reply within 24 hours" if Aless can actually hold
|
||||
> to it. If response time is more like 1-2 business days, soften to
|
||||
> "within one business day".
|
||||
|
||||
---
|
||||
|
||||
## High — direct conversion impact
|
||||
|
||||
- [x] **Hero primary CTA: "Learn more" → "Explore our services →"**
|
||||
- File: `src/lib/content/homepage.ts:38`
|
||||
- Why: "Learn more" is the lowest-intent CTA that exists.
|
||||
|
||||
- [x] **Promise CTA: "See our services" → "Book a free Meet & Greet"**
|
||||
- File: `src/lib/content/homepage.ts:59`
|
||||
- Also: target changed from `#services` to `/contact-us` so the CTA goes
|
||||
to the booking page instead of bouncing back up to a service list.
|
||||
- Why: After the value prop + happy-dog photo, sending visitors to the
|
||||
services list is a step backwards. Push them to book.
|
||||
|
||||
- [x] **Booking subtitle now states response time**
|
||||
- File: `src/lib/content/homepage.ts:159-162`
|
||||
- Old: *"...so we can reach out to arrange your free, no-obligation Meet & Greet."*
|
||||
- New: *"...We'll reply within 24 hours to arrange your free, no-obligation Meet & Greet."*
|
||||
- General-enquiry variant updated to match.
|
||||
- Why: Open-ended "we'll reach out" creates anxiety at submit time.
|
||||
|
||||
- [x] 1 **Pricing page — Google rating trust signal above plan grid**
|
||||
- File: `src/lib/components/PricingPage.svelte`
|
||||
- Implementation: Pill-styled trust badge inside the green hero,
|
||||
directly under the subtitle — five yellow stars + "30+ five-star
|
||||
Google reviews" label + arrow, links out to Google. Styled to read
|
||||
against the green hero (semi-transparent white pill) rather than
|
||||
reusing the cream IntroStrip, which would have clashed.
|
||||
- Why: Visitors land on pricing mid-decision; trust signal now appears
|
||||
before the plan grid.
|
||||
|
||||
- [x] 2 **Service plan CTAs — add free / no-obligation reassurance**
|
||||
- File: `src/lib/components/ServiceLandingPage.svelte`
|
||||
- Implementation: A subtle green pill *"Every booking starts with a
|
||||
free, no-obligation Meet & Greet."* (yellow shield-heart icon) sits
|
||||
centred directly under the plan grid on every service page, above the
|
||||
Extras block. Reuses the brand-tinted-pill aesthetic so it feels
|
||||
native, not tacked on.
|
||||
- Why: The "Book a Meet & Greet" buttons under each plan didn't carry
|
||||
risk-reversal phrasing in their immediate context. Now they do.
|
||||
|
||||
## Medium — trust + polish
|
||||
|
||||
- [x] 3 **Quantify the Google rating wherever it appears**
|
||||
- Files: `src/lib/content/homepage.ts:46`,
|
||||
`src/lib/components/Footer.svelte:89`,
|
||||
`src/lib/components/TestimonialsSection.svelte:200`,
|
||||
`src/lib/components/PricingPage.svelte` (new pricing-trust pill).
|
||||
- Implementation: "All 5 star reviews on Google!" → "30+ five-star
|
||||
Google reviews" everywhere. Aless confirmed 30+ as the count.
|
||||
- Why: A specific number is dramatically more credible than "all".
|
||||
|
||||
- [x] 4 **Lean into the "limited spots" angle**
|
||||
- File: `src/lib/content/pack-walks.ts` (added `scarcityNote` to the
|
||||
`pricing` block); `src/lib/types.ts` (added optional `scarcityNote?:
|
||||
string` to ServicePageContent.pricing); rendered in
|
||||
`src/lib/components/ServiceLandingPage.svelte` directly under the
|
||||
plan grid as a yellow-tinted pill with a clock icon.
|
||||
- Copy: *"We keep packs small (4-8 dogs) — popular days fill up fast."*
|
||||
- Only set on Pack Walks (the 4-8 number is specific to that service);
|
||||
the field is optional so 1:1 Walks and Puppy Visits get nothing.
|
||||
- Why: Real, honest scarcity. The 4-8 cap is already a fact; saying it
|
||||
out loud nudges decision-making.
|
||||
|
||||
- [ ] **About page — quantify Aless's expertise**
|
||||
- File: `src/lib/content/about.ts:29-30`
|
||||
- Why: "years of experience" is the weakest possible claim. Replace with
|
||||
concrete numbers Aless can stand behind: years operating, dogs in
|
||||
rotation, first-aid certification.
|
||||
|
||||
- [x] 5 **Pack Walks pricing intro — lead with the differentiator**
|
||||
- File: `src/lib/content/pack-walks.ts:23-24`
|
||||
- Implementation: Old intro led with "Our pack walks are a permanent
|
||||
booking of at least one walk day a week..." (commitment ask first).
|
||||
New intro leads with the benefits: *"Small packs of 4-8 dogs, 2-hour
|
||||
outings at Auckland's scenic dog parks and beaches, with free pick-up
|
||||
and drop-off included. We reinforce recall, car manners, and leash
|
||||
etiquette while your dog plays. Booked as a permanent weekly slot —
|
||||
gift your dog the best life!"*
|
||||
- Why: Buyers scan for benefits before commitments. Lead-with-policy
|
||||
framing creates resistance; lead-with-benefit framing builds desire.
|
||||
|
||||
- [ ] **FAQs — reframe from policy to reassurance**
|
||||
- File: `src/lib/content/homepage.ts:180-205`
|
||||
- Why: Answers are correct but read like terms & conditions. Lead with
|
||||
the *why* (the benefit/reassurance), then the *what*.
|
||||
|
||||
## Low — incremental polish
|
||||
|
||||
- [x] **Home services-card CTAs: "Learn more" → outcome-oriented**
|
||||
- File: `src/lib/components/ServicesSection.svelte:29`
|
||||
- Implementation: Visible label is now derived from the service title —
|
||||
*"See Pack Walks pricing →"*, *"See 1:1 Walks pricing →"*, *"See
|
||||
Puppy Visits pricing →"*. The previously-added screen-reader-only
|
||||
"about <Service>" span was removed since the visible label now carries
|
||||
that context for everyone, not just assistive tech users.
|
||||
- Why: "Learn more" was the lowest-intent CTA on the page; the new
|
||||
label states the destination and the next step.
|
||||
|
||||
- [x] **Testimonials intro blurb — sharper jobs-to-be-done framing**
|
||||
- File: `src/lib/components/TestimonialsSection.svelte:10-11`
|
||||
- Old: *"Happy owners, even happier dogs. Our Auckland dog walking
|
||||
clients love what the Tiny Gang brings to their dog's routine — and
|
||||
you can see why. Follow along on Instagram for daily adventures..."*
|
||||
- New: *"Busy parents get peace of mind. Dogs come home tired and
|
||||
happy. See why 30+ Auckland families trust the Tiny Gang — follow
|
||||
along on Instagram for daily adventures, wagging tails and the odd
|
||||
zoomie."*
|
||||
- Why: Leads with the two outcomes buyers actually care about (peace of
|
||||
mind for them, exercise for the dog), keeps the brand voice + 30+
|
||||
review proof point, then makes the Instagram nudge feel like a
|
||||
follow-on rather than the lead.
|
||||
|
||||
- [x] **Surface "Reliability / on-time" earlier**
|
||||
- File: `src/lib/content/homepage.ts:37` (hero subtitle)
|
||||
- Old: *"Trusted, professional dog walking across Auckland Central..."*
|
||||
- New: *"Trusted, on-time dog walking across Auckland Central..."*
|
||||
- Why: Reliability/punctuality is the #1 anxiety for busy parents
|
||||
booking a service that visits their home. Pulling "on-time" into the
|
||||
hero subtitle (one-word swap, no length cost) puts the reassurance
|
||||
above the fold.
|
||||
|
||||
## Deliberately not actioning
|
||||
|
||||
- **Pack Walks H1 rewrite to "Small-dog pack walks designed for calm,
|
||||
confident groups."** *"Join our Tiny Gang!"* is doing brand work — it's
|
||||
memorable and reinforces a phrase used everywhere else. Rewriting kills
|
||||
the most distinctive asset for marginal headline clarity.
|
||||
- **Booking submit button: "Send" → "Book my Meet & Greet".** The form
|
||||
also handles general enquiries, so a "book my…" label would feel wrong
|
||||
on a complaint email. Better fix would be to switch the label by
|
||||
`enquiryType` — keep "Send my booking" / "Send my enquiry" contextually.
|
||||
@@ -1,56 +0,0 @@
|
||||
# WebP Conversion (C5) — One-time setup
|
||||
|
||||
The hero `<picture>` element in `src/lib/components/HeroSection.svelte` now
|
||||
supports WebP sources. Once you generate the WebP files, add the URL fields
|
||||
to the hero content block — the markup will start serving WebP automatically
|
||||
to supporting browsers (every browser currently in use).
|
||||
|
||||
## 1. Generate WebP variants
|
||||
|
||||
From the project root, with `cwebp` installed (`brew install webp` /
|
||||
`choco install webp` / `apt install webp`):
|
||||
|
||||
```bash
|
||||
# Mobile hero (already 1536x1024 — optimise + convert)
|
||||
cwebp -q 82 static/images/maya-mascot.png -o static/images/maya-mascot.webp
|
||||
|
||||
# Three new untracked images visible in git status
|
||||
cwebp -q 82 static/images/happy-dogs-in-travel-ready-suv.jpg -o static/images/happy-dogs-in-travel-ready-suv.webp
|
||||
cwebp -q 82 static/images/playful-dog-pack-in-park.jpg -o static/images/playful-dog-pack-in-park.webp
|
||||
cwebp -q 82 static/images/testimonial-freddy-eating-stick-in-park.png -o static/images/testimonial-freddy-eating-stick-in-park.webp
|
||||
```
|
||||
|
||||
Target file sizes:
|
||||
- Hero (mobile): under 150 KB
|
||||
- Hero (desktop): under 300 KB
|
||||
- Testimonials/inline: under 80 KB
|
||||
|
||||
## 2. Wire up the WebP source
|
||||
|
||||
In `src/lib/content/homepage.ts`, uncomment and set:
|
||||
|
||||
```ts
|
||||
hero: {
|
||||
// ...existing fields
|
||||
imageWidth: 1536,
|
||||
imageHeight: 1024,
|
||||
imageWebpUrl: '/images/maya-mascot.webp'
|
||||
}
|
||||
```
|
||||
|
||||
If you also produce a desktop-specific variant, add `desktopImageUrl` and
|
||||
`desktopImageWebpUrl`.
|
||||
|
||||
## 3. Verify
|
||||
|
||||
Open the homepage in Chrome DevTools → Network → filter `Img`. You should
|
||||
see `maya-mascot.webp` being served with `Type: webp`. The `.png` is the
|
||||
fallback `<img>` and should only load if WebP is somehow unsupported.
|
||||
|
||||
## Why this matters
|
||||
|
||||
The codebase ships PNG/JPG hero assets. WebP at quality 82 typically lands
|
||||
30–50% smaller than the equivalent PNG/JPG with no perceptible quality
|
||||
difference. For the LCP element on a mobile homepage, that's a 0.5–1.5s
|
||||
improvement on slow 4G — directly relevant to the `largest-contentful-paint`
|
||||
Core Web Vital.
|
||||
@@ -27,6 +27,12 @@ RATE_LIMIT_MAX_PER_EMAIL=3
|
||||
RATE_LIMIT_MIN_INTERVAL_SECONDS=20
|
||||
EMAIL_SEND_TIMEOUT_SECONDS=20
|
||||
|
||||
# Shared secret for the post-deploy form smoke tests. The deploy script reads
|
||||
# this from the live remote .env and presents it via X-Deploy-Smoke; the
|
||||
# mail-api short-circuits matching requests before email/db side effects.
|
||||
# Rotate with: node -e "console.log(require('crypto').randomBytes(32).toString('hex'))"
|
||||
DEPLOY_SMOKE_SECRET=ed7261d3d7a5ac0a51e0cfb2bf4e2bd4009503605d2963d3ee766b7e885e76eb
|
||||
|
||||
# Security hardening — sensible defaults are in mail_api/config.py.
|
||||
# Override only if the public domains change or you need to allow extra origins.
|
||||
# CORS_ALLOWED_ORIGINS=https://goodwalk.co.nz,https://www.goodwalk.co.nz,https://clients.goodwalk.co.nz,https://cp.goodwalk.co.nz
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
[CmdletBinding()]
|
||||
[CmdletBinding()]
|
||||
param(
|
||||
[switch]$Force,
|
||||
[switch]$SkipSiteCheck,
|
||||
@@ -8,9 +8,94 @@ param(
|
||||
# postgres database, overwriting anything currently in admin_kv. After the
|
||||
# deploy completes the flag is automatically reset to 'auto' for subsequent
|
||||
# boots so we don't keep overwriting live data.
|
||||
[switch]$SeedAdminData
|
||||
[switch]$SeedAdminData,
|
||||
# Skip regenerating the legacy-clients-seed.json file. By default every
|
||||
# deploy refreshes it from data/legacy-{clients,onboarding,contracts}.json.
|
||||
# The mail-api merges this on boot, add-only — it never clobbers live data.
|
||||
[switch]$SkipLegacySeed
|
||||
)
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# UI helpers — colored output, step headers, timing
|
||||
# ---------------------------------------------------------------------------
|
||||
$script:UseColor = $Host.UI.SupportsVirtualTerminal -or [bool]$env:WT_SESSION -or [bool]$env:TERM_PROGRAM
|
||||
$script:ESC = [char]27
|
||||
|
||||
# Goodwalk brand palette + status colors (24-bit ANSI where supported)
|
||||
$script:CGreen = '38;2;122;170;122' # softened Goodwalk green for legibility on dark terminals
|
||||
$script:CYellow = '38;2;255;209;0' # #FFD100
|
||||
$script:CCyan = '96'
|
||||
$script:CDim = '90'
|
||||
$script:CGrey = '37'
|
||||
$script:COk = '92'
|
||||
$script:CWarn = '93'
|
||||
$script:CErr = '91'
|
||||
|
||||
function Format-Color {
|
||||
param([string]$Text, [string]$Code)
|
||||
if (-not $script:UseColor) { return $Text }
|
||||
return "$script:ESC[${Code}m$Text$script:ESC[0m"
|
||||
}
|
||||
|
||||
function Show-DeployBanner {
|
||||
$border = '──────────────────────────────────────────────────────' # 54 chars
|
||||
$pad22 = ' ' * 22
|
||||
Write-Host ''
|
||||
Write-Host (Format-Color " ╭$border╮" $script:CGreen)
|
||||
Write-Host -NoNewline (Format-Color ' │' $script:CGreen)
|
||||
Write-Host -NoNewline (Format-Color ' GoodWalk' $script:CYellow)
|
||||
Write-Host -NoNewline (Format-Color ' · production deploy' $script:CDim)
|
||||
Write-Host (Format-Color ($pad22 + '│') $script:CGreen)
|
||||
Write-Host (Format-Color " ╰$border╯" $script:CGreen)
|
||||
Write-Host ''
|
||||
}
|
||||
|
||||
function Write-Info {
|
||||
param([string]$Text)
|
||||
Write-Host (Format-Color " $Text" $script:CDim)
|
||||
}
|
||||
|
||||
function Write-Note {
|
||||
param([string]$Text)
|
||||
Write-Host (" " + (Format-Color '·' $script:CYellow) + " " + (Format-Color $Text $script:CGrey))
|
||||
}
|
||||
|
||||
function Write-Field {
|
||||
param([string]$Label, [string]$Value)
|
||||
$padded = $Label.PadRight(22)
|
||||
Write-Host -NoNewline (Format-Color " $padded" $script:CDim)
|
||||
Write-Host (Format-Color $Value $script:CGrey)
|
||||
}
|
||||
|
||||
function Write-Section {
|
||||
param([string]$Text)
|
||||
Write-Host ''
|
||||
Write-Host (Format-Color "── $Text " $script:CCyan)
|
||||
}
|
||||
|
||||
function Write-StepHeader {
|
||||
param([string]$Text)
|
||||
Write-Host ''
|
||||
Write-Host -NoNewline (Format-Color '▶ ' $script:CCyan)
|
||||
Write-Host (Format-Color $Text $script:CGrey)
|
||||
}
|
||||
|
||||
function Write-Ok {
|
||||
param([string]$Text, [double]$Sec = -1)
|
||||
$suffix = if ($Sec -ge 0) { Format-Color (' ({0:N1}s)' -f $Sec) $script:CDim } else { '' }
|
||||
Write-Host (" " + (Format-Color '✓' $script:COk) + " " + (Format-Color $Text $script:CGrey) + $suffix)
|
||||
}
|
||||
|
||||
function Write-Fail {
|
||||
param([string]$Text)
|
||||
Write-Host (" " + (Format-Color '✗' $script:CErr) + " " + (Format-Color $Text $script:CErr))
|
||||
}
|
||||
|
||||
function Write-WarnLine {
|
||||
param([string]$Text)
|
||||
Write-Host (" " + (Format-Color '!' $script:CWarn) + " " + (Format-Color $Text $script:CWarn))
|
||||
}
|
||||
|
||||
# ---------------------------------------------------------------------------
|
||||
# Goodwalk production deployment settings
|
||||
# Update these values before the first real deployment.
|
||||
@@ -154,6 +239,32 @@ function Export-HomepageContent {
|
||||
}
|
||||
}
|
||||
|
||||
function Build-LegacySeed {
|
||||
param(
|
||||
[string]$ProjectPath
|
||||
)
|
||||
|
||||
$scriptPath = Join-Path $ProjectPath 'scripts\build-legacy-seed.mjs'
|
||||
|
||||
if (-not (Test-Path -LiteralPath $scriptPath)) {
|
||||
throw "Legacy seed builder not found: $scriptPath"
|
||||
}
|
||||
|
||||
Push-Location $ProjectPath
|
||||
try {
|
||||
Invoke-External -FilePath 'node' -Arguments @($scriptPath)
|
||||
}
|
||||
finally {
|
||||
Pop-Location
|
||||
}
|
||||
|
||||
$outputPath = Join-Path $ProjectPath 'mail-api\legacy-clients-seed.json'
|
||||
if (-not (Test-Path -LiteralPath $outputPath)) {
|
||||
throw "Legacy seed builder did not produce expected output: $outputPath"
|
||||
}
|
||||
return $outputPath
|
||||
}
|
||||
|
||||
function New-UnixScriptCopy {
|
||||
param(
|
||||
[string]$SourcePath
|
||||
@@ -175,13 +286,13 @@ function New-UnixScriptCopy {
|
||||
function Invoke-SiteCheck {
|
||||
param([string]$Url)
|
||||
|
||||
Write-Host ''
|
||||
Write-Host "[deploy] Checking production site: $Url"
|
||||
Write-StepHeader 'Production site check'
|
||||
Write-Info $Url
|
||||
|
||||
try {
|
||||
$response = Invoke-WebRequest -Uri $Url -MaximumRedirection 5 -TimeoutSec 30
|
||||
if ($response.StatusCode -ge 200 -and $response.StatusCode -lt 400) {
|
||||
Write-Host "[deploy] Site responded with HTTP $($response.StatusCode)"
|
||||
Write-Ok "HTTP $($response.StatusCode) from production"
|
||||
return
|
||||
}
|
||||
throw "Unexpected HTTP $($response.StatusCode) from $Url"
|
||||
@@ -189,13 +300,136 @@ function Invoke-SiteCheck {
|
||||
catch {
|
||||
$message = "Post-deploy site check failed: $($_.Exception.Message). Verify URL: $Url"
|
||||
if ($SkipSiteCheck) {
|
||||
Write-Warning $message
|
||||
Write-WarnLine $message
|
||||
} else {
|
||||
Write-Fail $message
|
||||
throw $message
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function Get-RemoteSmokeSecret {
|
||||
param(
|
||||
[string]$SshTarget,
|
||||
[string[]]$SshArgList,
|
||||
[string]$DeployPath
|
||||
)
|
||||
|
||||
# Pulls DEPLOY_SMOKE_SECRET from the live .env that was just merged by the
|
||||
# deploy. Single source of truth — no need to sync template and live.
|
||||
$remoteCmd = "grep -E '^DEPLOY_SMOKE_SECRET=' '$DeployPath/.env' | tail -n1 | cut -d= -f2-"
|
||||
$output = & ssh @SshArgList $SshTarget $remoteCmd 2>$null
|
||||
if ($LASTEXITCODE -ne 0) { return $null }
|
||||
if ([string]::IsNullOrWhiteSpace($output)) { return $null }
|
||||
return ($output -join '').Trim()
|
||||
}
|
||||
|
||||
function Invoke-FormSmokeTest {
|
||||
param(
|
||||
[string]$Url,
|
||||
[string]$Secret,
|
||||
[hashtable]$Payload,
|
||||
[string]$Label
|
||||
)
|
||||
|
||||
$json = $Payload | ConvertTo-Json -Compress -Depth 6
|
||||
$headers = @{
|
||||
'X-Deploy-Smoke' = $Secret
|
||||
'Content-Type' = 'application/json'
|
||||
'User-Agent' = 'goodwalk-deploy-smoke/1.0'
|
||||
}
|
||||
|
||||
try {
|
||||
$response = Invoke-WebRequest -Uri $Url -Method Post -Headers $headers -Body $json -TimeoutSec 30 -MaximumRedirection 0
|
||||
}
|
||||
catch {
|
||||
throw "$Label smoke test failed: $($_.Exception.Message)"
|
||||
}
|
||||
|
||||
if ($response.StatusCode -ne 200) {
|
||||
throw "$Label smoke test returned HTTP $($response.StatusCode) (expected 200)"
|
||||
}
|
||||
|
||||
$body = $null
|
||||
try { $body = $response.Content | ConvertFrom-Json } catch { }
|
||||
if (-not $body -or -not $body.ok -or -not $body.smoke) {
|
||||
throw "$Label smoke test response did not include ok=true,smoke=true. Body: $($response.Content)"
|
||||
}
|
||||
|
||||
Write-Ok "$Label OK (request_id=$($body.request_id))"
|
||||
}
|
||||
|
||||
function Invoke-AllFormSmokeTests {
|
||||
param(
|
||||
[string]$SshTarget,
|
||||
[string[]]$SshArgList,
|
||||
[string]$DeployPath
|
||||
)
|
||||
|
||||
Write-StepHeader 'Production form smoke tests'
|
||||
|
||||
$secret = Get-RemoteSmokeSecret -SshTarget $SshTarget -SshArgList $SshArgList -DeployPath $DeployPath
|
||||
if ([string]::IsNullOrWhiteSpace($secret)) {
|
||||
$msg = "DEPLOY_SMOKE_SECRET not found in $DeployPath/.env on the server. Smoke tests cannot run."
|
||||
if ($SkipSiteCheck) {
|
||||
Write-WarnLine $msg
|
||||
return
|
||||
}
|
||||
Write-Fail $msg
|
||||
throw $msg
|
||||
}
|
||||
|
||||
# Payloads carry only the fields each Pydantic model requires. Validation,
|
||||
# rate limiting, honeypot, email send and DB writes are all bypassed by the
|
||||
# mail-api smoke shortcut, but Pydantic still parses the body — so this also
|
||||
# catches schema breakage.
|
||||
$smokeEmail = 'deploy-smoke@goodwalk.co.nz'
|
||||
$bookingPayload = @{
|
||||
fullName = 'Deploy Smoke'
|
||||
email = $smokeEmail
|
||||
phone = '+64-00-000-0000'
|
||||
enquiryType = 'booking'
|
||||
page = '/deploy-smoke'
|
||||
}
|
||||
$onboardingPayload = @{
|
||||
fullName = 'Deploy Smoke'
|
||||
email = $smokeEmail
|
||||
phone = '+64-00-000-0000'
|
||||
address = '1 Smoke St, Auckland'
|
||||
dogName = 'Smoke'
|
||||
dogBreed = 'Test'
|
||||
vetName = 'Dr Smoke'
|
||||
vetPhone = '+64-00-000-0000'
|
||||
emergencyContactName = 'Smoke Contact'
|
||||
emergencyContactPhone = '+64-00-000-0000'
|
||||
signatureDataUrl = 'data:image/png;base64,iVBORw0KGgo='
|
||||
page = '/deploy-smoke'
|
||||
}
|
||||
$contractPayload = @{
|
||||
fullName = 'Deploy Smoke'
|
||||
email = $smokeEmail
|
||||
phone = '+64-00-000-0000'
|
||||
address = '1 Smoke St, Auckland'
|
||||
dogName = 'Smoke'
|
||||
dogBreed = 'Test'
|
||||
serviceType = 'pack-walks'
|
||||
startDate = '2099-01-01'
|
||||
signatureDataUrl = 'data:image/png;base64,iVBORw0KGgo='
|
||||
page = '/deploy-smoke'
|
||||
}
|
||||
|
||||
# Each endpoint is exercised on the subdomain that actually hosts it
|
||||
# in production (see nginx/goodwalk.co.nz.svelte.conf.example):
|
||||
# /api/submit → www.goodwalk.co.nz
|
||||
# /api/onboarding-submit → clients.goodwalk.co.nz
|
||||
# /api/contract-submit → clients.goodwalk.co.nz
|
||||
Invoke-FormSmokeTest -Url 'https://www.goodwalk.co.nz/api/submit' -Secret $secret -Payload $bookingPayload -Label 'booking form (www /api/submit)'
|
||||
Invoke-FormSmokeTest -Url 'https://clients.goodwalk.co.nz/api/onboarding-submit' -Secret $secret -Payload $onboardingPayload -Label 'onboarding form (clients /api/onboarding-submit)'
|
||||
Invoke-FormSmokeTest -Url 'https://clients.goodwalk.co.nz/api/contract-submit' -Secret $secret -Payload $contractPayload -Label 'contract form (clients /api/contract-submit)'
|
||||
}
|
||||
|
||||
Show-DeployBanner
|
||||
|
||||
Assert-Command ssh
|
||||
Assert-Command scp
|
||||
Assert-Command tar
|
||||
@@ -243,75 +477,106 @@ $scpArchiveTarget = '{0}:{1}' -f $sshTarget, $RemoteArchivePath
|
||||
$scpHelperTarget = '{0}:{1}' -f $sshTarget, $RemoteHelperPath
|
||||
$sshArgs = Get-SshArgumentList
|
||||
|
||||
Write-Host '[deploy] Main Goodwalk website deployment'
|
||||
Write-Host "[deploy] Local project path: $LocalProjectPath"
|
||||
Write-Host "[deploy] Remote deployment path: $RemoteDeploymentPath"
|
||||
Write-Host "[deploy] Remote compose file: $ComposeFileName"
|
||||
Write-Host "[deploy] Docker project name: $DockerProjectName"
|
||||
Write-Host "[deploy] Shared nginx config: $NginxConfigTarget"
|
||||
Write-Host "[deploy] Shared nginx compose file: $NginxComposeFile"
|
||||
Write-Host "[deploy] Maintenance host dir: $MaintenanceHostDir (must be bind-mounted at /var/www/maintenance:ro)"
|
||||
Write-Host "[deploy] Maintenance flag path: $MaintenanceFlagPath"
|
||||
Write-Host "[deploy] SSH target: $sshTarget"
|
||||
Write-Host "[deploy] SSH config: $SshConfigPath"
|
||||
Write-Section 'Configuration'
|
||||
Write-Field 'Local project' $LocalProjectPath
|
||||
Write-Field 'Remote path' $RemoteDeploymentPath
|
||||
Write-Field 'Compose file' $ComposeFileName
|
||||
Write-Field 'Docker project' $DockerProjectName
|
||||
Write-Field 'Shared nginx conf' $NginxConfigTarget
|
||||
Write-Field 'Shared nginx compose' $NginxComposeFile
|
||||
Write-Field 'Maintenance dir' "$MaintenanceHostDir (mounted at /var/www/maintenance:ro)"
|
||||
Write-Field 'Maintenance flag' $MaintenanceFlagPath
|
||||
Write-Field 'SSH target' $sshTarget
|
||||
Write-Field 'SSH config' $SshConfigPath
|
||||
if (-not [string]::IsNullOrWhiteSpace($Service)) {
|
||||
Write-Host "[deploy] Target service: $Service"
|
||||
Write-Field 'Target service' $Service
|
||||
}
|
||||
if ($SeedAdminData) {
|
||||
Write-Host '[deploy] Admin data: seeding postgres from JSON on next mail-api boot'
|
||||
Write-Field 'Admin data' 'seed postgres from JSON on next mail-api boot'
|
||||
}
|
||||
if ($SkipLegacySeed) {
|
||||
Write-Field 'Legacy clients seed' 'SKIPPED (existing mail-api/legacy-clients-seed.json reused)'
|
||||
} else {
|
||||
Write-Field 'Legacy clients seed' 'rebuild from data/legacy-*.json (add-only merge on boot)'
|
||||
}
|
||||
if ([string]::IsNullOrWhiteSpace($SshKeyPath)) {
|
||||
Write-Host '[deploy] SSH auth: interactive password prompt'
|
||||
Write-Field 'SSH auth' 'interactive password prompt'
|
||||
} else {
|
||||
Write-Host "[deploy] SSH auth: key file $SshKeyPath"
|
||||
Write-Field 'SSH auth' "key file $SshKeyPath"
|
||||
}
|
||||
Write-Host ''
|
||||
Write-Host '[deploy] Safety notes:'
|
||||
Write-Host ' - Only the top-level Goodwalk compose project will be updated.'
|
||||
Write-Host ' - Legacy WordPress/onboarding compose files are not used.'
|
||||
Write-Host ' - Remote .env files are preserved because they are not uploaded.'
|
||||
Write-Host ' - No global Docker prune/stop/delete commands are used.'
|
||||
Write-Host ' - Shared nginx will be updated and reloaded with the Docker-DNS-based config.'
|
||||
Write-Host ' - Subdomains served by this stack:'
|
||||
Write-Host ' goodwalk.co.nz / www.goodwalk.co.nz (marketing)'
|
||||
Write-Host ' clients.goodwalk.co.nz (client onboarding + contracts)'
|
||||
Write-Host ' cp.goodwalk.co.nz (owner admin dashboard)'
|
||||
Write-Host ' onboarding/admin remain legacy redirect aliases'
|
||||
|
||||
Write-Section 'Safety notes'
|
||||
Write-Note 'Only the top-level Goodwalk compose project will be updated.'
|
||||
Write-Note 'Legacy WordPress/onboarding compose files are not used.'
|
||||
Write-Note 'Remote .env files are preserved because they are not uploaded.'
|
||||
Write-Note 'No global Docker prune/stop/delete commands are used.'
|
||||
Write-Note 'Shared nginx will be updated and reloaded with the Docker-DNS-based config.'
|
||||
|
||||
Write-Section 'Subdomains served by this stack'
|
||||
Write-Note 'goodwalk.co.nz / www.goodwalk.co.nz — marketing'
|
||||
Write-Note 'clients.goodwalk.co.nz — client onboarding + contracts'
|
||||
Write-Note 'cp.goodwalk.co.nz — owner admin dashboard'
|
||||
Write-Note 'onboarding/admin remain legacy redirect aliases'
|
||||
if ($SeedAdminData) {
|
||||
Write-Host ' - Admin data seed: mail-api will OVERWRITE postgres admin_kv from the JSON volume.'
|
||||
Write-Host ''
|
||||
Write-WarnLine 'Admin data seed: mail-api will OVERWRITE postgres admin_kv from the JSON volume.'
|
||||
}
|
||||
|
||||
if (-not $Force) {
|
||||
$confirmation = Read-Host "Type DEPLOY to continue with the remote path '$RemoteDeploymentPath'"
|
||||
Write-Host ''
|
||||
Write-Host -NoNewline (" " + (Format-Color '?' $script:CYellow) + " ")
|
||||
Write-Host -NoNewline (Format-Color 'Type ' $script:CGrey)
|
||||
Write-Host -NoNewline (Format-Color 'DEPLOY' $script:CYellow)
|
||||
Write-Host -NoNewline (Format-Color " to continue with remote path '" $script:CGrey)
|
||||
Write-Host -NoNewline (Format-Color $RemoteDeploymentPath $script:CCyan)
|
||||
Write-Host -NoNewline (Format-Color "' " $script:CGrey)
|
||||
$confirmation = Read-Host
|
||||
if ($confirmation -ne 'DEPLOY') {
|
||||
Write-Fail 'Deployment cancelled.'
|
||||
throw 'Deployment cancelled.'
|
||||
}
|
||||
}
|
||||
|
||||
$archivePath = $null
|
||||
$uploadHelperPath = $null
|
||||
$totalWatch = [System.Diagnostics.Stopwatch]::StartNew()
|
||||
$stepWatch = [System.Diagnostics.Stopwatch]::new()
|
||||
|
||||
try {
|
||||
Write-Host ''
|
||||
Write-Host '[deploy] Exporting current homepage content for PostgreSQL sync'
|
||||
Write-StepHeader 'Export homepage content for PostgreSQL sync'
|
||||
$stepWatch.Restart()
|
||||
Export-HomepageContent -ProjectPath $LocalProjectPath -OutputPath $GeneratedHomepageContentPath
|
||||
Write-Ok 'Homepage content exported' $stepWatch.Elapsed.TotalSeconds
|
||||
|
||||
Write-Host ''
|
||||
Write-Host '[deploy] Creating deployment archive'
|
||||
if (-not $SkipLegacySeed) {
|
||||
Write-StepHeader 'Build legacy clients seed for mail-api image'
|
||||
$stepWatch.Restart()
|
||||
$legacySeedPath = Build-LegacySeed -ProjectPath $LocalProjectPath
|
||||
Write-Info "Seed file: $legacySeedPath"
|
||||
Write-Ok 'Legacy clients seed ready' $stepWatch.Elapsed.TotalSeconds
|
||||
} else {
|
||||
Write-Note 'Legacy clients seed step skipped (-SkipLegacySeed).'
|
||||
}
|
||||
|
||||
Write-StepHeader 'Create deployment archive'
|
||||
$stepWatch.Restart()
|
||||
$archivePath = New-DeployArchive -ProjectPath $LocalProjectPath
|
||||
Write-Host "[deploy] Archive ready: $archivePath"
|
||||
Write-Info "Archive: $archivePath"
|
||||
Write-Ok 'Archive ready' $stepWatch.Elapsed.TotalSeconds
|
||||
|
||||
Write-Host ''
|
||||
Write-Host '[deploy] Uploading remote helper'
|
||||
Write-StepHeader 'Upload remote helper'
|
||||
$stepWatch.Restart()
|
||||
$uploadHelperPath = New-UnixScriptCopy -SourcePath $LocalRemoteHelperPath
|
||||
Invoke-External -FilePath 'scp' -Arguments ($sshArgs + @($uploadHelperPath, $scpHelperTarget))
|
||||
Write-Ok 'Helper uploaded' $stepWatch.Elapsed.TotalSeconds
|
||||
|
||||
Write-Host ''
|
||||
Write-Host '[deploy] Uploading application archive'
|
||||
Write-StepHeader 'Upload application archive'
|
||||
$stepWatch.Restart()
|
||||
Invoke-External -FilePath 'scp' -Arguments ($sshArgs + @($archivePath, $scpArchiveTarget))
|
||||
Write-Ok 'Archive uploaded' $stepWatch.Elapsed.TotalSeconds
|
||||
|
||||
Write-Host ''
|
||||
Write-Host '[deploy] Running remote deployment'
|
||||
Write-StepHeader 'Run remote deployment'
|
||||
$stepWatch.Restart()
|
||||
Invoke-External -FilePath 'ssh' -Arguments ($sshArgs + @(
|
||||
$sshTarget,
|
||||
'bash',
|
||||
@@ -338,9 +603,10 @@ try {
|
||||
$MaintenanceFlagPath
|
||||
) + $(if (-not [string]::IsNullOrWhiteSpace($Service)) { @('--service', $Service) } else { @() }) `
|
||||
+ $(if ($SeedAdminData) { @('--seed-admin-data') } else { @() }))
|
||||
Write-Ok 'Remote deployment finished' $stepWatch.Elapsed.TotalSeconds
|
||||
|
||||
Write-Host ''
|
||||
Write-Host '[deploy] Cleaning remote temporary files'
|
||||
Write-StepHeader 'Clean remote temporary files'
|
||||
$stepWatch.Restart()
|
||||
Invoke-External -FilePath 'ssh' -Arguments ($sshArgs + @(
|
||||
$sshTarget,
|
||||
'rm',
|
||||
@@ -348,13 +614,28 @@ try {
|
||||
$RemoteArchivePath,
|
||||
$RemoteHelperPath
|
||||
))
|
||||
Write-Ok 'Remote temp files cleaned' $stepWatch.Elapsed.TotalSeconds
|
||||
|
||||
if (-not $SkipSiteCheck) {
|
||||
Invoke-SiteCheck -Url $VerifyUrl
|
||||
Invoke-AllFormSmokeTests -SshTarget $sshTarget -SshArgList $sshArgs -DeployPath $RemoteDeploymentPath
|
||||
}
|
||||
|
||||
$totalWatch.Stop()
|
||||
$border = '──────────────────────────────────────────────────────'
|
||||
Write-Host ''
|
||||
Write-Host (Format-Color " ╭$border╮" $script:CGreen)
|
||||
Write-Host -NoNewline (Format-Color ' │' $script:CGreen)
|
||||
Write-Host -NoNewline (Format-Color ' ✓ ' $script:COk)
|
||||
Write-Host -NoNewline (Format-Color 'Deployment complete' $script:CYellow)
|
||||
$elapsedText = (' · {0:N1}s total' -f $totalWatch.Elapsed.TotalSeconds)
|
||||
Write-Host -NoNewline (Format-Color $elapsedText $script:CDim)
|
||||
$visibleLen = (' ✓ Deployment complete' + $elapsedText).Length
|
||||
$padLen = 54 - $visibleLen
|
||||
if ($padLen -lt 1) { $padLen = 1 }
|
||||
Write-Host (Format-Color ((' ' * $padLen) + '│') $script:CGreen)
|
||||
Write-Host (Format-Color " ╰$border╯" $script:CGreen)
|
||||
Write-Host ''
|
||||
Write-Host '[deploy] Deployment completed successfully'
|
||||
}
|
||||
finally {
|
||||
if ($archivePath -and (Test-Path -LiteralPath $archivePath)) {
|
||||
|
||||
@@ -56,6 +56,7 @@ services:
|
||||
CORS_ALLOWED_ORIGINS: ${CORS_ALLOWED_ORIGINS:-}
|
||||
TRUSTED_HOSTS: ${TRUSTED_HOSTS:-}
|
||||
MAX_REQUEST_BODY_BYTES: ${MAX_REQUEST_BODY_BYTES:-}
|
||||
DEPLOY_SMOKE_SECRET: ${DEPLOY_SMOKE_SECRET:-}
|
||||
PYTHONUNBUFFERED: '1'
|
||||
TZ: ${TZ:-Pacific/Auckland}
|
||||
expose:
|
||||
|
||||
@@ -0,0 +1,34 @@
|
||||
# Goodwalk docs
|
||||
|
||||
Reference, audit, and planning documents for the Goodwalk site. Project-level rules live in [`/CLAUDE.md`](../CLAUDE.md); everything else lives here.
|
||||
|
||||
## Product & brand
|
||||
|
||||
- [product.md](product.md) — users, purpose, brand personality, design principles
|
||||
- [marketing-principles.md](marketing-principles.md) — Chris Do / Debbie Millman principles, applied to the rebuild
|
||||
- [marketing-voice.md](marketing-voice.md) — copy/voice style guide (calm, certain, warm, specific)
|
||||
|
||||
## Design
|
||||
|
||||
- [design-language.md](design-language.md) — canonical visual system: colour, typography, spacing, motion, surface, photography direction
|
||||
- [design-audit.md](design-audit.md) — codebase-wide design drift audit + token/refactor changelog
|
||||
- [homepage-flow-audit.md](homepage-flow-audit.md) — section-by-section IA review of the homepage
|
||||
- [mobile-polish.md](mobile-polish.md) — mobile conversion & comfort tracker
|
||||
- [ux-polish.md](ux-polish.md) — conversion audit tracker
|
||||
|
||||
## SEO & growth
|
||||
|
||||
- [competitors.md](competitors.md) — Fetch / Sticks & Bones analysis + what to borrow
|
||||
- [comparison-page-plan.md](comparison-page-plan.md) — brief for the dog-walker-vs-daycare-vs-sitter page
|
||||
- [nz-citations.md](nz-citations.md) — canonical NAP block + directory submission list
|
||||
|
||||
## Ops
|
||||
|
||||
- [deployment.md](deployment.md) — production deploy flow, server layout, nginx cutover
|
||||
- [webp-conversion.md](webp-conversion.md) — one-time WebP setup for hero images
|
||||
- [onboarding.md](onboarding.md) — client onboarding flow, lifecycle status, legacy CSV migration, Postgres target schema
|
||||
|
||||
## Archive
|
||||
|
||||
- [archive/v3-design-audit.md](archive/v3-design-audit.md) — superseded by [design-audit.md](design-audit.md)
|
||||
- [archive/marketing-voice-v2-proposal.md](archive/marketing-voice-v2-proposal.md) — one-shot copy-change proposal against [marketing-voice.md](marketing-voice.md)
|
||||
@@ -0,0 +1,89 @@
|
||||
# Competitor analysis — Auckland dog walking
|
||||
|
||||
A look at what two of our closest competitors do well, and what's worth borrowing for Goodwalk. Captured 2026-05-20.
|
||||
|
||||
---
|
||||
|
||||
## Fetch Dog Walking — https://fetchdogwalking.co.nz
|
||||
|
||||
### What they do well
|
||||
|
||||
- **Founder-led narrative.** Andy Evans is named on the site and the philosophy ("freedom to explore in nature is an essential investment in our dog's health") is front and centre. Feels like a person, not a faceless service.
|
||||
- **Content depth.** They go beyond "book a walk" and publish:
|
||||
- Auckland **Dog Parks** guide
|
||||
- Basic **Training** tips
|
||||
- **FAQs** page
|
||||
- "Our Dogs" and "Our People" pages
|
||||
- **"Dog's Highlight of the Week"** — a recurring content hook that gives existing clients a reason to keep visiting the site and gives new visitors social proof.
|
||||
- **Diversified offering.** The 100-acre Waitakere Ranges farm-stay / boarding service is a real differentiator — they're not just selling walks, they're selling a destination.
|
||||
- **Positions on authority.** Tagline "Auckland's Leading Dog Walking Company" is bold and clear.
|
||||
- **Strong primary CTA.** Homepage contact form is short (company, dog name, phone, email, suburb, message) — low friction.
|
||||
|
||||
### What we could implement
|
||||
|
||||
| Idea | Effort | Why |
|
||||
|---|---|---|
|
||||
| **Founder bio / "About Matt" page** with photo and philosophy | S | Personality is a moat. Easy SEO win for branded searches. |
|
||||
| **"Dog of the Week" / Instagram-style spotlight on the homepage** | M | Gives the site an "alive" feel; reuses content we likely already post to social. Could pull from the existing client_profiles table. |
|
||||
| **Dog parks of Auckland** content page | M | Pure SEO play — high-intent local search ("best dog parks Auckland"). Drives top-of-funnel traffic. |
|
||||
| **Training tips / FAQ content hub** | M | Same SEO logic. Also reduces support load. |
|
||||
| **"Our pack" page showing all dogs we walk** (with owner consent) | S | Trust + community. We already collect dog photos in onboarding. |
|
||||
|
||||
---
|
||||
|
||||
## Sticks & Bones — https://sticksandbones.co.nz
|
||||
|
||||
### What they do well
|
||||
|
||||
- **Process is the product.** They sell their *enrollment process* as a trust signal:
|
||||
1. Free Meet & Greet at home
|
||||
2. **Minimum two Assessment Walks** before enrollment
|
||||
3. Handler matching
|
||||
4. App-based booking
|
||||
This makes the service feel professional and screened — they're not taking everyone.
|
||||
- **Real-time walk reports via app.** Clients get photos and a report after every walk. This is the single biggest differentiator and the most-mentioned feature in their testimonials.
|
||||
- **Concrete trust signals.**
|
||||
- Police-checked handlers
|
||||
- Secure key handling process documented
|
||||
- Handlers named individually ("Amanda walks our Cooper")
|
||||
- Maximum 4 dogs per Urban Adventure Walk (explicit cap = safety)
|
||||
- **Personalised client app** for scheduling, history, and reports — clients self-serve.
|
||||
- **In-home dog sitting**, but only for enrolled walking clients (creates retention + upsell).
|
||||
- **Tagline is benefit-led**, not feature-led: "dog walking made easy" + "active dogs behave better."
|
||||
|
||||
### What we could implement
|
||||
|
||||
| Idea | Effort | Why |
|
||||
|---|---|---|
|
||||
| **Post-walk reports** (photo + short note delivered by email or in the client portal) | L | This is the killer feature. Our admin dashboard already touches client_profiles — extending to walk records is a natural next step. Single biggest perceived-value lift available. |
|
||||
| **Document our enrollment process** as a 3-step visual on the homepage (Enquire → Meet & Greet → First Walk) | S | We already do a Meet & Greet — we're just not selling it. Pure copy/design change. |
|
||||
| **Cap group size publicly** (e.g. "max 4 dogs per pack") | S | Free trust signal if it matches reality. |
|
||||
| **Name the team** with photos on About page | S | Sticks & Bones gets specific mentions of handlers by name in reviews — that only happens when you make handlers visible. |
|
||||
| **Police-check / insurance / first-aid badges** in the footer or About page | S | Cheap credibility. |
|
||||
| **Client portal extensions** — walk history, upcoming bookings, ability to reschedule | L | We already have auth + onboarding + admin dashboard. The data model is most of the way there. |
|
||||
| **"Enrolled clients only" services** as a retention hook (e.g. boarding, sitting) | M | Mirrors S&B's sitting model. Existing clients get priority — gives a reason to stay engaged after onboarding. |
|
||||
|
||||
---
|
||||
|
||||
## Cross-cutting themes
|
||||
|
||||
Both competitors do these things; we should too:
|
||||
|
||||
1. **Named humans on the site.** Founder, team, dogs. The word "we" without faces is worth less.
|
||||
2. **Process as proof.** Both make their intake process visible — it doubles as marketing.
|
||||
3. **Service ladder, not a service.** Walks → assessments → boarding/sitting → premium options. Gives clients somewhere to grow into.
|
||||
4. **Locations as content.** Specific Auckland park / suburb mentions throughout — local SEO and "they know my area" trust.
|
||||
|
||||
## What Goodwalk has that they don't (lean into this)
|
||||
|
||||
- **Multi-step onboarding form** with signature, vet, emergency contacts, behavioural profile — already more thorough than what's visible on either competitor.
|
||||
- **Database-backed client profiles** — foundation for a portal richer than S&B's (their app is the obvious next ceiling; we have data they don't seem to capture).
|
||||
- **Brand-coloured design system** already in place (Goodwalk Green / Yellow) — distinct visual identity vs. the generic blue/green dog-walker templates.
|
||||
|
||||
## Suggested priority order
|
||||
|
||||
1. **Founder/team page + named handlers** (1 day, big trust lift)
|
||||
2. **Visualise the enrollment process on the homepage** (1 day, sells the Meet & Greet we already do)
|
||||
3. **Dog Parks of Auckland content page** (2-3 days, SEO compounding)
|
||||
4. **Post-walk reports** in the client portal (1-2 weeks, table-stakes vs. S&B)
|
||||
5. **Walk history / portal extensions** (2-3 weeks, retention moat)
|
||||
@@ -0,0 +1,252 @@
|
||||
# Onboarding
|
||||
|
||||
How a Goodwalk client comes into the system, how their record is shaped, and how
|
||||
that record evolves over time.
|
||||
|
||||
## Flow at a glance
|
||||
|
||||
1. **Enquiry / email claim** — the user signs in to `clients.goodwalk.co.nz`
|
||||
with a one-time code. A `client_profiles` entry is created keyed by the
|
||||
lower-cased email.
|
||||
2. **Onboarding form** (`src/lib/components/OnboardingPage.svelte`) — a 5-step
|
||||
form: Owner Details → Dog Details → Health → Behaviour → Sign. Every
|
||||
question must be answered; there is no skip / mark-complete shortcut.
|
||||
3. **Submit** — the form posts to `POST /api/onboarding-submit`. The backend
|
||||
validates, stores a `submissionSnapshot` on the profile, sends the owner an
|
||||
email + PDF, and flips `onboardingCompleted: true`.
|
||||
4. **Admin dashboard** (`src/lib/components/admin-dashboard/AdminDashboard.svelte`)
|
||||
surfaces pending and completed clients, and now lets the owner change a
|
||||
client's lifecycle status.
|
||||
|
||||
## Where data lives today
|
||||
|
||||
| Thing | Storage | Source of truth |
|
||||
| --- | --- | --- |
|
||||
| Client profiles | `mail-api/data/client_profiles.json` (mirrored into `admin_db` KV) | `_client_profiles` dict in `main.py` |
|
||||
| Onboarding drafts (in-progress, per email) | `mail-api/data/drafts.json` | `_drafts` dict |
|
||||
| Submitted onboarding snapshot | Inside the client profile as `onboardingSubmission` / `submissionSnapshot` | same |
|
||||
| Activity feed (admin actions) | SQLite via `admin_db.record_event` | same |
|
||||
| Legacy historical data (Gravity Forms exports) | `data/legacy-onboarding.json`, `data/legacy-contracts.json`, `data/legacy-clients.json` | regenerated from the CSVs in repo root |
|
||||
|
||||
The plan is to move the JSON file to Postgres later; the JSON shape is already
|
||||
relational-friendly (see *Postgres target* below).
|
||||
|
||||
## Client profile shape
|
||||
|
||||
Keyed by lower-cased email. Fields are merged over time as the client moves
|
||||
through the funnel.
|
||||
|
||||
```jsonc
|
||||
{
|
||||
"fullName": "Shelley McCabe",
|
||||
"phone": "+6421328907",
|
||||
"address": "...",
|
||||
"dogName": "Maisie",
|
||||
"dogBreed": "...",
|
||||
"dogAge": "2015-04-28",
|
||||
"services": ["Tiny Gang Pack Walks"],
|
||||
"onboardingCompleted": true,
|
||||
"onboardingSubmittedAt": "2026-03-14T16:14:20",
|
||||
"onboardingSubmission": { /* full 5-step snapshot */ },
|
||||
"welcomePackSentAt": "...",
|
||||
"welcomePackOffer": { "serviceType": "...", "priceDetails": "...", "startDate": "..." },
|
||||
"birthdayAutoSend": false,
|
||||
"birthdayEmailLastSentAt": "...",
|
||||
"lastEnquiryAt": "...",
|
||||
"lifecycle": {
|
||||
"status": "active", // active | paused | cancelled | archived
|
||||
"reason": "",
|
||||
"changedAt": "2026-05-20T10:11:00",
|
||||
"changedBy": "info@goodwalk.co.nz"
|
||||
},
|
||||
"lifecycleHistory": [
|
||||
{ "status": "active", "reason": "", "changedAt": "...", "changedBy": "..." }
|
||||
]
|
||||
}
|
||||
```
|
||||
|
||||
## Lifecycle status
|
||||
|
||||
Soft-delete only. **No client is ever removed**, so we keep the full history
|
||||
for retention / newsletter work.
|
||||
|
||||
| Status | Meaning | Outreach included? |
|
||||
| --- | --- | --- |
|
||||
| `active` | Currently walking with us. | Yes — pending + completed + birthdays |
|
||||
| `paused` | Temporarily not walking (holiday, recovery, etc.). | Yes — still on outreach lists |
|
||||
| `cancelled` | No longer a client. Kept on file. | **No** — excluded from outreach |
|
||||
| `archived` | Long-term inactive. Hidden from default views. | **No** — excluded from outreach |
|
||||
|
||||
Defaults: every client starts as `active`. A missing `lifecycle` block is
|
||||
treated as `active` by the API and the dashboard.
|
||||
|
||||
### Filters that respect lifecycle
|
||||
|
||||
`_client_is_reachable(profile)` in `mail-api/main.py` is the single gate.
|
||||
`cancelled` and `archived` are filtered out of:
|
||||
|
||||
- `GET /owner/pending-onboarding`
|
||||
- `GET /owner/completed-onboarding`
|
||||
- `GET /owner/birthdays`
|
||||
|
||||
`GET /owner/all-clients` returns **everyone**, regardless of status, so the
|
||||
admin can always find a former client. The status pill on each row makes the
|
||||
state obvious.
|
||||
|
||||
### Changing a client's status
|
||||
|
||||
Admin dashboard → Clients tab → row action **"Change status"**. Opens a modal
|
||||
with the four status options and an optional 500-char reason. Saving calls:
|
||||
|
||||
```
|
||||
POST /api/owner/client-status
|
||||
Authorization: Bearer <admin session token>
|
||||
Content-Type: application/json
|
||||
|
||||
{
|
||||
"email": "owner@example.com",
|
||||
"status": "paused",
|
||||
"reason": "Moving house, back in June"
|
||||
}
|
||||
```
|
||||
|
||||
The backend writes the new `lifecycle` block, appends to `lifecycleHistory`
|
||||
(capped at the last 50 entries), and records an
|
||||
`owner_client_status_changed` event in the activity feed.
|
||||
|
||||
## Legacy data migration
|
||||
|
||||
Two CSV exports from the old Gravity Forms install:
|
||||
|
||||
- `dog-enrolment-form-2026-05-20.csv` — 72 onboarding submissions (2022–2026).
|
||||
- `goodwalk-contract-2026-05-20.csv` — 39 contract signings (2022-06 → 2024-02).
|
||||
|
||||
Two cleansing scripts produce relational-friendly JSON:
|
||||
|
||||
```
|
||||
node scripts/clean-legacy-onboarding.mjs # → data/legacy-onboarding.json
|
||||
node scripts/clean-legacy-contracts.mjs # → data/legacy-contracts.json
|
||||
# also enriches legacy-onboarding.json
|
||||
# and writes data/legacy-clients.json
|
||||
```
|
||||
|
||||
### Importing legacy clients into production
|
||||
|
||||
The deploy pipeline ships these 36 clients into the live system on every
|
||||
deploy. The flow:
|
||||
|
||||
1. `deploy.ps1` runs `scripts/build-legacy-seed.mjs`, which folds the three
|
||||
legacy JSON files into one email-keyed dict shaped like the live
|
||||
`_client_profiles` entries. Output: `mail-api/legacy-clients-seed.json`.
|
||||
2. The mail-api `Dockerfile` `COPY`s that file to `/app/legacy-clients-seed.json`
|
||||
so it lives **outside** the persistent `mail_api_data` volume — every
|
||||
deploy carries a fresh copy.
|
||||
3. On boot, `_merge_legacy_seed_if_present()` adds every email that is not
|
||||
already in `_client_profiles`, persists the result to JSON + admin_kv,
|
||||
and never overwrites a live entry. Re-running is idempotent.
|
||||
|
||||
Each imported client is created with `lifecycle: { status: 'archived' }` so
|
||||
they don't pollute outreach lists. The admin can promote any of them back to
|
||||
`active` from the dashboard. Provenance (entry IDs, signature URL, PDF URLs)
|
||||
is preserved on the `legacy` field of every profile.
|
||||
|
||||
To skip the rebuild on a hotfix deploy use `deploy.ps1 -SkipLegacySeed`; the
|
||||
previous seed file is reused as-is. To regenerate locally:
|
||||
|
||||
```
|
||||
node scripts/build-legacy-seed.mjs
|
||||
```
|
||||
|
||||
The contracts script backfills owner email + postal address onto onboarding
|
||||
records where a contract row matches (by owner name, then dog name, then a
|
||||
unique surname). At present: **38 of 72** onboarding rows now have an email
|
||||
attached; the rest were submitted after the contract form was retired in
|
||||
2024-02 and will need an email-claim flow before they can sign in.
|
||||
|
||||
### Known gaps in the legacy data
|
||||
|
||||
- 34 onboarding rows have no email (mostly post-2024-02).
|
||||
- Dog **breed** was never collected by the old forms — `null` everywhere.
|
||||
- Owner **address** was only collected on the contract form, not on
|
||||
onboarding directly.
|
||||
- One vet phone number (`6307168`) couldn't be normalised to E.164 — kept
|
||||
as `phoneRaw` only.
|
||||
|
||||
## Postgres target
|
||||
|
||||
The JSON shapes already map cleanly onto a relational schema. Suggested tables
|
||||
once we migrate off the JSON file:
|
||||
|
||||
```sql
|
||||
clients (
|
||||
email TEXT PRIMARY KEY,
|
||||
first_name, last_name, phone, phone_raw,
|
||||
address, address_street, address_line2, address_suburb,
|
||||
address_city, address_postal_code, address_country,
|
||||
-- lifecycle
|
||||
lifecycle_status TEXT NOT NULL DEFAULT 'active', -- enum: active|paused|cancelled|archived
|
||||
lifecycle_reason TEXT,
|
||||
lifecycle_changed_at TIMESTAMPTZ,
|
||||
lifecycle_changed_by TEXT,
|
||||
created_at TIMESTAMPTZ NOT NULL DEFAULT now(),
|
||||
updated_at TIMESTAMPTZ NOT NULL DEFAULT now()
|
||||
);
|
||||
|
||||
client_status_history (
|
||||
id SERIAL PRIMARY KEY,
|
||||
client_email TEXT REFERENCES clients(email),
|
||||
status TEXT NOT NULL,
|
||||
reason TEXT,
|
||||
changed_at TIMESTAMPTZ NOT NULL,
|
||||
changed_by TEXT NOT NULL
|
||||
);
|
||||
|
||||
dogs (
|
||||
id SERIAL PRIMARY KEY,
|
||||
client_email TEXT REFERENCES clients(email),
|
||||
name TEXT, surname TEXT, date_of_birth DATE, breed TEXT
|
||||
);
|
||||
|
||||
onboarding_submissions (
|
||||
id SERIAL PRIMARY KEY,
|
||||
client_email TEXT REFERENCES clients(email),
|
||||
dog_id INT REFERENCES dogs(id),
|
||||
legacy_entry_id TEXT,
|
||||
legacy_pdf_url TEXT,
|
||||
signature_url TEXT,
|
||||
vet_name TEXT, vet_address TEXT, vet_phone TEXT, vet_phone_raw TEXT,
|
||||
emergency_contact_name TEXT, emergency_contact_phone TEXT,
|
||||
health JSONB, -- vaccinated / allergies / diet / medication
|
||||
behaviour JSONB, -- recall / reactivity / car / commands
|
||||
misc JSONB, -- additional notes / referrals / socials
|
||||
signed_on DATE,
|
||||
submitted_at TIMESTAMPTZ
|
||||
);
|
||||
|
||||
contracts (
|
||||
id SERIAL PRIMARY KEY,
|
||||
client_email TEXT REFERENCES clients(email),
|
||||
legacy_entry_id TEXT,
|
||||
signature_url TEXT,
|
||||
pdf_url TEXT,
|
||||
consent_text TEXT,
|
||||
signed_on DATE
|
||||
);
|
||||
```
|
||||
|
||||
`health` / `behaviour` / `misc` stay as `JSONB` rather than exploding every
|
||||
nullable yes/no into a column — it matches the existing
|
||||
`submissionSnapshot: dict[str, Any]` shape in
|
||||
`mail-api/mail_api/models.py` and makes the form easier to evolve.
|
||||
|
||||
## Things that intentionally don't exist yet
|
||||
|
||||
- **Hard delete** of a client. Use `archived` instead. Hard delete would also
|
||||
break the activity feed and the legacy import provenance.
|
||||
- **A "former client" notes log.** If we need anything richer than a 500-char
|
||||
reason, the next step is to add a `client_notes` table keyed by email — but
|
||||
hold off until the JSON → Postgres migration is done.
|
||||
- **Re-engagement / win-back emails.** The data is there (lifecycle history +
|
||||
email), so this is unblocked whenever marketing wants to send a newsletter
|
||||
to past clients. `cancelled` and `archived` rows are still in
|
||||
`/owner/all-clients`.
|
||||
@@ -0,0 +1,277 @@
|
||||
"Dog Name","Dog Surname","Dog's date of birth","Owner Name","Owner Surname","Owner Contact","Emergency Contact Name","Emergency Contact Number","Vet Name","Vet Address","Vet Contact Number","Is your dog vaccinated?","Does your dog have any food allergies?","Specify","Does your dog have any environmental allergy?","Specify","Is your dog on a special diet?","Specify","Is your dog taking any medication that could put him at risk during a walk","Specify","Is your dog well socialised?","How many dogs does your dog interact with weekly (excluding your family dogs)","Does your dog visit the beach?","Does your dog visit dog parks frequently - How many times a week?","Does your dog have a bite history?","Is your dog reactive to other dogs?","Is your dog reactive to other animals?","Is your dog reactive to children?","Is your dog desexed?","Is your dog registered?","Is your dog reactive to other people?","Is your dog leash trained?","Rate your dog's recall from 1 to 5, with one being the lowest score and 5 the highest.","Has your dog ran away before?","Please describe your dog's behaviour in the car","List of commands your dog understands","Anything else you'd like to let us know?","Social Media Account Name","How did you hear about Goodwalk?","Person's name who reffered Goodwalk to you","Signature","Date","Created By (User Id)","Entry Id","Entry Date","Date Updated","Source Url","Transaction Id","Payment Amount","Payment Date","Payment Status","Post Id","User Agent","User IP","PDF: DogOnboardingForm"
|
||||
"Maisie","McCabe","2015-04-28","Shelley","McCabe","021328907","Finn McCabe","02040319829","Auckland Pet Hospital","46 - 48 Pollen Street","09 360 0961","Yes","No","","No","","No","","No","","Yes","1-5","Yes, she loves the beach. Likes being in the water but not keen going in past top of her legs.","7 days per week","No","No","Cats","No","Yes","Yes","Yes","Yes","4","No","Very comfortable in the car. Does not need to be harnessed.","Sit, stay, stop, paw, walkies, come on, let's go for a walk","Nothing in particular","Maisie","Someone reffered me","Brigid McLean","https://onboarding.goodwalk.co.nz/index.php?gf-signature=69b4d28c1f80d3.92082912&form-id=4&field-id=50&hash=e252f70aca9398464a6ef16f21fdc18f4866fd6ba4804d4868373223f945e144","2026-03-14","","140","2026-03-14 16:14:20","2026-03-14 03:14:20","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/145.0.0.0 Safari/537.36 Edg/145.0.0.0","104.23.198.146","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/140/"
|
||||
"Pipi","Steuart","2013-04-13","Ellen","Graney","021740674","Patrick Steuart","0272235853","Royal Oak Vet care","649 Mt Albert Rd, Royal Oak","09 6259729","Yes","No","","No","","No","","No","","Yes","1-5","Yes","Weekly","No","Yes","Other","No","Yes","Yes","No","Yes","3","No","Settles down quickly and sleeps","Come, wait, stay, sit, down","",".","Someone reffered me","Paula Smith","https://onboarding.goodwalk.co.nz/index.php?gf-signature=699ce3df243213.09800906&form-id=4&field-id=50&hash=e84ff9c458b76d71f18cc977c19d0e45ce9866691863fe72b2e4c1b07b055a8a","2026-02-24","","139","2026-02-24 12:33:51","2026-02-23 23:33:51","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 18_7 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/26.2 Mobile/15E148 Safari/604.1","122.57.112.233","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/139/"
|
||||
"Poppy","Nam","2024-09-23","Yoori","Nam","021777306","Lily","0274880031","Auckland Pet Hospital","46/ pollen Street, Gray Lynn","093600961","Yes","No","","No","","No","","No","","No","1-5","Yes","twice a week","No","Yes","Cats","Yes","Yes","Yes","Yes","Yes","3","No","she is so quiet in the car","sit
|
||||
wait
|
||||
stay
|
||||
shake hands
|
||||
spin
|
||||
hi5","She loves her family and thrives on routine.
|
||||
she is stubborn,but once she gets used to someone, she becomes incredibly affectionate and friendly.
|
||||
she is very wary of other dogs, but she would never attack- she only barks.
|
||||
Honestly, barking is her only real issue.","Kiwiyuuri (instagram)","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=699cde43dc7b26.23456486&form-id=4&field-id=50&hash=553864499410acd217be476f35b4a07ba0269a1f74eba002932b6c642e366926","2026-02-24","","138","2026-02-24 12:09:55","2026-02-23 23:09:55","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.4.1 Safari/605.1.15","222.154.51.238","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/138/"
|
||||
"Ollie","Butt","2025-08-26","Maya","Steeper","0211148194","David Butt","021 722256","Balmoral Vets","482 Dominon Rd","6307168","Yes","No","","No","","No","","No","","Yes","1-5","yes","Not started visiting dog parks yet","No","No","Cats","No","No","Yes","No","Yes","5","No","He is good on car rides, normally buckled in with harness and curls up on seat","sit, down, stay, wait","","mayabutt on facebook","Someone reffered me","you did!","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6982533a56dfc8.61964352&form-id=4&field-id=50&hash=d34bd458652da66ae6c0ed90863ba454afac74fbb985f74fe5704fc8f237d297","2026-02-04","","137","2026-02-04 08:57:46","2026-02-03 19:57:46","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/144.0.0.0 Safari/537.36 Edg/144.0.0.0","223.165.69.9","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/137/"
|
||||
"Izzy","Gonzales -Keen","2023-01-02","Veronica","Gonzales","0225402906","Veronica Gonzales","0225402906","The Good Vet","726 New North Road Mount Albert Auckland, 1025","+64 27 297 5506","Yes","No","","Yes","Itchy paws during summer time, just needs a rinse after being in grass after walks","No","","No","","Yes","1-5","Onehunga reserve every morning and does go to the beach sometimes but does not like the water","Yes 4 times a week at least","No","No","Cats","Yes","Yes","Yes","No","Yes","4","No","Loves head out the window, will bark if people come close to car .","Sit, come, leave it, wait , no, yes, watch me, look.","As discussed Izzy is an anxious dog and can get overstimulated easily . It’s best to start the walk on lead and then after some time to then take her off lead . Has chased children before but has never bit them. However if she is with a pack of dogs it is less likely she will engage in that behavior .","Burritotoezz","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=69632786c6bf78.27345173&form-id=4&field-id=50&hash=8e111587451b694e01972381c28693db94261122e8dbb2cf6e9369f6da0c6b59","2026-01-11","","136","2026-01-11 17:31:02","2026-01-11 04:31:02","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_7_12 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.6.1 Mobile/15E148 Safari/604.1","118.148.194.197","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/136/"
|
||||
"Scotch","Maxwell","2021-04-22","Anna","Maxwell","0274261363","Benedict Casey","+64 27 879 0948","Emma at Auckland Pet Hospital","46-48 pollen street, grey lynn","(09) 360 0961","Yes","No","","No","","No","","No","","Yes","1-5","Yes - loves it. Will chase a stick into the water but otherwise not big on swimming - unless very hot!","He is off leash at the local parks twice a day. In terms of designated “dog parks” this is less frequent - 1 a month","No","No","Cats","No","Yes","Yes","No","Yes","3.5","No","Good. Loves the windows down! Loves being where he can see what’s happening. Will curl up and sleep too. Loves that he is being included on a journey.","This way, come, leave it, touch, a-way, that’s enough","Not reactive but can be bossy with big male dogs - particularly anything that looks remotely like a wolf -(German shepherd)
|
||||
He will protect humans from big dogs (even their owners)
|
||||
This only happens really with me, not my husband.","a.j.maxwell","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6928e3156d4662.20837423&form-id=4&field-id=50&hash=4337603b6b42e20a9146dba43a0efa517a162d8480145db43f396ee2e46825e6","2025-11-28","","135","2025-11-28 12:47:33","2025-11-27 23:47:33","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 18_6_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/142.0.7444.148 Mobile/15E148 Safari/604.1","122.63.71.137","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/135/"
|
||||
"Yuki","Manu","2025-01-25","Sue","Manu","0220781853","Chris Kasper","0272430925","Balmoral Veterinary Clinic","482 Dominion Rd, Mt Eden","09 6307168","Yes","No","","No","","No","","No","","Yes","1-5","He has been once. He loved it","3 -4","No","Yes","Cats","Yes","Yes","Yes","Yes","Yes","1","Yes","Travelling in car is a daily event.
|
||||
Settled in the car","Sit
|
||||
Down
|
||||
Stay
|
||||
Off
|
||||
Jump down
|
||||
Cross now","Training is still work in progress","Suemanu. Instigram","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=68b6d1bb5cfe90.22636463&form-id=4&field-id=50&hash=093613c10d52a1459f56e63db5b76eb830ca733204d9d46f91b253e8ce57b598","2025-09-02","","134","2025-09-02 23:15:07","2025-09-02 11:15:07","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/139.0.0.0 Mobile Safari/537.36","125.237.54.176","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/134/"
|
||||
"Donut","Todoroki","2023-04-02","Masaya","Todoroki","0273425437","Christine Todoroki","0278978907","Pakuranga Vets","7 Johns Lane, Pakuranga, Auckland","09 576 4108","Yes","No","","No","","No","","No","","No","1-5","Only once","5 - 7 (Victoria Park)","No","Yes","Birds","Yes","Yes","Yes","Yes","Yes","2","Yes","Anxious unless sitting on my lap","Sit
|
||||
Down
|
||||
Stay","Donut has only run away when staying at Noya's parent place. Her parents open the door and he snuck through. He has never tried to run away when with us.","@masayadt (instagram)","Someone reffered me","Noya Xing","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6848f4f12abdc9.30092404&form-id=4&field-id=50&hash=06a0bbfe59b9821440e1d4815c6944fb5945c6e9283bd25de97956a10e069285","2025-06-11","","133","2025-06-11 15:16:01","2025-06-11 03:16:01","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/134.0.0.0 Safari/537.36 OPR/119.0.0.0","125.237.229.122","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/133/"
|
||||
"Mimi","Xing","2020-01-20","Noya","Xing","0210751435","Masaya Todoroki","0273425437","Pakuranga Vets","7 Johns Lane, Pakuranga, Auckland 2010","09 576 4108","Yes","No","","No","","No","","No","","No","1-5","Only once, she enjoyed the beach","Never visited a dog park, just general parks like Victoria Park","No","Yes","Cats","No","Yes","Yes","No","Yes","2","No","Sits on the lap or backseat fine, crate is fine, calm. Stays quite still, or naps","Sit
|
||||
Stay
|
||||
Paw/shake
|
||||
No
|
||||
|
||||
Recall is only good with treats on hand, she is very attentive to treats.","Mimi is reactive to other dogs on walks, generally fine when given time to sniff them out and play, will proceed to walk normally along them after. She's wary of larger dogs.","@Masayadt (instagram)","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6848f158f06299.86914798&form-id=4&field-id=50&hash=8c70b1420a73f311da6f6cee5780b282a98a589a32ae0e64edb227cd5143fade","2025-06-11","","132","2025-06-11 15:00:41","2025-06-11 03:00:41","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/137.0.0.0 Safari/537.36","125.237.229.122","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/132/"
|
||||
"Billie Bunny","Osborne","2024-03-27","Niki","Osborne","0211337479","Stacy Hyland","02102820035","Remuera Veterinary Hospital","236 Orakei Road, Remuera, Auckland","09-529-2091","Yes","No","","No","","No","","No","","Yes","5-10","Yes - loves the beach. Takapuna, Okahu & Mission Bay are regular spots","10 x per week","No","Yes","Birds","No","Yes","Yes","No","Yes","4","No","Lays down and sleeps right away.","Sit, Come, Stay, No.","Her reactivity to other dogs isn't constant - some dogs she won't even acknowledge. Others (usually bigger dogs), she'll bark at like she wants to play with them.","nikioznz","Someone reffered me","Met Alessandra walking in the park.","https://onboarding.goodwalk.co.nz/index.php?gf-signature=683e73d44fca21.96072834&form-id=4&field-id=50&hash=a2de4a4ebcdca29e3ccec6be1ab18b9758b710c4dc430e61406db858f0da8a46","2025-06-03","","131","2025-06-03 16:02:28","2025-06-03 04:02:28","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/136.0.0.0 Safari/537.36 Edg/136.0.0.0","222.152.69.84","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/131/"
|
||||
"Teddy or Ted","Uhlenberg","2021-11-01","Clare","Uhlenberg","021 508 767 / 846 1820 wk","Hamish Cook","021 713 216","Pohutukawa Vets Beachlands","Wakelin Road, Beachlands","09 320 1472","Yes","No","","No","","No","","No","","Yes","5-10","Yes","Yes - Daily","No","Yes","Other","Yes","Yes","Yes","Yes","Yes","4","No","Pretty good - Likes his head out the window
|
||||
Will sometimes bark at motorbike riders on the motor way","sit/lay down/come etc","","face book - clare uhlenberg","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=682bf263299c64.69501352&form-id=4&field-id=50&hash=0d28d901c266d7d4ea39a526b07928571af76f882a37249605b59d18508fef13","2025-05-20","","130","2025-05-20 15:09:37","2025-05-20 03:09:37","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/136.0.0.0 Safari/537.36","219.89.199.64","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/130/"
|
||||
"Floyd","Carney","2020-08-21","Kylie","Gladwell","0223771926","Sean Carney","0223509475","Royal Oak Veterinary Care","649 Mount Albert Road, Royal Oak","09 625 9729","Yes","No","","No","","No","","No","","Yes","1-5","Yes, he loves the beach and getting in the water","1-2 times a week","No","No","Cats","No","Yes","Yes","No","Yes","3","Yes","He is pretty good in the car. He can get anxious if we are getting close to where we are going and will bark when we park - it's either excitement or making sure we don't leave him in the car - which we never do! He doesn't get car sick.","Floyd, come.
|
||||
Floyd, sit.
|
||||
Down.
|
||||
Ah a (meaning no)
|
||||
Floyd, drop. (doesn't work all the time, but with consistency it does).","Our main concern about Floyd is his anxiety and barking for attention. We would love to help him to be more calm and reduce his anxiety. We are open to advise and will try and be consistent in our approach - something we haven't done too well in the past.","kyliegladwell","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6819b912bbdad1.02837485&form-id=4&field-id=50&hash=a5f9905ebd0bb4bae9eaff6f0371f549f16198e71c88cd419a9632a4e6ffce8e","2025-05-06","","129","2025-05-06 19:24:02","2025-05-06 07:24:02","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/135.0.0.0 Safari/537.36","101.53.219.205","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/129/"
|
||||
"Mickey","Vu","2024-01-04","Sophie","Vu","02108416304","Sophie","02108416304","Normanby Vet","49 Normanby Road, Mt Eden","096388445","Yes","Yes","Beef and chicken - eye stains","No","","Yes","Raw diet","No","","No","1-5","Yes","Yes, 4-5 times per week","No","No","Other","No","Yes","Yes","No","Yes","4","No","He usually shakes in the car. That’s because of his anxiety.","Sit, down, paw, high five, stay, turn around","Mickey is a timid and shy pup but loves going for a walk and does not like to stay in one place for too long. He might be whining and making noise when going out. And we don't know why he is like that :((","mickeyinauckland","Instagram","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=68193e35e2ca84.35518672&form-id=4&field-id=50&hash=68801ef892bfc3579b3cd48a2058a6627357f0dd810d3f885497c72d67d09b82","2025-05-06","","128","2025-05-06 10:39:49","2025-05-05 22:39:49","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 18_4_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/18.4 Mobile/15E148 Safari/604.1","125.239.151.7","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/128/"
|
||||
"Jaxson","Yardley","2023-12-06","Layne","Yardley","0220145043","Jeremy (and Layne)","0220145045 or 0220145043","Vet North","45 Commercial Road, Helensvilles","09 420 8325","Yes","No","","No","","Yes","Generally on grain free but not fussed for training treats","No","","Yes","1-5","Yes, though doesnt like going in past chest.","Parks yes but not generally dog parks","Yes","Yes","Other","No","Yes","Yes","No","Yes","2","No","Just generally lies down and naps.
|
||||
|
||||
He can sometimes take awhile to get into the car (mooches around like he cannot get into the car physically and needs to be picked up) but is fine travelling.","Sit, Down, Nose, Spot, Come, Out!, Stay (for about 5m radius)","Bite History: Nibbles when excited to see you. Will generally only do so if Tryggr is around and getting overly excited.
|
||||
|
||||
Reactive to other dogs: Only when Tryggr is around he will get excited and want to be part of it.
|
||||
|
||||
Reactive to other animals: Not really, a bit scared of cats. Will jump around hedgehogs but only if Tryggr is going crazy.
|
||||
|
||||
Was diagnosed by Foster and Foster Vet as partially deaf (hearing high high and low low) but does seem to have the full realm of hearing (a bit belligerent!)","laynejburgess","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=67f360cb5d76a6.54558856&form-id=4&field-id=50&hash=e3ad1e680ee905a2d00960cff2f3fa1e6b59ac66eb0f0c7b64782c9159bdae2c","2025-04-07","","127","2025-04-07 17:21:15","2025-04-07 05:21:15","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/135.0.0.0 Safari/537.36 Edg/135.0.0.0","101.100.142.27","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/127/"
|
||||
"Tryggr","Yardley","2024-02-07","Layne","Yardley","0220145043","Jeremy (or Layne)","0220145045 or 0220145043","Vet North","45 Commerical Road, Helensville","09 420 8325","Yes","No","","No","","Yes","Generally we use grain free kibble but not fussed for training treats","No","","Yes","1-5","Yes, locally we take him. He likes to have a swim and fetch sticks.","Not dog parks but does go to various parks and interacts with dogs","No","Yes","Other","No","Yes","Yes","No","Yes","2","No","Generally will just lie down and go to sleep","Sit, Nose, Spot, Come","REactive to other dogs: Gets quite excited and 'whiney' trying to get close.
|
||||
|
||||
Reactive to other animals: Occasionally will chase a bird, not often. Hates hedgehogs and possums.","laynejburgess","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=67f35e0d2188f2.62328554&form-id=4&field-id=50&hash=686985b072ccbbe5eb48ba4ec0383f9293ca6c73643c46e81a38af4a1e6233b4","2025-04-07","","126","2025-04-07 17:09:33","2025-04-07 05:09:33","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/135.0.0.0 Safari/537.36 Edg/135.0.0.0","101.100.142.27","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/126/"
|
||||
"Fergus","Callender","2022-12-28","Geoff","Callender","022 318 8798","Steph Wilson","0210503830","Green","Green Bay Veterinary","098274625","Yes","Yes","Chicken","No","","Yes","Royal Canin DermaCare","No","","Yes","1-5","Yes - loves it","No","No","Yes","Birds","No","Yes","Yes","No","Yes","2","No","He likes looking out the windows and can bark at other dogs if he sees them. Otherwise he'll usually go to sleep.","Close
|
||||
Heel
|
||||
Sit
|
||||
Down
|
||||
Leave it
|
||||
Wait
|
||||
Stay (sometimes works)
|
||||
Inside
|
||||
Outside
|
||||
Off
|
||||
Come (Close works better)","Fergus has struggled with reactivity after he was attacked when he was a pup. He's super sweet, but it's quite anxious so will often bark. We're working in this with a trainer on this.","Steph.wilson or @FergusandMinnie","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=670ada87024785.23084565&form-id=4&field-id=50&hash=fa8a05c8943c81e576dbae15e8cf3fc1523eac2dbf0a3eb97a5c85f051ec68d2","2024-10-13","","125","2024-10-13 09:22:31","2024-10-12 20:22:31","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Android 14; Mobile; rv:131.0) Gecko/131.0 Firefox/131.0","115.188.37.188","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/125/"
|
||||
"Indy","Yang","2019-03-29","Erica","Mancilla","02041931790","James","0220165396","St Lukes Veterinary Centre","13 Morningside Drive, St lukes, Auckland 1025","098455573","Yes","No","","No","","No","","No","","No","1-5","Yes","Once every week","No","Yes","Birds","No","Yes","Yes","No","Yes","1","No","He loves car rides, he usually stays on my lap on passenger or he back in his bed. First half of the trip will be energetic, wants to look out, second half he will rest on my lap. He doesn't vomit, reactive to only motercycles.","Indy
|
||||
Hand
|
||||
Stay
|
||||
Down
|
||||
Up
|
||||
Wait
|
||||
Bad boy
|
||||
Good boy
|
||||
Come, outside, inside (with hand gesture)","Dominant, doesn't share squeaky toys well.
|
||||
He tugs and pulls during walks.
|
||||
Very persistent to where he wants to go.","Erica_tiffany","Instagram","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=670acbb024b3b3.41379744&form-id=4&field-id=50&hash=d156e39fae40efc2f87038bf19e797b7821aafc1784a86d7a3dde0820ea82344","2024-10-13","","124","2024-10-13 08:19:28","2024-10-12 19:19:28","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_2 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.2 Mobile/15E148 Safari/604.1","125.239.44.46","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/124/"
|
||||
"Lara","Macleod","2017-06-01","Chrisy","Macleod","021918814","Chrisy","021918814","Auckland Pet Hospital","Ponsonby","09 360 0961","Yes","No","","No","","No","","No","","Yes","5-10","Yes loves the beach and swimming","Meola Dog park once a fortnight","No","No","Cats","No","Yes","Yes","No","Yes","4","Yes","Happy in the car. WIll often stand up and look around","Sit
|
||||
Stay
|
||||
Bed
|
||||
Come","As Lara's master has just left she may be a little anxious and edgy. She will bark as she hears you on porch and at door. She wags her tail while barking.
|
||||
She is tempted by cats and sometimes barks or strains at leads when she sees one. Sometimes she is ok, she is improving!
|
||||
She does not like German Shepherds as she was attacked by one as a puppy. Sometimes she growls/barks at them.
|
||||
She is very good natured and loves company.","https://www.facebook.com/chrisy.macleod/","Someone reffered me","Fluffy Bums","https://onboarding.goodwalk.co.nz/index.php?gf-signature=66e8abf5687124.26063072&form-id=4&field-id=50&hash=14f89d41ff7e39a7b06af802327b4123894a9c69a960479d35a9cf8407b768f7","2024-09-17","","123","2024-09-17 10:06:45","2024-09-16 22:06:45","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/128.0.0.0 Safari/537.36","101.53.219.202","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/123/"
|
||||
"Bernie","","2021-02-22","Robin","Fryer","0279428816","Mark Fryer","021372109","CareVets Oratia","546 West Coast Road, Oratia, Auckland 0604","09 818 4104","Yes","No","","No","","No","","No","","Yes","5-10","Yes, she loves to run by the water","7","No","Yes","Birds","No","Yes","Yes","No","Yes","3","No","Fine","Sit, stay, come, stop. And some tricks","","ellislabel","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=66e68eec701c57.71195225&form-id=4&field-id=50&hash=004202b47300b83145766765695eed4f791cd79e9a7b78fd77aa333b1ef0b1db","2024-09-15","","122","2024-09-15 19:38:20","2024-09-15 07:38:20","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 17_6 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/128.0.6613.98 Mobile/15E148 Safari/604.1","118.92.99.79","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/122/"
|
||||
"Holly","Millman","2014-04-24","Emma","Millman","0272184446","(Other than me) Chris Millman","027 960 2911","Auckland Pet Hospital","46/48 Pollen Street, Grey Lynn, Auckland 1021","09 360 0961","Yes","No","","No","","No","","No","","Yes","1-5","Loves the beach and chasing sticks in the waves","7","No","Yes","Cats","No","Yes","Yes","No","Yes","5","No","Holly Will get into a scrap if a dog attacks her first or growl at her she will never be the 1st to take action though. In other words she won’t back down from a scuffle.","Understands English so comes to any command","Gets sore if runs too much","Emmaljudge","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=66c7d18060eee9.98659650&form-id=4&field-id=50&hash=1afe6ca11db4f392692f8b4259a6b9096b414cf063866936d4bf53e990b1a491","2024-08-23","","121","2024-08-23 12:02:08","2024-08-23 00:02:08","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 17_5_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.5 Mobile/15E148 Safari/604.1","104.28.29.65","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/121/"
|
||||
"Wilson","Storr","2014-12-07","Ashleigh","Storr","0211089688","Fran Storr","0211663821","The Vets","08 Manukau Road, Epsom, Auckland 1023","09 625 5556","Yes","No","","No","","No","","No","","Yes","5-10","Yes","10","No","No","Cats","No","Yes","Yes","No","Yes","4","No","Anxious","Sit, Stop, Stay, Come, Off, Down (i.e. lay down)","","@ashstorr","Someone reffered me","Rachel","https://onboarding.goodwalk.co.nz/index.php?gf-signature=66c25a44137723.11498121&form-id=4&field-id=50&hash=f6b3c42a3cd4111ebb5335d582fa6c95a559eed9106da9fcd486e3041cf2e100","2024-08-19","","120","2024-08-19 08:32:04","2024-08-18 20:32:04","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/127.0.0.0 Safari/537.36 Edg/127.0.0.0","103.242.69.153","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/120/"
|
||||
"Archie","Menezes","2023-10-01","Conal","Menezes","0210368104","Conal Menezes","0210368104","Balmoral Veterinary Care","482 Dominion Road, Mt Eden","096307168","Yes","No","","No","","No","","No","","Yes","1-5","Not yet, but we intend to in the summer","I used to take him once a week, but I have not taken him recently, as the behaviour of other dogs at our local dog park can be mixed. Often it is good and Archie is happy to be around them. Sometimes it is bad, with other dogs doing things to Archie that he does not enjoy. This is why we are seeking a pack walk with other dogs, to continue his socialisation in a controlled and structured environment.","No","Yes","Birds","Yes","Yes","Yes","Yes","Yes","3","No","Archie loves car rides and is reasonably well behaved in the car. We have a basket that we have tethered to the seat and Archie sits in that basket during car rides. He is tethered to the basket. The tether is attached to his harness that he wears in the car. We use a harness so that if there is an accident, he is not restrained by the neck.
|
||||
|
||||
Usually the command ""In"" will get him into the car and into his basket. This is sometimes not reliable if we are returning home and he would rather stay at the park. When this happens, I pick him up and put him in the car, and he will happily climb into his basket. While driving, he mostly sits in his basket. Sometimes he climbs out to look out the window, but the commands ""In"", ""Sit"" and ""Down"" get him back into the basket and sitting or lying down. If it is a longer ride, he usually curls up and goes to sleep in his basket. He has never thrown up in the car yet.","Sit - Sits down (index finger extended hand signal)
|
||||
Down - Lie's down (point at the ground)
|
||||
Wait - Wait until I give you the next command (hold palm out in a ""stop"" gesture)
|
||||
Stay - Stay where you are until I come back to you (hold fist up). this command is not very reliable.
|
||||
Come - Return to me
|
||||
Squeeze - Return to me, walk around me and stand between my legs","","@kiwi_h2oboy","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=669dacb8823b43.24584415&form-id=4&field-id=50&hash=d81ab401d4486e911d57e88349d1e42e6362a85d61772f3810aedb64d5d6764f","2024-07-22","","119","2024-07-22 12:50:00","2024-07-22 00:50:00","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/126.0.0.0 Safari/537.36","161.29.73.81","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/119/"
|
||||
"Moss","Lane","2021-12-03","Henry","Lane","0221308980","Henry","0221308980","Royal Oak Vet Clinic","649 Mt Albert Road, Royal Oak","09 625 9729","Yes","No","","No","","No","","No","","Yes","1-5","Occasionally","Yes, everyday","No","No","Other","No","Yes","Yes","No","Yes","4","No","Moss is excitable when we knows he is going to the park. He is otherwise content to sleep or look out the windo","Wait
|
||||
Sit
|
||||
OK
|
||||
Here Boy
|
||||
Drop
|
||||
No","","N/A","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=669d7d9015b428.67048631&form-id=4&field-id=50&hash=93da47387a79563fc48ca88069e7d44ba48235d69eedd8f14eb88bf0a7182d63","2024-07-22","","118","2024-07-22 09:28:48","2024-07-21 21:28:48","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/126.0.0.0 Safari/537.36","103.229.249.250","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/118/"
|
||||
"Pumpkin","Anderson","2023-07-07","Laree and Kevin","Anderson","0220 325121","Kevin Anderson","021679196","Balmoral Vet","482 dominion rd","096307168","Yes","No","","No","","No","","No","","Yes","1-5","Yes","1-2","No","No","Cats","Yes","Yes","Yes","Yes","Yes","1","Yes","She's fine in the car","Down, stay, wait. Come.","","Lareeandkevin","Someone reffered me","Anna Shaw","https://onboarding.goodwalk.co.nz/index.php?gf-signature=669c36f5a53f35.55401462&form-id=4&field-id=50&hash=721e0a06da4b27f76a6d582979cb17a8e68c04e3d5ab72c9d170a976cd5dd40f","2024-07-21","","117","2024-07-21 10:15:31","2024-07-20 22:15:31","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/126.0.0.0 Mobile Safari/537.36","125.239.167.102","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/117/"
|
||||
"Zola","Humphrey","2023-03-09","Monique","Humphrey","0212764358","Adam Begg","0211862322","Chris Laurenson","49 Normanby Road, Mt Eden","096388445","Yes","No","","Yes","Vet thinks she has an allergy to a particular grass/weed (so far unidentified) which causes her eyes to go gunky.","No","","No","","Yes","5-10","Yes - on holidays to Whangamata","None","No","No","Cats","Yes","Yes","Yes","No","Yes","5","No","Settled - enjoys looking out the window and sniffing out an open window.","Zola come
|
||||
Sit
|
||||
Down
|
||||
Look
|
||||
Wait
|
||||
Leave
|
||||
Where's Dad?
|
||||
Get (toy) e.g. octopus, duck
|
||||
Relax mat (still working on this one but getting there)","","moniquehumphrey31","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6690c2a50a3550.99332594&form-id=4&field-id=50&hash=d1920dd0952fb767ff4f8f7228cec5c8b4e849befcb03362651148ff941060cb","2024-07-12","","116","2024-07-12 17:44:18","2024-07-12 05:44:18","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/126.0.0.0 Safari/537.36","121.98.13.62","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/116/"
|
||||
"Louis","Shallard","2018-12-08","Annika","Shallard","021 2718707","Dini Ratcliffe","073876878","St Lukes Vet","13 Morningside Drive","098455573","Yes","No","","No","","No","","No","","No","1-5","Yes, on lead as his ears are painted on. He doesn't go into the water doesn't like it.","1 time per week plus goodwalk","No","No","Other","No","Yes","Yes","No","Yes","1","Yes","Louis has wandered.
|
||||
|
||||
Louis is experienced in the car. Enjoys being with you. No issues.","Ad hoc.
|
||||
|
||||
Come. Sit.","Hip and back issues need to be cared for.","AnnikaShallard","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=668b0be2701063.18070877&form-id=4&field-id=50&hash=81d8184fe718f887c40a9f41b29a283ffc538789216a821f190c4d935821c1b2","2024-07-08","","115","2024-07-08 09:42:58","2024-07-07 21:42:58","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) SamsungBrowser/25.0 Chrome/121.0.0.0 Mobile Safari/537.36","203.97.18.134","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/115/"
|
||||
"Moss","Lane","2020-12-03","Henry","Lane","0221308980","Henry Lane","0221308980","Royal Oak Vet Clinic","649 Mt Albert Road, Royal Oak 1024","096259729","Yes","No","","No","","No","","No","","Yes","1-5","Yes","7","No","No","Birds","No","Yes","Yes","No","Yes","4","No","Chilled. Happy to sleep or look out of the window. Is excited and will whine when he knows he is going to the park","“Here boy”
|
||||
“Wait”
|
||||
“OK”
|
||||
“Drop”
|
||||
“Stop”
|
||||
“Down”
|
||||
“Sit”","","Lydia Park (instagram)","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=666e7df19031a2.70237035&form-id=4&field-id=50&hash=19527f1b9dd630922ba8e6fe067c42a1dd2fee1fa2dea728b5e3c3fd6d5e13fe","2024-06-16","","114","2024-06-16 17:53:53","2024-06-16 05:53:53","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 17_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/126.0.6478.54 Mobile/15E148 Safari/604.1","121.73.188.128","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/114/"
|
||||
"Louis","Shallard","2018-12-08","Annika","Shallard","021 2718707","Dini Ratcliffe","0277126554","St Lukes Veterinary Centre","13 Morningside Drive","09 8455573","Yes","No","","No","","No","","No","","Yes","1-5","Yes - on lead, prevent physical over-exertion may affect joints hips","1","No","Yes","Other","No","Yes","Yes","No","Yes","1","No","Relaxed , travels often on the passenger seat without any issue.","Ears seem to be painted on. Louis has had previous training needs to be refreshed.","He can be very excited and also jump when meeting new human and doggy friends. Please avoid due to hip issues.","Annika Shallard","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=665bba626dfb29.88761325&form-id=4&field-id=50&hash=af56bc371f54af34378af3a47137f04fc74178f8e170d2e2ee8008ee27f308f4","2024-06-02","","113","2024-06-02 12:18:42","2024-06-02 00:18:42","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Linux; Android 10; K) AppleWebKit/537.36 (KHTML, like Gecko) SamsungBrowser/24.0 Chrome/117.0.0.0 Mobile Safari/537.36","49.224.89.46","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/113/"
|
||||
"Finn","Stock","2015-01-02","Cara","Telle","0212585984","Lueder Stock","021374459","MangereVet Clinic","95 Coronation Rd, Mangere","096366732","Yes","No","","No","","No","","No","","Yes","5-10","Yes","4 times per week","No","No","Cats","No","Yes","Yes","No","Yes","5","No","Sleeps or looks out the window","Heal, Come, Sit, Be Quick (for the toilet), Roll over, Down, Off, Car, Walk, Leave it, Dinner, Paw","Finn does not like male dogs who have not been desexed on our property","facebook Cara Telle","Instagram","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=664478578c97a7.90730553&form-id=4&field-id=50&hash=ee6fcf843fb7e6f15b4b491a5a27c4a650d32db0fd62e8527fd16fef2d14b6c0","2024-05-15","","112","2024-05-15 20:54:47","2024-05-15 08:54:47","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:125.0) Gecko/20100101 Firefox/125.0","161.29.229.57","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/112/"
|
||||
"Griffin","Taylor","2012-11-01","Sam","Taylor","0204202808","Tate Watson","02040182762","St Lukes Veterinary Centre","13 Morningside Drive, Mount Albert","09 845 5573","Yes","No","","No","","Yes","Dental biscuits","No","","No","1-5","Yes – loves the beach","Yes – less so now that Meola Rd is closed. We used to take him to Coyles Park most mornings off lead and he was fine with other dogs.","No","Yes","Cats","Yes","Yes","Yes","No","No","3.5","No","A bit anxious, a bit excited. He's very, very vocal and usually won't sit still. He's a lot more relaxed after a walk.
|
||||
|
||||
He's getting better in the car, but it's hit and miss. The trick is to make him feel secure in his seat.
|
||||
|
||||
Recently we got him a car seat. Because it has a lip, he feels a lot more secure in it. When we're in the van, Griffin sits between us in his seat and is normally pretty relaxed. Often he'll even lie down. But he rarely does this in other cars.","Sit – but he only really does it if there's food.","More details about the questions above:
|
||||
Griffin is ok with other dogs, he just gets annoyed with young or really active dogs who in his face and won't leave him alone. On walks, he happily goes up to other dogs to say hello and have a sniff and 90% of the time it's absolutely fine, particularly when he's off lead. When I say he's reactive, he always notices other dogs and wants to say hello. He won't just walk by another dog without taking notice.
|
||||
|
||||
He is very reactive to cats, but his eyesight isn't as good as it used to be, so if they're far away he likely won't notice them.
|
||||
|
||||
Small children (toddlers) make him nervous and he has nipped a few in the past. He's never nipped anybody off lead at a dog park, just when he's been stuck in a room with them. Still if there are small kids around, we always put him on a short lead until we're past them. Best not to risk it.
|
||||
|
||||
His recall is ok. Most of the time he comes back straight away, or he waits for you to catch up with him. But if he's having a really good time, he might choose to ignore you...","@taylormayd_ @lowercase.nz","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=65f093d9438b36.22712932&form-id=4&field-id=50&hash=a3cf79a8ca1dada2f2f073c7a9dee3f71c28c1bd42b0085ada1acc9f2d122218","2024-03-13","","111","2024-03-13 06:41:45","2024-03-12 17:41:45","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Safari/537.36","203.211.108.130","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/111/"
|
||||
"Hudson","Beever","2021-08-22","Lee","Beever","021655859","Lee Beever","021655859","Lee Beever","8 Hereford St, Freemans Bay","09 281 5815","Yes","No","","No","","Yes","He eats mushy food standing on a stool because he has megaoesohagus so his oesophagus doesn’t work properly and he will regurg food and cough if he eats dry kibble. He can eat small amounts of kibble for training, but not too much as it will take quite a while before it moves down to his stomach","No","","Yes","5-10","Yes. He does get ear infections if he puts his head under the water too much.","No","No","No","Birds","No","Yes","Yes","No","Yes","2","No","Travels well. No issues. Either looks out window or lies down.","Sit. Down. Come. Leave. Wait.","We’ve been working on his leash walking to try limit pulling - it’s currently a work in progress.
|
||||
His recall is much better if you have something he wants like a ball or treats.
|
||||
Regarding dry kibble treats - please see comment above.","Lbeever (instagram)","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=65e3d62dc175c1.20462780&form-id=4&field-id=50&hash=df70dd78431d6d977897a60098a1d3f2a2f27732752abfe0395cc3136b933c70","2024-03-03","","110","2024-03-03 14:45:17","2024-03-03 01:45:17","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 17_3_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.3.1 Mobile/15E148 Safari/604.1","115.189.90.224","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/110/"
|
||||
"Ruru","Short","2016-04-26","Emily","Short","0274124402","Emily Short","0274124402","Lee Beever","8 Hereford Street, Freemans Bay","09 281 5815","Yes","No","","No","","Yes","Hills sensitive stomach and skin","No","","No","1-5","Yes - will run out to water edge and just dip her chest into the water to cool down","None","No","Yes","Other","No","Yes","Yes","No","Yes","4","No","Fine in the car. Usually travels in the back of RAV4 behind a dog barrier.","Come. Sit. Lie down. Stay. Leave. Wait.","Ru is a very vocal Beardie, so barks and makes noises often. Usually this is due to excitement. She often will bark when she sees other dogs and would run towards them if she is off lead. Usually when she gets close she stops barking and they sniff each other and that’s fine. If she’s on lead I often lead her away from other dogs because her barking is loud. She does not growl and has never had a fight with another dog.
|
||||
She is a really sweet girl and I often just walk her through neighbourhood streets rather than going to dog parks because I think she finds that more calm, rather than interacting with lots of random dogs.","Emily Short","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=65e3d352cb6797.26378477&form-id=4&field-id=50&hash=9437cae00563f2b47658bb71143dbf7e35d18c2f6bc70ff7cf4f2d42f93ba5fb","2024-03-03","","109","2024-03-03 14:33:06","2024-03-03 01:33:06","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 17_3_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.3.1 Mobile/15E148 Safari/604.1","115.189.90.224","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/109/"
|
||||
"Louis","Hall","2022-03-21","Rachel","Hall","021777441","Peter Hall","0274222118","Pt Chev Vet","67 Pt Chevalier Road","098150696","Yes","No","","No","","No","","No","","Yes","5-10","Yes, he has been to Thorne Bay and Milford Beach","The Domain (1-2 times per week), Parnell Rose Gardens (1-2 times per week), Myers Park (daily), Albert Park (3-5 times week), Tahaki Reserve (Mt Eden) once a fortnight","No","No","Cats","Yes","Yes","Yes","No","Yes","4","Yes","Usually curls up and goes to sleep once going at 50km/h","Come, sit, wait (for when he sits and you move away and then call him), stay (for when he stays on the spot until you return to him), down, relax (down with back legs tucked to one side),stop (e.g. at road), leave it (not 100% if something irresistible like chicken bones), heel, gentle, up and over (to jump over low fence)","He is ok with children so long as they do not squeal and run up to him fast and then reach over his head. If a child does this he will run away from them or sometimes bark.","_rachel_hall_ (instagram)","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=65bdd6163dc8f5.22341806&form-id=4&field-id=50&hash=de7c4489d6bcdc1bbb0ad3c389a943a9402be5abaa8052a179a8a7ba533b1c14","2024-02-03","","107","2024-02-03 18:58:46","2024-02-03 05:58:46","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.6 Safari/605.1.15","115.189.128.39","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/107/"
|
||||
"Tinks","Ewen","2021-01-31","sue","Ewen","0274 524722","sue ewen","+64274524722","The Vets","608 Manukau Rd, Greenwoods Corner, Epsom","+64 9 625 5556","Yes","No","","No","","No","","No","","Yes","5-10","Yes","Every day","No","No","Cats","No","Yes","Yes","No","Yes","2","No","Generally quiet but does bark at dogs on the footpath","Sit
|
||||
Down
|
||||
Leave
|
||||
Heel","Interested in birds and cats - I say ""leave""
|
||||
Uses hand signals up - means sit
|
||||
hands pushing down means drop","i.paua.pearl","Someone reffered me","I met Aless in the park","https://onboarding.goodwalk.co.nz/index.php?gf-signature=65b0c5573d2637.72381668&form-id=4&field-id=50&hash=cef171b38a8f109e90675acc0c7b91c491e0ba4dd066928af98fd905af6a693f","2024-01-24","","105","2024-01-24 21:07:51","2024-01-24 08:07:51","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36 Edg/120.0.0.0","125.237.63.27","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/105/"
|
||||
"Honey Crumbles","Shaw","2020-11-08","Anna","Shaw","02102970532","Andrew SHaw","00275237374","St Lukes Veterinary Cllinic","13 Morningside Drive, St Lukes, Auckland 1025","098455573","Yes","No","","Yes","Nothiing major but gets a bit itchy, not sure what triggers it. It's managed well.","No","","No","","Yes","1-5","Yes, not often","No","No","No","Birds","No","Yes","Yes","No","Yes","3","Yes","Honey is content to be in the car. We have never had any issues. She does like to see what is going on, but that is a preference, not an issue.","Sit, Down, Stay (sometimes!) Up (for jump up), Come. Also sometimes ""no barking!!!'","No I think we covered verything in the meet and greet. Also please note that in the question about running away, Honey did escape a few times and has run but this is when she was a puppy. NOw if she gets out she just comes back to the front door and waits to be let in.","@themignonette","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=65459e30a3bcc8.74105744&form-id=4&field-id=50&hash=ebf8c00d85a89599214e6b0dddc91fc01e498f8170c87418ad4ed1f2aec0a6f2","2023-11-04","","103","2023-11-04 14:28:50","2023-11-04 01:28:50","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/118.0.0.0 Safari/537.36","123.255.41.22","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/103/"
|
||||
"Hank","O'Donnell","2021-05-14","Shelly","Haggas","0210425095","Zane O'Donnell","021734734","Your Mobile Vet","80d Main Road, Kumeu. 0810","0275590889","Yes","No","","No","","No","","No","","Yes","1-5","We visit Cornwall beach most weekends.","Not too frequently, maybe once every few weeks.","No","No","Birds","Yes","No","Yes","Yes","Yes","4","No","Can be quite unsettled on short journeys. But sometimes very settled, it's sporadic.","Sit. Touch. Down. Get in. Whistles. Middle. Wait","","@hankthekiwivizsla","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=653eb518c7bde8.66228884&form-id=4&field-id=50&hash=b9f295f1f69bdcdf23e3ff0546fec4d7abb924ec31a9b3a1c937e2cdd6c8b61d","2023-10-30","","101","2023-10-30 08:40:08","2023-10-29 19:40:08","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/118.0.0.0 Safari/537.36","203.118.148.26","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/101/"
|
||||
"Molly","Hickey","2014-12-11","Danielle","Barclay","0273972112","Matthew Hickey","021851489","Balmoral Vets","Dominion road","tel:+6496307168","Yes","No","","No","","No","","No","","Yes","5-10","Yes, loves Pt Chev, Takapuna in particular","Yes, 4 to 5 times a week. Monte Cecelia, big king","No","No","Cats","No","Yes","Yes","No","Yes","3","No","Molly is good in the car, usually sleeps","Molly come
|
||||
Wait
|
||||
Sit
|
||||
Down
|
||||
Shake","She likes to bark with excitement but quickly settles down","Danielle Barclay","Someone reffered me","Tiny Paws","https://onboarding.goodwalk.co.nz/index.php?gf-signature=650c9ec862e7f9.75096945&form-id=4&field-id=50&hash=c53404ed82e4c1856d770489779edd316d19d65155c9f202b2330f660212cf39","2023-09-22","","99","2023-09-22 07:51:45","2023-09-21 19:51:45","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_6 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/117.0.5938.108 Mobile/15E148 Safari/604.1","121.98.37.166","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/99/"
|
||||
"Otis","Batchelor","2022-02-11","Ross","Batchelor","021799704","Rosemary Batchelor","0274956005","Abbotts Way Veterinary Clinic","199 Abbotts Way, Remuera, Auckland 1050","09 524 8361","Yes","Yes","Possibly allergic to beef products.","No","","Yes","Strictly eats Acana Singles Free Run Duck biscuits only, at least until December 2023","No","","Yes","5-10","Yes, he loves swimming","Yes, every day","No","No","Cats","No","Yes","Yes","No","Yes","3","No","Very familiar with car travel, he usually rides in the boot of the car or sometimes on the front passenger seat with a seatbelt/collar clip.","Come, Wait, Sit, Toilet, Inside, Up, Cushion (bed)","","rossinopecorino (Instagram)","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6503b7198a9f77.22408968&form-id=4&field-id=50&hash=2573f783f06a20dec6ce6982c6def8f7e3e0b883a564f8ef88a93aa3efc5cdf6","2023-09-15","","97","2023-09-15 13:44:57","2023-09-15 01:44:57","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_6_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.6 Mobile/15E148 Safari/604.1","172.225.244.185","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/97/"
|
||||
"Ollie","Buckley","2017-09-27","Ann","Buckley","0274479293","Emily Peterson","0273407725","Grey Lynn Vet (Vetcare)","46-48 Pollen St","09 3600961","Yes","No","","No","","No","","No","","Yes","5-10","Yes","4-5","No","No","Cats","No","Yes","Yes","Yes","Yes","4","No","Ollie is a little reactive and protective at home and in the car - he will bark at especially cyclists","Wait, Stay, Come, Sit, Down, Leave, Walk with me (walk on leash beside me politely), High five, shake","No","None although I am on facebook","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=64bf5e8263bfa4.37532663&form-id=4&field-id=50&hash=108323a32dc92389b36f35f3f2b62fabb488d1cf4e9c9c142813d0fa42cca7f4","2023-07-27","","95","2023-07-25 17:32:50","2023-07-25 05:32:50","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Safari/537.36","210.246.17.164","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/95/"
|
||||
"Zola","Humphrey","2023-03-09","Monique","Humphrey","0212764358","Adam Begg","0211862322","Normanby Road Vets","49 Normanby Road, Mt Eden, 1024","096388445","Yes","No","","No","","No","","No","","Yes","1-5","No not yet but will in the future","No","No","No","Cats","No","No","Yes","Yes","Yes","2","Yes","A little anxious at the start and then settles down and curls up and rests.","Pup, pup, pup (for come)
|
||||
Zola
|
||||
Sit
|
||||
Close
|
||||
Down","","Monique Humphrey","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=648f8b0814a3c0.02837457&form-id=4&field-id=50&hash=97010e06e3b2cb0e650c3505b319534859c56305e935cce4ff94dfe30c316c04","2023-06-19","","93","2023-06-19 10:54:00","2023-06-18 22:54:00","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.2 Safari/605.1.15","121.98.13.62","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/93/"
|
||||
"Sunny","Lloyd","2021-09-03","Ruby","Tautuhi-Lloyd","02041461878","Kanoa Lloyd","021657788","Auckland Pet Hospital","46/48 Pollen Street, Grey Lynn","093600961","Yes","No","","No","","Yes","Royal canin digestion + supplements","No","","Yes","5-10","Yes","3-4","No","No","Cats","No","Yes","Yes","Yes","Yes","4","No","Calm and happy. May want to look out the window.","Sit, come, wait","Sunny has chronic rhianitus and has some breathing issues on and off. May flare up and be snotty from time to time. She is a little overweight at the moment and not as fit as usual","?","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=647f8d98a32e44.61071603&form-id=4&field-id=50&hash=6460be1bcbd773cbb5b389c7a1d9134b98815929b59b4582918f2b7a38b5c3d4","2023-06-07","","90","2023-06-07 07:48:40","2023-06-06 19:48:40","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/113.0.5672.121 Mobile/15E148 Safari/604.1","101.100.129.22","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/90/"
|
||||
"Donny","Kenny","2020-12-03","Megan and Adrian","Kenny","Meg 021676959 Adrian 021828902","Megan kenny","021676959","Normanby road vets","49 Normanby road","096388445","Yes","Yes","No red meat","No","","Yes","No raw food only biscuits","No","","Yes","1-5","Yes every weekend","Yes with his day carer 4 times a week","No","No","Other","No","Yes","Yes","No","Yes","3","No","He sits in his car seat or on the front seat he travels to Pauanui every week which is a three hour drive there and back very used to travelling in the car","Come Donny
|
||||
|
||||
Sit
|
||||
|
||||
Down
|
||||
Shake hand","If cold he needs a jumper or jacket
|
||||
If more than one dog runs up to him last he will definitely shy away
|
||||
We try and keep an eye on what he eats on the ground if he is off the lead as he can tend to eat extremely unhealthy things which could make him sick","meg.kenny","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6476ee0f7f6963.08004489&form-id=4&field-id=50&hash=b9bc50afd19e6222c4993f4929950f6d2484bcbfed5ed8e9ec6171b641a711c0","2023-05-31","","88","2023-05-31 18:49:51","2023-05-31 06:49:51","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/113.0.5672.121 Mobile/15E148 Safari/604.1","115.188.70.175","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/88/"
|
||||
"Maggie","Read","2021-10-20","Harriet","Read","0212340063","Sue read","02102446843","Glendowie Vets","Riddell Road st Heliers","09 575 7688","Yes","No","","No","","No","","No","","Yes","1-5","Yes","Yes 3","No","No","Birds","No","Yes","Yes","Yes","Yes","4","No","Very happy normally sleeps","Come sit stay","","No socials","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6465739a39fc42.86875372&form-id=4&field-id=50&hash=93533b1bd1928f98414fc8aafb44986fdcf214ecadb129f698b28859d1148a92","2023-05-18","","87","2023-05-18 12:38:50","2023-05-18 00:38:50","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/113.0.5672.69 Mobile/15E148 Safari/604.1","203.211.107.249","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/87/"
|
||||
"Monty","Rewiri","2020-11-15","Estelle","Rewiri","0275194228","Nick McGrath","+61 433 864 156","Massey Vets","256 Don Buck Road, Massey","09 832 4895","Yes","No","","No","","No","","No","","Yes","5-10","yes","4","No","No","Cats","No","Yes","Yes","Yes","No","3","No","Whines in the car mostly when the windows are up / reversing","Sit
|
||||
Stay
|
||||
Wait (food)
|
||||
Handshake
|
||||
Bonk
|
||||
Come
|
||||
Turn
|
||||
|
||||
Maori commands..","Doesn’t like his paws being touched / isn’t the biggest fan of men / high vis clothing","Mischief_ Monty","Instagram","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=645c1e471090d2.61524914&form-id=4&field-id=50&hash=2ae19a9423580c651cdf1e2d0070a8bd595fe64ef5daf41389b747997849d4fb","2023-05-11","","84","2023-05-11 10:44:31","2023-05-10 22:44:31","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 15_6_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.6.1 Mobile/15E148 Safari/604.1","115.189.88.240","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/84/"
|
||||
"Luna","Matthews","2020-04-16","Carl","Matthews","021737727","Vanessa Matthews","021 380 334","Chris at Normandy Rd Vet","49 Normanby Road, Mount Eden,","09 638 8445","Yes","No","","No","","No","","No","","Yes","5-10","Yes she loves the beach","2-3 times","No","No","Birds","Yes","Yes","Yes","Yes","Yes","4","Yes","She travels a lot in the car. Seems to prefer the boot as she can lean better.","Sit, stay, heel (sometimes), come, no. Good Girl (if she has responded to a command)","I put she is reactive to children and other people but only because she is very social and wants to play. She is never aggressive and will very rarely bark.
|
||||
About 3 times a week we go to the local school (Kowhai) and there is often up to 12 of her dog friends there to play with.
|
||||
|
||||
She has run away before when we left the door open but didn't go far. She does have our phone numbers on the collar.
|
||||
|
||||
If we are at a dog park she has never run away or gone to far from us and will come back when called.
|
||||
|
||||
Loves treats!","I don't have any social media.","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=645aa1326e9497.56635043&form-id=4&field-id=50&hash=01947a1ded47ed55e41ee9b5190fb28a25759adecebf22c082dff38df73eef9a","2023-05-10","","83","2023-05-10 07:38:26","2023-05-09 19:38:26","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/113.0.0.0 Safari/537.36","121.99.240.46","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/83/"
|
||||
"Sweep","","2013-12-17","Todd","Benton","021482670","Jose Rodriguez (co-owner)","021846685","Auckland Pet Hospital","46-48 Pollen Street, Grey Lynn, Auckland 1021","09 360 0961","Yes","Yes","Allergic to many different types of fish - please don't give fish based treats / food","Yes","Hayfever (very typical of Spanish water dogs) - he can get quite sneezy and rub his eyes but don't worry too much about it, I'm just telling you because I don't want you to worry too much about it","No","","No","","Yes","5-10","Yes - not often but he likes it (he will chase ducks etc)","14 minimum","No","Yes","Cats","No","Yes","Yes","No","Yes","4","No","He's very good in the car - he has travelled a lot (between UK & Spain and around Spain many times so very used to it). He does like to get out of the car after about 2.5 hours and stretch his legs but I don't think he will be with you in the car for longer than 2.5 hours continuously!","Sweep gets spoken to at home in both English and Spanish - English is fine but Spanish is good if you have any Spanish speaking staff! There's probably more, but the following are the key one's you are likely to use / need:
|
||||
|
||||
Sit - Sentate
|
||||
Come - Ven / ven aqui
|
||||
Cross (to cross the street) - Cruza
|
||||
Wait (if he wants to cross the street and it's not safe) - Espera
|
||||
Let's go - Vamos","I couldn't select above that he will react to both cats and birds - he will chase birds as well but it isn't a severe problem (the cats are worse). To add, when we walk around on the street, I always have him on a lead as if he sees a cat he will go straight after it and won't wait for any traffic (off-lead in the park, beach or other similar safe place is fine but not on the street).
|
||||
|
||||
As said on the phone, he doesn't like dogs jumping on him and isn't super good with small dogs that behave very aggressively ('small dog syndrome')toward him, as he was bitten by a small dog when he was younger.
|
||||
|
||||
As mentioned, he does have arthritis (which he does have medication for) - walking is fine but we try to avoid higher impact activities (chasing balls etc - he does love chasing balls so we need to keep an eye on him and make sure he doesn't go rushing off after other dog's balls in the park).","Let me get back to you on this (will email you) - Jose uses social media more than me","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=64570e76700919.28264836&form-id=4&field-id=50&hash=f161774d4c291638db6211700c14419473a773e4d625e02777d5b15517d5b40d","2023-05-07","","80","2023-05-07 14:35:34","2023-05-07 02:35:34","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/112.0.0.0 Safari/537.36 OPR/98.0.0.0","121.99.103.51","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/80/"
|
||||
"Bob","Smiyh","2021-06-01","Paula","Smith","021979995","Tony Browne","021505006","Balmoral Vets","Balmoral Road","+64 9 630 7168","Yes","No","","No","","No","","No","","Yes","1-5","He has been a couple of times","2 parks a week","No","No","Birds","No","Yes","Yes","Yes","Yes","4","No","Anxious- Whines and high pitched bark","Sit
|
||||
Stay
|
||||
Come
|
||||
No","","Smithy007","Someone reffered me","Joni Liggins - owner of Basil a cavoodle","https://onboarding.goodwalk.co.nz/index.php?gf-signature=64549b66b5eb99.40000704&form-id=4&field-id=50&hash=6787581b9df2e5d49139bdab8ac6db043ce11b3035c4d0f51d87b4185a1b447e","2023-05-05","","78","2023-05-05 18:00:06","2023-05-05 06:00:06","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_4_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.4 Mobile/15E148 Safari/604.1","125.237.205.159","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/78/"
|
||||
"Test","test","1986-10-04","Test","Test","t","test","test","test","test","test","Yes","No","","No","","No","","No","","No","1-5","1","1","No","No","Cats","No","No","Yes","Yes","Yes","5","Yes","test","test","test","test","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6454406fe18cc4.20548864&form-id=4&field-id=50&hash=54ae4c519482f955b5b884c1b7f901ea40c6a3e8a83694cfd7eec842c314e854","1986-10-04","1","77","2023-05-05 11:31:59","2023-05-04 23:31:59","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/112.0.0.0 Safari/537.36","101.98.31.253","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/77/"
|
||||
"Fergie","Taylor","2020-10-06","Caroline","Logan","0272217468","Matt Taylor","0272175951","Auckland Pet Hospital","Pollen Street, Ponsonby","093600961","Yes","No","","No","","No","","No","","Yes","5-10","Yes - she loves the beach","4","No","Yes","Cats","No","Yes","Yes","Yes","Yes","4","No","She is well behaved. She likes to look out the window and gets very excited when she realises where she is going.","Come, heel, leave it, down, sit, Fergie","","@carriestrells","Facebook","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=645430414a9eb8.15054547&form-id=4&field-id=50&hash=7c9cb76001b8ba9925f6d81aacd877fe600e3d15b5dbdd73721d00b6d164c538","2023-05-05","","76","2023-05-05 10:22:57","2023-05-04 22:22:57","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_3_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.3 Mobile/15E148 Safari/604.1","122.56.203.158","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/76/"
|
||||
"Buddy","Reynolds","2022-07-13","Lori","Reynolds","0223109861","Sarah Rothwell","021 998 531","Belmont vets","2 Egremont St Belmont Auckland 0622 New Zealand","09446 1155","Yes","No","","No","","No","","No","","Yes","5-10","Yes, weekend walks a few times a month.","Yes, approx 2","No","No","Birds","No","Yes","Yes","No","Yes","3","Yes","Good in the car and frequently rides with me. Loves to look out the window, stirs at times.","Sit and paw","","Instagram loriannreynolds","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6448d7241bccb3.97264724&form-id=4&field-id=50&hash=1f1318ecdf264abe17122ed5d4aa39bd7ff0885a9fce5abcb7ba3aa0b819a887","2023-05-26","","74","2023-04-26 19:48:02","2023-04-26 07:48:02","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_3_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.3 Mobile/15E148 Safari/604.1","172.225.244.185","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/74/"
|
||||
"Baylee","Curac","2022-04-24","Tony","Curac","0274432085","Maxine Curac","+64212611082","The Strand Vet","Textile Centre Kenwyn Street Parnell","09 3776667","Yes","No","","No","","No","","No","","No","1-5","Yes","2 to 5","No","No","Birds","Yes","Yes","Yes","Yes","Yes","3","Yes","Likes to look out the window, sleeps after a while","Sit, come, stay shake, no","We need some help with training as she is a constant chewer and her recall needs improving","Max and Muppy - Insta","Someone reffered me","Johnathan","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6448b179106070.98351970&form-id=4&field-id=50&hash=30380f4859624003ebd68f2cb529d47daef907a96df05cd3559e47f879599e89","2023-04-26","","73","2023-04-26 17:07:05","2023-04-26 05:07:05","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/112.0.0.0 Safari/537.36","115.188.146.39","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/73/"
|
||||
"Jake","Korucu","2021-12-01","Stephanie","Korucu","021989369","Murat Korucu","021975369","Pt Chev Veterinary Clinic","67 Point Chevalier Road, Point Chevalier, Auckland 1022","09 815 0696","Yes","No","","No","","No","","No","","Yes","5-10","Yes, he loves the beach and the water, but doesn't actually swim","Twice a day, most days","No","No","Cats","No","Yes","Yes","No","Yes","3","No","Excitable on the way out, calm on the way home! Generally good.","Come, sit, sit nicely, wait , this way (understands and generally obeys)
|
||||
Leave it, get down, stay (understands but variable as to whether he obeys)","Jake loves being off-leash for his morning walk - he enjoys running up and down the sides of the hills (Owairaka Domain is our usual). He knows a lot of the regular dog on the hill and there are several he meets and plays with on a regular basis. With some of his friends there is mutual fun in some fairly rough-looking wrestling play (Especially in the mornings you might encounter Sadie -yellow lab, Freddie -black boxer-labX, Miso -large yellow lab or Pearl -grey staffie - all of them love to play like this).","Steph Korucu (FB and Instagram)","Facebook","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=641d262642c5d7.69182113&form-id=4&field-id=50&hash=dd409536be64dcc6789788c2495fa916bffc40c9ea665d78354acad174a4b183","2023-03-24","","64","2023-03-24 17:25:10","2023-03-24 04:25:10","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/111.0.0.0 Safari/537.36","198.41.238.111","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/64/"
|
||||
"Archie","Dine","2013-01-02","Penny","Dine","0225280966","Jessica","0225280966","Cambridge Vet","41 EMPIRE STREET","0800226384","Yes","No","","No","","No","","No","","Yes","1-5","Yes","5","No","No","Other","No","Yes","Yes","Yes","Yes","4","No","Very calm just sits there","come","","same as FLo / Mine","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63ebdb3cddbc13.42446379&form-id=4&field-id=50&hash=71bee8bcc5ac73a3f57eabccf7f82d3c78b757ba7627a8e8d7c647c5012fff92","2023-02-15","","61","2023-02-15 08:04:28","2023-02-14 19:04:28","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/109.0.0.0 Safari/537.36","172.68.66.78","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/61/"
|
||||
"Freddie","Sullivan","2019-12-01","Monique","Sullivan","0275306046","Mark Sullivan","0211639997","Balmoral Vets","482 Dominion Rd, Mt Eden","630 7168","Yes","Yes","Dairy","Yes","Grass","Yes","Sensitive skin diet but can have treats","No","","Yes","5-10","Yes","3-4","No","No","Cats","No","Yes","Yes","No","Yes","4","No","Loves being in the car","Sit, down, come","Please keep Freddie off long grass that has gone to seed. He has needed two surgeries now to remove these seeds and we would like to try and prevent this happening again.","Instagram minksullivan","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63e43156c3cca5.95225021&form-id=4&field-id=50&hash=3bb68d2fa182db969b56928cc459ed91f4584c28a8f9af5cfecaa29741902768","2023-02-09","","59","2023-02-09 12:34:06","2023-02-08 23:34:06","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/110.0.0.0 Safari/537.36","198.41.238.110","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/59/"
|
||||
"Digby","Langdon-Baird","2021-11-21","Kate","Langdon","021411488","Sophie Baird","021931977","Auckland Pet Hospital","46-48 Pollen Street, Ponsonby","09 360 0961","Yes","No","","No","","No","","No","","Yes","1-5","Yes","7 times per week","No","No","Birds","No","Yes","Yes","No","Yes","3","Yes","He ran away once, when he saw scared off by another (huge dog).
|
||||
He is fine in the car. He sits in the boot of the car.","Sit
|
||||
Down
|
||||
Come
|
||||
Wait
|
||||
Up (to get in the car)","He is a boisterous (but friendly) boy who needs lots of exercise!","Kate Langdon (facebook) katelangdon_nz (insta)","Facebook","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63df001dd7e541.37838378&form-id=4&field-id=50&hash=e8f90d72d6f9d3cfc20ff42c9efd5f16088654a14d5e6ccbdbc6609d2392670a","2023-02-05","","57","2023-02-05 14:02:21","2023-02-05 01:02:21","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.6.1 Safari/605.1.15","172.68.146.3","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/57/"
|
||||
"Lulu","Levermore","2015-02-16","Matthew","Swinburne","02041249994","Victoria Levermore","02041196653","TBC","TBC","TBC","Yes","No","","No","","Yes","Hypoallergenic kibble","No","","No","1-5","Yes","No","Yes","Yes","Other","No","Yes","Yes","No","Yes","4","No","Lulu regularly travels in the boot of our car and has no issues.","Sit. Stay. Wait. Heel. Roll over. Down. Shake.","Lulu is nervous around other dogs and needs to be kept on leash at all times when walked.
|
||||
If she feels threatened by other dogs she can get aggressive.
|
||||
We recommend keeping space from dogs she is not familiar with.","@mattswinny (instagram)","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63605b3fcc6f81.39608116&form-id=4&field-id=50&hash=79743c51c5ae9f38787210f42fbe2cdd1a10f84a5138057d2ae676b0e54fcbb3","2022-11-01","","56","2022-11-01 12:33:19","2022-10-31 23:33:19","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/107.0.0.0 Safari/537.36","162.158.2.21","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/56/"
|
||||
"Wallace","Darrah","2022-07-22","Nina","Darrah","0224981015","Siobhan Connelly","02108639903","Pt Chev Vet Clinic","67 Point Chevalier Road, Point Chevalier, Auckland 1022","09 815 0696","Yes","No","","No","","No","","No","","Yes","1-5","Yes, put still on leash","0","No","No","Birds","No","No","Yes","No","No","1","No","She is pretty calm, I have her tethered to the seat so she cant roam far.","Sit
|
||||
Down
|
||||
|
||||
working on: Wait, Stay","","@nina.darrah","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=635ae10606d868.49435145&form-id=4&field-id=50&hash=3db725b1f841bd74ebff7d583cac7b4b032bea51541c88d086df92c4be0a01a3","2022-10-28","","53","2022-10-28 08:50:30","2022-10-27 19:50:30","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/106.0.0.0 Safari/537.36","172.68.66.3","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/53/"
|
||||
"otis","McLean","2020-04-23","Fiona","Clarke","021477546","Brigid McLean","+64 21 051 9770","St Luke’s vets","13 Morningside Drive Mount Albert Auckland 1025 New Zealand","+6498455573","Yes","No","","Yes","Allergic to wandering Jew","No","","No","","Yes","1-5","Yes","Not often","No","No","Other","No","Yes","No","No","Yes","3","No","Calm, but likes cuddles","Sit, stay, come, wait, no, good boy etc","","Feeclarkee","Someone reffered me","Brigid McLean","https://onboarding.goodwalk.co.nz/index.php?gf-signature=633be8142e7824.35905911&form-id=4&field-id=50&hash=1e4809781b32a909dbf95107fe4c759f1de0ff7397bb547f1995f74c04157baf","2022-10-04","","51","2022-10-04 21:01:29","2022-10-04 08:01:29","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 15_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.5 Mobile/15E148 Safari/604.1","172.68.146.2","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/51/"
|
||||
"Fergie","Taylor","2020-10-06","Caroline","Logan","0272217468","Matt Taylor","0272175951","Auckland Pet Hospital","46-48 Pollen Street, Grey Lynn, Aucklandd","09 360 0961","Yes","No","","No","","No","","No","","Yes","5-10","Yes regularly on and off leash","1-2 times per week","No","No","Birds","No","Yes","Yes","No","Yes","3","No","Relaxed. Happy to sit wherever but always keen to look out of the window where possible.","Sit, down, stay, come, leave it","I didn't quite understand the questions about being reactive. Fergie will look to chase cats and birds if given the opportunity, albeit has not totally run away previously in doing so. She is happy and confident around other dogs, people and children. She is particularly fond of playing with other dogs so will seek that out when off leash.
|
||||
|
||||
One call out would that she can be nervous walking on leash on pavements. Following a bad experience walking next to a building site during a demolition, she can be twitchy and look to pull away from buses, large trucks or loud noises from building sites etc.","@carriestrells & @matthewtaylorphoto on Instagram","Someone reffered me","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63328f1f08d1d9.93390027&form-id=4&field-id=50&hash=14ec5a99a530fa02e079e24e5a3480f2eb531de1c4689736af62fa056ff3e5b7","2022-09-27","","49","2022-09-27 18:51:12","2022-09-27 05:51:12","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/105.0.0.0 Safari/537.36","172.68.66.2","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/49/"
|
||||
"Flo","Dine","2020-12-01","Jessica","Dine","0225280966","Alex Little","0276888676","Normanby Road Vets","49 Normandy Road","09 6388445","Yes","No","","No","","No","","No","","Yes","1-5","Yes loves it!","6x oer week","No","No","Cats","No","Yes","Yes","Yes","Yes","2","No","Fine, will just sit on seat","Sit
|
||||
Come
|
||||
Down
|
||||
No
|
||||
Walk
|
||||
Treat
|
||||
Dinner","","n/a","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63280e157c1eb5.67652552&form-id=4&field-id=50&hash=8cb69d5e96254bc74621d32638a83a2c7c1954ad316cb1c81b81007e82e96759","2022-09-19","","47","2022-09-19 18:37:46","2022-09-19 06:37:46","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.0.0 Safari/537.36","172.68.66.36","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/47/"
|
||||
"Lenny","Stark","2020-07-21","Andy","Stark","0211926498","Andy","0211926498","St Luke Veterinary Clinic","13 Morningside Drive, St Lukes, Auckland 1025","09 845 5573","Yes","No","","Yes","The Mud at Meola Park causes bald patches. We avoid in the winter","No","","No","","Yes","5-10","Yes","one per week","No","No","Other","No","Yes","Yes","No","Yes","4","No","Very good. Loves the car and always relaxed.","Come
|
||||
Stay
|
||||
Sit
|
||||
Down
|
||||
Shake hands
|
||||
No....","","kylie.stark_ and andystark88","Someone reffered me","Nancy from Generator","https://onboarding.goodwalk.co.nz/index.php?gf-signature=631d8173f240b8.62865903&form-id=4&field-id=50&hash=115663b7f127b08654c731bfe60b509bb18f551a77d675edd30ce9561aa9fe37","2022-09-11","","45","2022-09-11 18:35:14","2022-09-11 06:35:14","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/103.0.0.0 Safari/537.36","172.69.62.30","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/45/"
|
||||
"Archie","Lawrence","2019-03-15","Kate","Cadzow","0212052720","Robert Lawrence","0275194579","Auckland Pet Hospital","46 - 48 Pollen Street, Grey Lynn","09 3600961","Yes","No","","No","","No","","No","","Yes","1-5","Yes in summer time. He is unsure about water, but will chase a stick in.","5","No","Yes","Cats","No","Yes","Yes","Yes","Yes","3","Yes","Archie's behavior in a car varies. He will usually just stand where ever he is put, but can get very whiny and try to sit on someone in the car. He is usually fine though.","Sit, Down, Come, Spin, Shake, Stay, Paws, Up, Off, Leave it","Whist off lead Archie can be quite reactive to prams, bikes and people running. He will chase them and bark, but he will not bite them. He is also very reactive to cats whilst on the lead. He can sometimes very rarely be reactive to other dogs sometimes on lead, but also if they are behind fences.","Kate Cadzow (Facebook)","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6317c9766f3ba5.04202227&form-id=4&field-id=50&hash=e2947bb13e0173e9439b5048518bfd588d4f3a3e2b4e699c7646b17f741a3d3b","2022-09-07","","43","2022-09-07 10:28:06","2022-09-06 22:28:06","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.0.0 Safari/537.36","172.68.210.84","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/43/"
|
||||
"Franky","Powell","2021-12-05","craig","powell","0223715249","Sean","0272889696","Balmoral vets","482 Dominion Road, Mount Eden, Auckland 1024","09 630 7168","Yes","No","","No","","No","","No","","Yes","5-10","Yes","at least 5 times a wekk","No","No","Other","No","No","Yes","No","Yes","4","No","He gets very excited as he knows hes going for a walk","Sit.
|
||||
|
||||
Stay.
|
||||
|
||||
Here
|
||||
|
||||
Paw.","Hes super friendly with everyone and all animals.
|
||||
|
||||
Will lick you to death if you let him.","craig powell","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63156c03a78f56.56615128&form-id=4&field-id=50&hash=4a623f4948d0dd41cd534843301a14ff3187ac2a53d45ee495bab3a9186d72d2","2022-09-05","","41","2022-09-05 15:24:51","2022-09-05 03:24:51","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.0.0 Safari/537.36","172.68.210.6","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/41/"
|
||||
"Digby","McLean","2021-09-11","Brigid","McLean","0210519770","David Kidd","0299765289","St Luke’s Vet","13 Morningside Drive, Mt Albert","09 845 5573","Yes","No","","No","","No","","No","","Yes","1-5","Yes, mainly Pt Chev","4 -5 times mainly Nixon, Grey Lynn Park and Cox’s Bay Reserve","No","No","Birds","No","Yes","Yes","No","Yes","3","Yes","He loves the car because he knows it’s leading to an adventure","Sit
|
||||
Recall","He only has run away once and on a group beach walk with other dogs when he was 8 months and got distracted by some birds. It was his second walk and hadn’t formed a bond with the walker","Brigid McLean","Facebook","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=630efe47eb1f48.98016769&form-id=4&field-id=50&hash=2c097f472e116334b53e543b9e8072082b6698782340692681861f245b7cc121","2022-08-31","","39","2022-08-31 18:23:03","2022-08-31 06:23:03","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 15_6_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.6.1 Mobile/15E148 Safari/604.1","172.68.66.22","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/39/"
|
||||
"Slava","","2022-01-27","Daria","Derecha","0223972856","Nataliya Shchetkova","0223972860","Normanby Vet Clinic","49 Normanby Road, Mount Eden","09 638 8445","Yes","No","","No","","No","","No","","Yes","1-5","She's been to the beach, but haven't swam yet","1-2","No","No","Cats","Yes","No","No","Yes","Yes","3","Yes","She behaves in the car, but during long drives she can get sick","- Sit
|
||||
- Down
|
||||
- No
|
||||
- Stay
|
||||
- Wait
|
||||
- Watch
|
||||
- Leave it
|
||||
- Roll","Slava will be kept outside when you come in and she'll have a vibration collar on.
|
||||
Could you please take the collar off when you take her for a walk and put it back on when you back?
|
||||
& also please make sure that the side door is closed shut when you leave, thank you :)","Daria Derecha","Facebook","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63082e2ad5ff32.85559674&form-id=4&field-id=50&hash=881863d12cd6127e227a40b851c1817d339386ee277691d5d0fd7fd865ee7cb2","2022-09-26","","38","2022-08-26 14:21:30","2022-08-26 02:21:30","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.0.0 Safari/537.36","172.68.146.4","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/38/"
|
||||
"Charlie","Le Harivel","2012-01-01","Caitlin","Le Harivel","022 525 7530","Gerardo","+64224084568","Auckland vet hospital","46-48 Pollen Street, Grey Lynn, Auckland 1021","+64 9 360 0961","Yes","Yes","No grains, soy, dairy","No","","Yes","Raw meat/vege","No","","No","1-5","Yes, loves it","3-5","No","Yes","Cats","No","Yes","Yes","Yes","Yes","3","No","Needs to be in cage, whines and doesn't like turns. Settles on the straight parts of road/after walk.","Sit, come, lie down, down. ""Bah"": deep guttural noise for behavior you don't want.","Quite barky. Better off leash with other dogs","Caitlin le harivel","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6304215c740272.35188451&form-id=4&field-id=50&hash=c7dacf3575b1c220bf9bb512f9737af239b3d2379e7afc71f25c2be86b971aef","2022-08-23","","36","2022-08-23 12:38:24","2022-08-23 00:38:24","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Linux; Android 10; SM-N960F) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.4951.41 Mobile Safari/537.36 (Ecosia android@101.0.4951.41)","172.68.146.4","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/36/"
|
||||
"Freddie","Douglas","2017-05-01","James","Douglas","0202040525283","Paula Lorgelly","02040646006","Auckland Pet Hospital","46 Pollen St, Grey Lynn, 1021","093600961","Yes","No","","No","","No","","No","","Yes","1-5","Yes, loves running on the beach, but doesn't like the water","5","No","No","Other","No","Yes","Yes","No","Yes","3","No","Likes the car, will generally sleep, he can jump into most cars to get in.","Sit
|
||||
Wait
|
||||
Cross (the road)
|
||||
Come (back)
|
||||
No","","Not applicalbe","Google / Google Ads","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6303f978daad47.67660995&form-id=4&field-id=50&hash=a5fe26196a6e39bdaf111206b1fb45288dac4cc2bdcc99f95f1b38c65dfefc92","2022-08-23","","34","2022-08-23 09:48:09","2022-08-22 21:48:09","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.6.1 Safari/605.1.15","172.68.146.16","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/34/"
|
||||
"Quest","Kedzlie","2009-08-01","Joanna","Kedzlie","0274484720","Aaron Kedzlie","0274962151","Balmoral Vets","482 Dominion Rd Mt Eden","096307168","Yes","Yes","Beef chicken lamb causes itching","No","","Yes","Venison / Fish varieties of Ziwi Pet dried food","No","","Yes","5-10","yes every 2-3 weeks","5 x week","No","No","Cats","No","Yes","Yes","No","Yes","4","No","sleepy .","Wait , Sit , Go( chase or go see the dog or bird or go run)","Lifting to and from vehicles / couches / beds because of hip joint risk ( left hip fragile )
|
||||
Sometimes not reactive to calling if far away ? getting deaf but knows her name and usually very responsive .
|
||||
If sees cats can chase uncontrollably","Instagram","","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=62f22657a8dc11.18756572&form-id=4&field-id=50&hash=4a42a8967744c968154fdf32da58f9442cae6e0a9155558bd60c98d7f2f29bb6","2022-08-09","","32","2022-08-09 21:31:24","2022-08-09 09:31:24","https://onboarding.goodwalk.co.nz/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 15_6 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.6 Mobile/15E148 Safari/604.1","172.68.146.14","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/32/"
|
||||
"Sadie","Louise","2021-11-13","Nancy","Louise","0276234406","Sam Moir","+64 22 411 5904","Auckland Pet Hospital","46-48 Pollen Street, Grey Lynn, Auckland 1021","093600961","Yes","No","","No","","No","","No","","Yes","1-5","Yes","2","No","No","Birds","Yes","No","","","Yes","3","No","Sadie likes to sit on laps during a drive, but will snooze. She can sometimes be a bit hyper in the backseat of a car.","","Wants to be the best girl ever, but is still learning. Fine walking off leash and has good recall, but sometimes turns her ears off. Working on it…","","","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=62c65baf861644.12326872&form-id=4&field-id=50&hash=9f5474c574064ad3fe1e3ed448a567efd5794d6cc1128fedef44d8183cdd085a","2022-07-07","","30","2022-07-07 16:06:25","2022-07-07 04:06:25","https://goodwalk.co.nz/onboarding-form/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 15_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.5 Mobile/15E148 Safari/604.1","172.68.210.25","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/30/"
|
||||
"Stella","Blows","2021-08-01","Adrienne","Blows","0212965171","Adrienne","Blows","Balmoral veterinary care","482 Dominion road Mt Edwn","09 6307168","Yes","No","","Select","","No","","No","","Yes","1-5","Yes","No","No","Yes","Birds","No","Yes","","","No","1","No","Moves around a bit and needs a restraint. Likes to stick her head out the window","","","","","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=62c3d5039fbde6.48018846&form-id=4&field-id=50&hash=1ec5ffc4a70ebe2961deb18a2522231b38fd73c21f017f1c6c50bc725e5d819c","2022-07-05","","27","2022-07-05 18:06:59","2022-07-05 06:06:59","https://goodwalk.co.nz/onboarding-form/","","","","","","Mozilla/5.0 (Linux; Android 12; SM-G988B Build/SP1A.210812.016; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/103.0.5060.70 Mobile Safari/537.36 EdgW/1.0","172.68.146.11","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/27/"
|
||||
"Frankie","Richards-Berry","2020-06-08","Renee","Richards-Berry","0275342790","Michele Richards-Berry","021432583","St Luke’s Vet","13 Morningside Dr, st Lukes","09 845 5573","Yes","No","","No","","No","","No","","Yes","5-10","Yes. Loves swimming and fetching sticks","Daily","No","No","Cats","No","No","","","Yes","4","Yes","Excited initially but chills out and goes to sleep when she realises it might be a long journey. She loves the car.","","She loves to chase swallows at parks and they seem to love to taunt her. She usually just sits down in the park when she’s too tired to run any more","","","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=62c0bc95013e88.79836255&form-id=4&field-id=50&hash=a5170df830c4836db17676312d81ece5adf1d7aa57a6c7c5d7c17131cef108e3","2022-07-02","","24","2022-07-03 09:45:57","2022-07-02 21:45:57","https://goodwalk.co.nz/onboarding-form/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 15_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.5 Mobile/15E148 Safari/604.1","172.68.66.85","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/24/"
|
||||
"Ted / Teddy","Jenkinson","2020-10-02","Kim","Jenkinson","0274490005","Kyle Brown (Brownie)","0274887496","Ellerslie Veterinary Clinic","199 Main Highway, Ellerslie","092813481","Yes","No","","Yes","Grass - but he has pills for that. Can help by hosing down his feet after a run around","No","","No","","Yes","1-5","Yes","four times","No","No","Birds","No","No","","","No","3","Yes","Loves it. Loves just being there for a ride. Can leave him in the car (with water/ ventilation) happily for up to 2 hours","","","","","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=62bb6c86903b49.56603473&form-id=4&field-id=50&hash=5cdeebf1018eca2ad26f9674dbd27ef095015e3ab9a90881d728ba1ca6de87f0","2022-06-29","","22","2022-06-29 09:03:02","2022-06-28 21:03:02","https://goodwalk.co.nz/onboarding-form/","","","","","","Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/102.0.0.0 Safari/537.36","202.3.88.174","https://onboarding.goodwalk.co.nz/pdf/62bad160c3d11/22/"
|
||||
|
@@ -0,0 +1,40 @@
|
||||
"Consent (Consent)","Consent (Text)","Consent (Description)","Owner Signature","GoodWalk Limited Signature","Date contract signed","Owners Name (Prefix)","Owners Name (First Name)","Owners Name (Middle)","Owners Name (Last Name/Surname)","Owners Name (Suffix)","Owner's email (Enter Email)","Phone","Dog's name (include surname)","Residential Address (Street Address)","Residential Address (Address Line 2)","Residential Address (City)","Residential Address (Suburb)","Residential Address (ZIP / Postal Code)","Residential Address (Country)","Created By (User Id)","Entry Id","Entry Date","Date Updated","Source Url","Transaction Id","Payment Amount","Payment Date","Payment Status","Post Id","User Agent","User IP","PDF: PDF Label"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=65bddbac696dc2.10945403&form-id=3&field-id=1&hash=4ba1e2e94457e06082630de6de930a252ecb39bd4c56b5e9ffe7093bb5a0d957","","2024-02-03","","Rachel","","Hall","","rachelhall@dimery.co.nz","","Louis Hall","303/26 Poynton Terrace","","Auckland","Auckland Central","","","","108","2024-02-03 19:22:36","2024-02-03 06:22:36","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.6 Safari/605.1.15","115.189.128.39","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/108/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=65b1a8726c9964.16114292&form-id=3&field-id=1&hash=c9ab0ac4b94a84044c5030e2e24afb4134107a75395c766cccc04c28381db6f8","","2024-01-25","","Sue","","Ewen","","sewen@xtra.co.nz","(642) 745-2472","Tinks Ewen","16 Simmonds Ave, Mount Roskill,","16 Simmonds Ave","Mount Roskill Auckland 1041","Mt Roskill / Three Kings","","","","106","2024-01-25 13:17:08","2024-01-25 00:17:08","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36 Edg/120.0.0.0","125.237.63.27","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/106/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=65459ee482d712.77412582&form-id=3&field-id=1&hash=5bcb04c9fe21643fb791372228114a63555f7d3cafaa5d7ef45240b2115ec97a","","2023-11-04","","Anna","","Shaw","","annalshaw@gmail.com","(021) 029-7053","Honey Crumbles Shaw","43 Kings Road","","Auckland","Mt Roskill","","","","104","2023-11-04 14:31:16","2023-11-04 01:31:16","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/118.0.0.0 Safari/537.36","123.255.41.22","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/104/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=653eb664cba249.42033046&form-id=3&field-id=1&hash=b297538263e5233fac01524703fa2cc4c79f8227dca70a551893cb09f273f194","","2023-10-30","","Shelly","","Haggas","","shelly@envoyconstruction.co.nz","(021) 042-5095","Hank O'Donnell","377 New North Road","Level 1","Aickland","Kingsland","","","","102","2023-10-30 08:45:40","2023-10-29 19:45:40","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/118.0.0.0 Safari/537.36","203.118.148.26","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/102/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=650c9f6be75ef7.65308375&form-id=3&field-id=1&hash=a95b743d7995570e67bcb685c374b5028771af177d079eeaa9b256fdffed2b9f","","2023-09-22","","Danielle","","Barclay","","danielle@weblurthelines.com","(027) 397-2112","Molly Hickey","29 Kensington Avenue","","Auckland","Mt Eden","","","","100","2023-09-22 07:54:19","2023-09-21 19:54:19","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_6 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/117.0.5938.108 Mobile/15E148 Safari/604.1","121.98.37.166","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/100/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6503b77ec85832.30852194&form-id=3&field-id=1&hash=623ca1485ecedfcb151957fad9f0fc7a7f0d38926897465de892f7afd227289c","","2023-09-15","","Ross","","Batchelor","","ross.batchelor@icloud.com","","Otis Batchelor","203/2 Finch Street","","Auckland","Morningside","","","","98","2023-09-15 13:46:38","2023-09-15 01:46:38","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_6_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.6 Mobile/15E148 Safari/604.1","172.225.244.185","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/98/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=64bf5fddc33ed7.06471243&form-id=3&field-id=1&hash=ce7a980f14ee232a171ce1ff4d43f0d24e7863d70c3016078c295c2f5a180503","","2023-07-25","","Ann","","Buckley","","ann.buckley9@gmail.com","(027) 447-9293","Ollie Buckley","","12 Stewart Rd","Auckland","Mt Albert","","","","96","2023-07-25 17:38:37","2023-07-25 05:38:37","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.0.0 Safari/537.36","210.246.17.164","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/96/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=648fd5294cb4d0.32639145&form-id=3&field-id=1&hash=ac2277457a879fd40401f0b29b9349c3bc8125798ca78b3e0051c4b45a17b822","","2023-06-19","","Monique","","Humphrey","","moniquehrtlb@gmail.com","(021) 276-4358","Zola Humphrey","90A Owens Road","","Auckland","Epsom, 1023","","","","94","2023-06-19 16:10:30","2023-06-19 04:10:30","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.2 Safari/605.1.15","121.98.13.62","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/94/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=64804b84af2975.29344074&form-id=3&field-id=1&hash=719768bc8e5f1cc180a2255ac84a3c0a32e54f987e0d26499b3210e522c5e721","","2023-06-07","","Ruby Kahurangi","","Tautuhi-Lloyd","","rubytlloyd@gmail.com","(020) 414-1878","Sunny Lloyd","6 Sussex Street","","Auckland","Grey Lynn","","","","92","2023-06-07 21:19:00","2023-06-07 09:19:00","https://onboarding.goodwalk.co.nz/contract/?","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/114.0.5735.99 Mobile/15E148 Safari/604.1","101.100.129.22","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/92/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=647fd5bb62cc48.21415458&form-id=3&field-id=1&hash=63c36109e5e7bb77911bbcb0acc0baf789a450618b8d1b04958a3694db9ae270","","2023-06-07","","Ruby Kahurangi","","Tautuhi-Lloyd","","rubytlloyd@gmail.com","(020) 414-1878","Sunny Lloyd","6 Sussex Street","","Auckland","Grey Lynn","","","","91","2023-06-07 12:56:27","2023-06-07 00:56:27","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/114.0.5735.99 Mobile/15E148 Safari/604.1","122.56.196.93","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/91/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=647d87282fb183.81737041&form-id=3&field-id=1&hash=84d5464dd9b51110878b940c8262e4dfcbcd612ac07feaf24c044f05a2a17b55","","2023-06-05","","Megan & Adrian","","Kenny","","info@kennyandharlow.co.nz","","Donny Kenny","31 b Normanby road","","Mount Eden","","","","","89","2023-06-05 18:57:51","2023-06-05 06:57:51","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/113.0.5672.121 Mobile/15E148 Safari/604.1","115.188.134.131","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/89/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6462c5bf460fe8.96828794&form-id=3&field-id=1&hash=b08513e4d2ca635b22a3d1ef52bc35dbf730c3786cac77abc16144edbe73d431","","2023-05-16","","Harriet","","Read","","harriet.a.read@gmail.com","(021) 234-0063","Maggie Read","10 parkdale road,","My Albert","Auckland","Mt Albert","","","","86","2023-05-16 13:44:38","2023-05-16 01:44:38","https://onboarding.goodwalk.co.nz/contract/?","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/113.0.5672.69 Mobile/15E148 Safari/604.1","122.56.5.204","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/86/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6462c5bf460fe8.96828794&form-id=3&field-id=1&hash=b08513e4d2ca635b22a3d1ef52bc35dbf730c3786cac77abc16144edbe73d431","","2023-05-16","","Harriet","","Read","","harriet.a.read@gmail.com","(021) 234-0063","Maggie Read","10 parkdale road,","My Albert","Auckland","Mt Albert","","","","85","2023-05-16 11:52:50","2023-05-15 23:52:50","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_3 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) CriOS/113.0.5672.69 Mobile/15E148 Safari/604.1","122.56.5.204","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/85/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=645a9f6a95a969.46711675&form-id=3&field-id=1&hash=4f9b59d32b0c077ada66b5cb44c2aea16d395ea5cd0c20705e2ea2571c2f4722","","2023-05-10","","Carl","","Matthews","","carl@devmark.co.nz","(642) 173-7727","Luna Matthews","57 Marlborough Street","","Mount Eden, AUCKLAND","","","","","82","2023-05-10 07:30:58","2023-05-09 19:30:58","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/113.0.0.0 Safari/537.36","121.99.240.46","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/82/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=64589d00752470.47731501&form-id=3&field-id=1&hash=1f578bd344c4e39ca0eb0c962abd3ae4ecb5529f1f9a90017af8b9dd7a8c6cfa","","2023-05-08","","Todd","","Benton","","toddbenton@outlook.com","","Sweep","3/61 Hepburn Street","","Auckland","Freemans Bay","","","","81","2023-05-08 18:56:00","2023-05-08 06:56:00","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/112.0.0.0 Safari/537.36 OPR/98.0.0.0","121.99.103.51","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/81/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=64549cabb86241.61504100&form-id=3&field-id=1&hash=a3aedd8ed8cac157b00bfe7d0ed2da461469981031d5a476c276b06d5eb569da","","2023-05-05","","Paula","","Smith","","smithy007@mac.com","","Bob Smith","178 Mount Albert Road","","Sandringham","Auckland","","","","79","2023-05-05 18:05:31","2023-05-05 06:05:31","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_4_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.4 Mobile/15E148 Safari/604.1","125.237.205.159","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/79/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6448d984e1f712.46902795&form-id=3&field-id=1&hash=0fe483a8e365b4bfa9555bfe9b327c846d9abd40e6f9af8884b8770b4665774f","","2023-04-26","","Lori","","Reynolds","","lorireynolds5@gmail.com","(022) 310-9861","Buddy Reynolds","14b Lidcombe Place","Avondale","Auckland","Avondale","","","","75","2023-04-26 19:57:56","2023-04-26 07:57:56","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 16_3_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.3 Mobile/15E148 Safari/604.1","172.225.244.181","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/75/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=641d2924990540.87574330&form-id=3&field-id=1&hash=bdfc49b3876c2de4e6f00efd38adb61ce3bbc36e35ca9337da875f00f901886b","","2023-03-24","","Stephanie","","Korucu","","korucusteph@gmail.com","","Jake Korucu","1/98 Owairaka Ave","Mount Albert","Auckland","","","","","65","2023-03-24 17:37:56","2023-03-24 04:37:56","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/111.0.0.0 Safari/537.36","198.41.238.107","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/65/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=641a1be974e772.11405043&form-id=3&field-id=1&hash=25d65613131edf5cd825ef16c4031163b34f1117d8053f7d5e81e1474758cb08","","2023-03-22","","Estelle","","Rewiri","","estelle.rewiri@mainfreight.com","(027) 519-4228","Monty Rewiri","11A Sandringham Road","Mt Eden","Auckland","","","","","63","2023-03-22 10:04:41","2023-03-21 21:04:41","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 15_6_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.6.1 Mobile/15E148 Safari/604.1","198.41.238.28","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/63/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63ebdb8add55a1.53333768&form-id=3&field-id=1&hash=2349b49c8b1a7a13da3898071653d63de9ce2b754a647728f5ae459ef5f5fb4b","","2023-02-15","","Penny","","Dine","","jess.dine@gmail.com","(022) 528-0966","Archie Dine","57 Bright Street","Eden Terrace","Auckland","Eden Terrace","","","","62","2023-02-15 08:06:04","2023-02-14 19:06:04","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/109.0.0.0 Safari/537.36","198.41.238.110","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/62/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63e4328e530e96.01203691&form-id=3&field-id=1&hash=4283da25a29c8eece570f4683e0cfcd5b393ac347b8fef10bb0faca02467a99f","","2023-02-09","","Monique","","Sullivan","","msullivan@mpm.co.nz","(027) 530-6046","Freddie Sullivan","3 Shaw St","Sandringham","Auckland","Sandringham","","","","60","2023-02-09 12:38:54","2023-02-08 23:38:54","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/110.0.0.0 Safari/537.36","198.41.238.111","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/60/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63df01e41a56e9.54070037&form-id=3&field-id=1&hash=81945645899ec7d3be0002a858fae4ec90474ac1f96ac45a6eb451dc52510740","","2023-02-05","","Kate","","Langdon","","kate@katelangdon.com","","Digby Langdon-Baird","13 Buchanan Street","Kingsland","Auckland","Kingsland","","","","58","2023-02-05 14:09:56","2023-02-05 01:09:56","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.6.1 Safari/605.1.15","172.68.210.34","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/58/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=636058c639eb93.61392483&form-id=3&field-id=1&hash=1e8946795c3b839220656619ba8dff93772d372dce5ddef915b3a593cdc717e0","","2022-11-01","","Matthew","","Swinburne","","mattswinburne@gmail.com","","Lulu Levermore","12 Herringson Ave","","Auckland 1021","Grey Lynn","","","","55","2022-11-01 12:23:01","2022-10-31 23:23:01","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/107.0.0.0 Safari/537.36","162.158.2.13","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/55/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=635b0152493df6.85330069&form-id=3&field-id=1&hash=b81d54dcafff595f974d09a77ae5c6ea92c56500fe8a024515175098f48c5c72","","2022-10-28","","Nina","","Darrah","","nina.darrah@gmail.com","(022) 498-1015","Wallace Darrah","5 Cadman Ave","Waterview","Auckland","Waterview","","","","54","2022-10-28 11:08:18","2022-10-27 22:08:18","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/106.0.0.0 Safari/537.36","172.68.146.15","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/54/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=633bec2129d2c8.71863735&form-id=3&field-id=1&hash=fe95c8059b6cb3b8b97246c5e0f06c5ac10b54193680f3440040365772140fee","","2022-10-04","","Fiona","","Clarke","","feeclarke@gmail.com","","Otis McLean","33 walters road","Mt eden","","Auckland","","","","52","2022-10-04 21:18:24","2022-10-04 08:18:24","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 15_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.5 Mobile/15E148 Safari/604.1","172.68.66.34","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/52/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63329aab7c9154.72679599&form-id=3&field-id=1&hash=a62256fed96d754da7294f5675ecaede73ec45ac890e053dcf0b2c9f8271f498","","2022-09-27","","Caroline","","Logan","","caroline.a.logan@gmail.com","(027) 221-7468","Fergie Taylor","5 Alexander Street","","Auckland","Kingsland","","","","50","2022-09-27 19:39:39","2022-09-27 06:39:39","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 15_6_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.6.1 Mobile/15E148 Safari/604.1","172.68.210.84","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/50/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63280e8d9c03e0.76063236&form-id=3&field-id=1&hash=3f0ce8d19b1b1667500bff4b993e292b58244b4c137c30e2bbae407b28fe933f","","2022-09-19","","Jessica","","Dine","","jess.dine@gmail.com","(022) 528-0966","Florence Dine","57 Bright Street","Eden Terrace","Auckland","Eden Terrace","","","","48","2022-09-19 18:39:09","2022-09-19 06:39:09","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_14_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.0.0 Safari/537.36","172.68.210.84","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/48/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=631d82d24472c5.01871772&form-id=3&field-id=1&hash=67496d58daa653a7fe5360b95f73f89a89a0ad2b9a677b2a2d4cb354875a69f5","","2022-09-11","","Andy","","Stark","","andystark88@gmail.com","(021) 192-6498","Lenny Stark","67 Kiwi Road","","Auckland","Point Chevalier","","","","46","2022-09-11 18:40:18","2022-09-11 06:40:18","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/103.0.0.0 Safari/537.36","172.68.66.36","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/46/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6317cb0f063948.03054426&form-id=3&field-id=1&hash=50186e66c8e14daf4cb01de0ad3682ac9afeaa76497bdbb841b6846356248bba","","2022-09-07","","Robert","","Lawrence","","robert.a.lawrence@hotmail.com","(027) 519-4579","Archie Lawrence","4/52 Meadowbank Road","","Auckland","Meadowbank","","","","44","2022-09-07 10:34:55","2022-09-06 22:34:55","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.0.0 Safari/537.36","172.68.66.26","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/44/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=63156c4dec6ed5.48344366&form-id=3&field-id=1&hash=77ec6a504b0353f0266c32ee35784668fae2c44040c986a12dccfad54863c1e1","","2022-09-05","","craig","","powell","","c.powell1985@outlook.com","(022) 371-5249","Franky Powell","46 Burnley Terrace","Mount Eden","auckland","","","","","42","2022-09-05 15:26:05","2022-09-05 03:26:05","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/104.0.0.0 Safari/537.36","172.68.210.6","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/42/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=630eff5a769a86.20761083&form-id=3&field-id=1&hash=4cf1e7afd2e705ed2b18d9df15278703946add6ec431cef43d011c4745b2f96d","","2022-08-31","","Brigid","","McLean","","brigidmclean@gmail.com","(021) 051-9770","Digby McLean","Apt 413 / 8 Kingsland Terrace","8 Kingsland Terrace","Kingsland","Auckland","","","","40","2022-08-31 18:27:38","2022-08-31 06:27:38","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 15_6_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.6.1 Mobile/15E148 Safari/604.1","172.68.66.26","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/40/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=630422ba814ef5.81658908&form-id=3&field-id=1&hash=3c10ceb165845ec67c854f4220a05db9fa9c327d95f5285ce8a7938ede432e88","","2022-08-23","","Caitlin","","Le Harivel","","Caitlinleh@gmail.com","(022) 525-7530","Charlie le harivel","412/145 nelson street","","Auckland","Auckland Central","","","","37","2022-08-23 12:43:38","2022-08-23 00:43:38","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Linux; Android 10; SM-N960F) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.4951.41 Mobile Safari/537.36 (Ecosia android@101.0.4951.41)","172.69.62.30","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/37/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=6303fa4e4c0848.05382829&form-id=3&field-id=1&hash=ae71e833c9195588975ef592d666f59cb24215987b531be7be4744f09a1c990c","","2022-08-23","","James","","Douglas","","james_douglas@me.com","(020) 405-2528","Freddie Douglas","16 Rossgrove Terrace","Mount Albert","Auckland","Mount Albert, 1025","","","","35","2022-08-23 09:51:10","2022-08-22 21:51:10","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.6.1 Safari/605.1.15","172.68.210.56","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/35/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=62f5d231bca2a3.30220644&form-id=3&field-id=1&hash=141bd60c7b10b46e0b3efd8bab2cefa474ff07289a751621df98770396058b62","","2022-08-12","","Kedzlie","","Joanna","","kedzlie@xtra.co.nz","(027) 448-4720","Quest Kedzlie","24 A","Woodside Road","Mt Eden","Auckland","","","","33","2022-08-12 16:08:17","2022-08-12 04:08:17","https://onboarding.goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 15_6 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.6 Mobile/15E148 Safari/604.1","172.69.62.30","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/33/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=62c65c66e18d38.45439060&form-id=3&field-id=1&hash=826e6dd0c397424665e72f5f38de6bcd4caccebde17f4f8060fa5f68f967ceaf","","2022-07-07","","Nancy","","Louise","","nancy.louise@icloud.com","","Sadie Louise","414 Scenic Drive, Waiatarua","","Auckland","Waiatarua","","","","31","2022-07-07 16:09:10","2022-07-07 04:09:10","https://goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 15_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.5 Mobile/15E148 Safari/604.1","172.68.210.25","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/31/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=62c52b76d0d3a0.71873578&form-id=3&field-id=1&hash=924287d3ac145b67103e00154588808d23f37b85bf6757baf11493fedb946403","","2022-07-06","","Maddie","","Thomas","","thomasmaddie@ymail.com","","Doug Thomas","12 Dryden Street","DRYDEN STREET","Grey Lynn","Grey Lynn","","","","29","2022-07-06 18:29:48","2022-07-06 06:29:48","https://goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_6) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/14.1.2 Safari/605.1.15","172.68.146.45","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/29/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=62c4c9cf2d2015.66413113&form-id=3&field-id=1&hash=41bb2215be098a3e0cec6f8d294dc1b36a22b9204bc2566294ed2e7c56b20305","","2022-07-06","","Adrienne","","Blows","","adrienneblows@hotmail.com","(021) 296-5171","Stella Blows","1/107 Taylors Road","Mt Albert","Auckland","Mt Albert","","","","28","2022-07-06 11:31:27","2022-07-05 23:31:27","https://goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Linux; Android 12; SM-G988B Build/SP1A.210812.016; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/103.0.5060.70 Mobile Safari/537.36 EdgW/1.0","172.68.210.25","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/28/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=62c0bdfa32cf79.13628927&form-id=3&field-id=1&hash=98f77c6b8498e390689018ee1d0dc63e5c8a0ccfadd23a7c3dbd6932fd3d6e0d","","2022-07-02","","Renee","","Richards-Berry","","reneeberryrenee@gmail.com","(027) 534-2790","Frankie Richards-Berry","66 Fourth Ave","Kingsland","Auckland 1021","","","","","25","2022-07-03 09:51:54","2022-07-02 21:51:54","https://goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (iPhone; CPU iPhone OS 15_5 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.5 Mobile/15E148 Safari/604.1","172.69.62.73","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/25/"
|
||||
"Checked","I agree that I have fully read and understood the terms and conditions","","https://onboarding.goodwalk.co.nz/index.php?gf-signature=62bb6d456d0595.78626579&form-id=3&field-id=1&hash=1df30f89b71b90ec341a48975474708e25fbe822f939e4b2f73fa959a94a0dd2","","2022-06-29","","Kim","","Jenkinson","","kimajenkinson@gmail.com","(027) 449-0005","Ted Jenkinson","61 Aranui Road,","","Auckland","Mt Wellington","","","","23","2022-06-29 09:06:13","2022-06-28 21:06:13","https://goodwalk.co.nz/contract/","","","","","","Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/102.0.0.0 Safari/537.36","202.3.88.174","https://onboarding.goodwalk.co.nz/pdf/62b7e36382d03/23/"
|
||||
|
@@ -26,4 +26,11 @@ RUN pip install --no-cache-dir -r requirements.txt
|
||||
COPY main.py db.py ./
|
||||
COPY mail_api ./mail_api
|
||||
|
||||
# Legacy clients seed, generated by scripts/build-legacy-seed.mjs (called from
|
||||
# deploy.ps1). Lives OUTSIDE the data volume so every deploy carries a fresh
|
||||
# copy. The mail-api merges this on boot, add-only — it never overwrites a
|
||||
# live entry. Must always be present in the build context; for local dev run
|
||||
# `node scripts/build-legacy-seed.mjs` once, or `echo {} > legacy-clients-seed.json`.
|
||||
COPY legacy-clients-seed.json /app/legacy-clients-seed.json
|
||||
|
||||
CMD ["uvicorn", "main:app", "--host", "0.0.0.0", "--port", "8000"]
|
||||
|
||||
@@ -1436,3 +1436,4 @@ resend.exceptions.ResendError: API key is invalid
|
||||
19/05/2026 23:32:40 New Zealand Standard Time ERROR mail-api: Startup smoke: WeasyPrint UNAVAILABLE — PDF attachments will be skipped (No module named 'weasyprint')
|
||||
19/05/2026 23:32:40 New Zealand Standard Time WARNING mail-api: Startup smoke: postgres disabled — activity/submissions will NOT be recorded
|
||||
19/05/2026 23:32:40 New Zealand Standard Time INFO mail-api: Startup test email skipped: OWNER_BCC is not set to a real address
|
||||
19/05/2026 23:37:58 New Zealand Standard Time INFO mail-api: [e8c74fa0] GET /auth/verify → 401 (2ms)
|
||||
|
||||
@@ -211,6 +211,13 @@ _DATA_DIR = Path(os.environ.get("DATA_DIR", "data"))
|
||||
ALLOWED_EMAILS_FILE = _DATA_DIR / "allowed_emails.json"
|
||||
CLIENT_PROFILES_FILE = _DATA_DIR / "client_profiles.json"
|
||||
DRAFTS_FILE = _DATA_DIR / "drafts.json"
|
||||
# Legacy seed lives OUTSIDE the data volume — it's shipped in the image so a
|
||||
# fresh deploy always carries the same baked-in copy. On boot the mail-api
|
||||
# merges this dict into _client_profiles, adding any emails that aren't
|
||||
# already present. It never overwrites a live entry.
|
||||
LEGACY_SEED_FILE = Path(
|
||||
os.environ.get("LEGACY_SEED_FILE", "/app/legacy-clients-seed.json")
|
||||
)
|
||||
|
||||
LOGO_URL = "https://www.goodwalk.co.nz/images/goodwalk-auckland-dog-walking-logo.png"
|
||||
|
||||
@@ -235,6 +242,12 @@ EMAIL_SEND_TIMEOUT_SECONDS = settings.email_send_timeout_seconds
|
||||
# Owner-BCC placeholder used by deploy.env.template; treat it as "unset" for the smoke email.
|
||||
STARTUP_TEST_RECIPIENT = OWNER_BCC if OWNER_BCC and OWNER_BCC.lower() != "example@example.com" else ""
|
||||
|
||||
# Shared secret presented by the deploy script's post-deploy form smoke tests.
|
||||
# When set, requests carrying a matching X-Deploy-Smoke header short-circuit the
|
||||
# form handlers before any email/db side effects so production submissions stay
|
||||
# clean. When unset, the bypass is fully disabled.
|
||||
DEPLOY_SMOKE_SECRET = (os.environ.get("DEPLOY_SMOKE_SECRET") or "").strip()
|
||||
|
||||
logger.info(
|
||||
"Mail API config: version=%r timezone=%r from=%r reply_to=%r owner=%r cp_admins=%r owner_bcc=%r client_bcc=%r general_enquiries=%r max_attempts=%d form_min=%ss form_max=%ss rate_window=%ss per_ip=%d per_email=%d min_interval=%ss send_timeout=%ss",
|
||||
APP_VERSION,
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
from typing import Any
|
||||
from typing import Any, Literal
|
||||
|
||||
from pydantic import BaseModel, EmailStr
|
||||
|
||||
@@ -70,6 +70,17 @@ class BirthdayAutoSendRequest(BaseModel):
|
||||
enabled: bool
|
||||
|
||||
|
||||
# Client lifecycle status. Soft-delete only — every client stays on file so the
|
||||
# history is preserved for future newsletter / retention work.
|
||||
ClientLifecycleStatus = Literal["active", "paused", "cancelled", "archived"]
|
||||
|
||||
|
||||
class ClientStatusUpdate(BaseModel):
|
||||
email: EmailStr
|
||||
status: ClientLifecycleStatus
|
||||
reason: str = ""
|
||||
|
||||
|
||||
class ContractSubmission(BaseSubmission):
|
||||
address: str
|
||||
dogName: str
|
||||
|
||||
@@ -36,6 +36,7 @@ from mail_api.config import (
|
||||
CLIENT_PROFILES_FILE as _CLIENT_PROFILES_FILE,
|
||||
CORS_ALLOWED_ORIGINS,
|
||||
CP_ADMIN_EMAILS,
|
||||
DEPLOY_SMOKE_SECRET,
|
||||
DEV_MODE,
|
||||
DRAFTS_FILE as _DRAFTS_FILE,
|
||||
EMAIL_SEND_TIMEOUT_SECONDS,
|
||||
@@ -43,6 +44,7 @@ from mail_api.config import (
|
||||
FORM_MAX_SECONDS,
|
||||
FORM_MIN_SECONDS,
|
||||
FROM_EMAIL,
|
||||
LEGACY_SEED_FILE as _LEGACY_SEED_FILE,
|
||||
LOGO_URL,
|
||||
MAX_REQUEST_BODY_BYTES,
|
||||
MAX_SEND_ATTEMPTS,
|
||||
@@ -62,6 +64,7 @@ from mail_api.models import (
|
||||
BirthdayAutoSendRequest,
|
||||
BirthdayEmailRequest,
|
||||
BookingSubmission,
|
||||
ClientStatusUpdate,
|
||||
ContractSubmission,
|
||||
OnboardingSubmission,
|
||||
RenderMessageRequest,
|
||||
@@ -227,6 +230,70 @@ async def _seed_admin_state_from_json_if_needed() -> None:
|
||||
logger.warning("Admin seed from JSON failed: %s", exc)
|
||||
|
||||
|
||||
async def _merge_legacy_seed_if_present() -> None:
|
||||
"""Merge the shipped legacy-clients-seed.json into _client_profiles.
|
||||
|
||||
Add-only: never overwrites an email that already exists in the live data.
|
||||
Idempotent: re-running on every boot is a no-op once the entries are in.
|
||||
|
||||
Writes the updated profiles back to the JSON file + admin_kv so the merged
|
||||
state survives container restarts.
|
||||
"""
|
||||
global _client_profiles
|
||||
|
||||
seed_path = _LEGACY_SEED_FILE
|
||||
if not seed_path.exists():
|
||||
return
|
||||
|
||||
try:
|
||||
seed = json.loads(seed_path.read_text(encoding="utf-8"))
|
||||
except Exception as exc:
|
||||
logger.warning("Legacy seed file unreadable (%s): %s", seed_path, exc)
|
||||
return
|
||||
|
||||
if not isinstance(seed, dict) or not seed:
|
||||
return
|
||||
|
||||
added: list[str] = []
|
||||
skipped_existing = 0
|
||||
|
||||
for raw_email, profile in seed.items():
|
||||
if not isinstance(raw_email, str) or not isinstance(profile, dict):
|
||||
continue
|
||||
email = raw_email.strip().lower()
|
||||
if not email:
|
||||
continue
|
||||
if email == OWNER_EMAIL.strip().lower():
|
||||
continue
|
||||
if email in _client_profiles:
|
||||
skipped_existing += 1
|
||||
continue
|
||||
_client_profiles[email] = profile
|
||||
added.append(email)
|
||||
|
||||
if not added:
|
||||
logger.info(
|
||||
"Legacy seed already merged (existing=%d, candidates=%d).",
|
||||
skipped_existing, len(seed),
|
||||
)
|
||||
return
|
||||
|
||||
snapshot = dict(_client_profiles)
|
||||
try:
|
||||
await asyncio.to_thread(_save_client_profiles_file, snapshot)
|
||||
except Exception as exc:
|
||||
logger.warning("Could not save client_profiles after legacy merge: %s", exc)
|
||||
try:
|
||||
await _persist_admin_state("client_profiles", snapshot)
|
||||
except Exception as exc:
|
||||
logger.warning("Could not persist client_profiles to postgres after legacy merge: %s", exc)
|
||||
|
||||
logger.info(
|
||||
"Legacy seed merged: added=%d skipped_existing=%d total_after=%d",
|
||||
len(added), skipped_existing, len(_client_profiles),
|
||||
)
|
||||
|
||||
|
||||
async def _load_allowed_emails_async() -> set[str]:
|
||||
if admin_db.is_enabled():
|
||||
data = await admin_db.get_kv("allowed_emails")
|
||||
@@ -308,6 +375,16 @@ async def _register_email(email: str) -> None:
|
||||
logger.info("Auth: registered new allowed email: %s", normalized)
|
||||
|
||||
|
||||
def _client_is_reachable(profile: dict) -> bool:
|
||||
"""True if outreach (welcome pack, birthday email, etc.) should still target
|
||||
this client. Excludes lifecycle states that mean the relationship has ended.
|
||||
"""
|
||||
lifecycle = profile.get("lifecycle")
|
||||
if not isinstance(lifecycle, dict):
|
||||
return True
|
||||
return lifecycle.get("status") not in {"cancelled", "archived"}
|
||||
|
||||
|
||||
async def _store_client_profile(email: str, profile: dict) -> None:
|
||||
normalized = email.strip().lower()
|
||||
if not normalized:
|
||||
@@ -458,6 +535,21 @@ def _get_ip(request: Request) -> str:
|
||||
return request.client.host if request.client else "unknown"
|
||||
|
||||
|
||||
def _is_deploy_smoke(request: Request) -> bool:
|
||||
"""True when the request carries a matching X-Deploy-Smoke header.
|
||||
|
||||
Used by the deploy script to verify the form endpoints are reachable and
|
||||
parse a valid payload, without producing a real submission. Disabled
|
||||
entirely when DEPLOY_SMOKE_SECRET is unset.
|
||||
"""
|
||||
if not DEPLOY_SMOKE_SECRET:
|
||||
return False
|
||||
presented = request.headers.get("x-deploy-smoke") or ""
|
||||
if not presented:
|
||||
return False
|
||||
return secrets.compare_digest(presented, DEPLOY_SMOKE_SECRET)
|
||||
|
||||
|
||||
_submit_attempts_by_ip: dict[str, deque[float]] = {}
|
||||
_submit_attempts_by_email: dict[str, deque[float]] = {}
|
||||
_submit_rate_limit_lock = asyncio.Lock()
|
||||
@@ -1477,6 +1569,213 @@ def _birthday_ics_attachment(dog_name: str, dog_birth_date: str, owner_name: str
|
||||
}
|
||||
|
||||
|
||||
def _pdf_escape(value: Any) -> str:
|
||||
if value is None:
|
||||
return ""
|
||||
text = str(value)
|
||||
return (
|
||||
text.replace("&", "&")
|
||||
.replace("<", "<")
|
||||
.replace(">", ">")
|
||||
.replace("\n", "<br>")
|
||||
)
|
||||
|
||||
|
||||
def owner_onboarding_pdf_html(data: OnboardingSubmission) -> str:
|
||||
"""Clean, full-width, black-and-white, Arial HTML for the printable onboarding PDF."""
|
||||
submitted_at = datetime.now().strftime("%d %b %Y at %I:%M %p").lstrip("0")
|
||||
|
||||
snapshot = data.submissionSnapshot or {}
|
||||
sections = snapshot.get("sections") if isinstance(snapshot, dict) else None
|
||||
|
||||
def render_value(value: Any) -> str:
|
||||
if isinstance(value, list):
|
||||
items = [_pdf_escape(item) for item in value if str(item).strip()]
|
||||
return ", ".join(items) if items else "—"
|
||||
text = _pdf_escape(value).strip()
|
||||
return text if text else "—"
|
||||
|
||||
sections_html_parts: list[str] = []
|
||||
|
||||
if isinstance(sections, list) and sections:
|
||||
for section in sections:
|
||||
if not isinstance(section, dict):
|
||||
continue
|
||||
title = _pdf_escape(section.get("title", ""))
|
||||
fields = section.get("fields") or []
|
||||
rows_html = ""
|
||||
for field in fields:
|
||||
if not isinstance(field, dict):
|
||||
continue
|
||||
label = _pdf_escape(field.get("label", ""))
|
||||
rows_html += (
|
||||
"<tr>"
|
||||
f"<th>{label}</th>"
|
||||
f"<td>{render_value(field.get('value'))}</td>"
|
||||
"</tr>"
|
||||
)
|
||||
if rows_html:
|
||||
sections_html_parts.append(
|
||||
f"<section class='pdf-section'>"
|
||||
f"<h2>{title}</h2>"
|
||||
f"<table class='pdf-table'><tbody>{rows_html}</tbody></table>"
|
||||
f"</section>"
|
||||
)
|
||||
else:
|
||||
# Fallback if snapshot is missing — render the core fields directly.
|
||||
def row(label: str, value: Any) -> str:
|
||||
return f"<tr><th>{_pdf_escape(label)}</th><td>{render_value(value)}</td></tr>"
|
||||
|
||||
owner_rows = (
|
||||
row("Name", data.fullName)
|
||||
+ row("Email", str(data.email))
|
||||
+ row("Phone", data.phone)
|
||||
+ row("Address", data.address)
|
||||
)
|
||||
dog_rows = (
|
||||
row("Dog", data.dogName)
|
||||
+ row("Breed", data.dogBreed)
|
||||
+ row("Date of birth", data.dogAge or "")
|
||||
+ row("Services", data.servicesNeeded)
|
||||
+ row("Temperament / routine", data.temperament)
|
||||
+ row("Medical notes", data.medicalNotes)
|
||||
+ row("Home access", data.accessInstructions)
|
||||
)
|
||||
safety_rows = (
|
||||
row("Vet clinic", data.vetName)
|
||||
+ row("Vet phone", data.vetPhone)
|
||||
+ row("Emergency contact", data.emergencyContactName)
|
||||
+ row("Emergency phone", data.emergencyContactPhone)
|
||||
+ row("Council registration", "Confirmed" if data.councilRegistrationConfirmed else "Not confirmed")
|
||||
+ row("Vaccinations", "Confirmed" if data.vaccinationsConfirmed else "Not confirmed")
|
||||
+ row("Emergency vet consent", "Confirmed" if data.emergencyVetConsent else "Not confirmed")
|
||||
+ row("Declaration", "Signed" if data.termsAccepted else "Not signed")
|
||||
)
|
||||
sections_html_parts.append(
|
||||
f"<section class='pdf-section'><h2>Owner Details</h2><table class='pdf-table'><tbody>{owner_rows}</tbody></table></section>"
|
||||
f"<section class='pdf-section'><h2>Dog Details</h2><table class='pdf-table'><tbody>{dog_rows}</tbody></table></section>"
|
||||
f"<section class='pdf-section'><h2>Safety</h2><table class='pdf-table'><tbody>{safety_rows}</tbody></table></section>"
|
||||
)
|
||||
|
||||
signature_html = ""
|
||||
if data.signatureDataUrl:
|
||||
signature_html = (
|
||||
"<section class='pdf-section pdf-signature'>"
|
||||
"<h2>Signature</h2>"
|
||||
f"<img src='{data.signatureDataUrl}' alt='Client signature'>"
|
||||
f"<div class='pdf-signed-line'>Signed by {_pdf_escape(data.fullName)} on {_pdf_escape(submitted_at)}</div>"
|
||||
"</section>"
|
||||
)
|
||||
|
||||
body_html = "".join(sections_html_parts) + signature_html
|
||||
|
||||
return f"""<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<title>Goodwalk onboarding form — {_pdf_escape(data.fullName)}</title>
|
||||
<style>
|
||||
@page {{
|
||||
size: A4;
|
||||
margin: 14mm 12mm 14mm 12mm;
|
||||
}}
|
||||
* {{ box-sizing: border-box; }}
|
||||
html, body {{
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
background: #ffffff;
|
||||
color: #000000;
|
||||
font-family: Arial, Helvetica, sans-serif;
|
||||
font-size: 10.5pt;
|
||||
line-height: 1.4;
|
||||
}}
|
||||
.pdf-doc {{ width: 100%; }}
|
||||
.pdf-header {{
|
||||
width: 100%;
|
||||
border-bottom: 1.5pt solid #000;
|
||||
padding-bottom: 8pt;
|
||||
margin-bottom: 14pt;
|
||||
}}
|
||||
.pdf-header h1 {{
|
||||
font-size: 18pt;
|
||||
font-weight: 700;
|
||||
margin: 0 0 4pt 0;
|
||||
letter-spacing: 0.5pt;
|
||||
text-transform: uppercase;
|
||||
}}
|
||||
.pdf-header .pdf-meta {{
|
||||
font-size: 9.5pt;
|
||||
color: #000;
|
||||
}}
|
||||
.pdf-section {{
|
||||
width: 100%;
|
||||
margin: 0 0 14pt 0;
|
||||
page-break-inside: avoid;
|
||||
}}
|
||||
.pdf-section h2 {{
|
||||
font-size: 11pt;
|
||||
font-weight: 700;
|
||||
text-transform: uppercase;
|
||||
letter-spacing: 0.6pt;
|
||||
margin: 0 0 6pt 0;
|
||||
padding: 0 0 3pt 0;
|
||||
border-bottom: 0.75pt solid #000;
|
||||
}}
|
||||
table.pdf-table {{
|
||||
width: 100%;
|
||||
border-collapse: collapse;
|
||||
table-layout: fixed;
|
||||
}}
|
||||
table.pdf-table th,
|
||||
table.pdf-table td {{
|
||||
text-align: left;
|
||||
vertical-align: top;
|
||||
padding: 5pt 8pt;
|
||||
border-bottom: 0.4pt solid #000;
|
||||
word-wrap: break-word;
|
||||
overflow-wrap: break-word;
|
||||
}}
|
||||
table.pdf-table th {{
|
||||
width: 34%;
|
||||
font-weight: 700;
|
||||
background: #ffffff;
|
||||
}}
|
||||
table.pdf-table td {{
|
||||
width: 66%;
|
||||
font-weight: 400;
|
||||
}}
|
||||
.pdf-signature img {{
|
||||
display: block;
|
||||
max-width: 70%;
|
||||
max-height: 60mm;
|
||||
height: auto;
|
||||
border: 0.5pt solid #000;
|
||||
padding: 4pt;
|
||||
margin: 4pt 0 6pt 0;
|
||||
background: #fff;
|
||||
}}
|
||||
.pdf-signed-line {{
|
||||
font-size: 9.5pt;
|
||||
border-top: 0.4pt solid #000;
|
||||
padding-top: 4pt;
|
||||
margin-top: 4pt;
|
||||
}}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="pdf-doc">
|
||||
<header class="pdf-header">
|
||||
<h1>Goodwalk Onboarding Form</h1>
|
||||
<div class="pdf-meta">
|
||||
<strong>{_pdf_escape(data.fullName)}</strong> · {_pdf_escape(data.dogName)} · Submitted {_pdf_escape(submitted_at)}
|
||||
</div>
|
||||
</header>
|
||||
{body_html}
|
||||
</div>
|
||||
</body>
|
||||
</html>"""
|
||||
|
||||
|
||||
def _render_pdf_sync(html: str) -> bytes:
|
||||
from weasyprint import HTML # imported lazily so unit tests don't require the native libs
|
||||
return HTML(string=html).write_pdf()
|
||||
@@ -1678,6 +1977,9 @@ async def _startup_mail_check() -> None:
|
||||
except Exception:
|
||||
logger.exception("Admin state refresh from postgres failed; using JSON snapshot")
|
||||
|
||||
# 3. Merge any shipped legacy seed (add-only — never clobbers live entries).
|
||||
await _merge_legacy_seed_if_present()
|
||||
|
||||
try:
|
||||
await _send_startup_test_email()
|
||||
except Exception:
|
||||
@@ -2728,6 +3030,42 @@ async def owner_render_message(data: RenderMessageRequest, request: Request):
|
||||
return {"ok": True, "html": html}
|
||||
|
||||
|
||||
@app.post("/owner/render-welcome-pack")
|
||||
async def owner_render_welcome_pack(data: WelcomePackEmailRequest, request: Request):
|
||||
"""Render the welcome pack email as HTML for in-modal preview."""
|
||||
await _require_owner_email(request)
|
||||
|
||||
email = str(data.email).strip().lower()
|
||||
profile = _client_profiles.get(email, {})
|
||||
owner_name = str(profile.get("fullName", "")).strip()
|
||||
dog_name = str(profile.get("dogName", "")).strip()
|
||||
|
||||
html = _welcome_pack_email_html(
|
||||
owner_name,
|
||||
dog_name,
|
||||
_trimmed(data.serviceType),
|
||||
_trimmed(data.priceDetails),
|
||||
_trimmed(data.startDate),
|
||||
)
|
||||
return {"ok": True, "html": html}
|
||||
|
||||
|
||||
@app.post("/owner/render-birthday-email")
|
||||
async def owner_render_birthday_email(data: BirthdayEmailRequest, request: Request):
|
||||
"""Render the birthday email as HTML for in-modal preview."""
|
||||
await _require_owner_email(request)
|
||||
|
||||
email = str(data.email).strip().lower()
|
||||
profile = _client_profiles.get(email, {})
|
||||
if not profile:
|
||||
raise HTTPException(status_code=404, detail="Client profile not found.")
|
||||
|
||||
owner_name = str(profile.get("fullName", "")).strip()
|
||||
dog_name = str(profile.get("dogName", "")).strip()
|
||||
html = _birthday_email_html(owner_name, dog_name)
|
||||
return {"ok": True, "html": html}
|
||||
|
||||
|
||||
@app.post("/owner/send-message")
|
||||
async def owner_send_message(data: SendMessageRequest, request: Request):
|
||||
request_id = getattr(request.state, "request_id", uuid.uuid4().hex[:8])
|
||||
@@ -2896,6 +3234,8 @@ async def owner_pending_onboarding(request: Request):
|
||||
continue
|
||||
if profile.get("onboardingCompleted"):
|
||||
continue
|
||||
if not _client_is_reachable(profile):
|
||||
continue
|
||||
|
||||
pending_clients.append({
|
||||
"email": email,
|
||||
@@ -2947,6 +3287,8 @@ async def owner_completed_onboarding(request: Request):
|
||||
continue
|
||||
if not profile.get("onboardingCompleted"):
|
||||
continue
|
||||
if not _client_is_reachable(profile):
|
||||
continue
|
||||
|
||||
completed_clients.append({
|
||||
"email": email,
|
||||
@@ -3011,6 +3353,7 @@ async def owner_all_clients(request: Request):
|
||||
if email == OWNER_EMAIL.strip().lower():
|
||||
continue
|
||||
|
||||
lifecycle = profile.get("lifecycle") if isinstance(profile.get("lifecycle"), dict) else None
|
||||
clients.append({
|
||||
"email": email,
|
||||
"fullName": profile.get("fullName", ""),
|
||||
@@ -3018,6 +3361,7 @@ async def owner_all_clients(request: Request):
|
||||
"dogName": profile.get("dogName", ""),
|
||||
"dogBreed": profile.get("dogBreed", ""),
|
||||
"status": "completed" if profile.get("onboardingCompleted") else "pending",
|
||||
"lifecycle": lifecycle or {"status": "active", "reason": "", "changedAt": "", "changedBy": ""},
|
||||
"lastActivityAt": profile.get("onboardingSubmittedAt", "") or profile.get("lastEnquiryAt", "") or profile.get("welcomePackSentAt", ""),
|
||||
"welcomePackSentAt": profile.get("welcomePackSentAt", ""),
|
||||
})
|
||||
@@ -3068,6 +3412,8 @@ async def owner_birthdays(request: Request):
|
||||
continue
|
||||
if not profile.get("onboardingCompleted"):
|
||||
continue
|
||||
if not _client_is_reachable(profile):
|
||||
continue
|
||||
|
||||
upcoming = _upcoming_birthday_date(str(profile.get("dogAge", "")), today)
|
||||
if not upcoming:
|
||||
@@ -3255,6 +3601,56 @@ async def owner_send_birthday_email(data: BirthdayEmailRequest, request: Request
|
||||
return {"ok": True, "sentAt": datetime.now().isoformat(timespec="seconds"), "preview": bool(data.preview)}
|
||||
|
||||
|
||||
@app.post("/owner/client-status")
|
||||
async def owner_client_status(data: ClientStatusUpdate, request: Request):
|
||||
"""Set a client's lifecycle status (active / paused / cancelled / archived).
|
||||
|
||||
Soft-delete only: no client record is ever removed. Each change is recorded
|
||||
in the profile's lifecycleHistory list and the global activity feed.
|
||||
"""
|
||||
request_id = getattr(request.state, "request_id", uuid.uuid4().hex[:8])
|
||||
owner_email = await _require_owner_email(request)
|
||||
|
||||
email = str(data.email).strip().lower()
|
||||
profile = _client_profiles.get(email)
|
||||
if not profile:
|
||||
raise HTTPException(status_code=404, detail="Client not found.")
|
||||
|
||||
reason = (data.reason or "").strip()[:500]
|
||||
now_iso = datetime.now().isoformat(timespec="seconds")
|
||||
|
||||
existing_history = profile.get("lifecycleHistory")
|
||||
history: list[dict[str, Any]] = list(existing_history) if isinstance(existing_history, list) else []
|
||||
history.append({
|
||||
"status": data.status,
|
||||
"reason": reason,
|
||||
"changedAt": now_iso,
|
||||
"changedBy": owner_email,
|
||||
})
|
||||
# Cap history to a sensible size so the JSON file doesn't grow unbounded.
|
||||
history = history[-50:]
|
||||
|
||||
lifecycle = {
|
||||
"status": data.status,
|
||||
"reason": reason,
|
||||
"changedAt": now_iso,
|
||||
"changedBy": owner_email,
|
||||
}
|
||||
|
||||
await _store_client_profile(email, {
|
||||
"lifecycle": lifecycle,
|
||||
"lifecycleHistory": history,
|
||||
})
|
||||
|
||||
await admin_db.record_event(
|
||||
event_type="owner_client_status_changed",
|
||||
request_id=request_id, actor_email=owner_email, ip=_get_ip(request), status="ok",
|
||||
detail={"clientEmail": email, "status": data.status, "reason": reason},
|
||||
)
|
||||
logger.info("[%s] owner: %s set %s -> %s", request_id, owner_email, email, data.status)
|
||||
return {"ok": True, "email": email, "lifecycle": lifecycle}
|
||||
|
||||
|
||||
@app.post("/owner/birthday-auto-send")
|
||||
async def owner_birthday_auto_send(data: BirthdayAutoSendRequest, request: Request):
|
||||
owner_email = await _require_owner_email(request)
|
||||
@@ -3285,6 +3681,10 @@ async def submit_booking(data: BookingSubmission, request: Request):
|
||||
ip = _get_ip(request)
|
||||
browser = _parse_ua(request.headers.get("user-agent", ""))
|
||||
|
||||
if _is_deploy_smoke(request):
|
||||
logger.info("[%s] /submit deploy-smoke bypass (no email, no db write)", request_id)
|
||||
return {"ok": True, "request_id": request_id, "smoke": True}
|
||||
|
||||
await _enforce_submit_rate_limits(request_id, ip, str(data.email))
|
||||
_enforce_form_timing(request_id, data)
|
||||
|
||||
@@ -3625,6 +4025,10 @@ async def submit_onboarding(data: OnboardingSubmission, request: Request):
|
||||
ip = _get_ip(request)
|
||||
browser = _parse_ua(request.headers.get("user-agent", ""))
|
||||
|
||||
if _is_deploy_smoke(request):
|
||||
logger.info("[%s] /onboarding-submit deploy-smoke bypass (no email, no db write)", request_id)
|
||||
return {"ok": True, "request_id": request_id, "smoke": True}
|
||||
|
||||
await _enforce_submit_rate_limits(request_id, ip, str(data.email))
|
||||
_enforce_form_timing(request_id, data)
|
||||
|
||||
@@ -3670,7 +4074,8 @@ async def submit_onboarding(data: OnboardingSubmission, request: Request):
|
||||
if birthday_attachment:
|
||||
attachments.append(birthday_attachment)
|
||||
if ONBOARDING_PDF_ATTACHMENT_ENABLED:
|
||||
pdf_attachment = await _signed_form_pdf_attachment(owner_html, data.fullName, "onboarding", request_id)
|
||||
pdf_html = owner_onboarding_pdf_html(data)
|
||||
pdf_attachment = await _signed_form_pdf_attachment(pdf_html, data.fullName, "onboarding", request_id)
|
||||
if pdf_attachment:
|
||||
attachments.append(pdf_attachment)
|
||||
if attachments:
|
||||
@@ -3755,6 +4160,10 @@ async def submit_contract(data: ContractSubmission, request: Request):
|
||||
ip = _get_ip(request)
|
||||
browser = _parse_ua(request.headers.get("user-agent", ""))
|
||||
|
||||
if _is_deploy_smoke(request):
|
||||
logger.info("[%s] /contract-submit deploy-smoke bypass (no email, no db write)", request_id)
|
||||
return {"ok": True, "request_id": request_id, "smoke": True}
|
||||
|
||||
await _enforce_submit_rate_limits(request_id, ip, str(data.email))
|
||||
_enforce_form_timing(request_id, data)
|
||||
|
||||
|
||||
@@ -1,422 +0,0 @@
|
||||
# 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 `<title>` / `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 `<title>` / `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.
|
||||
@@ -1,197 +0,0 @@
|
||||
# Goodwalk Marketing Voice
|
||||
|
||||
A practical guide for writing site copy that sells without sounding like it's selling.
|
||||
|
||||
## The voice in one line
|
||||
|
||||
**A trusted neighbour who happens to be brilliant at this.** Calm, certain, warm, specific. Not corporate. Not chirpy. Not over-promising.
|
||||
|
||||
## What we're borrowing from Apple
|
||||
|
||||
Apple's marketing works because it does three things ruthlessly:
|
||||
|
||||
1. **Leads with the outcome, not the process.** "A thousand songs in your pocket" — not "5GB of solid-state storage."
|
||||
2. **Makes the decision feel small.** Confident, declarative sentences. No hedging.
|
||||
3. **Cuts every word the meaning doesn't need.** Short. Then one longer line for texture. Then short again.
|
||||
|
||||
For a service business, the equivalent is selling the **evening** (calm dog, settled house, no guilt), not the **walk** (60 minutes, pickup included, group size 4–8).
|
||||
|
||||
## Voice attributes
|
||||
|
||||
| Attribute | What it means | What it isn't |
|
||||
|---|---|---|
|
||||
| **Calm** | Even cadence. No exclamation marks. No "amazing!" or "incredible!" | Hyped, sales-y |
|
||||
| **Certain** | "We do X." Not "We try to X" or "We may be able to X." | Arrogant, brash |
|
||||
| **Warm** | Real feeling for dogs and owners. "Your dog comes home tired and happy." | Saccharine, cutesy ("fur babies", "pawsome") |
|
||||
| **Specific** | Names suburbs, parks, times. Numbers when they help. | Vague ("various", "a wide range", "we offer") |
|
||||
| **Honest** | If a service isn't right for a dog, we say so. | False scarcity, manipulative urgency |
|
||||
|
||||
## Principles
|
||||
|
||||
### 1. Lead with the customer's win, not your feature
|
||||
|
||||
Open every section with what the **owner gets** or what the **dog feels**. The mechanism comes second.
|
||||
|
||||
> ❌ "Tiny Gang Pack Walks are built for Auckland Central owners of small and medium dogs who want a reliable weekly routine."
|
||||
>
|
||||
> ✅ "Your dog comes home tired and happy. You stop worrying through the workday. That's the whole point."
|
||||
|
||||
### 2. Cut every hedge
|
||||
|
||||
Search-and-destroy these words: *can, may, might, try to, more, genuinely, properly, generally, often, typically, possibly.* Each one quietly weakens the sentence.
|
||||
|
||||
> ❌ "Walks tailored to your dog's pace, confidence, and routine."
|
||||
>
|
||||
> ✅ "Built around your dog. Their pace. Their walk."
|
||||
|
||||
### 3. Short. Then long. Then short.
|
||||
|
||||
Vary the rhythm. A wall of medium-length sentences is the most boring possible cadence.
|
||||
|
||||
> ❌ "Goodwalk Tiny Gang Pack Walks are built for Auckland Central owners of small and medium dogs who want a reliable weekly routine, a well-exercised dog, and more peace of mind during the workday."
|
||||
>
|
||||
> ✅ "A walk your dog looks forward to. A routine you don't have to manage. Pickup, walk, drop-off, photo update — every time, without you having to ask."
|
||||
|
||||
### 4. Active voice, present tense
|
||||
|
||||
Things happen. We do them. Your dog enjoys them. Avoid "are designed to" / "is intended for" / "can be tailored."
|
||||
|
||||
> ❌ "Our visits are intended to provide enrichment and support during the day."
|
||||
>
|
||||
> ✅ "We visit. We play. We feed. You get a photo when we leave."
|
||||
|
||||
### 5. Replace abstract nouns with concrete verbs
|
||||
|
||||
"Provide structure" → "settles them." "Build confidence" → "they stop pulling on the lead." "Ensure consistency" → "same walker, every time."
|
||||
|
||||
### 6. Specifics build trust faster than adjectives
|
||||
|
||||
"A well-loved local park" tells me nothing. "Western Springs at 9:15, Cornwall Park on Wednesdays" tells me you're real.
|
||||
|
||||
### 7. Sell the relief
|
||||
|
||||
Owners aren't buying a walk. They're buying: a quieter evening, a guilt-free workday, one fewer thing to manage. Name those.
|
||||
|
||||
### 8. One idea per sentence
|
||||
|
||||
If you wrote a comma, ask whether it should be a full stop.
|
||||
|
||||
## Patterns we use
|
||||
|
||||
### Headlines
|
||||
|
||||
Two flavours, used purposefully:
|
||||
|
||||
- **Outcome line:** "Come home to a calm, happy dog."
|
||||
- **Definitional line:** "Pack walks for small dogs that actually suit small dogs."
|
||||
|
||||
Avoid: "Welcome to Goodwalk." / "About Us." / "Our Services."
|
||||
|
||||
### Subheads / leads
|
||||
|
||||
One sentence. Says what the section delivers, not what it is.
|
||||
|
||||
> ❌ "About our pack walks"
|
||||
>
|
||||
> ✅ "Four to eight dogs. Same walker every time. Home by mid-afternoon."
|
||||
|
||||
### Body copy
|
||||
|
||||
- 1–3 short paragraphs max per section
|
||||
- Lead sentence is the most important; treat it like a headline
|
||||
- One link or CTA per paragraph, max
|
||||
|
||||
### CTAs
|
||||
|
||||
Action + outcome, never just "Submit" or "Learn more."
|
||||
|
||||
| ❌ | ✅ |
|
||||
|---|---|
|
||||
| Submit | Book a free Meet & Greet |
|
||||
| Learn more | See if Tiny Gang fits your dog |
|
||||
| Contact us | Talk to Aless |
|
||||
| Get started | Start with a Meet & Greet |
|
||||
|
||||
### FAQ answers
|
||||
|
||||
- First sentence answers the question completely
|
||||
- Second sentence (if any) adds the texture
|
||||
- No "Great question!" / "Glad you asked"
|
||||
|
||||
> Q: How big are the pack walks?
|
||||
>
|
||||
> ✅ "4–8 dogs, carefully matched on size and energy. We never run oversized packs — the small group size is the whole point."
|
||||
|
||||
## Words and phrases
|
||||
|
||||
### Use
|
||||
|
||||
- **You / your dog** — far more than "owners" or "clients"
|
||||
- **We** — direct, owned. Not "the team" or "our walkers"
|
||||
- **Walk, visit, pickup, drop-off** — the customer's words
|
||||
- **Tiny Gang** — our signature, use sparingly so it stays distinct
|
||||
- **Auckland Central** — anchors local intent
|
||||
- Concrete park names, suburb names, times
|
||||
|
||||
### Avoid
|
||||
|
||||
- "Solutions" — never. We're not enterprise software.
|
||||
- "Services" as a noun in body copy — too distant. Name the thing.
|
||||
- "Pet parents" / "fur babies" / "pup parents" — twee
|
||||
- "Pawsome" / "pawfect" / any pun — never
|
||||
- "We are passionate about" — show, don't tell
|
||||
- "Industry-leading" / "best in class" / "premium" — empty
|
||||
- "Reach out" — say "email" or "text" or "call"
|
||||
- Exclamation marks in headlines or body copy
|
||||
|
||||
## Sentence length budget
|
||||
|
||||
- **Headlines:** ≤ 8 words
|
||||
- **Subheads:** ≤ 14 words
|
||||
- **Body sentences:** average 12–16 words, max ~24
|
||||
- **First sentence of any section:** ≤ 12 words
|
||||
|
||||
If you wrote a 30-word sentence, it's two sentences.
|
||||
|
||||
## Before / after, from the live site
|
||||
|
||||
### Hero subtitle (homepage)
|
||||
|
||||
> ❌ "Reliable dog walking for busy Auckland owners who want happier dogs, calmer evenings, and a team they can trust."
|
||||
>
|
||||
> ✅ "Reliable dog walking across Auckland Central. Happier dogs. Quieter evenings."
|
||||
|
||||
### Pack walks intro paragraph
|
||||
|
||||
> ❌ "Goodwalk Tiny Gang Pack Walks are built for Auckland Central owners of small and medium dogs who want a reliable weekly routine, a well-exercised dog, and more peace of mind during the workday."
|
||||
>
|
||||
> ✅ "Tiny Gang is built for small and medium dogs who like the right kind of company. Small groups. Same walker. A real walk, every time."
|
||||
|
||||
### Puppy visits subtitle
|
||||
|
||||
> ❌ "Toilet breaks, play, feeding, and calm one-on-one attention — at home, while you're out."
|
||||
>
|
||||
> ✅ "While you're at work, your puppy is fed, played with, and looked after. At home."
|
||||
|
||||
### Benefits-section intro
|
||||
|
||||
> ❌ "Small, compatible groups give dogs the exercise, confidence, and routine they need without the chaos of oversized pack walks."
|
||||
>
|
||||
> ✅ "Small groups. Compatible dogs. No chaos. That's why it works."
|
||||
|
||||
## A 60-second editing pass
|
||||
|
||||
Before any new copy ships, run it through this:
|
||||
|
||||
1. **Cut 20%.** If you can't, cut 10%.
|
||||
2. **First sentence test.** Could it be a headline? If not, rewrite.
|
||||
3. **Hedge sweep.** Delete every *can/may/might/try to/generally/typically* and re-read. Most are improvements.
|
||||
4. **Active voice check.** Search for "is/are [verb-ed] by" or "is intended to" and rewrite.
|
||||
5. **Specific vs vague.** Replace one vague phrase per paragraph with a real name, number, or detail.
|
||||
6. **Read it aloud.** If you take a breath mid-sentence, it's too long.
|
||||
|
||||
## When to break these rules
|
||||
|
||||
- **Legal pages, contracts, privacy.** Be precise and complete, not punchy.
|
||||
- **Onboarding instructions.** Clarity > rhythm.
|
||||
- **Genuine warmth moments.** A short, slightly longer line about a dog or a moment is allowed — it's the texture. Just don't make it the default.
|
||||
@@ -225,6 +225,21 @@ server {
|
||||
proxy_set_header X-Forwarded-Proto $scheme;
|
||||
}
|
||||
|
||||
location /api/contract-submit {
|
||||
if (-f /etc/nginx/conf.d/maintenance.flag) {
|
||||
return 503;
|
||||
}
|
||||
|
||||
set $goodwalk_mail_api goodwalk_svelte_mail_api:8000;
|
||||
limit_req zone=goodwalk_limit burst=10 nodelay;
|
||||
proxy_pass http://$goodwalk_mail_api/contract-submit;
|
||||
proxy_http_version 1.1;
|
||||
proxy_set_header Host $host;
|
||||
proxy_set_header X-Real-IP $remote_addr;
|
||||
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||
proxy_set_header X-Forwarded-Proto $scheme;
|
||||
}
|
||||
|
||||
location /api/auth/ {
|
||||
set $goodwalk_mail_api goodwalk_svelte_mail_api:8000;
|
||||
limit_req zone=goodwalk_limit burst=10 nodelay;
|
||||
|
||||
@@ -0,0 +1,234 @@
|
||||
#!/usr/bin/env node
|
||||
// Builds mail-api/legacy-clients-seed.json from the three cleaned legacy JSON
|
||||
// files. The seed file is shipped into the mail-api Docker image and merged
|
||||
// into _client_profiles on first boot — never overwriting a live entry.
|
||||
//
|
||||
// Inputs:
|
||||
// data/legacy-clients.json
|
||||
// data/legacy-onboarding.json
|
||||
// data/legacy-contracts.json
|
||||
//
|
||||
// Output:
|
||||
// mail-api/legacy-clients-seed.json (email -> profile dict)
|
||||
//
|
||||
// The output shape mirrors the live _client_profiles entries that the mail-api
|
||||
// already serializes to JSON, so the merge path doesn't need to translate
|
||||
// anything.
|
||||
|
||||
import { readFileSync, writeFileSync, existsSync } from 'node:fs';
|
||||
import { dirname, resolve } from 'node:path';
|
||||
import { fileURLToPath } from 'node:url';
|
||||
|
||||
const __dirname = dirname(fileURLToPath(import.meta.url));
|
||||
const ROOT = resolve(__dirname, '..');
|
||||
const CLIENTS_FILE = resolve(ROOT, 'data/legacy-clients.json');
|
||||
const ONBOARDING_FILE = resolve(ROOT, 'data/legacy-onboarding.json');
|
||||
const CONTRACTS_FILE = resolve(ROOT, 'data/legacy-contracts.json');
|
||||
const OUTPUT = resolve(ROOT, 'mail-api/legacy-clients-seed.json');
|
||||
|
||||
for (const p of [CLIENTS_FILE, ONBOARDING_FILE, CONTRACTS_FILE]) {
|
||||
if (!existsSync(p)) {
|
||||
console.error(`Required legacy file missing: ${p}`);
|
||||
process.exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
const clientsDoc = JSON.parse(readFileSync(CLIENTS_FILE, 'utf8'));
|
||||
const onboardingDoc = JSON.parse(readFileSync(ONBOARDING_FILE, 'utf8'));
|
||||
const contractsDoc = JSON.parse(readFileSync(CONTRACTS_FILE, 'utf8'));
|
||||
|
||||
const onboardingByEntryId = new Map();
|
||||
for (const rec of onboardingDoc.records ?? []) {
|
||||
if (rec?.legacy?.entryId) onboardingByEntryId.set(String(rec.legacy.entryId), rec);
|
||||
}
|
||||
|
||||
const contractsByEntryId = new Map();
|
||||
for (const rec of contractsDoc.records ?? []) {
|
||||
if (rec?.legacy?.entryId) contractsByEntryId.set(String(rec.legacy.entryId), rec);
|
||||
}
|
||||
|
||||
const importedAt = new Date().toISOString();
|
||||
|
||||
function pickOnboarding(client) {
|
||||
// Take the latest onboarding entry referenced for this client. Entry IDs
|
||||
// are numeric — higher == newer.
|
||||
const ids = (client.onboardingEntryIds ?? [])
|
||||
.map((id) => String(id))
|
||||
.filter((id) => onboardingByEntryId.has(id))
|
||||
.sort((a, b) => Number(b) - Number(a));
|
||||
return ids.length ? onboardingByEntryId.get(ids[0]) : null;
|
||||
}
|
||||
|
||||
function pickContract(client) {
|
||||
const ids = (client.contractEntryIds ?? [])
|
||||
.map((id) => String(id))
|
||||
.filter((id) => contractsByEntryId.has(id))
|
||||
.sort((a, b) => Number(b) - Number(a));
|
||||
return ids.length ? contractsByEntryId.get(ids[0]) : null;
|
||||
}
|
||||
|
||||
function nonEmpty(v) {
|
||||
return typeof v === 'string' ? v.trim() : v;
|
||||
}
|
||||
|
||||
function fullName(client) {
|
||||
const parts = [client.firstName, client.lastName].map(nonEmpty).filter(Boolean);
|
||||
return parts.join(' ');
|
||||
}
|
||||
|
||||
function buildProfile(client) {
|
||||
const onboarding = pickOnboarding(client);
|
||||
const contract = pickContract(client);
|
||||
|
||||
const dog = (client.dogs || [])[0] || {};
|
||||
const dogName = dog.name || onboarding?.dog?.name || '';
|
||||
const dogBreed = dog.breed || onboarding?.dog?.breed || '';
|
||||
const dogDob = dog.dateOfBirth || onboarding?.dog?.dateOfBirth || '';
|
||||
|
||||
// Pick a submittedAt timestamp from the best signal we have.
|
||||
const submittedAt =
|
||||
onboarding?.declaration?.signedOn ||
|
||||
onboarding?.legacy?.entryDate ||
|
||||
contract?.consent?.signedOn ||
|
||||
contract?.legacy?.entryDate ||
|
||||
'';
|
||||
|
||||
// Build the snapshot the admin dashboard expects when viewing a completed
|
||||
// client. The shape matches what OnboardingPage.svelte writes today:
|
||||
// { submittedAt, sections: [{ title, icon, fields: [{label, value}] }] }.
|
||||
const sections = [];
|
||||
if (onboarding) {
|
||||
const yn = (v) => (v === true ? 'Yes' : v === false ? 'No' : '—');
|
||||
const fv = (v) => (typeof v === 'string' && v.trim()) ? v.trim() : '—';
|
||||
sections.push({
|
||||
title: 'Owner Details',
|
||||
icon: 'fas fa-user',
|
||||
fields: [
|
||||
{ label: 'Owner First Name', value: fv(onboarding.owner?.firstName) },
|
||||
{ label: 'Owner Surname', value: fv(onboarding.owner?.lastName) },
|
||||
{ label: 'Email', value: fv(client.email) },
|
||||
{ label: 'Contact Number', value: fv(client.phoneRaw || client.phone) },
|
||||
{ label: 'Home Address', value: fv(client.address) },
|
||||
],
|
||||
});
|
||||
sections.push({
|
||||
title: 'Dog Details',
|
||||
icon: 'fas fa-paw',
|
||||
fields: [
|
||||
{ label: 'Dog Name', value: fv(onboarding.dog?.name) },
|
||||
{ label: 'Dog Surname', value: fv(onboarding.dog?.surname) },
|
||||
{ label: 'Breed', value: fv(onboarding.dog?.breed) },
|
||||
{ label: 'Date of Birth', value: fv(onboarding.dog?.dateOfBirth) },
|
||||
],
|
||||
});
|
||||
sections.push({
|
||||
title: 'Vet & Emergency',
|
||||
icon: 'fas fa-heart-pulse',
|
||||
fields: [
|
||||
{ label: 'Vet Name', value: fv(onboarding.vet?.name) },
|
||||
{ label: 'Vet Address', value: fv(onboarding.vet?.address) },
|
||||
{ label: 'Vet Phone', value: fv(onboarding.vet?.phoneRaw || onboarding.vet?.phone) },
|
||||
{ label: 'Emergency Contact Name', value: fv(onboarding.emergencyContact?.name) },
|
||||
{ label: 'Emergency Contact Phone', value: fv(onboarding.emergencyContact?.phoneRaw || onboarding.emergencyContact?.phone) },
|
||||
],
|
||||
});
|
||||
sections.push({
|
||||
title: 'Health',
|
||||
icon: 'fas fa-stethoscope',
|
||||
fields: [
|
||||
{ label: 'Vaccinations current?', value: yn(onboarding.health?.vaccinated) },
|
||||
{ label: 'Food allergies', value: onboarding.health?.foodAllergy?.present ? fv(onboarding.health.foodAllergy.detail) : yn(onboarding.health?.foodAllergy?.present) },
|
||||
{ label: 'Environmental allergies', value: onboarding.health?.environmentalAllergy?.present ? fv(onboarding.health.environmentalAllergy.detail) : yn(onboarding.health?.environmentalAllergy?.present) },
|
||||
{ label: 'Special diet', value: onboarding.health?.specialDiet?.present ? fv(onboarding.health.specialDiet.detail) : yn(onboarding.health?.specialDiet?.present) },
|
||||
{ label: 'Medication', value: onboarding.health?.medication?.present ? fv(onboarding.health.medication.detail) : yn(onboarding.health?.medication?.present) },
|
||||
],
|
||||
});
|
||||
sections.push({
|
||||
title: 'Behaviour',
|
||||
icon: 'fas fa-bone',
|
||||
fields: [
|
||||
{ label: 'Well socialised?', value: yn(onboarding.behaviour?.wellSocialised) },
|
||||
{ label: 'Dogs interacted with weekly', value: fv(onboarding.behaviour?.dogsInteractedWeekly) },
|
||||
{ label: 'Beach notes', value: fv(onboarding.behaviour?.beachNotes) },
|
||||
{ label: 'Dog park notes', value: fv(onboarding.behaviour?.dogParkNotes) },
|
||||
{ label: 'Bite history?', value: yn(onboarding.behaviour?.biteHistory) },
|
||||
{ label: 'Reactive to dogs?', value: yn(onboarding.behaviour?.reactiveToDogs) },
|
||||
{ label: 'Reactive to animals?', value: onboarding.behaviour?.reactiveToAnimals?.reactive ? fv(onboarding.behaviour.reactiveToAnimals.detail) : yn(onboarding.behaviour?.reactiveToAnimals?.reactive) },
|
||||
{ label: 'Reactive to children?', value: yn(onboarding.behaviour?.reactiveToChildren) },
|
||||
{ label: 'Reactive to people?', value: yn(onboarding.behaviour?.reactiveToPeople) },
|
||||
{ label: 'Desexed?', value: yn(onboarding.behaviour?.desexed) },
|
||||
{ label: 'Council registered?', value: yn(onboarding.behaviour?.councilRegistered) },
|
||||
{ label: 'Leash trained?', value: yn(onboarding.behaviour?.leashTrained) },
|
||||
{ label: 'Recall rating', value: onboarding.behaviour?.recallRating != null ? String(onboarding.behaviour.recallRating) : '—' },
|
||||
{ label: 'Ever run away?', value: yn(onboarding.behaviour?.ranAwayBefore) },
|
||||
{ label: 'Behaviour in car', value: fv(onboarding.behaviour?.carBehaviour) },
|
||||
{ label: 'Known commands', value: fv(onboarding.behaviour?.knownCommands) },
|
||||
],
|
||||
});
|
||||
sections.push({
|
||||
title: 'Other',
|
||||
icon: 'fas fa-file-signature',
|
||||
fields: [
|
||||
{ label: 'Additional notes', value: fv(onboarding.misc?.additionalNotes) },
|
||||
{ label: 'Social media', value: fv(onboarding.misc?.socialMediaAccount) },
|
||||
{ label: 'How did you hear?', value: fv(onboarding.misc?.howDidYouHear) },
|
||||
{ label: 'Referred by', value: fv(onboarding.misc?.referredBy) },
|
||||
],
|
||||
});
|
||||
}
|
||||
|
||||
const profile = {
|
||||
fullName: fullName(client),
|
||||
phone: client.phone || '',
|
||||
address: client.address || '',
|
||||
dogName: dogName || '',
|
||||
dogBreed: dogBreed || '',
|
||||
dogAge: dogDob || '',
|
||||
onboardingCompleted: Boolean(onboarding),
|
||||
onboardingSubmittedAt: submittedAt || '',
|
||||
onboardingSubmission: sections.length ? { submittedAt: submittedAt || '', sections } : undefined,
|
||||
// Mark legacy imports as archived by default — they pre-date the new
|
||||
// system and shouldn't pollute "active" outreach lists. The admin can
|
||||
// promote any of them back to 'active' from the dashboard.
|
||||
lifecycle: {
|
||||
status: 'archived',
|
||||
reason: 'Imported from legacy Gravity Forms data',
|
||||
changedAt: importedAt,
|
||||
changedBy: 'legacy-import',
|
||||
},
|
||||
// Provenance so we can always trace what came from the migration.
|
||||
legacy: {
|
||||
source: 'gravity-forms-csv',
|
||||
importedAt,
|
||||
onboardingEntryIds: client.onboardingEntryIds ?? [],
|
||||
contractEntryIds: client.contractEntryIds ?? [],
|
||||
onboardingPdfUrl: onboarding?.legacy?.pdfUrl ?? null,
|
||||
contractPdfUrl: contract?.legacy?.pdfUrl ?? null,
|
||||
signatureUrl: onboarding?.declaration?.signatureUrl ?? contract?.legacy?.signatureUrl ?? null,
|
||||
},
|
||||
};
|
||||
|
||||
// Drop any keys that are empty strings so the mail-api merge path doesn't
|
||||
// store noisy blanks.
|
||||
for (const k of Object.keys(profile)) {
|
||||
if (profile[k] === '' || profile[k] === undefined) delete profile[k];
|
||||
}
|
||||
return profile;
|
||||
}
|
||||
|
||||
const out = {};
|
||||
let written = 0;
|
||||
for (const client of clientsDoc.clients ?? []) {
|
||||
const email = String(client.email || '').trim().toLowerCase();
|
||||
if (!email) continue;
|
||||
out[email] = buildProfile(client);
|
||||
written++;
|
||||
}
|
||||
|
||||
writeFileSync(OUTPUT, JSON.stringify(out, null, 2) + '\n', 'utf8');
|
||||
console.log(JSON.stringify({
|
||||
ok: true,
|
||||
emails: written,
|
||||
outputPath: OUTPUT,
|
||||
note: 'Ship this file in the mail-api Docker image. Merged on boot — never overwrites live entries.',
|
||||
}, null, 2));
|
||||
@@ -0,0 +1,360 @@
|
||||
#!/usr/bin/env node
|
||||
// Cleans the legacy Gravity Forms contract CSV into structured JSON, then
|
||||
// enriches data/legacy-onboarding.json with the owner email + postal address
|
||||
// (and missing phone) wherever an onboarding row matches a contract row.
|
||||
//
|
||||
// Input:
|
||||
// goodwalk-contract-2026-05-20.csv (repo root)
|
||||
// data/legacy-onboarding.json (produced by clean-legacy-onboarding.mjs)
|
||||
//
|
||||
// Output:
|
||||
// data/legacy-contracts.json (cleaned contracts)
|
||||
// data/legacy-onboarding.json (enriched in place)
|
||||
// data/legacy-clients.json (owner-email-keyed merged view)
|
||||
//
|
||||
// Run: node scripts/clean-legacy-contracts.mjs
|
||||
|
||||
import { readFileSync, writeFileSync, mkdirSync, existsSync } from 'node:fs';
|
||||
import { dirname, resolve } from 'node:path';
|
||||
import { fileURLToPath } from 'node:url';
|
||||
|
||||
const __dirname = dirname(fileURLToPath(import.meta.url));
|
||||
const ROOT = resolve(__dirname, '..');
|
||||
const CONTRACT_CSV = resolve(ROOT, 'goodwalk-contract-2026-05-20.csv');
|
||||
const ONBOARDING_JSON = resolve(ROOT, 'data/legacy-onboarding.json');
|
||||
const CONTRACTS_OUT = resolve(ROOT, 'data/legacy-contracts.json');
|
||||
const CLIENTS_OUT = resolve(ROOT, 'data/legacy-clients.json');
|
||||
|
||||
function parseCsv(text) {
|
||||
const rows = [];
|
||||
let cur = [];
|
||||
let val = '';
|
||||
let inQ = false;
|
||||
for (let i = 0; i < text.length; i++) {
|
||||
const c = text[i];
|
||||
if (inQ) {
|
||||
if (c === '"') {
|
||||
if (text[i + 1] === '"') { val += '"'; i++; }
|
||||
else inQ = false;
|
||||
} else val += c;
|
||||
} else {
|
||||
if (c === '"') inQ = true;
|
||||
else if (c === ',') { cur.push(val); val = ''; }
|
||||
else if (c === '\n') { cur.push(val); rows.push(cur); cur = []; val = ''; }
|
||||
else if (c === '\r') { /* skip */ }
|
||||
else val += c;
|
||||
}
|
||||
}
|
||||
if (val.length || cur.length) { cur.push(val); rows.push(cur); }
|
||||
return rows;
|
||||
}
|
||||
|
||||
const trimOrNull = (v) => {
|
||||
const s = (v ?? '').trim();
|
||||
return s ? s : null;
|
||||
};
|
||||
|
||||
const lowerKey = (v) => (v ?? '').trim().toLowerCase().replace(/\s+/g, ' ');
|
||||
|
||||
function normalizePhone(raw) {
|
||||
const original = (raw ?? '').trim();
|
||||
if (!original) return { raw: null, e164: null };
|
||||
let digits = original.replace(/[^\d+]/g, '');
|
||||
if (digits.startsWith('+')) return { raw: original, e164: '+' + digits.slice(1).replace(/\D/g, '') };
|
||||
digits = digits.replace(/\D/g, '');
|
||||
if (!digits) return { raw: original, e164: null };
|
||||
if (digits.startsWith('64')) return { raw: original, e164: '+' + digits };
|
||||
if (digits.startsWith('0')) return { raw: original, e164: '+64' + digits.slice(1) };
|
||||
return { raw: original, e164: null };
|
||||
}
|
||||
|
||||
function composeAddress({ street, line2, city, suburb, postal, country }) {
|
||||
const parts = [street, line2, suburb, city, postal, country]
|
||||
.map((p) => (p ?? '').trim())
|
||||
.filter(Boolean);
|
||||
// De-dupe consecutive identical fragments (suburb sometimes duplicates city).
|
||||
const deduped = [];
|
||||
for (const p of parts) {
|
||||
if (!deduped.length || deduped[deduped.length - 1].toLowerCase() !== p.toLowerCase()) {
|
||||
deduped.push(p);
|
||||
}
|
||||
}
|
||||
return deduped.length ? deduped.join(', ') : null;
|
||||
}
|
||||
|
||||
// -- Parse contract CSV --------------------------------------------------------
|
||||
const raw = readFileSync(CONTRACT_CSV, 'utf8').replace(/^/, '');
|
||||
const rows = parseCsv(raw);
|
||||
const headers = rows[0].map((h) => h.trim());
|
||||
const data = rows.slice(1).filter((r) => r.some((c) => (c ?? '').trim() !== ''));
|
||||
const idx = Object.fromEntries(headers.map((h, i) => [h, i]));
|
||||
const col = (row, name) => row[idx[name]] ?? '';
|
||||
|
||||
const contracts = data.map((row) => {
|
||||
const first = trimOrNull(col(row, 'Owners Name (First Name)'));
|
||||
const middle = trimOrNull(col(row, 'Owners Name (Middle)'));
|
||||
const last = trimOrNull(col(row, 'Owners Name (Last Name/Surname)'));
|
||||
const fullName = [first, middle, last].filter(Boolean).join(' ') || null;
|
||||
const phone = normalizePhone(col(row, 'Phone'));
|
||||
|
||||
return {
|
||||
legacy: {
|
||||
entryId: trimOrNull(col(row, 'Entry Id')),
|
||||
entryDate: trimOrNull(col(row, 'Entry Date')),
|
||||
dateUpdated: trimOrNull(col(row, 'Date Updated')),
|
||||
createdByUserId: trimOrNull(col(row, 'Created By (User Id)')),
|
||||
sourceUrl: trimOrNull(col(row, 'Source Url')),
|
||||
userAgent: trimOrNull(col(row, 'User Agent')),
|
||||
userIp: trimOrNull(col(row, 'User IP')),
|
||||
pdfUrl: trimOrNull(col(row, 'PDF: PDF Label')),
|
||||
signatureUrl: trimOrNull(col(row, 'Owner Signature')),
|
||||
},
|
||||
owner: {
|
||||
firstName: first,
|
||||
middleName: middle,
|
||||
lastName: last,
|
||||
fullName,
|
||||
email: trimOrNull(col(row, "Owner's email (Enter Email)")),
|
||||
phone: phone.e164,
|
||||
phoneRaw: phone.raw,
|
||||
address: composeAddress({
|
||||
street: col(row, 'Residential Address (Street Address)'),
|
||||
line2: col(row, 'Residential Address (Address Line 2)'),
|
||||
city: col(row, 'Residential Address (City)'),
|
||||
suburb: col(row, 'Residential Address (Suburb)'),
|
||||
postal: col(row, 'Residential Address (ZIP / Postal Code)'),
|
||||
country: col(row, 'Residential Address (Country)'),
|
||||
}),
|
||||
addressParts: {
|
||||
street: trimOrNull(col(row, 'Residential Address (Street Address)')),
|
||||
line2: trimOrNull(col(row, 'Residential Address (Address Line 2)')),
|
||||
suburb: trimOrNull(col(row, 'Residential Address (Suburb)')),
|
||||
city: trimOrNull(col(row, 'Residential Address (City)')),
|
||||
postalCode: trimOrNull(col(row, 'Residential Address (ZIP / Postal Code)')),
|
||||
country: trimOrNull(col(row, 'Residential Address (Country)')),
|
||||
},
|
||||
},
|
||||
dog: {
|
||||
fullName: trimOrNull(col(row, "Dog's name (include surname)")),
|
||||
},
|
||||
consent: {
|
||||
checked: (col(row, 'Consent (Consent)') ?? '').trim().toLowerCase() === 'checked',
|
||||
text: trimOrNull(col(row, 'Consent (Text)')),
|
||||
signedOn: trimOrNull(col(row, 'Date contract signed')),
|
||||
},
|
||||
};
|
||||
});
|
||||
|
||||
// -- Build contract lookup -----------------------------------------------------
|
||||
// Some owners appear twice (re-signing) — keep the highest entryId per key.
|
||||
function keepNewer(map, key, contract) {
|
||||
const existing = map.get(key);
|
||||
if (!existing) { map.set(key, contract); return; }
|
||||
const a = Number(contract.legacy.entryId) || 0;
|
||||
const b = Number(existing.legacy.entryId) || 0;
|
||||
if (a > b) map.set(key, contract);
|
||||
}
|
||||
|
||||
const byNameKey = new Map(); // "last|first"
|
||||
const byLastKey = new Map(); // "last"
|
||||
const byDogKey = new Map(); // dog full name lowercased
|
||||
const byDogFirstWord = new Map(); // first token of dog name (handles surname mismatches)
|
||||
|
||||
for (const c of contracts) {
|
||||
const last = lowerKey(c.owner.lastName);
|
||||
const first = lowerKey(c.owner.firstName);
|
||||
if (last && first) keepNewer(byNameKey, `${last}|${first}`, c);
|
||||
if (last) keepNewer(byLastKey, last, c);
|
||||
if (c.dog.fullName) {
|
||||
keepNewer(byDogKey, lowerKey(c.dog.fullName), c);
|
||||
const firstToken = lowerKey(c.dog.fullName).split(/\s+/)[0];
|
||||
if (firstToken) keepNewer(byDogFirstWord, `${firstToken}|${last}`, c);
|
||||
}
|
||||
}
|
||||
|
||||
// -- Enrich onboarding ---------------------------------------------------------
|
||||
if (!existsSync(ONBOARDING_JSON)) {
|
||||
console.error(`Onboarding JSON not found at ${ONBOARDING_JSON}. Run clean-legacy-onboarding.mjs first.`);
|
||||
process.exit(1);
|
||||
}
|
||||
const onboardingPayload = JSON.parse(readFileSync(ONBOARDING_JSON, 'utf8'));
|
||||
|
||||
let matched = 0;
|
||||
let backfilledEmail = 0;
|
||||
let backfilledAddress = 0;
|
||||
let backfilledPhone = 0;
|
||||
const unmatched = [];
|
||||
|
||||
for (const rec of onboardingPayload.records) {
|
||||
const last = lowerKey(rec.owner.lastName);
|
||||
const first = lowerKey(rec.owner.firstName);
|
||||
const dogFull = lowerKey([rec.dog.name, rec.dog.surname].filter(Boolean).join(' '));
|
||||
const dogFirst = lowerKey(rec.dog.name);
|
||||
|
||||
let match = null;
|
||||
let matchedBy = null;
|
||||
if (last && first && byNameKey.has(`${last}|${first}`)) {
|
||||
match = byNameKey.get(`${last}|${first}`);
|
||||
matchedBy = 'owner_name';
|
||||
} else if (dogFull && byDogKey.has(dogFull)) {
|
||||
match = byDogKey.get(dogFull);
|
||||
matchedBy = 'dog_full_name';
|
||||
} else if (dogFirst && last && byDogFirstWord.has(`${dogFirst}|${last}`)) {
|
||||
match = byDogFirstWord.get(`${dogFirst}|${last}`);
|
||||
matchedBy = 'dog_first_owner_last';
|
||||
} else if (last && byLastKey.has(last)) {
|
||||
// Last-resort: lone surname match. Only accept if surname is unique enough
|
||||
// (i.e. only one contract has it).
|
||||
const candidates = contracts.filter((c) => lowerKey(c.owner.lastName) === last);
|
||||
if (candidates.length === 1) {
|
||||
match = candidates[0];
|
||||
matchedBy = 'owner_last_only';
|
||||
}
|
||||
}
|
||||
|
||||
if (!match) {
|
||||
unmatched.push({
|
||||
onboardingEntryId: rec.legacy.entryId,
|
||||
owner: rec.owner.fullName,
|
||||
dog: [rec.dog.name, rec.dog.surname].filter(Boolean).join(' '),
|
||||
});
|
||||
continue;
|
||||
}
|
||||
|
||||
matched++;
|
||||
if (!rec.owner.email && match.owner.email) {
|
||||
rec.owner.email = match.owner.email;
|
||||
backfilledEmail++;
|
||||
}
|
||||
if (!rec.owner.address && match.owner.address) {
|
||||
rec.owner.address = match.owner.address;
|
||||
rec.owner.addressParts = match.owner.addressParts;
|
||||
backfilledAddress++;
|
||||
}
|
||||
if (!rec.owner.phone && match.owner.phone) {
|
||||
rec.owner.phone = match.owner.phone;
|
||||
rec.owner.phoneRaw = match.owner.phoneRaw;
|
||||
backfilledPhone++;
|
||||
}
|
||||
rec.legacy.contractMatch = {
|
||||
entryId: match.legacy.entryId,
|
||||
matchedBy,
|
||||
signedOn: match.consent.signedOn,
|
||||
contractPdfUrl: match.legacy.pdfUrl,
|
||||
};
|
||||
}
|
||||
|
||||
// -- Write outputs -------------------------------------------------------------
|
||||
mkdirSync(dirname(CONTRACTS_OUT), { recursive: true });
|
||||
|
||||
writeFileSync(
|
||||
CONTRACTS_OUT,
|
||||
JSON.stringify({
|
||||
exportedAt: new Date().toISOString(),
|
||||
source: { file: 'goodwalk-contract-2026-05-20.csv', rows: data.length, columns: headers.length },
|
||||
records: contracts,
|
||||
}, null, 2) + '\n',
|
||||
'utf8',
|
||||
);
|
||||
|
||||
onboardingPayload.enrichedAt = new Date().toISOString();
|
||||
onboardingPayload.notes = [
|
||||
...(onboardingPayload.notes ?? []),
|
||||
'Enriched from goodwalk-contract-2026-05-20.csv: owner email + postal address backfilled where a contract row matched.',
|
||||
];
|
||||
writeFileSync(ONBOARDING_JSON, JSON.stringify(onboardingPayload, null, 2) + '\n', 'utf8');
|
||||
|
||||
// -- Build a clients view keyed by email ---------------------------------------
|
||||
// This is the shape that maps most naturally to a Postgres `clients` table.
|
||||
const clientsByEmail = new Map();
|
||||
|
||||
function upsertClient(email, partial) {
|
||||
const key = (email ?? '').toLowerCase().trim();
|
||||
if (!key) return;
|
||||
const existing = clientsByEmail.get(key) ?? {
|
||||
email: key,
|
||||
firstName: null,
|
||||
lastName: null,
|
||||
phone: null,
|
||||
phoneRaw: null,
|
||||
address: null,
|
||||
addressParts: null,
|
||||
dogs: [],
|
||||
onboardingEntryIds: [],
|
||||
contractEntryIds: [],
|
||||
};
|
||||
for (const [k, v] of Object.entries(partial)) {
|
||||
if (v == null) continue;
|
||||
if (k === 'dogs') {
|
||||
for (const dog of v) {
|
||||
if (!existing.dogs.find((d) => (d.name ?? '').toLowerCase() === (dog.name ?? '').toLowerCase())) {
|
||||
existing.dogs.push(dog);
|
||||
}
|
||||
}
|
||||
} else if (k === 'onboardingEntryIds' || k === 'contractEntryIds') {
|
||||
for (const id of v) if (!existing[k].includes(id)) existing[k].push(id);
|
||||
} else if (existing[k] == null) {
|
||||
existing[k] = v;
|
||||
}
|
||||
}
|
||||
clientsByEmail.set(key, existing);
|
||||
}
|
||||
|
||||
for (const c of contracts) {
|
||||
if (!c.owner.email) continue;
|
||||
upsertClient(c.owner.email, {
|
||||
firstName: c.owner.firstName,
|
||||
lastName: c.owner.lastName,
|
||||
phone: c.owner.phone,
|
||||
phoneRaw: c.owner.phoneRaw,
|
||||
address: c.owner.address,
|
||||
addressParts: c.owner.addressParts,
|
||||
dogs: c.dog.fullName ? [{ name: c.dog.fullName, source: 'contract' }] : [],
|
||||
contractEntryIds: c.legacy.entryId ? [c.legacy.entryId] : [],
|
||||
});
|
||||
}
|
||||
|
||||
for (const rec of onboardingPayload.records) {
|
||||
if (!rec.owner.email) continue;
|
||||
const dogName = [rec.dog.name, rec.dog.surname].filter(Boolean).join(' ');
|
||||
upsertClient(rec.owner.email, {
|
||||
firstName: rec.owner.firstName,
|
||||
lastName: rec.owner.lastName,
|
||||
phone: rec.owner.phone,
|
||||
phoneRaw: rec.owner.phoneRaw,
|
||||
address: rec.owner.address,
|
||||
addressParts: rec.owner.addressParts,
|
||||
dogs: dogName ? [{
|
||||
name: dogName,
|
||||
dateOfBirth: rec.dog.dateOfBirth,
|
||||
breed: rec.dog.breed,
|
||||
source: 'onboarding',
|
||||
}] : [],
|
||||
onboardingEntryIds: rec.legacy.entryId ? [rec.legacy.entryId] : [],
|
||||
});
|
||||
}
|
||||
|
||||
writeFileSync(
|
||||
CLIENTS_OUT,
|
||||
JSON.stringify({
|
||||
exportedAt: new Date().toISOString(),
|
||||
note: 'Owner-email-keyed merged view. Maps 1:1 to a Postgres `clients` table; the `dogs` array maps to a `dogs` table with a clients_id FK.',
|
||||
clients: [...clientsByEmail.values()].sort((a, b) => a.email.localeCompare(b.email)),
|
||||
}, null, 2) + '\n',
|
||||
'utf8',
|
||||
);
|
||||
|
||||
// -- Summary -------------------------------------------------------------------
|
||||
console.log(JSON.stringify({
|
||||
contracts: contracts.length,
|
||||
contractsWithEmail: contracts.filter((c) => c.owner.email).length,
|
||||
onboardingRecords: onboardingPayload.records.length,
|
||||
matched,
|
||||
unmatched: unmatched.length,
|
||||
backfilledEmail,
|
||||
backfilledAddress,
|
||||
backfilledPhone,
|
||||
uniqueClientsByEmail: clientsByEmail.size,
|
||||
unmatchedSample: unmatched.slice(0, 10),
|
||||
outputs: { CONTRACTS_OUT, ONBOARDING_JSON, CLIENTS_OUT },
|
||||
}, null, 2));
|
||||
@@ -0,0 +1,269 @@
|
||||
#!/usr/bin/env node
|
||||
// Cleans the legacy Gravity Forms onboarding CSV export into a structured JSON
|
||||
// file ready for later import into Postgres.
|
||||
//
|
||||
// Input: dog-enrolment-form-2026-05-20.csv (repo root)
|
||||
// Output: data/legacy-onboarding.json
|
||||
//
|
||||
// Run: node scripts/clean-legacy-onboarding.mjs
|
||||
|
||||
import { readFileSync, writeFileSync, mkdirSync } from 'node:fs';
|
||||
import { dirname, resolve } from 'node:path';
|
||||
import { fileURLToPath } from 'node:url';
|
||||
|
||||
const __dirname = dirname(fileURLToPath(import.meta.url));
|
||||
const ROOT = resolve(__dirname, '..');
|
||||
const INPUT = resolve(ROOT, 'dog-enrolment-form-2026-05-20.csv');
|
||||
const OUTPUT = resolve(ROOT, 'data/legacy-onboarding.json');
|
||||
|
||||
function parseCsv(text) {
|
||||
const rows = [];
|
||||
let cur = [];
|
||||
let val = '';
|
||||
let inQ = false;
|
||||
for (let i = 0; i < text.length; i++) {
|
||||
const c = text[i];
|
||||
if (inQ) {
|
||||
if (c === '"') {
|
||||
if (text[i + 1] === '"') { val += '"'; i++; }
|
||||
else inQ = false;
|
||||
} else val += c;
|
||||
} else {
|
||||
if (c === '"') inQ = true;
|
||||
else if (c === ',') { cur.push(val); val = ''; }
|
||||
else if (c === '\n') { cur.push(val); rows.push(cur); cur = []; val = ''; }
|
||||
else if (c === '\r') { /* skip */ }
|
||||
else val += c;
|
||||
}
|
||||
}
|
||||
if (val.length || cur.length) { cur.push(val); rows.push(cur); }
|
||||
return rows;
|
||||
}
|
||||
|
||||
const trimOrNull = (v) => {
|
||||
const s = (v ?? '').trim();
|
||||
if (!s) return null;
|
||||
if (s.toLowerCase() === 'select') return null;
|
||||
return s;
|
||||
};
|
||||
|
||||
const yesNo = (v) => {
|
||||
const s = (v ?? '').trim().toLowerCase();
|
||||
if (!s || s === 'select') return null;
|
||||
if (s === 'yes') return true;
|
||||
if (s === 'no') return false;
|
||||
return null;
|
||||
};
|
||||
|
||||
// reactive-to-animals style: 'Yes'/'No'/'' or a free-text species ('Cats', 'Birds', 'Other').
|
||||
const reactiveField = (v) => {
|
||||
const s = (v ?? '').trim();
|
||||
if (!s) return { reactive: null, detail: null };
|
||||
const lower = s.toLowerCase();
|
||||
if (lower === 'yes') return { reactive: true, detail: null };
|
||||
if (lower === 'no') return { reactive: false, detail: null };
|
||||
if (lower === 'select') return { reactive: null, detail: null };
|
||||
return { reactive: true, detail: s };
|
||||
};
|
||||
|
||||
const parseFloatOrNull = (v) => {
|
||||
const s = (v ?? '').trim();
|
||||
if (!s) return null;
|
||||
const n = Number(s);
|
||||
return Number.isFinite(n) ? n : null;
|
||||
};
|
||||
|
||||
// Best-effort NZ phone normalisation. Keeps the raw, adds an e164 when we can
|
||||
// confidently produce one. Doesn't try to invent missing data.
|
||||
function normalizePhone(raw) {
|
||||
const original = (raw ?? '').trim();
|
||||
if (!original) return { raw: null, e164: null };
|
||||
let digits = original.replace(/[^\d+]/g, '');
|
||||
if (digits.startsWith('+')) {
|
||||
return { raw: original, e164: '+' + digits.slice(1).replace(/\D/g, '') };
|
||||
}
|
||||
digits = digits.replace(/\D/g, '');
|
||||
if (!digits) return { raw: original, e164: null };
|
||||
if (digits.startsWith('64')) return { raw: original, e164: '+' + digits };
|
||||
if (digits.startsWith('0')) return { raw: original, e164: '+64' + digits.slice(1) };
|
||||
return { raw: original, e164: null };
|
||||
}
|
||||
|
||||
const isoDateOrNull = (v) => {
|
||||
const s = (v ?? '').trim();
|
||||
if (!s) return null;
|
||||
// CSV already uses YYYY-MM-DD or 'YYYY-MM-DD HH:MM:SS'
|
||||
return s;
|
||||
};
|
||||
|
||||
function emailGuess(ownerName, ownerSurname) {
|
||||
// Legacy CSV doesn't include owner email. Leaving null so the Postgres
|
||||
// import treats these as needing email-claim before they can sign in.
|
||||
return null;
|
||||
}
|
||||
|
||||
const raw = readFileSync(INPUT, 'utf8').replace(/^/, '');
|
||||
const rows = parseCsv(raw);
|
||||
const headers = rows[0];
|
||||
const data = rows.slice(1).filter((r) => r.some((cell) => (cell ?? '').trim() !== ''));
|
||||
|
||||
// Column index lookups (defensive in case header order shifts).
|
||||
const idx = Object.fromEntries(headers.map((h, i) => [h.trim(), i]));
|
||||
const col = (row, name) => row[idx[name]] ?? '';
|
||||
|
||||
const records = data.map((row) => {
|
||||
const hasFoodAllergy = yesNo(col(row, 'Does your dog have any food allergies?'));
|
||||
const foodAllergyDetail = trimOrNull(col(row, 'Specify'));
|
||||
// Two columns are labelled 'Specify'. The first is food, then env, then diet,
|
||||
// then medication — pick them by position.
|
||||
const specifyCols = headers
|
||||
.map((h, i) => ({ h, i }))
|
||||
.filter((x) => x.h.trim() === 'Specify')
|
||||
.map((x) => x.i);
|
||||
const [foodSpecifyIdx, envSpecifyIdx, dietSpecifyIdx, medSpecifyIdx] = specifyCols;
|
||||
|
||||
const phone = normalizePhone(col(row, 'Owner Contact'));
|
||||
const emergencyPhone = normalizePhone(col(row, 'Emergency Contact Number'));
|
||||
const vetPhone = normalizePhone(col(row, 'Vet Contact Number'));
|
||||
|
||||
const dogBirth = isoDateOrNull(col(row, "Dog's date of birth"));
|
||||
const signedOn = isoDateOrNull(col(row, 'Date'));
|
||||
const entryDate = isoDateOrNull(col(row, 'Entry Date'));
|
||||
const updatedDate = isoDateOrNull(col(row, 'Date Updated'));
|
||||
|
||||
const ownerFirst = trimOrNull(col(row, 'Owner Name'));
|
||||
const ownerLast = trimOrNull(col(row, 'Owner Surname'));
|
||||
|
||||
return {
|
||||
legacy: {
|
||||
entryId: trimOrNull(col(row, 'Entry Id')),
|
||||
entryDate,
|
||||
dateUpdated: updatedDate,
|
||||
createdByUserId: trimOrNull(col(row, 'Created By (User Id)')),
|
||||
sourceUrl: trimOrNull(col(row, 'Source Url')),
|
||||
signatureUrl: trimOrNull(col(row, 'Signature')),
|
||||
pdfUrl: trimOrNull(col(row, 'PDF: DogOnboardingForm')),
|
||||
userAgent: trimOrNull(col(row, 'User Agent')),
|
||||
userIp: trimOrNull(col(row, 'User IP')),
|
||||
},
|
||||
owner: {
|
||||
firstName: ownerFirst,
|
||||
lastName: ownerLast,
|
||||
fullName: [ownerFirst, ownerLast].filter(Boolean).join(' ') || null,
|
||||
email: emailGuess(ownerFirst, ownerLast),
|
||||
phone: phone.e164,
|
||||
phoneRaw: phone.raw,
|
||||
address: null, // not collected in legacy form
|
||||
},
|
||||
emergencyContact: {
|
||||
name: trimOrNull(col(row, 'Emergency Contact Name')),
|
||||
phone: emergencyPhone.e164,
|
||||
phoneRaw: emergencyPhone.raw,
|
||||
},
|
||||
dog: {
|
||||
name: trimOrNull(col(row, 'Dog Name')),
|
||||
surname: trimOrNull(col(row, 'Dog Surname')),
|
||||
dateOfBirth: dogBirth,
|
||||
breed: null, // not collected in legacy form
|
||||
},
|
||||
vet: {
|
||||
name: trimOrNull(col(row, 'Vet Name')),
|
||||
address: trimOrNull(col(row, 'Vet Address')),
|
||||
phone: vetPhone.e164,
|
||||
phoneRaw: vetPhone.raw,
|
||||
},
|
||||
health: {
|
||||
vaccinated: yesNo(col(row, 'Is your dog vaccinated?')),
|
||||
foodAllergy: {
|
||||
present: hasFoodAllergy,
|
||||
detail: hasFoodAllergy === true ? trimOrNull(row[foodSpecifyIdx]) : null,
|
||||
},
|
||||
environmentalAllergy: (() => {
|
||||
const present = yesNo(col(row, 'Does your dog have any environmental allergy?'));
|
||||
return {
|
||||
present,
|
||||
detail: present === true ? trimOrNull(row[envSpecifyIdx]) : null,
|
||||
};
|
||||
})(),
|
||||
specialDiet: (() => {
|
||||
const present = yesNo(col(row, 'Is your dog on a special diet?'));
|
||||
return {
|
||||
present,
|
||||
detail: present === true ? trimOrNull(row[dietSpecifyIdx]) : null,
|
||||
};
|
||||
})(),
|
||||
medication: (() => {
|
||||
const present = yesNo(col(row, 'Is your dog taking any medication that could put him at risk during a walk'));
|
||||
return {
|
||||
present,
|
||||
detail: present === true ? trimOrNull(row[medSpecifyIdx]) : null,
|
||||
};
|
||||
})(),
|
||||
},
|
||||
behaviour: {
|
||||
wellSocialised: yesNo(col(row, 'Is your dog well socialised?')),
|
||||
dogsInteractedWeekly: trimOrNull(col(row, 'How many dogs does your dog interact with weekly (excluding your family dogs)')),
|
||||
beachNotes: trimOrNull(col(row, 'Does your dog visit the beach?')),
|
||||
dogParkNotes: trimOrNull(col(row, 'Does your dog visit dog parks frequently - How many times a week?')),
|
||||
biteHistory: yesNo(col(row, 'Does your dog have a bite history?')),
|
||||
reactiveToDogs: yesNo(col(row, 'Is your dog reactive to other dogs?')),
|
||||
reactiveToAnimals: reactiveField(col(row, 'Is your dog reactive to other animals?')),
|
||||
reactiveToChildren: yesNo(col(row, 'Is your dog reactive to children?')),
|
||||
reactiveToPeople: yesNo(col(row, 'Is your dog reactive to other people?')),
|
||||
desexed: yesNo(col(row, 'Is your dog desexed?')),
|
||||
councilRegistered: yesNo(col(row, 'Is your dog registered?')),
|
||||
leashTrained: yesNo(col(row, 'Is your dog leash trained?')),
|
||||
recallRating: parseFloatOrNull(col(row, "Rate your dog's recall from 1 to 5, with one being the lowest score and 5 the highest.")),
|
||||
ranAwayBefore: yesNo(col(row, 'Has your dog ran away before?')),
|
||||
carBehaviour: trimOrNull(col(row, "Please describe your dog's behaviour in the car")),
|
||||
knownCommands: trimOrNull(col(row, 'List of commands your dog understands')),
|
||||
},
|
||||
misc: {
|
||||
additionalNotes: trimOrNull(col(row, "Anything else you'd like to let us know?")),
|
||||
socialMediaAccount: trimOrNull(col(row, 'Social Media Account Name')),
|
||||
howDidYouHear: trimOrNull(col(row, 'How did you hear about Goodwalk?')),
|
||||
referredBy: trimOrNull(col(row, "Person's name who reffered Goodwalk to you")),
|
||||
},
|
||||
declaration: {
|
||||
signedOn,
|
||||
signatureUrl: trimOrNull(col(row, 'Signature')),
|
||||
// Legacy submissions implicitly accepted T&Cs when submitted.
|
||||
termsAccepted: true,
|
||||
emergencyVetConsent: true,
|
||||
},
|
||||
};
|
||||
});
|
||||
|
||||
mkdirSync(dirname(OUTPUT), { recursive: true });
|
||||
|
||||
const payload = {
|
||||
exportedAt: new Date().toISOString(),
|
||||
source: {
|
||||
file: 'dog-enrolment-form-2026-05-20.csv',
|
||||
rows: data.length,
|
||||
columns: headers.length,
|
||||
},
|
||||
notes: [
|
||||
'Legacy Gravity Forms export. Owner email/address were not collected by the old form.',
|
||||
'Dog breed not collected by the old form.',
|
||||
'Phone numbers retained as raw + best-effort E.164 (+64). Verify before dialling.',
|
||||
"Yes/No fields parsed to booleans; 'Select' and blanks become null.",
|
||||
"'Reactive to other animals' values like Cats/Birds/Other map to { reactive: true, detail }.",
|
||||
],
|
||||
records,
|
||||
};
|
||||
|
||||
writeFileSync(OUTPUT, JSON.stringify(payload, null, 2) + '\n', 'utf8');
|
||||
|
||||
// Print a small summary so the operator can sanity-check.
|
||||
const summary = {
|
||||
records: records.length,
|
||||
withOwnerPhone: records.filter((r) => r.owner.phone).length,
|
||||
withVetPhone: records.filter((r) => r.vet.phone).length,
|
||||
withEmergencyPhone: records.filter((r) => r.emergencyContact.phone).length,
|
||||
withSignature: records.filter((r) => r.declaration.signatureUrl).length,
|
||||
vaccinatedTrue: records.filter((r) => r.health.vaccinated === true).length,
|
||||
councilRegistered: records.filter((r) => r.behaviour.councilRegistered === true).length,
|
||||
output: OUTPUT,
|
||||
};
|
||||
console.log(JSON.stringify(summary, null, 2));
|
||||
|
After Width: | Height: | Size: 351 KiB |
|
After Width: | Height: | Size: 429 KiB |
|
After Width: | Height: | Size: 362 KiB |
@@ -0,0 +1,98 @@
|
||||
"""Drive the homepage BookingWizard in a mobile viewport and check the success modal appears."""
|
||||
from playwright.sync_api import sync_playwright
|
||||
import sys, io
|
||||
sys.stdout = io.TextIOWrapper(sys.stdout.buffer, encoding='utf-8')
|
||||
|
||||
BASE = "http://127.0.0.1:5180"
|
||||
|
||||
def run():
|
||||
with sync_playwright() as p:
|
||||
browser = p.chromium.launch()
|
||||
ctx = browser.new_context(
|
||||
viewport={"width": 390, "height": 844},
|
||||
user_agent="Mozilla/5.0 (iPhone; CPU iPhone OS 17_0 like Mac OS X) AppleWebKit/605.1.15 Mobile/15E148 Safari/604.1",
|
||||
is_mobile=True,
|
||||
has_touch=True,
|
||||
device_scale_factor=3,
|
||||
)
|
||||
page = ctx.new_page()
|
||||
console_logs = []
|
||||
page.on("console", lambda msg: console_logs.append(f"[{msg.type}] {msg.text}"))
|
||||
page.on("pageerror", lambda err: console_logs.append(f"[pageerror] {err}"))
|
||||
|
||||
def handle_submit(route):
|
||||
print(f" intercepted /api/submit -> returning 200")
|
||||
route.fulfill(status=200, content_type="application/json", body='{"ok":true}')
|
||||
page.route("**/api/submit", handle_submit)
|
||||
|
||||
print("[1] Loading homepage…")
|
||||
page.goto(BASE + "/#newlead", wait_until="networkidle", timeout=30000)
|
||||
page.locator("#newlead").scroll_into_view_if_needed()
|
||||
page.wait_for_timeout(400)
|
||||
|
||||
print("[2] Step 1 — dog details")
|
||||
# Pet name (avoid honeypot which uses autocomplete="new-password")
|
||||
page.locator('.wiz-form input[placeholder*="Teddy"]').fill("Rex")
|
||||
# Pick first service
|
||||
page.locator('.wiz-service').first.click()
|
||||
# Message (textarea)
|
||||
page.locator('.wiz-form textarea').first.fill("Friendly dog, two years old, recall fine.")
|
||||
|
||||
# Continue
|
||||
cont = page.get_by_role("button", name="Continue")
|
||||
print(f" Continue button visible: {cont.is_visible()}")
|
||||
cont.click()
|
||||
page.wait_for_timeout(1200)
|
||||
page.locator("#newlead").screenshot(path="scripts/mobile-step2.png")
|
||||
errs = page.locator('.wiz-error').all()
|
||||
print(f" visible errors after Continue: {len(errs)}")
|
||||
for e in errs:
|
||||
try: print(f" -> {e.inner_text()}")
|
||||
except: pass
|
||||
print(f" inputs after Continue: {page.locator('.wiz-form input').count()}")
|
||||
for inp in page.locator('.wiz-form input').all():
|
||||
print(f" autocomplete={inp.get_attribute('autocomplete')!r} type={inp.get_attribute('type')!r} visible={inp.is_visible()}")
|
||||
|
||||
print("[3] Step 2 — owner details")
|
||||
page.locator('.wiz-form input[autocomplete="name"]').fill("Test User")
|
||||
page.locator('.wiz-form input[type="email"]').fill("test@example.com")
|
||||
page.locator('.wiz-form input[type="tel"]').fill("0211234567")
|
||||
page.locator('.wiz-form input[autocomplete="address-level2"]').fill("Mt Eden")
|
||||
|
||||
print("[4] Submit")
|
||||
page.get_by_role("button", name="Send my details").click()
|
||||
|
||||
print("[5] Waiting for success modal…")
|
||||
try:
|
||||
page.wait_for_selector('[role="dialog"]', timeout=8000, state="attached")
|
||||
print(" PASS: dialog attached")
|
||||
except Exception:
|
||||
print(" FAIL: dialog never attached")
|
||||
return 1
|
||||
|
||||
# Snapshot immediately
|
||||
page.screenshot(path="scripts/mobile-modal-immediate.png", full_page=False)
|
||||
c0 = page.locator('canvas').count()
|
||||
d0 = page.locator('[role="dialog"]').count()
|
||||
print(f" t=0ms: dialogs={d0} canvases={c0}")
|
||||
|
||||
for delay in (200, 500, 1000, 2000):
|
||||
page.wait_for_timeout(delay)
|
||||
d = page.locator('[role="dialog"]').count()
|
||||
c = page.locator('canvas').count()
|
||||
print(f" after +{delay}ms: dialogs={d} canvases={c}")
|
||||
if d == 0:
|
||||
print(" Modal vanished!")
|
||||
page.screenshot(path=f"scripts/mobile-vanished-{delay}.png", full_page=False)
|
||||
break
|
||||
|
||||
page.screenshot(path="scripts/mobile-modal-final.png", full_page=False)
|
||||
|
||||
print("\nConsole tail:")
|
||||
for line in console_logs[-12:]:
|
||||
print(f" {line}")
|
||||
|
||||
browser.close()
|
||||
return 0
|
||||
|
||||
sys.exit(run())
|
||||
@@ -0,0 +1,87 @@
|
||||
"""Reproduce: testimonials cards blank after SPA navigation from another page."""
|
||||
from playwright.sync_api import sync_playwright
|
||||
import sys, io
|
||||
sys.stdout = io.TextIOWrapper(sys.stdout.buffer, encoding='utf-8')
|
||||
|
||||
BASE = "https://www.goodwalk.co.nz"
|
||||
|
||||
def run():
|
||||
with sync_playwright() as p:
|
||||
browser = p.chromium.launch()
|
||||
for label, ctx_opts in [
|
||||
("mobile", {"viewport": {"width": 390, "height": 844}, "is_mobile": True, "has_touch": True,
|
||||
"user_agent": "Mozilla/5.0 (iPhone; CPU iPhone OS 17_0 like Mac OS X) AppleWebKit/605.1.15 Mobile/15E148 Safari/604.1"}),
|
||||
]:
|
||||
print(f"\n=== {label} ===")
|
||||
ctx = browser.new_context(**ctx_opts)
|
||||
page = ctx.new_page()
|
||||
logs = []
|
||||
page.on("console", lambda m, L=logs: L.append(f"[{m.type}] {m.text}"))
|
||||
page.on("pageerror", lambda e, L=logs: L.append(f"[pageerror] {e}"))
|
||||
|
||||
print(" Loading homepage…")
|
||||
page.goto(BASE + "/", wait_until="networkidle", timeout=30000)
|
||||
page.wait_for_timeout(800)
|
||||
all_links = page.locator('a[href*="testimonials"]').all()
|
||||
print(f" testimonials links found: {len(all_links)}")
|
||||
for a in all_links[:5]:
|
||||
print(f" href={a.get_attribute('href')!r} visible={a.is_visible()}")
|
||||
|
||||
# Try opening mobile menu (hamburger) then click testimonials
|
||||
burger = page.locator('button[aria-label*="menu" i], button.header-burger, [aria-label*="open" i]').first
|
||||
if burger.count() and burger.is_visible():
|
||||
burger.click()
|
||||
page.wait_for_timeout(400)
|
||||
# Click first now-visible testimonials link (SPA nav, no reload)
|
||||
visible_link = None
|
||||
for a in page.locator('a[href*="testimonials"]').all():
|
||||
if a.is_visible():
|
||||
visible_link = a
|
||||
break
|
||||
if visible_link is None:
|
||||
print(" no visible link to click")
|
||||
return
|
||||
print(" clicking testimonials link…")
|
||||
with page.expect_navigation(wait_until="load", timeout=5000) as nav_info:
|
||||
visible_link.click()
|
||||
print(f" nav type detected: same-document={nav_info.value is None}")
|
||||
page.wait_for_timeout(2000)
|
||||
|
||||
page.wait_for_timeout(2500)
|
||||
print(f" url now: {page.url}")
|
||||
|
||||
section = page.locator(".testimonials-page-grid-section")
|
||||
print(f" section count: {section.count()}")
|
||||
if section.count():
|
||||
classes = section.first.get_attribute("class")
|
||||
print(f" section classes: {classes!r}")
|
||||
|
||||
cards = page.locator(".testimonials-page-card")
|
||||
print(f" cards count: {cards.count()}")
|
||||
if cards.count():
|
||||
first = cards.first
|
||||
computed = first.evaluate("el => { const cs = getComputedStyle(el); return { opacity: cs.opacity, transform: cs.transform }; }")
|
||||
print(f" first card computed: {computed}")
|
||||
last = cards.last
|
||||
last_comp = last.evaluate("el => { const cs = getComputedStyle(el); return { opacity: cs.opacity }; }")
|
||||
print(f" last card computed: {last_comp}")
|
||||
|
||||
page.screenshot(path=f"scripts/testimonials-spa-{label}.png", full_page=True)
|
||||
print(f" screenshot: scripts/testimonials-spa-{label}.png")
|
||||
|
||||
# Now scroll and recheck
|
||||
page.mouse.wheel(0, 600)
|
||||
page.wait_for_timeout(500)
|
||||
if cards.count():
|
||||
op_after = cards.first.evaluate("el => getComputedStyle(el).opacity")
|
||||
print(f" first card opacity after scroll 600px: {op_after}")
|
||||
page.screenshot(path=f"scripts/testimonials-spa-scrolled-{label}.png", full_page=True)
|
||||
|
||||
if logs:
|
||||
print(" console:")
|
||||
for l in logs[-12:]:
|
||||
print(f" {l}")
|
||||
ctx.close()
|
||||
browser.close()
|
||||
|
||||
run()
|
||||
@@ -0,0 +1,63 @@
|
||||
"""Load /testimonials and report what's actually visible."""
|
||||
from playwright.sync_api import sync_playwright
|
||||
import sys, io
|
||||
sys.stdout = io.TextIOWrapper(sys.stdout.buffer, encoding='utf-8')
|
||||
|
||||
BASE = "http://127.0.0.1:5180"
|
||||
|
||||
def run():
|
||||
with sync_playwright() as p:
|
||||
browser = p.chromium.launch()
|
||||
for label, ctx_opts in [
|
||||
("desktop", {"viewport": {"width": 1280, "height": 900}}),
|
||||
("mobile", {"viewport": {"width": 390, "height": 844}, "is_mobile": True, "has_touch": True,
|
||||
"user_agent": "Mozilla/5.0 (iPhone; CPU iPhone OS 17_0 like Mac OS X) AppleWebKit/605.1.15 Mobile/15E148 Safari/604.1"}),
|
||||
]:
|
||||
print(f"\n=== {label} ===")
|
||||
ctx = browser.new_context(**ctx_opts)
|
||||
page = ctx.new_page()
|
||||
logs = []
|
||||
page.on("console", lambda m, L=logs: L.append(f"[{m.type}] {m.text}"))
|
||||
page.on("pageerror", lambda e, L=logs: L.append(f"[pageerror] {e}"))
|
||||
|
||||
page.goto(BASE + "/testimonials", wait_until="networkidle", timeout=30000)
|
||||
page.wait_for_timeout(600)
|
||||
|
||||
section = page.locator(".testimonials-page-grid-section")
|
||||
print(f" section count: {section.count()}")
|
||||
classes = section.first.get_attribute("class") if section.count() else None
|
||||
print(f" section classes: {classes!r}")
|
||||
|
||||
cards = page.locator(".testimonials-page-card")
|
||||
print(f" cards count: {cards.count()}")
|
||||
|
||||
if cards.count():
|
||||
first = cards.first
|
||||
computed = first.evaluate("el => { const cs = getComputedStyle(el); return { opacity: cs.opacity, transform: cs.transform, display: cs.display, visibility: cs.visibility }; }")
|
||||
print(f" first card computed: {computed}")
|
||||
bb = first.bounding_box()
|
||||
print(f" first card bbox: {bb}")
|
||||
txt = first.inner_text()
|
||||
print(f" first card text: {txt[:200]!r}")
|
||||
|
||||
# Scroll into view and re-check
|
||||
if section.count():
|
||||
section.first.scroll_into_view_if_needed()
|
||||
page.wait_for_timeout(800)
|
||||
classes2 = section.first.get_attribute("class")
|
||||
print(f" section classes after scroll: {classes2!r}")
|
||||
if cards.count():
|
||||
op2 = cards.first.evaluate("el => getComputedStyle(el).opacity")
|
||||
print(f" first card opacity after scroll: {op2}")
|
||||
|
||||
page.screenshot(path=f"scripts/testimonials-{label}.png", full_page=True)
|
||||
print(f" screenshot: scripts/testimonials-{label}.png")
|
||||
|
||||
if logs:
|
||||
print(" console:")
|
||||
for l in logs[-15:]:
|
||||
print(f" {l}")
|
||||
ctx.close()
|
||||
browser.close()
|
||||
|
||||
run()
|
||||
|
After Width: | Height: | Size: 3.3 MiB |
|
After Width: | Height: | Size: 1.3 MiB |
|
After Width: | Height: | Size: 1.2 MiB |
|
After Width: | Height: | Size: 1.3 MiB |
@@ -28,8 +28,12 @@ describe('reveal action', () => {
|
||||
vi.unstubAllGlobals();
|
||||
});
|
||||
|
||||
it('toggles visibility as the element enters and leaves the viewport', () => {
|
||||
it('reveals the element when the observer reports intersection', () => {
|
||||
vi.stubGlobal('IntersectionObserver', TestIntersectionObserver);
|
||||
vi.spyOn(window, 'requestAnimationFrame').mockImplementation((cb) => {
|
||||
cb(0);
|
||||
return 0;
|
||||
});
|
||||
vi.spyOn(window, 'matchMedia').mockReturnValue({
|
||||
matches: false,
|
||||
media: '(prefers-reduced-motion: reduce)',
|
||||
@@ -67,9 +71,7 @@ describe('reveal action', () => {
|
||||
|
||||
observer.trigger(node, true);
|
||||
expect(node.classList.contains('reveal-visible')).toBe(true);
|
||||
|
||||
observer.trigger(node, false);
|
||||
expect(node.classList.contains('reveal-visible')).toBe(false);
|
||||
expect(observer.unobserve).toHaveBeenCalledWith(node);
|
||||
|
||||
action.destroy();
|
||||
expect(observer.disconnect).toHaveBeenCalledTimes(1);
|
||||
|
||||
@@ -34,6 +34,17 @@ export function reveal(node: HTMLElement, options: RevealOptions = {}) {
|
||||
let observer: IntersectionObserver | null = null;
|
||||
let cancelled = false;
|
||||
|
||||
// Failsafe: if hydration mismatch or any other edge case prevents the rAF
|
||||
// check and IntersectionObserver from running, force visibility after a
|
||||
// short delay so content never stays stuck at opacity 0.
|
||||
const failsafe = window.setTimeout(() => {
|
||||
if (cancelled) return;
|
||||
if (!node.classList.contains('reveal-visible')) {
|
||||
node.classList.add('reveal-visible');
|
||||
observer?.disconnect();
|
||||
}
|
||||
}, 1200);
|
||||
|
||||
// Defer the layout-reading initial check + observer setup to the next
|
||||
// frame. With 20+ `use:reveal` instances mounting in a row, calling
|
||||
// getBoundingClientRect() synchronously after a class mutation forces
|
||||
@@ -72,6 +83,7 @@ export function reveal(node: HTMLElement, options: RevealOptions = {}) {
|
||||
return {
|
||||
destroy() {
|
||||
cancelled = true;
|
||||
window.clearTimeout(failsafe);
|
||||
observer?.disconnect();
|
||||
}
|
||||
};
|
||||
|
||||
@@ -35,12 +35,12 @@
|
||||
{#each standardSections as section}
|
||||
{@const enhanced = getEnhancedImage(section.imageUrl)}
|
||||
<section
|
||||
use:reveal
|
||||
use:reveal={{ distance: 0 }}
|
||||
class="about-section reveal-block"
|
||||
class:about-section-gradient={section.accent === 'gradient'}
|
||||
>
|
||||
<div class="page-inner about-section-grid" class:about-section-reverse={section.reverse}>
|
||||
<div class="about-copy">
|
||||
<div class="about-copy fade-up">
|
||||
{#if section.eyebrow}
|
||||
<span class="eyebrow about-eyebrow">{section.eyebrow}</span>
|
||||
{/if}
|
||||
@@ -49,7 +49,7 @@
|
||||
<p>{paragraph}</p>
|
||||
{/each}
|
||||
</div>
|
||||
<div class="about-media">
|
||||
<div class="about-media scale-soft">
|
||||
{#if enhanced}
|
||||
<enhanced:img src={enhanced} alt={section.imageAlt} loading="lazy" decoding="async" />
|
||||
{:else}
|
||||
@@ -63,9 +63,9 @@
|
||||
<!-- ── Founder section ── -->
|
||||
{#if founderSection}
|
||||
{@const founderEnhanced = getEnhancedImage(founderSection.imageUrl)}
|
||||
<section use:reveal={{ delay: 50 }} class="about-founder reveal-block">
|
||||
<section use:reveal={{ delay: 50, distance: 0 }} class="about-founder reveal-block">
|
||||
<div class="page-inner about-founder-grid">
|
||||
<div class="about-founder-media">
|
||||
<div class="about-founder-media scale-soft">
|
||||
{#if founderEnhanced}
|
||||
<enhanced:img
|
||||
src={founderEnhanced}
|
||||
@@ -82,7 +82,7 @@
|
||||
/>
|
||||
{/if}
|
||||
</div>
|
||||
<div class="about-founder-copy">
|
||||
<div class="about-founder-copy fade-up">
|
||||
<article class="about-founder-note">
|
||||
{#if founderSection.eyebrow}
|
||||
<span class="eyebrow about-eyebrow about-founder-kicker">{founderSection.eyebrow}</span>
|
||||
@@ -122,7 +122,7 @@
|
||||
<span>If you are unsure about anything, feel free to email me anytime.</span>
|
||||
</a>
|
||||
|
||||
<a href="/contact-us" class="btn btn-green btn-mobile-center about-founder-cta">
|
||||
<a href="/contact-us" class="btn btn-green btn-mobile-center about-founder-cta cta-shimmer">
|
||||
Book a free Meet & Greet
|
||||
</a>
|
||||
</article>
|
||||
@@ -133,13 +133,13 @@
|
||||
|
||||
<!-- ── FAQs ── -->
|
||||
{#if pageContent.faqs && pageContent.faqs.length}
|
||||
<section use:reveal={{ delay: 30 }} class="about-faq reveal-block">
|
||||
<section use:reveal={{ delay: 30, distance: 0 }} class="about-faq reveal-block">
|
||||
<div class="page-inner">
|
||||
<div class="about-faq-header">
|
||||
<div class="about-faq-header fade-up">
|
||||
<span class="eyebrow about-eyebrow">FAQ</span>
|
||||
<h2>{pageContent.faqTitle ?? 'Common questions'}</h2>
|
||||
</div>
|
||||
<div use:accordion class="faq about-faq-list">
|
||||
<div use:accordion class="faq about-faq-list fade-up">
|
||||
{#each pageContent.faqs as item}
|
||||
<details>
|
||||
<summary>{item.question}</summary>
|
||||
@@ -571,4 +571,15 @@
|
||||
animation: none;
|
||||
}
|
||||
}
|
||||
|
||||
/* Per-section choreography — copy leads, then the media settles after a
|
||||
short beat. Repeated across each about-section instance. */
|
||||
@media (prefers-reduced-motion: no-preference) {
|
||||
:global(.reveal-visible) .about-copy { transition-delay: 40ms; }
|
||||
:global(.reveal-visible) .about-media { transition-delay: 200ms; }
|
||||
:global(.reveal-visible) .about-founder-media { transition-delay: 60ms; }
|
||||
:global(.reveal-visible) .about-founder-copy { transition-delay: 220ms; }
|
||||
:global(.reveal-visible) .about-faq-header { transition-delay: 40ms; }
|
||||
:global(.reveal-visible) .about-faq-list { transition-delay: 180ms; }
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -10,21 +10,20 @@ async function fillOwnerStep() {
|
||||
await fireEvent.input(screen.getByLabelText(/^Email/i), {
|
||||
target: { value: 'alex@example.com' }
|
||||
});
|
||||
await fireEvent.input(screen.getByLabelText(/Contact #/i), {
|
||||
await fireEvent.input(screen.getByLabelText(/Phone number/i), {
|
||||
target: { value: '021 123 4567' }
|
||||
});
|
||||
}
|
||||
|
||||
async function fillDogStep() {
|
||||
await fireEvent.click(screen.getByLabelText('Tiny Gang Pack Walks'));
|
||||
await fireEvent.click(screen.getByLabelText('Other Services'));
|
||||
await fireEvent.input(screen.getByLabelText(/Dog's Name/i), {
|
||||
target: { value: 'Maya' }
|
||||
});
|
||||
await fireEvent.input(screen.getByLabelText(/Location/i), {
|
||||
await fireEvent.input(screen.getByLabelText(/Your suburb/i), {
|
||||
target: { value: 'Kingsland' }
|
||||
});
|
||||
await fireEvent.input(screen.getByLabelText(/Tiny Gang Pack Walks fit/i), {
|
||||
await fireEvent.click(screen.getByLabelText('Tiny Gang Pack Walks'));
|
||||
await fireEvent.input(screen.getByLabelText(/A bit about your dog/i), {
|
||||
target: { value: 'Loves small group walks.' }
|
||||
});
|
||||
}
|
||||
@@ -103,7 +102,7 @@ describe('BookingSection', () => {
|
||||
petName: 'Maya',
|
||||
location: 'Kingsland',
|
||||
message: 'Loves small group walks.',
|
||||
services: ['Tiny Gang Pack Walks', 'Other Services'],
|
||||
services: ['Tiny Gang Pack Walks'],
|
||||
website: '',
|
||||
referrer: 'https://www.google.com/',
|
||||
stepChanges: 1,
|
||||
@@ -144,7 +143,7 @@ describe('BookingSection', () => {
|
||||
await fireEvent.click(container.querySelector('.booking-next-button')!);
|
||||
expect(screen.getByText('Please tell us how we can help')).toBeInTheDocument();
|
||||
|
||||
await fireEvent.input(screen.getByLabelText(/Your Message/i), {
|
||||
await fireEvent.input(screen.getByLabelText(/How can we help/i), {
|
||||
target: { value: 'I would like to discuss a business partnership.' }
|
||||
});
|
||||
|
||||
|
||||
@@ -22,20 +22,20 @@
|
||||
$: founderStoryEnhanced = getEnhancedImage(founderStory.imageUrl);
|
||||
</script>
|
||||
|
||||
<section id="promise" use:reveal={{ delay: 20 }} class="reveal-block">
|
||||
<section id="promise" use:reveal={{ delay: 20, distance: 0 }} class="reveal-block">
|
||||
<div class="founder-inner">
|
||||
<article class="founder-note">
|
||||
<div class="founder-intro">
|
||||
<div class="founder-intro fade-up">
|
||||
<span class="eyebrow founder-kicker">A note from Aless</span>
|
||||
<span class="founder-greeting">Hi, Aless from Goodwalk <span class="founder-greeting-wave" aria-hidden="true">👋</span></span>
|
||||
</div>
|
||||
|
||||
<h2 class="founder-heading">
|
||||
<h2 class="founder-heading fade-up">
|
||||
<span class="founder-heading-main">{founderStory.title}</span>
|
||||
<span class="founder-heading-sub">Goodwalk is built around trust.</span>
|
||||
</h2>
|
||||
|
||||
<div class="founder-trust-strip" aria-label="What owners can expect from Goodwalk">
|
||||
<div class="founder-trust-strip fade-up" aria-label="What owners can expect from Goodwalk">
|
||||
<span class="founder-trust-label">What owners notice first</span>
|
||||
<ul class="founder-trust-list">
|
||||
{#each founderTrustNotes as note}
|
||||
@@ -44,29 +44,29 @@
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="founder-body">
|
||||
<div class="founder-body fade-up">
|
||||
{#each founderStoryParagraphs as paragraph}
|
||||
<p>{paragraph}</p>
|
||||
{/each}
|
||||
</div>
|
||||
|
||||
<p class="founder-closing">
|
||||
<p class="founder-closing fade-up">
|
||||
Ready to <strong>{founderStory.emphasis}</strong>
|
||||
</p>
|
||||
|
||||
<div class="founder-actions">
|
||||
<div class="founder-actions fade-up">
|
||||
<a class="founder-contact-note" href="mailto:info@goodwalk.co.nz" aria-label="Email Aless at Goodwalk">
|
||||
<span class="founder-contact-wave" aria-hidden="true">👋</span>
|
||||
<span>If you are unsure about anything, feel free to email, call, or send me an Instagram DM anytime.</span>
|
||||
</a>
|
||||
|
||||
<a href={founderStory.cta.href} class="btn btn-green btn-with-arrow btn-hide-arrow-mobile founder-cta">
|
||||
<a href={founderStory.cta.href} class="btn btn-green btn-with-arrow btn-hide-arrow-mobile founder-cta cta-shimmer">
|
||||
{founderStory.cta.label}
|
||||
<Icon name="fas fa-arrow-right" />
|
||||
</a>
|
||||
</div>
|
||||
|
||||
<div class="founder-signoff">
|
||||
<div class="founder-signoff fade-up">
|
||||
<div class="founder-signoff-copy">
|
||||
<p class="founder-signoff-name">Aless, founder of Goodwalk</p>
|
||||
<p class="founder-signoff-line">The same calm face at the door, the same trusted routine for your dog.</p>
|
||||
@@ -102,6 +102,19 @@
|
||||
contain-intrinsic-size: 980px;
|
||||
}
|
||||
|
||||
/* Narrative cascade — the article unfolds top-to-bottom in a steady,
|
||||
unhurried rhythm. Intentionally slow steps (~90ms) so the eye reads
|
||||
each block as it lands, not as a rush. */
|
||||
@media (prefers-reduced-motion: no-preference) {
|
||||
:global(.reveal-visible) .founder-intro { transition-delay: 40ms; }
|
||||
:global(.reveal-visible) .founder-heading { transition-delay: 130ms; }
|
||||
:global(.reveal-visible) .founder-trust-strip { transition-delay: 220ms; }
|
||||
:global(.reveal-visible) .founder-body { transition-delay: 310ms; }
|
||||
:global(.reveal-visible) .founder-closing { transition-delay: 400ms; }
|
||||
:global(.reveal-visible) .founder-actions { transition-delay: 490ms; }
|
||||
:global(.reveal-visible) .founder-signoff { transition-delay: 580ms; }
|
||||
}
|
||||
|
||||
.founder-inner {
|
||||
max-width: 880px;
|
||||
margin: 0 auto;
|
||||
|
||||
@@ -12,10 +12,10 @@
|
||||
];
|
||||
</script>
|
||||
|
||||
<section id="how-it-works" use:reveal={{ delay: 30 }} class="reveal-block">
|
||||
<section id="how-it-works" use:reveal={{ delay: 30, distance: 0 }} class="reveal-block">
|
||||
<div class="hiw-inner">
|
||||
|
||||
<div class="section-header hiw-header">
|
||||
<div class="section-header hiw-header fade-up">
|
||||
<span class="eyebrow hiw-eyebrow">Getting started</span>
|
||||
<h2 class="section-heading">{content.title}</h2>
|
||||
{#if content.intro}
|
||||
@@ -23,18 +23,18 @@
|
||||
{/if}
|
||||
</div>
|
||||
|
||||
<div class="hiw-journey-bar" aria-label="How getting started works">
|
||||
<div class="hiw-journey-bar stagger-children" aria-label="How getting started works">
|
||||
{#each journeyChips as chip}
|
||||
<span class="hiw-journey-pill">
|
||||
<span class="hiw-journey-pill fade-up">
|
||||
<Icon name={chip.icon} className="hiw-journey-icon" />
|
||||
{chip.label}
|
||||
</span>
|
||||
{/each}
|
||||
</div>
|
||||
|
||||
<div class="hiw-steps">
|
||||
<div class="hiw-steps stagger-children">
|
||||
{#each content.steps as step, index}
|
||||
<div class="hiw-step">
|
||||
<div class="hiw-step fade-up">
|
||||
<div class="hiw-step-meta">
|
||||
<span class="hiw-phase">{step.phase}</span>
|
||||
<span class="hiw-num">0{index + 1}</span>
|
||||
@@ -54,8 +54,8 @@
|
||||
{/each}
|
||||
</div>
|
||||
|
||||
<div class="hiw-cta">
|
||||
<a href="#newlead" class="btn btn-green btn-mobile-center btn-with-arrow">
|
||||
<div class="hiw-cta fade-up">
|
||||
<a href="#newlead" class="btn btn-green btn-mobile-center btn-with-arrow cta-shimmer">
|
||||
Book your free Meet & Greet
|
||||
<Icon name="fas fa-arrow-right" />
|
||||
</a>
|
||||
@@ -364,4 +364,28 @@
|
||||
opacity: 1;
|
||||
transform: translate3d(0, 0, 0);
|
||||
}
|
||||
|
||||
/* Choreography: when the section reveal fires, the inner tiers come in
|
||||
as a hierarchy — header settles first, journey pills follow, then
|
||||
step cards cascade, then the CTA tail. Each tier's own stagger still
|
||||
applies on top of these base offsets, so within a tier the children
|
||||
ripple as expected. Kept inside @media so reduced-motion users skip
|
||||
the cascade and see everything at rest immediately. */
|
||||
@media (prefers-reduced-motion: no-preference) {
|
||||
:global(.reveal-visible) .hiw-header {
|
||||
transition-delay: 40ms;
|
||||
}
|
||||
|
||||
:global(.reveal-visible) .hiw-journey-bar > :nth-child(1) { transition-delay: 160ms; }
|
||||
:global(.reveal-visible) .hiw-journey-bar > :nth-child(2) { transition-delay: 220ms; }
|
||||
:global(.reveal-visible) .hiw-journey-bar > :nth-child(3) { transition-delay: 280ms; }
|
||||
|
||||
:global(.reveal-visible) .hiw-steps > :nth-child(1) { transition-delay: 340ms; }
|
||||
:global(.reveal-visible) .hiw-steps > :nth-child(2) { transition-delay: 420ms; }
|
||||
:global(.reveal-visible) .hiw-steps > :nth-child(3) { transition-delay: 500ms; }
|
||||
|
||||
:global(.reveal-visible) .hiw-cta {
|
||||
transition-delay: 600ms;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -4,24 +4,40 @@
|
||||
|
||||
export let instagram: HomePageContent['instagram'];
|
||||
|
||||
// Editorial layout — full-bleed photo with a dark scrim on the left.
|
||||
// Photo source is the existing tiny-gang pack walk image already used
|
||||
// elsewhere on the site, so no new assets are required.
|
||||
const editorialPhoto = '/images/goodwalk-tiny-gang-pack-walk-small-dogs-auckland.webp';
|
||||
const editorialPhotoAlt = 'Goodwalk Tiny Gang on a pack walk in Auckland';
|
||||
</script>
|
||||
|
||||
<aside id="instagram" aria-label="Follow Goodwalk on Instagram">
|
||||
<div class="instagram-stage">
|
||||
<div class="instagram-panel">
|
||||
<div class="instagram-copy">
|
||||
<span class="eyebrow instagram-kicker">Daily walks, happy dogs</span>
|
||||
<h2>{instagram.title}</h2>
|
||||
<p class="instagram-blurb">See our dogs in action — walks, play, and happy pups</p>
|
||||
<a href={instagram.href} target="_blank" rel="noopener" class="btn btn-green instagram-button">
|
||||
<Icon name="fab fa-instagram" />
|
||||
{instagram.label}
|
||||
</a>
|
||||
</div>
|
||||
<div class="ig-editorial">
|
||||
<div class="ig-editorial-photo">
|
||||
<img src={editorialPhoto} alt={editorialPhotoAlt} loading="lazy" decoding="async" />
|
||||
</div>
|
||||
|
||||
<div class="instagram-dog-wrap" aria-hidden="true">
|
||||
<img src="/images/goodwalk-instagram-dog-cutout.webp" alt="" class="instagram-dog" loading="lazy" decoding="async" />
|
||||
<div class="ig-editorial-scrim">
|
||||
<div class="ig-editorial-copy">
|
||||
<span class="ig-editorial-source">
|
||||
<Icon name="fab fa-instagram" />
|
||||
<span>{instagram.label}</span>
|
||||
</span>
|
||||
|
||||
<h2>{instagram.title}</h2>
|
||||
<p>Daily walks, real dogs, no filter. Park runs, pack moments, and the kind of updates you'd actually want to scroll.</p>
|
||||
|
||||
<a
|
||||
href={instagram.href}
|
||||
target={instagram.external ? '_blank' : undefined}
|
||||
rel={instagram.external ? 'noopener' : undefined}
|
||||
class="ig-editorial-cta"
|
||||
>
|
||||
<Icon name="fab fa-instagram" />
|
||||
<span>Follow {instagram.label}</span>
|
||||
<Icon name="fas fa-arrow-right" className="ig-editorial-arrow" />
|
||||
</a>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</aside>
|
||||
@@ -30,133 +46,186 @@
|
||||
#instagram {
|
||||
content-visibility: auto;
|
||||
contain-intrinsic-size: 360px;
|
||||
padding: 32px 24px 56px;
|
||||
}
|
||||
|
||||
.ig-editorial {
|
||||
position: relative;
|
||||
overflow: hidden;
|
||||
padding-bottom: 40px;
|
||||
}
|
||||
|
||||
.instagram-stage {
|
||||
position: relative;
|
||||
max-width: 1040px;
|
||||
max-width: 1180px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.instagram-panel {
|
||||
position: relative;
|
||||
min-height: 150px;
|
||||
padding: 24px 320px 24px 44px;
|
||||
border-radius: 28px;
|
||||
background:
|
||||
radial-gradient(circle at top left, rgba(255, 255, 255, 0.52), transparent 42%),
|
||||
linear-gradient(135deg, rgba(255, 252, 242, 0.98), rgba(255, 243, 198, 0.96));
|
||||
box-shadow:
|
||||
inset 0 0 0 1px rgba(17, 20, 24, 0.06),
|
||||
0 26px 50px rgba(106, 80, 16, 0.14);
|
||||
overflow: visible;
|
||||
aspect-ratio: 2.6 / 1;
|
||||
min-height: 280px;
|
||||
box-shadow: 0 30px 70px rgba(17, 20, 24, 0.16);
|
||||
}
|
||||
|
||||
.instagram-copy {
|
||||
max-width: 580px;
|
||||
margin: 0;
|
||||
text-align: left;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
}
|
||||
|
||||
.instagram-kicker {
|
||||
/* All visual styling comes from the shared .eyebrow utility. */
|
||||
display: inline-block;
|
||||
margin: 0 0 10px;
|
||||
}
|
||||
|
||||
.instagram-copy :global(h2) {
|
||||
max-width: none;
|
||||
margin: 0 0 6px;
|
||||
font-size: 22px;
|
||||
line-height: 1.2;
|
||||
}
|
||||
|
||||
.instagram-blurb {
|
||||
max-width: 520px;
|
||||
margin: 0;
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
.instagram-button {
|
||||
margin-top: 14px;
|
||||
position: relative;
|
||||
z-index: 1;
|
||||
}
|
||||
|
||||
.instagram-dog-wrap {
|
||||
.ig-editorial-photo {
|
||||
position: absolute;
|
||||
right: -56px;
|
||||
bottom: -14px;
|
||||
display: flex;
|
||||
align-items: flex-end;
|
||||
justify-content: center;
|
||||
width: 380px;
|
||||
pointer-events: none;
|
||||
inset: 0;
|
||||
z-index: 0;
|
||||
}
|
||||
|
||||
.instagram-dog {
|
||||
display: block;
|
||||
.ig-editorial-photo img {
|
||||
width: 100%;
|
||||
height: auto;
|
||||
height: 100%;
|
||||
object-fit: cover;
|
||||
object-position: 60% 40%;
|
||||
transform: scale(1.02);
|
||||
transition: transform 0.8s cubic-bezier(0.22, 1, 0.36, 1);
|
||||
}
|
||||
|
||||
@media (min-width: 1800px) {
|
||||
.instagram-stage {
|
||||
max-width: 1144px;
|
||||
}
|
||||
.ig-editorial:hover .ig-editorial-photo img {
|
||||
transform: scale(1.06);
|
||||
}
|
||||
|
||||
.ig-editorial-scrim {
|
||||
position: absolute;
|
||||
inset: 0;
|
||||
z-index: 1;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
background: linear-gradient(
|
||||
100deg,
|
||||
rgba(15, 24, 15, 0.86) 0%,
|
||||
rgba(15, 24, 15, 0.74) 36%,
|
||||
rgba(15, 24, 15, 0.18) 70%,
|
||||
rgba(15, 24, 15, 0) 100%
|
||||
);
|
||||
padding: 48px clamp(28px, 4vw, 56px);
|
||||
}
|
||||
|
||||
.ig-editorial-copy {
|
||||
max-width: 480px;
|
||||
color: #fff;
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
.ig-editorial-source {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
padding: 6px 12px 6px 10px;
|
||||
margin-bottom: 18px;
|
||||
border-radius: 999px;
|
||||
background: rgba(255, 255, 255, 0.14);
|
||||
backdrop-filter: blur(6px);
|
||||
-webkit-backdrop-filter: blur(6px);
|
||||
color: #fff;
|
||||
font-family: var(--font-head);
|
||||
font-size: 12px;
|
||||
font-weight: 700;
|
||||
letter-spacing: 0.02em;
|
||||
}
|
||||
|
||||
.ig-editorial-source :global(.icon) {
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
/* Specificity bump: #instagram h2 (id+element) lives in the global
|
||||
typography.css and would otherwise win. Prefixing #instagram here
|
||||
matches that specificity so the editorial sizing applies. */
|
||||
#instagram .ig-editorial-copy h2 {
|
||||
margin: 0 0 12px;
|
||||
font-family: var(--font-head);
|
||||
font-size: clamp(26px, 3vw, 36px);
|
||||
font-weight: 800;
|
||||
letter-spacing: -0.025em;
|
||||
line-height: 1.08;
|
||||
color: #fff;
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
.ig-editorial-copy p {
|
||||
margin: 0 0 26px;
|
||||
color: rgba(255, 255, 255, 0.78);
|
||||
font-size: 15px;
|
||||
line-height: 1.55;
|
||||
max-width: 36ch;
|
||||
}
|
||||
|
||||
.ig-editorial-cta {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 10px;
|
||||
padding: 13px 22px;
|
||||
border-radius: 999px;
|
||||
background: var(--yellow);
|
||||
color: var(--gw-green);
|
||||
font-family: var(--font-head);
|
||||
font-size: 14px;
|
||||
font-weight: 800;
|
||||
text-decoration: none;
|
||||
box-shadow: 0 12px 24px rgba(255, 209, 0, 0.32);
|
||||
transition: transform 0.22s ease, box-shadow 0.22s ease;
|
||||
}
|
||||
|
||||
.ig-editorial-cta:hover {
|
||||
transform: translateY(-2px);
|
||||
box-shadow: 0 18px 32px rgba(255, 209, 0, 0.38);
|
||||
}
|
||||
|
||||
.ig-editorial-cta :global(.ig-editorial-arrow) {
|
||||
transition: transform 0.22s cubic-bezier(0.22, 1, 0.36, 1);
|
||||
}
|
||||
|
||||
.ig-editorial-cta:hover :global(.ig-editorial-arrow) {
|
||||
transform: translateX(3px);
|
||||
}
|
||||
|
||||
/* Mobile — stack the photo above the copy so neither gets crushed.
|
||||
Copy centers inside the green panel for a calmer vertical rhythm. */
|
||||
@media (max-width: 768px) {
|
||||
#instagram {
|
||||
padding-bottom: 120px;
|
||||
padding: 24px 16px 40px;
|
||||
}
|
||||
|
||||
.instagram-panel {
|
||||
padding: 30px 24px 180px;
|
||||
border-radius: 28px;
|
||||
.ig-editorial {
|
||||
aspect-ratio: auto;
|
||||
min-height: 0;
|
||||
border-radius: 24px;
|
||||
}
|
||||
|
||||
.instagram-copy {
|
||||
.ig-editorial-photo {
|
||||
position: relative;
|
||||
aspect-ratio: 4 / 3;
|
||||
}
|
||||
|
||||
.ig-editorial-scrim {
|
||||
position: relative;
|
||||
inset: auto;
|
||||
background: var(--gw-green);
|
||||
padding: 28px 22px 32px;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.ig-editorial-copy {
|
||||
max-width: none;
|
||||
margin: 0 auto;
|
||||
text-align: center;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.instagram-copy :global(h2) {
|
||||
max-width: none;
|
||||
font-size: inherit;
|
||||
line-height: inherit;
|
||||
#instagram .ig-editorial-copy h2 {
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.instagram-blurb {
|
||||
max-width: none;
|
||||
font-size: inherit;
|
||||
.ig-editorial-copy p {
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
}
|
||||
}
|
||||
|
||||
.instagram-button {
|
||||
margin-top: 24px;
|
||||
@media (prefers-reduced-motion: reduce) {
|
||||
.ig-editorial-photo img,
|
||||
.ig-editorial-cta {
|
||||
transition: none;
|
||||
transform: none;
|
||||
}
|
||||
|
||||
.instagram-dog-wrap {
|
||||
left: 50%;
|
||||
right: auto;
|
||||
bottom: -96px;
|
||||
width: min(300px, calc(100% - 32px));
|
||||
transform: translateX(-50%);
|
||||
}
|
||||
|
||||
.instagram-dog {
|
||||
width: 100%;
|
||||
.ig-editorial-cta:hover {
|
||||
transform: none;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -155,8 +155,14 @@
|
||||
});
|
||||
}
|
||||
|
||||
function fieldValue(value: string): string {
|
||||
return value.trim() || '—';
|
||||
function toStr(value: unknown): string {
|
||||
if (typeof value === 'string') return value;
|
||||
if (value == null) return '';
|
||||
return String(value);
|
||||
}
|
||||
|
||||
function fieldValue(value: unknown): string {
|
||||
return toStr(value).trim() || '—';
|
||||
}
|
||||
|
||||
function resetIdle() {
|
||||
@@ -528,33 +534,7 @@
|
||||
window.scrollTo({ top: 0, behavior: 'smooth' });
|
||||
}
|
||||
|
||||
function clearOptionalHealthFields() {
|
||||
noteInteraction();
|
||||
accessInstructions = '';
|
||||
hasFoodAllergies = 'no';
|
||||
foodAllergiesDetail = '';
|
||||
hasEnvAllergies = 'no';
|
||||
envAllergiesDetail = '';
|
||||
onSpecialDiet = 'no';
|
||||
specialDietDetail = '';
|
||||
onMedication = 'no';
|
||||
medicationDetail = '';
|
||||
|
||||
errors = {
|
||||
...errors,
|
||||
accessInstructions: '',
|
||||
hasFoodAllergies: '',
|
||||
foodAllergiesDetail: '',
|
||||
hasEnvAllergies: '',
|
||||
envAllergiesDetail: '',
|
||||
onSpecialDiet: '',
|
||||
specialDietDetail: '',
|
||||
onMedication: '',
|
||||
medicationDetail: '',
|
||||
};
|
||||
}
|
||||
|
||||
function buildSubmissionSnapshot(submittedAt: string): SubmissionSummary {
|
||||
function buildSubmissionSnapshot(submittedAt: string): SubmissionSummary {
|
||||
return {
|
||||
submittedAt,
|
||||
sections: [
|
||||
@@ -651,22 +631,23 @@
|
||||
}
|
||||
|
||||
submitting = true;
|
||||
signedOnDate = new Date().toISOString();
|
||||
const submissionSnapshot = buildSubmissionSnapshot(signedOnDate);
|
||||
|
||||
const medicalNotes = [
|
||||
temperament.trim(),
|
||||
hasFoodAllergies === 'yes' ? `Food allergies: ${foodAllergiesDetail.trim()}` : '',
|
||||
hasEnvAllergies === 'yes' ? `Environmental allergies: ${envAllergiesDetail.trim()}` : '',
|
||||
onSpecialDiet === 'yes' ? `Special diet: ${specialDietDetail.trim()}` : '',
|
||||
onMedication === 'yes' ? `Medication: ${medicationDetail.trim()}` : '',
|
||||
additionalNotes.trim(),
|
||||
].filter(Boolean).join('\n');
|
||||
|
||||
const controller = new AbortController();
|
||||
const timeoutId = window.setTimeout(() => controller.abort(), 25000);
|
||||
|
||||
try {
|
||||
signedOnDate = new Date().toISOString();
|
||||
const submissionSnapshot = buildSubmissionSnapshot(signedOnDate);
|
||||
|
||||
const medicalNotes = [
|
||||
toStr(temperament).trim(),
|
||||
hasFoodAllergies === 'yes' ? `Food allergies: ${toStr(foodAllergiesDetail).trim()}` : '',
|
||||
hasEnvAllergies === 'yes' ? `Environmental allergies: ${toStr(envAllergiesDetail).trim()}` : '',
|
||||
onSpecialDiet === 'yes' ? `Special diet: ${toStr(specialDietDetail).trim()}` : '',
|
||||
onMedication === 'yes' ? `Medication: ${toStr(medicationDetail).trim()}` : '',
|
||||
toStr(additionalNotes).trim(),
|
||||
].filter(Boolean).join('\n');
|
||||
|
||||
const response = await fetch('/api/onboarding-submit', {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
@@ -1148,13 +1129,6 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="panel-subsection-actions">
|
||||
<button class="subsection-quick-action" type="button" on:click={clearOptionalHealthFields}>
|
||||
<Icon name="fas fa-check" />
|
||||
Clear optional health details
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<label class="field">
|
||||
<span>Home access instructions</span>
|
||||
<textarea
|
||||
@@ -2060,12 +2034,6 @@
|
||||
margin-bottom: 18px;
|
||||
}
|
||||
|
||||
.panel-subsection-actions {
|
||||
display: flex;
|
||||
justify-content: flex-start;
|
||||
margin: -2px 0 4px;
|
||||
}
|
||||
|
||||
.panel-subsection-head h3 {
|
||||
margin: 0;
|
||||
color: #171b20;
|
||||
@@ -2097,40 +2065,7 @@
|
||||
font-size: 15px;
|
||||
}
|
||||
|
||||
.subsection-quick-action {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
min-height: 42px;
|
||||
padding: 0 14px;
|
||||
border: 1px solid rgba(33, 48, 33, 0.12);
|
||||
border-radius: 999px;
|
||||
background: rgba(255, 255, 255, 0.92);
|
||||
color: rgba(33, 48, 33, 0.84);
|
||||
font: inherit;
|
||||
font-size: 13px;
|
||||
font-weight: 700;
|
||||
cursor: pointer;
|
||||
transition:
|
||||
border-color 0.18s ease,
|
||||
background 0.18s ease,
|
||||
transform 0.18s cubic-bezier(0.22, 1, 0.36, 1),
|
||||
box-shadow 0.18s ease;
|
||||
}
|
||||
|
||||
.subsection-quick-action:hover {
|
||||
border-color: rgba(33, 48, 33, 0.22);
|
||||
background: #fff;
|
||||
transform: translateY(-1px);
|
||||
box-shadow: 0 8px 18px rgba(17, 20, 24, 0.05);
|
||||
}
|
||||
|
||||
.subsection-quick-action:focus-visible {
|
||||
outline: none;
|
||||
box-shadow: 0 0 0 4px rgba(255, 209, 0, 0.14);
|
||||
}
|
||||
|
||||
.onboarding-step-icon {
|
||||
.onboarding-step-icon {
|
||||
flex: none;
|
||||
width: 48px;
|
||||
height: 48px;
|
||||
@@ -2803,8 +2738,7 @@
|
||||
.confirm-row,
|
||||
.yn-btn,
|
||||
.recall-btn,
|
||||
.chip-option,
|
||||
.subsection-quick-action {
|
||||
.chip-option {
|
||||
transition: none !important;
|
||||
}
|
||||
|
||||
@@ -2965,10 +2899,6 @@
|
||||
margin-bottom: 16px;
|
||||
}
|
||||
|
||||
.panel-subsection-actions {
|
||||
margin-bottom: 6px;
|
||||
}
|
||||
|
||||
.panel-subsection-head h3 {
|
||||
font-size: 17px;
|
||||
}
|
||||
@@ -2985,12 +2915,7 @@
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
.subsection-quick-action {
|
||||
width: 100%;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.signature-header {
|
||||
.signature-header {
|
||||
align-items: start;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
@@ -54,36 +54,36 @@ describe('OnboardingPage', () => {
|
||||
await waitFor(() => expect(screen.getByPlaceholderText('First name')).toBeInTheDocument());
|
||||
|
||||
await fireEvent.input(screen.getByPlaceholderText('First name'), { target: { value: 'Alex' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('Surname'), { target: { value: 'Walker' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('Last name'), { target: { value: 'Walker' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('you@example.com'), { target: { value: 'alex@example.com' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('021 234 5678'), { target: { value: '0212345678' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('Street address'), { target: { value: '1 Test Street' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('Street, suburb, city'), { target: { value: '1 Test Street' } });
|
||||
await fireEvent.click(screen.getByRole('button', { name: /save and continue/i }));
|
||||
|
||||
await waitFor(() => expect(screen.getByPlaceholderText('Dog name')).toBeInTheDocument());
|
||||
await fireEvent.input(screen.getByPlaceholderText('Dog name'), { target: { value: 'Milo' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('Breed'), { target: { value: 'Spoodle' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('e.g. Labrador, Cavoodle, mixed'), { target: { value: 'Spoodle' } });
|
||||
await fireEvent.input(screen.getByLabelText(/date of birth/i), { target: { value: '2020-01-01' } });
|
||||
await clickService('Tiny Gang Pack Walks');
|
||||
await fireEvent.click(screen.getByRole('button', { name: /save and continue/i }));
|
||||
|
||||
await waitFor(() => expect(screen.getByPlaceholderText('Vet clinic or vet name')).toBeInTheDocument());
|
||||
await fireEvent.input(screen.getByPlaceholderText('Vet clinic or vet name'), { target: { value: 'Grey Lynn Vets' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('Vet phone number'), { target: { value: '099999999' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('Vet address'), { target: { value: '2 Vet Street' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('Emergency contact name'), { target: { value: 'Jamie Walker' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('Emergency contact number'), { target: { value: '0211111111' } });
|
||||
await chooseOption(/is your dog vaccinated/i, 'Yes');
|
||||
await chooseOption(/does your dog have any food allergies/i, 'No');
|
||||
await chooseOption(/does your dog have any environmental allergies/i, 'No');
|
||||
await chooseOption(/is your dog on a special diet/i, 'No');
|
||||
await chooseOption(/is your dog taking any medication/i, 'No');
|
||||
await waitFor(() => expect(screen.getByPlaceholderText('Clinic name or your usual vet')).toBeInTheDocument());
|
||||
await fireEvent.input(screen.getByPlaceholderText('Clinic name or your usual vet'), { target: { value: 'Grey Lynn Vets' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('Clinic phone number'), { target: { value: '099999999' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('Clinic address'), { target: { value: '2 Vet Street' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('Full name'), { target: { value: 'Jamie Walker' } });
|
||||
await fireEvent.input(screen.getByPlaceholderText('Phone number'), { target: { value: '0211111111' } });
|
||||
await chooseOption(/vaccinations up to date/i, 'Yes');
|
||||
await chooseOption(/^Any food allergies\?$/i, 'No');
|
||||
await chooseOption(/^Any environmental allergies\?$/i, 'No');
|
||||
await chooseOption(/^On a special diet\?$/i, 'No');
|
||||
await chooseOption(/medication that could affect walks/i, 'No');
|
||||
await fireEvent.click(screen.getByRole('button', { name: /save and continue/i }));
|
||||
|
||||
await waitFor(() => expect(screen.getByText(/registered with council/i)).toBeInTheDocument());
|
||||
await chooseOption(/registered with council/i, 'Yes');
|
||||
await fireEvent.click(screen.getByRole('button', { name: /save and continue/i }));
|
||||
|
||||
await waitFor(() => expect(screen.getByText(/anything else you'd like us to know/i)).toBeInTheDocument());
|
||||
await waitFor(() => expect(screen.getByText(/anything else we should know/i)).toBeInTheDocument());
|
||||
});
|
||||
});
|
||||
|
||||
@@ -41,18 +41,18 @@
|
||||
/>
|
||||
|
||||
{#if pageContent.comparison}
|
||||
<section use:reveal id="pricing-chooser" class="pricing-guide reveal-block">
|
||||
<section use:reveal={{ distance: 0 }} id="pricing-chooser" class="pricing-guide reveal-block">
|
||||
<div class="page-inner">
|
||||
<div class="pricing-guide-heading">
|
||||
<div class="pricing-guide-heading fade-up">
|
||||
<span class="eyebrow">{pageContent.comparison.title}</span>
|
||||
{#if pageContent.comparison.intro}
|
||||
<h2>{pageContent.comparison.intro}</h2>
|
||||
{/if}
|
||||
</div>
|
||||
|
||||
<div class="pricing-chooser-grid">
|
||||
<div class="pricing-chooser-grid stagger-children">
|
||||
{#each pageContent.sections as section, sectionIndex}
|
||||
<article class="pricing-chooser-card">
|
||||
<article class="pricing-chooser-card fade-up">
|
||||
{#if section.imageUrl}
|
||||
<div class="pricing-chooser-image-wrap">
|
||||
<img class="pricing-chooser-image" src={section.imageUrl} alt={section.imageAlt ?? section.title} loading="lazy" decoding="async" />
|
||||
@@ -103,19 +103,19 @@
|
||||
|
||||
{#each pageContent.sections as section, sectionIndex}
|
||||
<section
|
||||
use:reveal
|
||||
use:reveal={{ distance: 0 }}
|
||||
id={`pricing-section-${sectionIndex}`}
|
||||
class:pricing-detail--alt={sectionIndex % 2 === 1}
|
||||
class="pricing-detail reveal-block"
|
||||
>
|
||||
<div class="page-inner pricing-detail-inner">
|
||||
<div class:pricing-detail-media--right={sectionIndex % 2 === 1} class="pricing-detail-media">
|
||||
<div class:pricing-detail-media--right={sectionIndex % 2 === 1} class="pricing-detail-media scale-soft">
|
||||
{#if section.imageUrl}
|
||||
<img src={section.imageUrl} alt={section.imageAlt ?? section.title} loading="lazy" decoding="async" />
|
||||
{/if}
|
||||
</div>
|
||||
|
||||
<div class="pricing-detail-copy">
|
||||
<div class="pricing-detail-copy fade-up">
|
||||
{#if section.eyebrow}
|
||||
<p class="eyebrow pricing-detail-eyebrow">{section.eyebrow}</p>
|
||||
{/if}
|
||||
@@ -158,14 +158,14 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="page-inner pricing-plan-shell">
|
||||
<div class="page-inner pricing-plan-shell fade-up">
|
||||
<div class:pricing-plan-grid-three={section.plans.length === 3} class="pricing-plan-grid">
|
||||
{#each decoratePlans(section.plans) as plan}
|
||||
<PricingPlanCard {plan} variant="pricing" hideCtaOnMobile={true} />
|
||||
{/each}
|
||||
</div>
|
||||
|
||||
<a class="btn btn-yellow pricing-section-mobile-cta" href="#newlead">
|
||||
<a class="btn btn-yellow pricing-section-mobile-cta cta-shimmer" href="#newlead">
|
||||
Book a Meet & Greet
|
||||
<Icon name="fas fa-arrow-right" />
|
||||
</a>
|
||||
@@ -173,8 +173,8 @@
|
||||
</section>
|
||||
{/each}
|
||||
|
||||
<section use:reveal class="pricing-worth reveal-block" aria-labelledby="pricing-worth-heading">
|
||||
<div class="page-inner pricing-worth-inner">
|
||||
<section use:reveal={{ distance: 0 }} class="pricing-worth reveal-block" aria-labelledby="pricing-worth-heading">
|
||||
<div class="page-inner pricing-worth-inner fade-up">
|
||||
<span class="eyebrow eyebrow--accent pricing-worth-kicker">On the price</span>
|
||||
<h2 id="pricing-worth-heading">A better walk costs more.</h2>
|
||||
|
||||
@@ -188,16 +188,16 @@
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<section use:reveal class="pricing-included reveal-block">
|
||||
<section use:reveal={{ distance: 0 }} class="pricing-included reveal-block">
|
||||
<div class="page-inner">
|
||||
<div class="pricing-included-heading">
|
||||
<div class="pricing-included-heading fade-up">
|
||||
<span class="eyebrow">Every booking</span>
|
||||
<h2>Easy to start. Easy to trust.</h2>
|
||||
</div>
|
||||
|
||||
<ul class="pricing-included-list" aria-label="What every Goodwalk booking includes">
|
||||
<ul class="pricing-included-list stagger-children" aria-label="What every Goodwalk booking includes">
|
||||
{#each includedPromises as promise}
|
||||
<li>
|
||||
<li class="fade-up">
|
||||
<span class="pricing-included-icon" aria-hidden="true">
|
||||
<Icon name={promise.icon} />
|
||||
</span>
|
||||
|
||||
@@ -179,7 +179,7 @@
|
||||
/>
|
||||
|
||||
{#if pageContent.highlight}
|
||||
<section use:reveal class="service-highlight reveal-block">
|
||||
<section use:reveal={{ distance: 0 }} class="service-highlight reveal-block">
|
||||
<div class="page-inner">
|
||||
<div class:service-highlight-layout-points={pageContent.highlight.points?.length} class="service-highlight-layout">
|
||||
<div class="service-highlight-copy">
|
||||
@@ -187,9 +187,9 @@
|
||||
<h2>{pageContent.highlight.title}</h2>
|
||||
|
||||
{#if pageContent.highlight.points?.length}
|
||||
<div class="service-highlight-points">
|
||||
<div class="service-highlight-points stagger-children">
|
||||
{#each pageContent.highlight.points as point}
|
||||
<article class="service-highlight-point">
|
||||
<article class="service-highlight-point fade-up">
|
||||
<h3>{point.title}</h3>
|
||||
<p>{point.body}</p>
|
||||
</article>
|
||||
@@ -234,7 +234,7 @@
|
||||
</section>
|
||||
{/if}
|
||||
|
||||
<section use:reveal class="service-benefits reveal-block">
|
||||
<section use:reveal={{ distance: 0 }} class="service-benefits reveal-block">
|
||||
<div class="page-inner">
|
||||
<div class="service-section-heading">
|
||||
<h2>{pageContent.benefits.title}</h2>
|
||||
@@ -244,12 +244,12 @@
|
||||
</div>
|
||||
|
||||
<div class="service-benefit-shell">
|
||||
<div bind:this={benefitScroller} class:service-benefit-grid-simple-swipe={useSimpleBenefitSwipe} class="service-benefit-grid">
|
||||
<div bind:this={benefitScroller} class:service-benefit-grid-simple-swipe={useSimpleBenefitSwipe} class="service-benefit-grid stagger-children">
|
||||
{#each benefitCards as benefit, index}
|
||||
<article
|
||||
class:active={index === activeBenefitIndex}
|
||||
class:service-benefit-card-featured={benefit.featured}
|
||||
class={`service-benefit-card ${benefit.tintClass}`}
|
||||
class={`service-benefit-card fade-up ${benefit.tintClass}`}
|
||||
>
|
||||
<div class="service-benefit-icon" aria-hidden="true">
|
||||
<Icon name={benefit.icon ?? 'fas fa-paw'} />
|
||||
@@ -344,7 +344,7 @@
|
||||
{/if}
|
||||
|
||||
{#if pageContent.decision}
|
||||
<section class:service-decision-pack={isPackWalks} class:service-decision-dog={isDogWalking} use:reveal class="service-decision reveal-block" aria-labelledby="service-decision-heading">
|
||||
<section class:service-decision-pack={isPackWalks} class:service-decision-dog={isDogWalking} use:reveal={{ distance: 0 }} class="service-decision reveal-block" aria-labelledby="service-decision-heading">
|
||||
<div class="page-inner">
|
||||
<div class="service-decision-card">
|
||||
<div class="service-decision-header">
|
||||
@@ -356,8 +356,8 @@
|
||||
{decisionIntro}
|
||||
</p>
|
||||
</div>
|
||||
<div class="service-decision-grid">
|
||||
<div class="service-decision-col service-decision-col-fit">
|
||||
<div class="service-decision-grid stagger-children">
|
||||
<div class="service-decision-col service-decision-col-fit fade-up">
|
||||
<span class="service-decision-col-kicker service-decision-col-kicker-fit">{decisionFitKicker}</span>
|
||||
<h3>{pageContent.decision.fitTitle ?? 'A good fit if your dog:'}</h3>
|
||||
<ul>
|
||||
@@ -371,7 +371,7 @@
|
||||
{/each}
|
||||
</ul>
|
||||
</div>
|
||||
<div class="service-decision-col service-decision-col-not">
|
||||
<div class="service-decision-col service-decision-col-not fade-up">
|
||||
<span class="service-decision-col-kicker service-decision-col-kicker-not">{decisionNotKicker}</span>
|
||||
<h3>{pageContent.decision.notFitTitle ?? 'Probably not a fit if:'}</h3>
|
||||
<ul>
|
||||
@@ -395,7 +395,7 @@
|
||||
{/if}
|
||||
|
||||
<section
|
||||
use:reveal
|
||||
use:reveal={{ distance: 0 }}
|
||||
class:service-pricing-immediate-mobile={useSimpleBenefitSwipe}
|
||||
class:service-pricing-pack={isPackWalks}
|
||||
class="service-pricing reveal-block"
|
||||
@@ -410,7 +410,7 @@
|
||||
|
||||
<div class:service-plan-grid-three={pageContent.pricing.plans.length === 3} class="service-plan-grid">
|
||||
{#each pricingPlans as plan}
|
||||
<PricingPlanCard {plan} variant="service" />
|
||||
<PricingPlanCard {plan} variant="service" hideCtaOnMobile={true} />
|
||||
{/each}
|
||||
</div>
|
||||
|
||||
@@ -426,7 +426,7 @@
|
||||
Every booking starts with a free, no-obligation Meet & Greet.
|
||||
</p>
|
||||
|
||||
<a class="btn btn-yellow btn-mobile-center service-plan-mobile-cta" href="#newlead">Book a Meet & Greet</a>
|
||||
<a class="btn btn-yellow btn-mobile-center service-plan-mobile-cta cta-shimmer" href="#newlead">Book a Meet & Greet</a>
|
||||
|
||||
{#if pageContent.pricing.extras?.length}
|
||||
<div class="service-extras" aria-labelledby="service-extras-heading">
|
||||
@@ -437,9 +437,9 @@
|
||||
</span>
|
||||
<h3 id="service-extras-heading" class="service-extras-title">Optional extras</h3>
|
||||
</div>
|
||||
<div class="service-extras-grid">
|
||||
<div class="service-extras-grid stagger-children">
|
||||
{#each pageContent.pricing.extras as extra}
|
||||
<article class="service-extra-card">
|
||||
<article class="service-extra-card fade-up">
|
||||
<div class="service-extra-card-body">
|
||||
<span class="service-extra-card-label">{extra.label}</span>
|
||||
{#if extra.note}
|
||||
@@ -515,15 +515,15 @@
|
||||
{/if}
|
||||
|
||||
{#if showRelatedServices}
|
||||
<section use:reveal class="service-related reveal-block" aria-label="Other services">
|
||||
<section use:reveal={{ distance: 0 }} class="service-related reveal-block" aria-label="Other services">
|
||||
<div class="page-inner">
|
||||
<div class="service-section-heading">
|
||||
<h2>Explore our other services</h2>
|
||||
</div>
|
||||
|
||||
<div class="service-related-grid">
|
||||
<div class="service-related-grid stagger-children">
|
||||
{#each relatedCards as card, i}
|
||||
<a class="service-related-card service-related-tint-{i % 4}" href={card.href}>
|
||||
<a class="service-related-card fade-up service-related-tint-{i % 4}" href={card.href}>
|
||||
<div class="service-related-icon" aria-hidden="true">
|
||||
<Icon name={card.icon} />
|
||||
</div>
|
||||
|
||||
@@ -67,22 +67,22 @@
|
||||
.map(({ service }) => service);
|
||||
</script>
|
||||
|
||||
<section id="services" use:reveal={{ delay: 20 }} class="reveal-block">
|
||||
<section id="services" use:reveal={{ delay: 20, distance: 0 }} class="reveal-block">
|
||||
<div class="services-inner">
|
||||
<div class="section-header">
|
||||
<div class="section-header fade-up">
|
||||
<h2 class="section-heading">{heading}</h2>
|
||||
<p class="section-intro services-intro">{intro}</p>
|
||||
</div>
|
||||
|
||||
|
||||
|
||||
<div class="services-grid">
|
||||
|
||||
<div class="services-grid stagger-children">
|
||||
{#each orderedServices as service}
|
||||
{@const meta = serviceMeta[service.title]}
|
||||
<a
|
||||
href={service.href}
|
||||
class:service-card-featured={meta?.featured}
|
||||
class="service-card"
|
||||
class="service-card fade-up"
|
||||
aria-label={`${service.title} — view service page`}
|
||||
>
|
||||
<div class="service-card-media">
|
||||
@@ -452,4 +452,15 @@
|
||||
opacity: 1;
|
||||
transform: translate3d(0, 0, 0);
|
||||
}
|
||||
|
||||
/* Tier choreography — header settles, then the three cards cascade. */
|
||||
@media (prefers-reduced-motion: no-preference) {
|
||||
:global(.reveal-visible) .section-header {
|
||||
transition-delay: 40ms;
|
||||
}
|
||||
|
||||
:global(.reveal-visible) .services-grid > :nth-child(1) { transition-delay: 180ms; }
|
||||
:global(.reveal-visible) .services-grid > :nth-child(2) { transition-delay: 260ms; }
|
||||
:global(.reveal-visible) .services-grid > :nth-child(3) { transition-delay: 340ms; }
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -62,7 +62,7 @@
|
||||
cta={{ label: 'Book a Meet & Greet', href: '#newlead', variant: 'yellow' }}
|
||||
/>
|
||||
|
||||
<section use:reveal class="testimonials-page-grid-section reveal-block">
|
||||
<section use:reveal={{ threshold: 0.02 }} class="testimonials-page-grid-section reveal-block">
|
||||
<div class="page-inner">
|
||||
<div class="testimonials-page-grid">
|
||||
{#each testimonialCards as testimonial}
|
||||
@@ -399,43 +399,41 @@
|
||||
opacity: 1;
|
||||
transform: translateY(0);
|
||||
transition:
|
||||
opacity 0.56s cubic-bezier(0.22, 1, 0.36, 1),
|
||||
transform 0.56s cubic-bezier(0.22, 1, 0.36, 1),
|
||||
box-shadow 0.28s cubic-bezier(0.22, 1, 0.36, 1),
|
||||
border-color 0.24s ease;
|
||||
opacity 0.42s cubic-bezier(0.22, 1, 0.36, 1),
|
||||
transform 0.42s cubic-bezier(0.22, 1, 0.36, 1);
|
||||
}
|
||||
|
||||
:global(.reveal-visible.reveal-block).testimonials-page-grid-section .testimonials-page-card:nth-child(1) {
|
||||
transition-delay: 60ms;
|
||||
transition-delay: 0ms;
|
||||
}
|
||||
|
||||
:global(.reveal-visible.reveal-block).testimonials-page-grid-section .testimonials-page-card:nth-child(2) {
|
||||
transition-delay: 140ms;
|
||||
transition-delay: 50ms;
|
||||
}
|
||||
|
||||
:global(.reveal-visible.reveal-block).testimonials-page-grid-section .testimonials-page-card:nth-child(3) {
|
||||
transition-delay: 220ms;
|
||||
transition-delay: 100ms;
|
||||
}
|
||||
|
||||
:global(.reveal-visible.reveal-block).testimonials-page-grid-section .testimonials-page-card:nth-child(4) {
|
||||
transition-delay: 300ms;
|
||||
transition-delay: 150ms;
|
||||
}
|
||||
|
||||
:global(.reveal-visible.reveal-block).testimonials-page-grid-section .testimonials-page-card:nth-child(5) {
|
||||
transition-delay: 380ms;
|
||||
transition-delay: 200ms;
|
||||
}
|
||||
|
||||
:global(.reveal-visible.reveal-block).testimonials-page-grid-section .testimonials-page-card:nth-child(6) {
|
||||
transition-delay: 460ms;
|
||||
transition-delay: 250ms;
|
||||
}
|
||||
|
||||
:global(.reveal-visible.reveal-block).testimonials-page-grid-section .testimonials-page-google-cta-wrap {
|
||||
opacity: 1;
|
||||
transform: translateY(0);
|
||||
transition:
|
||||
opacity 0.56s cubic-bezier(0.22, 1, 0.36, 1),
|
||||
transform 0.56s cubic-bezier(0.22, 1, 0.36, 1);
|
||||
transition-delay: 240ms;
|
||||
opacity 0.42s cubic-bezier(0.22, 1, 0.36, 1),
|
||||
transform 0.42s cubic-bezier(0.22, 1, 0.36, 1);
|
||||
transition-delay: 140ms;
|
||||
}
|
||||
|
||||
@media (hover: hover) {
|
||||
@@ -482,7 +480,7 @@
|
||||
}
|
||||
}
|
||||
|
||||
@media (min-width: 1400px) {
|
||||
@media (min-width: 1180px) {
|
||||
.testimonials-page-grid {
|
||||
grid-template-columns: repeat(3, minmax(0, 1fr));
|
||||
max-width: 1500px;
|
||||
|
||||
@@ -181,10 +181,10 @@
|
||||
});
|
||||
</script>
|
||||
|
||||
<section id="testimonials" use:reveal={{ delay: 40 }} class="reveal-block">
|
||||
<section id="testimonials" use:reveal={{ delay: 40, distance: 0 }} class="reveal-block">
|
||||
<div class="testimonials-inner">
|
||||
<div class="testimonials-header">
|
||||
<div class="testimonials-header-main">
|
||||
<div class="testimonials-header-main fade-up">
|
||||
<span class="eyebrow testimonials-eyebrow">
|
||||
<Icon name="fas fa-star" className="testimonials-eyebrow-star" />
|
||||
{eyebrow}
|
||||
@@ -192,13 +192,13 @@
|
||||
<h2 class="section-heading">{heading}</h2>
|
||||
</div>
|
||||
|
||||
<div class="testimonials-header-side">
|
||||
<div class="testimonials-header-side fade-up">
|
||||
<div class="testimonials-intro">
|
||||
<p>{blurb}</p>
|
||||
</div>
|
||||
|
||||
<div class="testimonials-cta-row">
|
||||
<a href={testimonialsHref} class="btn btn-yellow testimonials-cta testimonials-cta-primary">
|
||||
<a href={testimonialsHref} class="btn btn-yellow testimonials-cta testimonials-cta-primary cta-shimmer">
|
||||
<Icon name="fas fa-comment-dots" />
|
||||
<span class="testimonials-cta-label-desktop">All testimonials</span>
|
||||
<span class="testimonials-cta-label-mobile">Testimonials</span>
|
||||
@@ -215,7 +215,7 @@
|
||||
{#if slides.length}
|
||||
<div
|
||||
bind:this={carouselEl}
|
||||
class="testimonials-carousel"
|
||||
class="testimonials-carousel scale-soft"
|
||||
role="region"
|
||||
aria-label="Customer testimonials"
|
||||
on:mouseenter={() => (paused = true)}
|
||||
@@ -527,6 +527,12 @@
|
||||
transform: translate3d(0, 0, 0);
|
||||
}
|
||||
|
||||
@media (prefers-reduced-motion: no-preference) {
|
||||
:global(.reveal-visible) .testimonials-header-main { transition-delay: 40ms; }
|
||||
:global(.reveal-visible) .testimonials-header-side { transition-delay: 140ms; }
|
||||
:global(.reveal-visible) .testimonials-carousel { transition-delay: 260ms; }
|
||||
}
|
||||
|
||||
.testimonial-stage {
|
||||
position: relative;
|
||||
overflow: hidden;
|
||||
|
||||
@@ -84,15 +84,15 @@
|
||||
.map(({ value }) => value);
|
||||
</script>
|
||||
|
||||
<section id="values" use:reveal={{ delay: 30 }} class="reveal-block">
|
||||
<section id="values" use:reveal={{ delay: 30, distance: 0 }} class="reveal-block">
|
||||
<div class="values-inner">
|
||||
<div class="section-header">
|
||||
<div class="section-header fade-up">
|
||||
<h2 class="section-heading">Calmer dogs. Calmer evenings.</h2>
|
||||
</div>
|
||||
|
||||
<div class="values-photo-grid" aria-label="Goodwalk client dogs">
|
||||
<div class="values-photo-grid stagger-children stagger-tight" aria-label="Goodwalk client dogs">
|
||||
{#each clientPhotoCards as photo, index}
|
||||
<figure class:values-photo-card-featured={index === 0} class="values-photo-card">
|
||||
<figure class:values-photo-card-featured={index === 0} class="values-photo-card fade-up">
|
||||
{#if photo.enhanced}
|
||||
<enhanced:img
|
||||
class="values-photo-image"
|
||||
@@ -120,9 +120,9 @@
|
||||
{/each}
|
||||
</div>
|
||||
|
||||
<div class="values-bento values-contrast">
|
||||
<div class="values-bento values-contrast stagger-children">
|
||||
{#each stakes as stake, index}
|
||||
<article class:values-contrast-cell-good={index === 1} class="values-contrast-cell">
|
||||
<article class:values-contrast-cell-good={index === 1} class="values-contrast-cell fade-up">
|
||||
<div class="values-contrast-head">
|
||||
<span class:values-contrast-label-good={index === 1} class="values-contrast-label">
|
||||
{stake.label}
|
||||
@@ -149,13 +149,13 @@
|
||||
{/each}
|
||||
</div>
|
||||
|
||||
<div class="values-points-header">
|
||||
<div class="values-points-header fade-up">
|
||||
<h3 class="values-points-title">Things we don't treat as extras</h3>
|
||||
</div>
|
||||
|
||||
<div class="values-bento values-points">
|
||||
<div class="values-bento values-points stagger-children">
|
||||
{#each orderedValues as value}
|
||||
<div class="values-point">
|
||||
<div class="values-point fade-up">
|
||||
<div class="values-point-icon">
|
||||
<Icon name={value.icon} className="values-point-glyph" />
|
||||
</div>
|
||||
@@ -627,4 +627,25 @@
|
||||
opacity: 1;
|
||||
transform: translate3d(0, 0, 0);
|
||||
}
|
||||
|
||||
/* Tier choreography — header, then photo gallery cascades tight,
|
||||
contrast cells settle, then the bottom-row "extras" cascade. */
|
||||
@media (prefers-reduced-motion: no-preference) {
|
||||
:global(.reveal-visible) .values-inner > .section-header { transition-delay: 40ms; }
|
||||
|
||||
:global(.reveal-visible) .values-photo-grid > :nth-child(1) { transition-delay: 160ms; }
|
||||
:global(.reveal-visible) .values-photo-grid > :nth-child(2) { transition-delay: 220ms; }
|
||||
:global(.reveal-visible) .values-photo-grid > :nth-child(3) { transition-delay: 280ms; }
|
||||
:global(.reveal-visible) .values-photo-grid > :nth-child(4) { transition-delay: 340ms; }
|
||||
:global(.reveal-visible) .values-photo-grid > :nth-child(5) { transition-delay: 400ms; }
|
||||
|
||||
:global(.reveal-visible) .values-contrast > :nth-child(1) { transition-delay: 460ms; }
|
||||
:global(.reveal-visible) .values-contrast > :nth-child(2) { transition-delay: 540ms; }
|
||||
|
||||
:global(.reveal-visible) .values-points-header { transition-delay: 620ms; }
|
||||
:global(.reveal-visible) .values-points > :nth-child(1) { transition-delay: 680ms; }
|
||||
:global(.reveal-visible) .values-points > :nth-child(2) { transition-delay: 740ms; }
|
||||
:global(.reveal-visible) .values-points > :nth-child(3) { transition-delay: 800ms; }
|
||||
:global(.reveal-visible) .values-points > :nth-child(n + 4) { transition-delay: 860ms; }
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -26,6 +26,15 @@
|
||||
};
|
||||
};
|
||||
|
||||
type LifecycleStatus = 'active' | 'paused' | 'cancelled' | 'archived';
|
||||
|
||||
type Lifecycle = {
|
||||
status: LifecycleStatus;
|
||||
reason: string;
|
||||
changedAt: string;
|
||||
changedBy: string;
|
||||
};
|
||||
|
||||
type AllClient = {
|
||||
email: string;
|
||||
fullName: string;
|
||||
@@ -33,10 +42,22 @@
|
||||
dogName: string;
|
||||
dogBreed: string;
|
||||
status: 'pending' | 'completed';
|
||||
lifecycle: Lifecycle;
|
||||
lastActivityAt: string;
|
||||
welcomePackSentAt: string;
|
||||
};
|
||||
|
||||
const lifecycleOptions: { value: LifecycleStatus; label: string; description: string }[] = [
|
||||
{ value: 'active', label: 'Active', description: 'Currently a client. Included in outreach.' },
|
||||
{ value: 'paused', label: 'Paused', description: 'Temporarily not walking. Still included in outreach.' },
|
||||
{ value: 'cancelled', label: 'Cancelled', description: 'No longer a client. Excluded from outreach but kept on file.' },
|
||||
{ value: 'archived', label: 'Archived', description: 'Long-term inactive. Hidden from outreach but kept for records.' },
|
||||
];
|
||||
|
||||
function lifecycleLabel(status: LifecycleStatus): string {
|
||||
return lifecycleOptions.find((o) => o.value === status)?.label ?? status;
|
||||
}
|
||||
|
||||
type BirthdayClient = {
|
||||
email: string;
|
||||
fullName: string;
|
||||
@@ -126,6 +147,118 @@
|
||||
let confirmPreview = false;
|
||||
let confirmBusy = false;
|
||||
let confirmError = '';
|
||||
let confirmPreviewHtml = '';
|
||||
let confirmPreviewBusy = false;
|
||||
let confirmPreviewToken = 0;
|
||||
|
||||
async function fetchConfirmPreview(ctx: ConfirmContext) {
|
||||
const token = getToken();
|
||||
if (!token) return;
|
||||
const stamp = ++confirmPreviewToken;
|
||||
confirmPreviewBusy = true;
|
||||
try {
|
||||
let url = '';
|
||||
let body: Record<string, unknown> = {};
|
||||
if (ctx.kind === 'welcome') {
|
||||
url = '/api/owner/render-welcome-pack';
|
||||
body = {
|
||||
email: ctx.client.email,
|
||||
serviceType: ctx.serviceType,
|
||||
priceDetails: ctx.priceDetails,
|
||||
startDate: ctx.startDate,
|
||||
};
|
||||
} else if (ctx.kind === 'birthday') {
|
||||
url = '/api/owner/render-birthday-email';
|
||||
body = { email: ctx.client.email };
|
||||
} else {
|
||||
// 'message' uses the live preview already rendered in the messaging tab.
|
||||
if (stamp === confirmPreviewToken) confirmPreviewHtml = msgPreviewHtml || '';
|
||||
return;
|
||||
}
|
||||
const res = await fetch(url, {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json', Authorization: `Bearer ${token}` },
|
||||
body: JSON.stringify(body),
|
||||
});
|
||||
const data = await res.json().catch(() => null);
|
||||
if (stamp !== confirmPreviewToken) return;
|
||||
if (res.ok && typeof data?.html === 'string') {
|
||||
confirmPreviewHtml = data.html;
|
||||
} else {
|
||||
confirmPreviewHtml = '';
|
||||
}
|
||||
} catch {
|
||||
if (stamp === confirmPreviewToken) confirmPreviewHtml = '';
|
||||
} finally {
|
||||
if (stamp === confirmPreviewToken) confirmPreviewBusy = false;
|
||||
}
|
||||
}
|
||||
|
||||
function openConfirmPreviewInNewTab() {
|
||||
if (!confirmPreviewHtml) return;
|
||||
const blob = new Blob([confirmPreviewHtml], { type: 'text/html' });
|
||||
const url = URL.createObjectURL(blob);
|
||||
window.open(url, '_blank', 'noopener,noreferrer');
|
||||
setTimeout(() => URL.revokeObjectURL(url), 60_000);
|
||||
}
|
||||
|
||||
$: if (confirmCtx) {
|
||||
void fetchConfirmPreview(confirmCtx);
|
||||
} else {
|
||||
confirmPreviewHtml = '';
|
||||
confirmPreviewBusy = false;
|
||||
}
|
||||
|
||||
let statusCtx: AllClient | null = null;
|
||||
let statusDraft: LifecycleStatus = 'active';
|
||||
let statusReason = '';
|
||||
let statusBusy = false;
|
||||
let statusError = '';
|
||||
|
||||
function openStatusEditor(client: AllClient) {
|
||||
statusCtx = client;
|
||||
statusDraft = client.lifecycle?.status ?? 'active';
|
||||
statusReason = client.lifecycle?.reason ?? '';
|
||||
statusError = '';
|
||||
}
|
||||
|
||||
function closeStatusEditor() {
|
||||
if (statusBusy) return;
|
||||
statusCtx = null;
|
||||
statusError = '';
|
||||
}
|
||||
|
||||
async function saveStatus() {
|
||||
if (!statusCtx) return;
|
||||
const token = getToken();
|
||||
if (!token) {
|
||||
statusError = 'Your session has expired. Please sign in again.';
|
||||
return;
|
||||
}
|
||||
statusBusy = true;
|
||||
statusError = '';
|
||||
try {
|
||||
const res = await fetch('/api/owner/client-status', {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json', Authorization: `Bearer ${token}` },
|
||||
body: JSON.stringify({ email: statusCtx.email, status: statusDraft, reason: statusReason.trim() }),
|
||||
});
|
||||
const data = await res.json().catch(() => null);
|
||||
if (!res.ok) throw new Error(data?.detail?.message ?? data?.detail ?? 'Could not update status.');
|
||||
|
||||
const updated = data?.lifecycle as Lifecycle | undefined;
|
||||
if (updated) {
|
||||
allClients = allClients.map((c) =>
|
||||
c.email === statusCtx!.email ? { ...c, lifecycle: updated } : c,
|
||||
);
|
||||
}
|
||||
statusCtx = null;
|
||||
} catch (error) {
|
||||
statusError = error instanceof Error ? error.message : 'Could not update status.';
|
||||
} finally {
|
||||
statusBusy = false;
|
||||
}
|
||||
}
|
||||
|
||||
function getToken() {
|
||||
return window.localStorage.getItem('gw_onboarding_session') ?? '';
|
||||
@@ -825,6 +958,13 @@
|
||||
<svelte:head>
|
||||
<title>{header.title} | Goodwalk Admin</title>
|
||||
<meta name="robots" content="noindex, nofollow" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1, viewport-fit=cover" />
|
||||
<link rel="preconnect" href="https://fonts.googleapis.com" />
|
||||
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin="anonymous" />
|
||||
<link
|
||||
href="https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700;800&display=swap"
|
||||
rel="stylesheet"
|
||||
/>
|
||||
</svelte:head>
|
||||
|
||||
<main class="owner-page">
|
||||
@@ -1067,9 +1207,17 @@
|
||||
|
||||
<div class="owner-cell">
|
||||
<span class="owner-mobile-label">Status</span>
|
||||
<span class:owner-status-complete={client.status === 'completed'} class="owner-status-pill">
|
||||
{client.status === 'completed' ? 'Completed' : 'Pending'}
|
||||
</span>
|
||||
<div class="owner-status-stack">
|
||||
<span class:owner-status-complete={client.status === 'completed'} class="owner-status-pill">
|
||||
{client.status === 'completed' ? 'Completed' : 'Pending'}
|
||||
</span>
|
||||
<span class="owner-lifecycle-pill" data-lifecycle={client.lifecycle?.status ?? 'active'}>
|
||||
{lifecycleLabel(client.lifecycle?.status ?? 'active')}
|
||||
</span>
|
||||
{#if client.lifecycle?.reason}
|
||||
<small class="owner-lifecycle-reason" title={client.lifecycle.reason}>{client.lifecycle.reason}</small>
|
||||
{/if}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="owner-cell">
|
||||
@@ -1079,12 +1227,16 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="owner-cell owner-action-cell">
|
||||
<div class="owner-cell owner-action-cell owner-action-stack">
|
||||
<span class="owner-mobile-label">Action</span>
|
||||
<button class="owner-secondary-btn" type="button" on:click={() => openEnquiry(client.email)}>
|
||||
<Icon name="fas fa-file-lines" />
|
||||
View enquiry
|
||||
</button>
|
||||
<button class="owner-secondary-btn" type="button" on:click={() => openStatusEditor(client)}>
|
||||
<Icon name="fas fa-circle-half-stroke" />
|
||||
Change status
|
||||
</button>
|
||||
</div>
|
||||
</article>
|
||||
{/each}
|
||||
@@ -1473,8 +1625,8 @@
|
||||
</section>
|
||||
|
||||
{#if confirmCtx}
|
||||
<div class="owner-modal-backdrop" role="presentation" on:click={closeConfirm}>
|
||||
<div class="owner-modal" role="dialog" aria-modal="true" aria-label="Confirm send" on:click|stopPropagation>
|
||||
<div class="owner-modal-backdrop" role="presentation" on:click|self={closeConfirm}>
|
||||
<div class="owner-modal" role="dialog" aria-modal="true" aria-label="Confirm send" tabindex="-1">
|
||||
<div class="owner-modal-head">
|
||||
<h2>
|
||||
{#if confirmCtx.kind === 'welcome'}
|
||||
@@ -1506,9 +1658,34 @@
|
||||
{/if}
|
||||
</p>
|
||||
|
||||
<div class="owner-modal-preview-shell">
|
||||
{#if confirmPreviewHtml}
|
||||
<iframe
|
||||
class="owner-modal-preview-iframe"
|
||||
title="Email preview"
|
||||
sandbox=""
|
||||
srcdoc={confirmPreviewHtml}
|
||||
></iframe>
|
||||
<div class="owner-modal-preview-actions">
|
||||
<small>This is exactly how the email will look.</small>
|
||||
<button type="button" class="owner-modal-preview-link" on:click={openConfirmPreviewInNewTab}>
|
||||
<Icon name="fas fa-up-right-from-square" />
|
||||
Open full preview
|
||||
</button>
|
||||
</div>
|
||||
{:else}
|
||||
<div class="owner-modal-preview-empty">
|
||||
{confirmPreviewBusy ? 'Rendering preview…' : 'Preview unavailable.'}
|
||||
</div>
|
||||
{/if}
|
||||
{#if confirmPreviewBusy && confirmPreviewHtml}
|
||||
<span class="owner-modal-preview-busy">Refreshing…</span>
|
||||
{/if}
|
||||
</div>
|
||||
|
||||
<label class="owner-preview-toggle">
|
||||
<input type="checkbox" bind:checked={confirmPreview} disabled={confirmBusy} />
|
||||
<span>Send a preview to me ({ownerEmail}) instead</span>
|
||||
<span>Also send a test copy to me ({ownerEmail}) instead of the client</span>
|
||||
</label>
|
||||
|
||||
{#if confirmError}
|
||||
@@ -1531,9 +1708,62 @@
|
||||
</div>
|
||||
{/if}
|
||||
|
||||
{#if statusCtx}
|
||||
<div class="owner-modal-backdrop" role="presentation" on:click|self={closeStatusEditor}>
|
||||
<div class="owner-modal" role="dialog" aria-modal="true" aria-label="Change client status" tabindex="-1">
|
||||
<div class="owner-modal-head">
|
||||
<h2>Change status for {statusCtx.fullName || statusCtx.email}</h2>
|
||||
<button type="button" class="owner-modal-close" on:click={closeStatusEditor} aria-label="Close" disabled={statusBusy}>
|
||||
<Icon name="fas fa-xmark" />
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<p class="owner-modal-status">
|
||||
Soft-change only. The client stays on file in case we want to reach out later
|
||||
(e.g. a newsletter). Cancelled and archived clients are excluded from outreach.
|
||||
</p>
|
||||
|
||||
<div class="status-options">
|
||||
{#each lifecycleOptions as opt}
|
||||
<label class:status-option-active={statusDraft === opt.value} class="status-option">
|
||||
<input type="radio" name="lifecycle-status" value={opt.value} bind:group={statusDraft} disabled={statusBusy} />
|
||||
<div>
|
||||
<strong>{opt.label}</strong>
|
||||
<small>{opt.description}</small>
|
||||
</div>
|
||||
</label>
|
||||
{/each}
|
||||
</div>
|
||||
|
||||
<label class="status-reason-field">
|
||||
<span>Reason (optional)</span>
|
||||
<textarea bind:value={statusReason} rows="3" maxlength="500" disabled={statusBusy}
|
||||
placeholder="e.g. moved overseas, switched walkers, on hold until spring…"></textarea>
|
||||
</label>
|
||||
|
||||
{#if statusCtx.lifecycle?.changedAt}
|
||||
<p class="owner-modal-status">
|
||||
Last changed {formatDateLabel(statusCtx.lifecycle.changedAt)}{statusCtx.lifecycle.changedBy ? ` by ${statusCtx.lifecycle.changedBy}` : ''}.
|
||||
</p>
|
||||
{/if}
|
||||
|
||||
{#if statusError}
|
||||
<div class="owner-inline-error">{statusError}</div>
|
||||
{/if}
|
||||
|
||||
<div class="owner-compose-actions">
|
||||
<button class="owner-cancel-btn" type="button" on:click={closeStatusEditor} disabled={statusBusy}>Cancel</button>
|
||||
<button class="owner-send-btn" type="button" on:click={saveStatus} disabled={statusBusy}>
|
||||
{statusBusy ? 'Saving…' : 'Save status'}
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{/if}
|
||||
|
||||
{#if enquiryOpen}
|
||||
<div class="owner-modal-backdrop" role="presentation" on:click={closeEnquiry}>
|
||||
<div class="owner-modal" role="dialog" aria-modal="true" aria-label="Client enquiry" on:click|stopPropagation>
|
||||
<div class="owner-modal-backdrop" role="presentation" on:click|self={closeEnquiry}>
|
||||
<div class="owner-modal" role="dialog" aria-modal="true" aria-label="Client enquiry" tabindex="-1">
|
||||
<div class="owner-modal-head">
|
||||
<h2>Website enquiry</h2>
|
||||
<button type="button" class="owner-modal-close" on:click={closeEnquiry} aria-label="Close">
|
||||
@@ -1585,6 +1815,18 @@
|
||||
|
||||
<style>
|
||||
.owner-page {
|
||||
/* Scope a clean Inter type system to the admin only — leaves the public
|
||||
site's Unbounded / Readex Pro stack untouched. */
|
||||
--admin-font: 'Inter', -apple-system, BlinkMacSystemFont, 'Segoe UI', system-ui, sans-serif;
|
||||
--font-head: var(--admin-font);
|
||||
--font-body: var(--admin-font);
|
||||
font-family: var(--admin-font);
|
||||
font-feature-settings: 'cv11', 'ss01', 'ss03';
|
||||
letter-spacing: -0.005em;
|
||||
-webkit-font-smoothing: antialiased;
|
||||
-moz-osx-font-smoothing: grayscale;
|
||||
-webkit-text-size-adjust: 100%;
|
||||
|
||||
min-height: 100vh;
|
||||
background: #f7f6f1;
|
||||
color: var(--gw-green);
|
||||
@@ -1592,6 +1834,13 @@
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.owner-page :global(input),
|
||||
.owner-page :global(textarea),
|
||||
.owner-page :global(select),
|
||||
.owner-page :global(button) {
|
||||
font-family: inherit;
|
||||
}
|
||||
|
||||
.owner-shell {
|
||||
max-width: 1120px;
|
||||
margin: 0 auto;
|
||||
@@ -1719,29 +1968,8 @@
|
||||
to { transform: rotate(360deg); }
|
||||
}
|
||||
|
||||
.owner-hero {
|
||||
padding: 16px 0 10px;
|
||||
}
|
||||
|
||||
.owner-hero h1 {
|
||||
margin: 0 0 6px;
|
||||
color: rgba(17, 23, 27, 0.9);
|
||||
font-family: var(--font-head);
|
||||
font-size: clamp(22px, 3.4vw, 30px);
|
||||
line-height: 1.1;
|
||||
letter-spacing: -0.03em;
|
||||
}
|
||||
|
||||
.owner-hero p {
|
||||
margin: 0;
|
||||
max-width: 62ch;
|
||||
color: rgba(33, 48, 33, 0.66);
|
||||
font-size: 14px;
|
||||
line-height: 1.55;
|
||||
}
|
||||
|
||||
.owner-content {
|
||||
padding: 16px 0 112px;
|
||||
padding: 16px 0 calc(108px + env(safe-area-inset-bottom, 0px));
|
||||
}
|
||||
|
||||
.owner-summary-grid {
|
||||
@@ -1801,7 +2029,6 @@
|
||||
justify-content: flex-end;
|
||||
}
|
||||
|
||||
.owner-section-head h2,
|
||||
.owner-message-card h2 {
|
||||
margin: 0;
|
||||
color: #171b20;
|
||||
@@ -1811,7 +2038,6 @@
|
||||
letter-spacing: -0.02em;
|
||||
}
|
||||
|
||||
.owner-section-head p,
|
||||
.owner-message-card p {
|
||||
margin: 6px 0 0;
|
||||
color: rgba(33, 48, 33, 0.64);
|
||||
@@ -1932,24 +2158,6 @@
|
||||
overflow-wrap: anywhere;
|
||||
}
|
||||
|
||||
.owner-chip-list {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.owner-chip {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
min-height: 32px;
|
||||
padding: 0 10px;
|
||||
border-radius: 999px;
|
||||
background: rgba(33, 48, 33, 0.08);
|
||||
color: #213021;
|
||||
font-size: 12px;
|
||||
font-weight: 700;
|
||||
}
|
||||
|
||||
.owner-status-pill {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
@@ -1971,6 +2179,122 @@
|
||||
color: #213021;
|
||||
}
|
||||
|
||||
.owner-status-stack {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 6px;
|
||||
}
|
||||
|
||||
.owner-lifecycle-pill {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
min-height: 26px;
|
||||
padding: 0 10px;
|
||||
border-radius: 999px;
|
||||
font-size: 10.5px;
|
||||
font-weight: 700;
|
||||
letter-spacing: 0.08em;
|
||||
text-transform: uppercase;
|
||||
background: rgba(33, 48, 33, 0.08);
|
||||
color: rgba(33, 48, 33, 0.72);
|
||||
}
|
||||
|
||||
.owner-lifecycle-pill[data-lifecycle='active'] {
|
||||
background: rgba(60, 140, 75, 0.16);
|
||||
color: #1f5a2c;
|
||||
}
|
||||
|
||||
.owner-lifecycle-pill[data-lifecycle='paused'] {
|
||||
background: rgba(255, 209, 0, 0.22);
|
||||
color: #6a4a00;
|
||||
}
|
||||
|
||||
.owner-lifecycle-pill[data-lifecycle='cancelled'] {
|
||||
background: rgba(190, 60, 60, 0.14);
|
||||
color: #8a2a2a;
|
||||
}
|
||||
|
||||
.owner-lifecycle-pill[data-lifecycle='archived'] {
|
||||
background: rgba(33, 48, 33, 0.12);
|
||||
color: rgba(33, 48, 33, 0.62);
|
||||
}
|
||||
|
||||
.owner-lifecycle-reason {
|
||||
color: rgba(33, 48, 33, 0.6);
|
||||
font-size: 11px;
|
||||
line-height: 1.35;
|
||||
max-width: 220px;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.status-options {
|
||||
display: grid;
|
||||
gap: 8px;
|
||||
margin: 6px 0 10px;
|
||||
}
|
||||
|
||||
.status-option {
|
||||
display: flex;
|
||||
align-items: flex-start;
|
||||
gap: 10px;
|
||||
padding: 10px 12px;
|
||||
border-radius: 14px;
|
||||
border: 1px solid rgba(33, 48, 33, 0.12);
|
||||
cursor: pointer;
|
||||
background: rgba(255, 255, 255, 0.6);
|
||||
}
|
||||
|
||||
.status-option input {
|
||||
margin-top: 3px;
|
||||
}
|
||||
|
||||
.status-option strong {
|
||||
display: block;
|
||||
font-size: 13px;
|
||||
color: #171b20;
|
||||
}
|
||||
|
||||
.status-option small {
|
||||
display: block;
|
||||
margin-top: 2px;
|
||||
color: rgba(33, 48, 33, 0.62);
|
||||
font-size: 12px;
|
||||
line-height: 1.4;
|
||||
}
|
||||
|
||||
.status-option-active {
|
||||
border-color: rgba(33, 48, 33, 0.36);
|
||||
background: rgba(255, 209, 0, 0.12);
|
||||
}
|
||||
|
||||
.status-reason-field {
|
||||
display: grid;
|
||||
gap: 6px;
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.status-reason-field span {
|
||||
font-size: 12px;
|
||||
font-weight: 700;
|
||||
color: rgba(33, 48, 33, 0.72);
|
||||
letter-spacing: 0.04em;
|
||||
text-transform: uppercase;
|
||||
}
|
||||
|
||||
.status-reason-field textarea {
|
||||
width: 100%;
|
||||
padding: 10px 12px;
|
||||
border-radius: 12px;
|
||||
border: 1px solid rgba(33, 48, 33, 0.16);
|
||||
font: inherit;
|
||||
font-size: 13px;
|
||||
resize: vertical;
|
||||
min-height: 70px;
|
||||
}
|
||||
|
||||
.owner-open-btn,
|
||||
.owner-send-btn,
|
||||
.owner-cancel-btn,
|
||||
@@ -2175,43 +2499,51 @@
|
||||
|
||||
.owner-bottom-nav {
|
||||
position: fixed;
|
||||
left: 12px;
|
||||
right: 12px;
|
||||
bottom: calc(12px + env(safe-area-inset-bottom, 0px));
|
||||
left: 10px;
|
||||
right: 10px;
|
||||
bottom: calc(10px + env(safe-area-inset-bottom, 0px));
|
||||
z-index: 25;
|
||||
display: grid;
|
||||
grid-template-columns: repeat(4, minmax(0, 1fr));
|
||||
gap: 6px;
|
||||
grid-template-columns: repeat(5, minmax(0, 1fr));
|
||||
gap: 2px;
|
||||
padding: 6px;
|
||||
border-radius: 16px;
|
||||
background: rgba(255, 253, 250, 0.95);
|
||||
backdrop-filter: blur(12px);
|
||||
border: 1px solid rgba(17, 20, 24, 0.08);
|
||||
box-shadow: 0 6px 18px rgba(17, 20, 24, 0.08);
|
||||
border-radius: 20px;
|
||||
background: rgba(255, 255, 255, 0.92);
|
||||
backdrop-filter: saturate(180%) blur(18px);
|
||||
-webkit-backdrop-filter: saturate(180%) blur(18px);
|
||||
border: 1px solid rgba(17, 20, 24, 0.06);
|
||||
box-shadow: 0 10px 28px rgba(17, 20, 24, 0.12), 0 1px 0 rgba(255, 255, 255, 0.6) inset;
|
||||
}
|
||||
|
||||
.owner-bottom-tab {
|
||||
min-height: 48px;
|
||||
min-height: 54px;
|
||||
border: 0;
|
||||
border-radius: 12px;
|
||||
border-radius: 14px;
|
||||
background: transparent;
|
||||
display: grid;
|
||||
justify-items: center;
|
||||
align-content: center;
|
||||
gap: 3px;
|
||||
color: rgba(33, 48, 33, 0.58);
|
||||
gap: 4px;
|
||||
color: rgba(33, 48, 33, 0.6);
|
||||
font: inherit;
|
||||
cursor: pointer;
|
||||
transition: background 0.18s ease, color 0.18s ease, transform 0.18s ease;
|
||||
-webkit-tap-highlight-color: transparent;
|
||||
}
|
||||
|
||||
.owner-bottom-tab:active {
|
||||
transform: scale(0.96);
|
||||
}
|
||||
|
||||
.owner-bottom-tab span {
|
||||
font-size: 10px;
|
||||
font-size: 11px;
|
||||
font-weight: 600;
|
||||
line-height: 1.1;
|
||||
letter-spacing: -0.005em;
|
||||
}
|
||||
|
||||
.owner-bottom-tab :global(i) {
|
||||
font-size: 13px;
|
||||
font-size: 16px;
|
||||
}
|
||||
|
||||
.owner-bottom-tab-active {
|
||||
@@ -2219,8 +2551,11 @@
|
||||
color: #fff;
|
||||
}
|
||||
|
||||
.owner-bottom-tab-active span {
|
||||
font-weight: 700;
|
||||
}
|
||||
|
||||
@media (hover: hover) {
|
||||
.owner-tab:hover,
|
||||
.owner-open-btn:hover,
|
||||
.owner-send-btn:hover,
|
||||
.owner-cancel-btn:hover,
|
||||
@@ -2398,10 +2733,10 @@
|
||||
color: rgba(33, 48, 33, 0.4);
|
||||
}
|
||||
|
||||
.rte-editor p { margin: 0 0 12px; }
|
||||
.rte-editor ul,
|
||||
.rte-editor ol { margin: 0 0 12px; padding-left: 22px; }
|
||||
.rte-editor a { color: #c0392b; text-decoration: underline; }
|
||||
.rte-editor :global(p) { margin: 0 0 12px; }
|
||||
.rte-editor :global(ul),
|
||||
.rte-editor :global(ol) { margin: 0 0 12px; padding-left: 22px; }
|
||||
.rte-editor :global(a) { color: #c0392b; text-decoration: underline; }
|
||||
|
||||
.msg-form {
|
||||
display: grid;
|
||||
@@ -2563,25 +2898,6 @@
|
||||
gap: 10px;
|
||||
}
|
||||
|
||||
.owner-field textarea {
|
||||
width: 100%;
|
||||
min-height: 110px;
|
||||
padding: 12px 15px;
|
||||
border: 1px solid rgba(17, 20, 24, 0.1);
|
||||
border-radius: 14px;
|
||||
background: #fffdfa;
|
||||
font: inherit;
|
||||
color: #213021;
|
||||
outline: none;
|
||||
box-sizing: border-box;
|
||||
resize: vertical;
|
||||
}
|
||||
|
||||
.owner-field textarea:focus {
|
||||
border-color: rgba(33, 48, 33, 0.42);
|
||||
box-shadow: 0 0 0 4px rgba(255, 209, 0, 0.12);
|
||||
}
|
||||
|
||||
.msg-recipient-modes {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
@@ -2662,35 +2978,7 @@
|
||||
}
|
||||
}
|
||||
|
||||
.owner-footer-desktop {
|
||||
display: none;
|
||||
}
|
||||
|
||||
@media (min-width: 900px) {
|
||||
.owner-footer-desktop {
|
||||
display: block;
|
||||
position: fixed;
|
||||
left: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
z-index: 20;
|
||||
}
|
||||
|
||||
.owner-footer-desktop :global(.ob-footer-inner) {
|
||||
height: 38px;
|
||||
padding: 0 28px;
|
||||
}
|
||||
|
||||
.owner-footer-desktop :global(.ob-footer-back),
|
||||
.owner-footer-desktop :global(.ob-footer-email),
|
||||
.owner-footer-desktop :global(.ob-footer-logout) {
|
||||
font-size: 11px;
|
||||
}
|
||||
|
||||
.owner-footer-desktop :global(.ob-footer-logout) {
|
||||
padding: 4px 10px;
|
||||
}
|
||||
|
||||
.owner-content {
|
||||
padding-bottom: 60px;
|
||||
}
|
||||
@@ -2703,24 +2991,97 @@
|
||||
.owner-modal-backdrop {
|
||||
position: fixed;
|
||||
inset: 0;
|
||||
background: rgba(17, 20, 24, 0.45);
|
||||
backdrop-filter: blur(2px);
|
||||
background: rgba(17, 20, 24, 0.5);
|
||||
backdrop-filter: blur(4px);
|
||||
-webkit-backdrop-filter: blur(4px);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
padding: 16px;
|
||||
padding: 14px;
|
||||
padding-bottom: calc(14px + env(safe-area-inset-bottom, 0px));
|
||||
padding-top: calc(14px + env(safe-area-inset-top, 0px));
|
||||
z-index: 60;
|
||||
}
|
||||
|
||||
.owner-modal {
|
||||
width: 100%;
|
||||
max-width: 480px;
|
||||
max-height: 90vh;
|
||||
max-width: 560px;
|
||||
max-height: min(92vh, 800px);
|
||||
overflow-y: auto;
|
||||
overscroll-behavior: contain;
|
||||
-webkit-overflow-scrolling: touch;
|
||||
background: #fff;
|
||||
border-radius: 16px;
|
||||
padding: 18px;
|
||||
box-shadow: 0 24px 60px rgba(17, 20, 24, 0.25);
|
||||
border-radius: 18px;
|
||||
padding: 20px;
|
||||
box-shadow: 0 24px 60px rgba(17, 20, 24, 0.28);
|
||||
}
|
||||
|
||||
.owner-modal-preview-shell {
|
||||
margin: 6px 0 14px;
|
||||
border-radius: 14px;
|
||||
border: 1px solid rgba(17, 20, 24, 0.08);
|
||||
background: #f7f6f1;
|
||||
overflow: hidden;
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.owner-modal-preview-iframe {
|
||||
display: block;
|
||||
width: 100%;
|
||||
height: 320px;
|
||||
border: 0;
|
||||
background: #fff;
|
||||
}
|
||||
|
||||
.owner-modal-preview-busy {
|
||||
position: absolute;
|
||||
top: 8px;
|
||||
right: 10px;
|
||||
font-size: 11px;
|
||||
font-weight: 600;
|
||||
color: rgba(33, 48, 33, 0.6);
|
||||
background: rgba(255, 255, 255, 0.85);
|
||||
padding: 3px 8px;
|
||||
border-radius: 999px;
|
||||
}
|
||||
|
||||
.owner-modal-preview-empty {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
min-height: 120px;
|
||||
color: rgba(33, 48, 33, 0.5);
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.owner-modal-preview-actions {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
gap: 8px;
|
||||
padding: 8px 10px;
|
||||
background: #fff;
|
||||
border-top: 1px solid rgba(17, 20, 24, 0.06);
|
||||
}
|
||||
|
||||
.owner-modal-preview-actions small {
|
||||
color: rgba(33, 48, 33, 0.55);
|
||||
font-size: 12px;
|
||||
}
|
||||
|
||||
.owner-modal-preview-link {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 6px;
|
||||
padding: 6px 12px;
|
||||
border-radius: 999px;
|
||||
border: 1px solid rgba(33, 48, 33, 0.16);
|
||||
background: #fff;
|
||||
color: #213021;
|
||||
font-size: 12px;
|
||||
font-weight: 700;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.owner-modal-head {
|
||||
@@ -2832,21 +3193,56 @@
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.owner-page {
|
||||
/* Prevent any element from breaking horizontal scroll on a phone. */
|
||||
overflow-x: hidden;
|
||||
}
|
||||
|
||||
.owner-shell {
|
||||
padding-left: max(16px, env(safe-area-inset-left, 0px));
|
||||
padding-right: max(16px, env(safe-area-inset-right, 0px));
|
||||
}
|
||||
|
||||
.owner-topbar-inner {
|
||||
min-height: 56px;
|
||||
padding-top: 10px;
|
||||
padding-bottom: 10px;
|
||||
}
|
||||
|
||||
.owner-logo img {
|
||||
height: 22px;
|
||||
}
|
||||
|
||||
.owner-pill {
|
||||
font-size: 10px;
|
||||
padding: 4px 9px;
|
||||
}
|
||||
|
||||
.owner-topbar-logout {
|
||||
width: 36px;
|
||||
height: 36px;
|
||||
padding: 0;
|
||||
justify-content: center;
|
||||
border-radius: 999px;
|
||||
}
|
||||
|
||||
.owner-summary-grid {
|
||||
gap: 6px;
|
||||
}
|
||||
|
||||
.owner-summary-pill {
|
||||
padding: 8px 10px;
|
||||
gap: 8px;
|
||||
padding: 10px 12px;
|
||||
gap: 10px;
|
||||
border-radius: 14px;
|
||||
}
|
||||
|
||||
.owner-summary-pill strong {
|
||||
font-size: 16px;
|
||||
font-size: 18px;
|
||||
letter-spacing: -0.01em;
|
||||
}
|
||||
|
||||
.owner-summary-pill span {
|
||||
font-size: 10px;
|
||||
font-size: 11px;
|
||||
}
|
||||
|
||||
.owner-section-head {
|
||||
@@ -2859,6 +3255,16 @@
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.owner-table-row {
|
||||
border-radius: 16px;
|
||||
padding: 14px;
|
||||
gap: 10px;
|
||||
}
|
||||
|
||||
.owner-cell h3 {
|
||||
font-size: 16px;
|
||||
}
|
||||
|
||||
.owner-pagination {
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
@@ -2867,19 +3273,45 @@
|
||||
}
|
||||
|
||||
.owner-pagination .owner-page-btn {
|
||||
min-height: 34px;
|
||||
padding: 0 12px;
|
||||
font-size: 12px;
|
||||
min-height: 40px;
|
||||
padding: 0 14px;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.owner-pagination .owner-page-indicator {
|
||||
font-size: 12px;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.owner-enquiry-list > div {
|
||||
grid-template-columns: 1fr;
|
||||
gap: 2px;
|
||||
}
|
||||
|
||||
.owner-modal {
|
||||
max-width: 100%;
|
||||
max-height: calc(100dvh - 90px - env(safe-area-inset-bottom, 0px));
|
||||
border-radius: 20px 20px 16px 16px;
|
||||
padding: 18px 16px;
|
||||
}
|
||||
|
||||
.owner-modal-head h2 {
|
||||
font-size: 17px;
|
||||
}
|
||||
|
||||
.owner-modal-close {
|
||||
width: 36px;
|
||||
height: 36px;
|
||||
}
|
||||
|
||||
.owner-modal-preview-iframe {
|
||||
height: 260px;
|
||||
}
|
||||
|
||||
.owner-send-btn,
|
||||
.owner-cancel-btn {
|
||||
min-height: 46px;
|
||||
font-size: 14px;
|
||||
}
|
||||
}
|
||||
|
||||
/* ── Activity tab ── */
|
||||
|
||||
@@ -43,6 +43,23 @@ export function buildAreaServed(locations = locationPages) {
|
||||
}));
|
||||
}
|
||||
|
||||
export const goodwalkProviderNode = {
|
||||
'@type': ['LocalBusiness', 'PetCareService'],
|
||||
'@id': `${siteUrl}/#business`,
|
||||
name: 'Goodwalk',
|
||||
url: siteUrl,
|
||||
logo: `${siteUrl}/images/goodwalk-auckland-dog-walking-logo.webp`,
|
||||
image: `${siteUrl}/images/goodwalk-auckland-dog-walking-logo.webp`,
|
||||
email: 'info@goodwalk.co.nz',
|
||||
telephone: '+64226421011',
|
||||
address: {
|
||||
'@type': 'PostalAddress',
|
||||
addressLocality: 'Auckland Central',
|
||||
addressRegion: 'Auckland',
|
||||
addressCountry: 'NZ'
|
||||
}
|
||||
} as const;
|
||||
|
||||
function getLocationSeoImage(location: LocationPageContent) {
|
||||
return location.seo?.image ?? location.parks.find((park) => park.image?.src)?.image?.src ?? defaultLocationImage;
|
||||
}
|
||||
@@ -81,9 +98,7 @@ export function buildLocationSeo(location: LocationPageContent) {
|
||||
name: serviceName,
|
||||
description,
|
||||
serviceType,
|
||||
provider: {
|
||||
'@id': `${siteUrl}/#business`
|
||||
},
|
||||
provider: goodwalkProviderNode,
|
||||
areaServed: {
|
||||
'@type': 'Place',
|
||||
name: `${location.suburb}, Auckland, New Zealand`
|
||||
|
||||
@@ -25,6 +25,11 @@ img {
|
||||
display: block;
|
||||
max-width: 100%;
|
||||
height: auto;
|
||||
/* Hide alt text flash before pixels paint. Alt is still exposed to
|
||||
screen readers; this only suppresses the visual text on the
|
||||
broken-image placeholder shown by some browsers mid-load. */
|
||||
color: transparent;
|
||||
font-size: 0;
|
||||
}
|
||||
|
||||
a {
|
||||
|
||||
@@ -55,7 +55,7 @@ header {
|
||||
|
||||
@media (min-width: 769px) {
|
||||
.nav-ribbon::after {
|
||||
animation: nav-ribbon-shine 8s ease-in-out 2.8s infinite;
|
||||
animation: nav-ribbon-shine 8s ease-in-out 2.8s 2;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -0,0 +1,202 @@
|
||||
/* ─────────────────────────────────────────────────────────────────────────
|
||||
Motion system
|
||||
|
||||
Reusable, opt-in animation utilities. One easing family, one set of
|
||||
distances, one set of durations. Designed to pair with the existing
|
||||
`use:reveal` action: apply the action on a parent section, then add
|
||||
motion utilities to inner elements.
|
||||
|
||||
Naming
|
||||
.fade-up opacity + translateY
|
||||
.fade-in opacity only
|
||||
.scale-soft opacity + subtle scale-in (0.985 → 1)
|
||||
.stagger-children cascades children at --motion-stagger-step
|
||||
.hover-lift elevates on hover (hover-capable devices only)
|
||||
.cta-shimmer slow, subtle sheen across a CTA (one-off, not loop)
|
||||
.subtle-glow soft, breathing focus halo for accent surfaces
|
||||
.image-parallax gentle drift; CSS scroll-driven where supported
|
||||
|
||||
All motion is gated behind prefers-reduced-motion: no-preference.
|
||||
───────────────────────────────────────────────────────────────────────── */
|
||||
|
||||
@media (prefers-reduced-motion: no-preference) {
|
||||
/* ── Reveal utilities ─────────────────────────────────────────────────
|
||||
These start in an "off" state and animate when an ancestor carries
|
||||
`.reveal-visible` (set by the existing reveal action). That keeps
|
||||
trigger logic centralized — no extra observers per element. */
|
||||
.fade-up,
|
||||
.fade-in,
|
||||
.scale-soft {
|
||||
opacity: 0;
|
||||
transition:
|
||||
opacity var(--motion-duration-md) var(--ease-out-quiet),
|
||||
transform var(--motion-duration-md) var(--ease-out-quiet);
|
||||
will-change: opacity, transform;
|
||||
}
|
||||
|
||||
.fade-up {
|
||||
transform: translate3d(0, var(--motion-distance-md), 0);
|
||||
}
|
||||
|
||||
.scale-soft {
|
||||
transform: scale(0.985);
|
||||
transform-origin: center;
|
||||
}
|
||||
|
||||
/* Reveal trigger: when a parent reveal block becomes visible, or the
|
||||
element itself carries `.is-in` (for non-reveal contexts). */
|
||||
.reveal-visible .fade-up,
|
||||
.reveal-visible .fade-in,
|
||||
.reveal-visible .scale-soft,
|
||||
.fade-up.is-in,
|
||||
.fade-in.is-in,
|
||||
.scale-soft.is-in {
|
||||
opacity: 1;
|
||||
transform: none;
|
||||
}
|
||||
|
||||
/* Once the animation has played, drop will-change to free the layer. */
|
||||
.reveal-visible .fade-up,
|
||||
.reveal-visible .fade-in,
|
||||
.reveal-visible .scale-soft {
|
||||
will-change: auto;
|
||||
}
|
||||
|
||||
/* ── Stagger ──────────────────────────────────────────────────────────
|
||||
Direct-child cascade. Up to 8 children covered explicitly; beyond
|
||||
that, children inherit the 8th delay (graceful fallback for grids). */
|
||||
.stagger-children > * {
|
||||
transition-delay: 0ms;
|
||||
}
|
||||
.stagger-children > :nth-child(1) { transition-delay: calc(var(--motion-stagger-step) * 0); }
|
||||
.stagger-children > :nth-child(2) { transition-delay: calc(var(--motion-stagger-step) * 1); }
|
||||
.stagger-children > :nth-child(3) { transition-delay: calc(var(--motion-stagger-step) * 2); }
|
||||
.stagger-children > :nth-child(4) { transition-delay: calc(var(--motion-stagger-step) * 3); }
|
||||
.stagger-children > :nth-child(5) { transition-delay: calc(var(--motion-stagger-step) * 4); }
|
||||
.stagger-children > :nth-child(6) { transition-delay: calc(var(--motion-stagger-step) * 5); }
|
||||
.stagger-children > :nth-child(7) { transition-delay: calc(var(--motion-stagger-step) * 6); }
|
||||
.stagger-children > :nth-child(n + 8) { transition-delay: calc(var(--motion-stagger-step) * 7); }
|
||||
|
||||
/* Tighter cascade for dense grids (8+ cells). Opt-in. */
|
||||
.stagger-children.stagger-tight > * {
|
||||
--motion-stagger-step: 45ms;
|
||||
}
|
||||
|
||||
/* ── Hover lift ───────────────────────────────────────────────────────
|
||||
Quiet 2px elevation. Only on hover-capable devices so touch users
|
||||
don't get a sticky transformed state after tap. */
|
||||
.hover-lift {
|
||||
transition:
|
||||
transform var(--motion-duration-sm) var(--ease-out-soft),
|
||||
box-shadow var(--motion-duration-sm) var(--ease-out-soft);
|
||||
}
|
||||
|
||||
@media (hover: hover) {
|
||||
.hover-lift:hover {
|
||||
transform: translateY(-2px);
|
||||
}
|
||||
}
|
||||
|
||||
/* Focus-visible matches hover for keyboard parity. */
|
||||
.hover-lift:focus-visible {
|
||||
transform: translateY(-2px);
|
||||
outline: none;
|
||||
}
|
||||
|
||||
/* ── CTA shimmer ──────────────────────────────────────────────────────
|
||||
A single, soft sheen sweep that fires once on hover. Quiet, not a
|
||||
looping ad-banner. Requires the host to be position: relative and
|
||||
overflow: hidden, which most pill buttons already are. */
|
||||
.cta-shimmer {
|
||||
position: relative;
|
||||
overflow: hidden;
|
||||
isolation: isolate;
|
||||
}
|
||||
|
||||
.cta-shimmer::after {
|
||||
content: '';
|
||||
position: absolute;
|
||||
inset: 0;
|
||||
background: linear-gradient(
|
||||
115deg,
|
||||
transparent 30%,
|
||||
rgba(255, 255, 255, 0.32) 50%,
|
||||
transparent 70%
|
||||
);
|
||||
transform: translateX(-110%);
|
||||
transition: transform var(--motion-duration-lg) var(--ease-in-out-soft);
|
||||
pointer-events: none;
|
||||
z-index: 1;
|
||||
}
|
||||
|
||||
@media (hover: hover) {
|
||||
.cta-shimmer:hover::after {
|
||||
transform: translateX(110%);
|
||||
}
|
||||
}
|
||||
|
||||
/* ── Subtle glow ──────────────────────────────────────────────────────
|
||||
A slow, breathing accent halo. Calibrated to be barely perceptible —
|
||||
the kind of thing you notice only after watching for a few seconds. */
|
||||
.subtle-glow {
|
||||
animation: motion-subtle-glow 5.2s var(--ease-in-out-soft) infinite;
|
||||
}
|
||||
|
||||
@keyframes motion-subtle-glow {
|
||||
0%, 100% {
|
||||
box-shadow: 0 0 0 0 rgba(255, 209, 0, 0);
|
||||
}
|
||||
50% {
|
||||
box-shadow: 0 0 0 6px rgba(255, 209, 0, 0.08);
|
||||
}
|
||||
}
|
||||
|
||||
/* ── Image parallax ───────────────────────────────────────────────────
|
||||
Gentle vertical drift on scroll. Uses CSS scroll-driven animations
|
||||
where supported (Chrome 115+, Edge 115+). Older browsers get a still
|
||||
image — perfectly fine fallback, no jank, no JS. */
|
||||
@supports (animation-timeline: view()) {
|
||||
.image-parallax {
|
||||
animation: motion-image-parallax linear both;
|
||||
animation-timeline: view();
|
||||
animation-range: cover 0% cover 100%;
|
||||
will-change: transform;
|
||||
}
|
||||
|
||||
@keyframes motion-image-parallax {
|
||||
from { transform: translate3d(0, -3%, 0); }
|
||||
to { transform: translate3d(0, 3%, 0); }
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/* ── Reduced motion ─────────────────────────────────────────────────────
|
||||
Strip every transition and animation that this system introduces. The
|
||||
reveal utilities collapse to their visible state immediately so layout
|
||||
stays correct. */
|
||||
@media (prefers-reduced-motion: reduce) {
|
||||
.fade-up,
|
||||
.fade-in,
|
||||
.scale-soft {
|
||||
opacity: 1;
|
||||
transform: none;
|
||||
transition: none;
|
||||
}
|
||||
|
||||
.stagger-children > * {
|
||||
transition-delay: 0 !important;
|
||||
}
|
||||
|
||||
.hover-lift,
|
||||
.cta-shimmer::after,
|
||||
.subtle-glow,
|
||||
.image-parallax {
|
||||
transition: none !important;
|
||||
animation: none !important;
|
||||
transform: none !important;
|
||||
}
|
||||
|
||||
.cta-shimmer::after {
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
@@ -761,12 +761,71 @@
|
||||
}
|
||||
}
|
||||
|
||||
/* 14" laptop range: dog still clips the centered headline at the existing
|
||||
54% / -200px settings. Shrink the footprint and pull it further left so
|
||||
the mascot fully clears the text column. */
|
||||
@media (min-width: 1024px) and (max-width: 1439px) {
|
||||
.hero-img img {
|
||||
width: 46%;
|
||||
transform: translateX(-260px);
|
||||
}
|
||||
}
|
||||
|
||||
@media (min-width: 1200px) {
|
||||
body:has(#hero) .hero-inner {
|
||||
padding-bottom: 120px;
|
||||
}
|
||||
}
|
||||
|
||||
/* Laptop screens (13–14"): the hero is otherwise capped at 680px, which
|
||||
leaves the text crowding the top of the section. Give it more room so
|
||||
the bottom-anchored text sits lower in the viewport. */
|
||||
@media (min-width: 1024px) and (max-width: 1439px) {
|
||||
#hero {
|
||||
min-height: 92svh;
|
||||
max-height: 920px;
|
||||
}
|
||||
}
|
||||
|
||||
/* ── 13/14" laptop polish ───────────────────────────────────────────────────
|
||||
The default type and spacing scales were tuned on a large external
|
||||
monitor — their caps land at sizes that feel oversized on the shorter
|
||||
720–900px viewports typical of 13–14" laptops. Pull the caps down so
|
||||
headings, hero text, and section rhythm sit comfortably above the fold
|
||||
without changing the visual identity. Tokens flow through every section
|
||||
that reads from them. ── */
|
||||
@media (min-width: 1024px) and (max-width: 1439px) {
|
||||
:root {
|
||||
/* Type scale — trim caps ~15% so headings stop dominating the viewport */
|
||||
--text-display: clamp(38px, 3.8vw, 46px);
|
||||
--text-h1: clamp(30px, 3.1vw, 38px);
|
||||
--text-h2: clamp(24px, 2.4vw, 30px);
|
||||
--text-h3: 20px;
|
||||
|
||||
/* Section rhythm — keep generous breathing room but stop sections
|
||||
eating ~60% of the viewport on a single scroll step */
|
||||
--space-section-featured-y: clamp(var(--space-11), 6.2vw, var(--space-13));
|
||||
--space-section-support-y: clamp(var(--space-10), 5.4vw, var(--space-12));
|
||||
--space-section-form-y: clamp(var(--space-10), 5.4vw, var(--space-12));
|
||||
--space-section-page-y: clamp(var(--space-10), 5vw, var(--space-12));
|
||||
--space-hero-inner-bottom: clamp(var(--space-6), 3.6vw, var(--space-8));
|
||||
}
|
||||
|
||||
/* Hero heading reads --text-display (now 46px max); explicitly tune
|
||||
line-height and margin so the two-line headline stays tight. */
|
||||
.hero-text h1 {
|
||||
line-height: 1.04;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
/* Subtitle: drop a hair so the lead doesn't compete with the heading */
|
||||
.hero-subtitle,
|
||||
.hero-subtitle-desktop {
|
||||
font-size: 16px;
|
||||
line-height: 1.5;
|
||||
}
|
||||
}
|
||||
|
||||
/* ── Ultrawide (≥1800px) ────────────────────────────────────────────────────
|
||||
Only add rules here that are genuinely ultrawide-specific.
|
||||
Normal desktop, tablet, and mobile breakpoints are handled above. ── */
|
||||
|
||||
@@ -152,6 +152,21 @@
|
||||
--motion-reveal-opacity: 0.3s ease;
|
||||
--motion-reveal-transform: 0.45s cubic-bezier(0.2, 0.8, 0.2, 1);
|
||||
|
||||
/* Motion system — shared timing, easing, and distance tokens.
|
||||
A single curve family (refined ease-out, gentle ease-in-out) keeps
|
||||
the whole site speaking one motion language. */
|
||||
--ease-out-soft: cubic-bezier(0.22, 1, 0.36, 1);
|
||||
--ease-out-quiet: cubic-bezier(0.2, 0.8, 0.2, 1);
|
||||
--ease-in-out-soft: cubic-bezier(0.45, 0, 0.2, 1);
|
||||
--motion-duration-xs: 180ms;
|
||||
--motion-duration-sm: 260ms;
|
||||
--motion-duration-md: 420ms;
|
||||
--motion-duration-lg: 620ms;
|
||||
--motion-distance-xs: 6px;
|
||||
--motion-distance-sm: 12px;
|
||||
--motion-distance-md: 18px;
|
||||
--motion-stagger-step: 70ms;
|
||||
|
||||
/* Layout */
|
||||
--max-w: 1280px;
|
||||
--space-container-x: clamp(var(--space-6), 4vw, var(--space-9));
|
||||
|
||||
@@ -28,6 +28,7 @@
|
||||
import '$lib/styles/buttons.css';
|
||||
import '$lib/styles/forms.css';
|
||||
import '$lib/styles/sections.css';
|
||||
import '$lib/styles/motion.css';
|
||||
import '$lib/styles/responsive.css';
|
||||
|
||||
const mobileCtaButtonEnabled = isMobileCtaButtonEnabled();
|
||||
|
||||
@@ -16,7 +16,7 @@
|
||||
import { privacyPolicyContent } from '$lib/content/privacy-policy';
|
||||
import { puppyVisitsContent } from '$lib/content/puppy-visits';
|
||||
import { termsAndConditionsContent } from '$lib/content/terms-and-conditions';
|
||||
import { buildAreaServed, buildBreadcrumb, absoluteUrl } from '$lib/seo';
|
||||
import { buildAreaServed, buildBreadcrumb, absoluteUrl, goodwalkProviderNode } from '$lib/seo';
|
||||
import type { PageData } from './$types';
|
||||
|
||||
export let data: PageData;
|
||||
@@ -47,15 +47,31 @@
|
||||
};
|
||||
}
|
||||
|
||||
const sharedServiceRating = {
|
||||
'@type': 'AggregateRating',
|
||||
ratingValue: '5.0',
|
||||
bestRating: '5',
|
||||
worstRating: '1',
|
||||
reviewCount: '30'
|
||||
};
|
||||
function buildServiceNode(base: {
|
||||
name: string;
|
||||
description: string;
|
||||
serviceType: string;
|
||||
image: string;
|
||||
url: string;
|
||||
offers: ReturnType<typeof aggregateOfferSchema>;
|
||||
}) {
|
||||
const node: Record<string, unknown> = {
|
||||
'@context': 'https://schema.org',
|
||||
'@type': 'Service',
|
||||
name: base.name,
|
||||
description: base.description,
|
||||
serviceType: base.serviceType,
|
||||
provider: goodwalkProviderNode,
|
||||
areaServed,
|
||||
image: base.image,
|
||||
url: base.url
|
||||
};
|
||||
if (base.offers) {
|
||||
node.offers = base.offers;
|
||||
}
|
||||
return node;
|
||||
}
|
||||
|
||||
const expertProvider = { '@id': `${siteUrl}/#alessandra` };
|
||||
const areaServed = buildAreaServed();
|
||||
|
||||
let seoImage = defaultSeoImage;
|
||||
@@ -86,21 +102,14 @@
|
||||
seoImage = packWalksContent.hero.imageUrl;
|
||||
seoImageAlt = packWalksContent.hero.imageAlt;
|
||||
pageStructuredData = [
|
||||
{
|
||||
'@context': 'https://schema.org',
|
||||
'@type': 'Service',
|
||||
buildServiceNode({
|
||||
name: 'Goodwalk Tiny Gang Pack Walks',
|
||||
description: data.page.description,
|
||||
serviceType: 'Pack walks for small and medium dogs',
|
||||
provider: { '@id': 'https://www.goodwalk.co.nz/#business' },
|
||||
areaServed,
|
||||
image: absoluteUrl(seoImage),
|
||||
url: `${siteUrl}${data.page.canonicalPath}`,
|
||||
offers: aggregateOfferSchema(packWalksContent.pricing.plans),
|
||||
aggregateRating: sharedServiceRating,
|
||||
author: expertProvider,
|
||||
dateModified: packWalksContent.lastUpdated
|
||||
},
|
||||
offers: aggregateOfferSchema(packWalksContent.pricing.plans)
|
||||
}),
|
||||
buildBreadcrumb([
|
||||
{ name: 'Home', url: siteUrl },
|
||||
{ name: 'Tiny Gang Pack Walks', path: data.page.canonicalPath }
|
||||
@@ -111,21 +120,14 @@
|
||||
seoImage = dogWalkingContent.hero.imageUrl;
|
||||
seoImageAlt = dogWalkingContent.hero.imageAlt;
|
||||
pageStructuredData = [
|
||||
{
|
||||
'@context': 'https://schema.org',
|
||||
'@type': 'Service',
|
||||
buildServiceNode({
|
||||
name: 'Goodwalk Solo Dog Walks',
|
||||
description: data.page.description,
|
||||
serviceType: 'One-on-one dog walking',
|
||||
provider: { '@id': 'https://www.goodwalk.co.nz/#business' },
|
||||
areaServed,
|
||||
image: absoluteUrl(seoImage),
|
||||
url: `${siteUrl}${data.page.canonicalPath}`,
|
||||
offers: aggregateOfferSchema(dogWalkingContent.pricing.plans),
|
||||
aggregateRating: sharedServiceRating,
|
||||
author: expertProvider,
|
||||
dateModified: dogWalkingContent.lastUpdated
|
||||
},
|
||||
offers: aggregateOfferSchema(dogWalkingContent.pricing.plans)
|
||||
}),
|
||||
buildBreadcrumb([
|
||||
{ name: 'Home', url: siteUrl },
|
||||
{ name: 'Solo Walks', path: data.page.canonicalPath }
|
||||
@@ -136,21 +138,14 @@
|
||||
seoImage = puppyVisitsContent.hero.imageUrl;
|
||||
seoImageAlt = puppyVisitsContent.hero.imageAlt;
|
||||
pageStructuredData = [
|
||||
{
|
||||
'@context': 'https://schema.org',
|
||||
'@type': 'Service',
|
||||
buildServiceNode({
|
||||
name: 'Goodwalk Puppy Visits',
|
||||
description: data.page.description,
|
||||
serviceType: 'In-home puppy visits',
|
||||
provider: { '@id': 'https://www.goodwalk.co.nz/#business' },
|
||||
areaServed,
|
||||
image: absoluteUrl(seoImage),
|
||||
url: `${siteUrl}${data.page.canonicalPath}`,
|
||||
offers: aggregateOfferSchema(puppyVisitsContent.pricing.plans),
|
||||
aggregateRating: sharedServiceRating,
|
||||
author: expertProvider,
|
||||
dateModified: puppyVisitsContent.lastUpdated
|
||||
},
|
||||
offers: aggregateOfferSchema(puppyVisitsContent.pricing.plans)
|
||||
}),
|
||||
buildBreadcrumb([
|
||||
{ name: 'Home', url: siteUrl },
|
||||
{ name: 'Puppy Visits', path: data.page.canonicalPath }
|
||||
|
||||
@@ -17,7 +17,7 @@ describe('static slug route page', () => {
|
||||
['our-pricing', ourPricingContent.subtitle ?? ourPricingContent.title],
|
||||
['about', aboutPageContent.sections[0].title],
|
||||
['testimonials', 'Client Testimonials'],
|
||||
['contact-us', "Let's meet!"],
|
||||
['contact-us', 'Contact Us'],
|
||||
['terms-and-conditions', '1. Application of Terms'],
|
||||
['privacy-policy', 'How we collect your information']
|
||||
] as const)('renders the %s page branch', (slug, expectedText) => {
|
||||
@@ -49,26 +49,21 @@ describe('static slug route page', () => {
|
||||
data: createStaticRouteData(slug)
|
||||
});
|
||||
|
||||
expect(screen.queryByText("Let's meet!")).not.toBeInTheDocument();
|
||||
expect(screen.queryByText(/Tell us about your dog/i)).not.toBeInTheDocument();
|
||||
expect(screen.queryByText('What our clients say')).not.toBeInTheDocument();
|
||||
}
|
||||
);
|
||||
|
||||
it('shows the general enquiry option on the contact page only', () => {
|
||||
const { rerender } = render(SlugPage, {
|
||||
it('does not surface the general-enquiry option even when the flag is on', () => {
|
||||
render(SlugPage, {
|
||||
data: {
|
||||
...createStaticRouteData('contact-us'),
|
||||
generalEnquiryEnabled: true
|
||||
}
|
||||
});
|
||||
|
||||
expect(screen.getByRole('button', { name: /General enquiry/i })).toBeInTheDocument();
|
||||
|
||||
rerender({
|
||||
data: createStaticRouteData('pack-walks')
|
||||
});
|
||||
|
||||
expect(screen.queryByRole('button', { name: /General enquiry/i })).not.toBeInTheDocument();
|
||||
expect(screen.queryByLabelText(/General enquiry/i)).not.toBeInTheDocument();
|
||||
});
|
||||
|
||||
it('renders the shared FAQ section on the contact page', () => {
|
||||
|
||||
@@ -21,7 +21,9 @@ describe('home page route', () => {
|
||||
const booking = document.getElementById('newlead');
|
||||
|
||||
expect(screen.getAllByText(homepageContent.hero.highlight).length).toBeGreaterThan(0);
|
||||
expect(screen.getByText(homepageContent.intro.text)).toBeInTheDocument();
|
||||
const introHeadlineText = homepageContent.intro.text.replace(/\.$/, '');
|
||||
const introHeading = document.querySelector('#intro .intro-headline');
|
||||
expect(introHeading?.textContent?.replace(/\s+/g, ' ').trim()).toBe(introHeadlineText);
|
||||
expect(screen.getByText('Calmer dogs. Clearer routines. Less worry.')).toBeInTheDocument();
|
||||
expect(screen.getByText(homepageContent.howItWorks.title)).toBeInTheDocument();
|
||||
expect(screen.getByText(homepageContent.info.title)).toBeInTheDocument();
|
||||
|
||||
@@ -8,6 +8,7 @@ export const GET: RequestHandler = () => {
|
||||
'Disallow: /contract',
|
||||
'Disallow: /meet-greet-v2',
|
||||
'Disallow: /variants-contact-form',
|
||||
'Disallow: /variants/',
|
||||
'',
|
||||
'# AI crawlers — explicitly permitted',
|
||||
'User-agent: GPTBot',
|
||||
|
||||
@@ -0,0 +1,881 @@
|
||||
<script lang="ts">
|
||||
import Icon from '$lib/components/Icon.svelte';
|
||||
|
||||
const instagramHref = 'https://www.instagram.com/goodwalk.nz/';
|
||||
|
||||
// Thumbnails for the polaroid stack + phone mosaic come from the existing
|
||||
// imagery in /static/images so this page works without new uploads.
|
||||
const gridImages = [
|
||||
'/images/goodwalk-tiny-gang-pack-walk-auckland.webp',
|
||||
'/images/goodwalk-tiny-gang-mt-albert-park-auckland.webp',
|
||||
'/images/goodwalk-tiny-gang-finishing-walk-suv-auckland.webp',
|
||||
'/images/goodwalk-client-dogs-mt-cecila-auckland.webp',
|
||||
'/images/goodwalk-tiny-gang-pack-walk-small-dogs-auckland.webp',
|
||||
'/images/goodwalk-dogs-group-outing-auckland.webp',
|
||||
'/images/archie-goodwalk-dog-walking-review-auckland.webp',
|
||||
'/images/monty-goodwalk-dog-walking-review-auckland.webp',
|
||||
'/images/otis-goodwalk-dog-walking-review-auckland.webp'
|
||||
];
|
||||
|
||||
const variants = [
|
||||
{
|
||||
id: 'editorial',
|
||||
index: '01',
|
||||
label: 'Photo-led editorial',
|
||||
kicker: 'Magazine feel — image does the work, type stays quiet'
|
||||
},
|
||||
{
|
||||
id: 'phone-mosaic',
|
||||
index: '02',
|
||||
label: 'Phone mosaic',
|
||||
kicker: 'Native Instagram preview — see exactly what you’ll get'
|
||||
},
|
||||
{
|
||||
id: 'quiet-split',
|
||||
index: '03',
|
||||
label: 'Quiet split',
|
||||
kicker: 'Apple-style restraint — no yellow background, no dog cutout'
|
||||
}
|
||||
];
|
||||
</script>
|
||||
|
||||
<svelte:head>
|
||||
<title>Instagram CTA — variants</title>
|
||||
<meta name="robots" content="noindex, nofollow" />
|
||||
</svelte:head>
|
||||
|
||||
<main class="variants-page">
|
||||
<header class="variants-header">
|
||||
<span class="variants-eyebrow">Variants</span>
|
||||
<h1>Follow-on-Instagram CTA</h1>
|
||||
<p>
|
||||
Three reference takes on the existing yellow banner. All link to <code>@goodwalk.nz</code>.
|
||||
No copy, branding, or palette changes — same content, three different ways to frame it.
|
||||
</p>
|
||||
|
||||
<nav class="variants-nav" aria-label="Jump to variant">
|
||||
{#each variants as variant}
|
||||
<a href={`#${variant.id}`} class="variants-nav-chip">
|
||||
<span class="variants-nav-index">{variant.index}</span>
|
||||
<span class="variants-nav-label">{variant.label}</span>
|
||||
</a>
|
||||
{/each}
|
||||
</nav>
|
||||
</header>
|
||||
|
||||
<!-- ─── Variant 01: Photo-led editorial ─────────────────────────────────── -->
|
||||
<section id="editorial" class="variant-showcase">
|
||||
<div class="variant-meta">
|
||||
<span class="variant-index">01</span>
|
||||
<div>
|
||||
<h2>Photo-led editorial</h2>
|
||||
<p>Full-bleed dog photo. Type sits in a dark scrim on the left. A small Instagram source pill anchors the corner. Yellow CTA the only saturated element.</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="variant-stage variant-stage-editorial">
|
||||
<aside class="ig-editorial" aria-label="Follow Goodwalk on Instagram">
|
||||
<div class="ig-editorial-photo">
|
||||
<img
|
||||
src="/images/goodwalk-tiny-gang-pack-walk-small-dogs-auckland.webp"
|
||||
alt=""
|
||||
loading="lazy"
|
||||
decoding="async"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="ig-editorial-scrim">
|
||||
<div class="ig-editorial-copy">
|
||||
<span class="ig-editorial-source">
|
||||
<Icon name="fab fa-instagram" />
|
||||
<span>@goodwalk.nz</span>
|
||||
</span>
|
||||
<h2>Follow the Tiny Gang on Instagram.</h2>
|
||||
<p>Daily walks, real dogs, no filter. Park runs, pack moments, and the kind of updates you'd actually want to scroll.</p>
|
||||
|
||||
<a href={instagramHref} target="_blank" rel="noopener" class="ig-editorial-cta">
|
||||
<Icon name="fab fa-instagram" />
|
||||
<span>Follow @goodwalk.nz</span>
|
||||
<Icon name="fas fa-arrow-right" className="ig-editorial-arrow" />
|
||||
</a>
|
||||
</div>
|
||||
</div>
|
||||
</aside>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<!-- ─── Variant 02: Phone mosaic ────────────────────────────────────────── -->
|
||||
<section id="phone-mosaic" class="variant-showcase">
|
||||
<div class="variant-meta">
|
||||
<span class="variant-index">02</span>
|
||||
<div>
|
||||
<h2>Phone mosaic</h2>
|
||||
<p>Goodwalk green panel, copy left. Right side carries a subtle phone frame showing a 3×3 Instagram grid. Visitors see the product before they tap.</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="variant-stage variant-stage-mosaic">
|
||||
<aside class="ig-mosaic" aria-label="Follow Goodwalk on Instagram">
|
||||
<div class="ig-mosaic-copy">
|
||||
<span class="ig-mosaic-eyebrow">
|
||||
<Icon name="fab fa-instagram" />
|
||||
<span>@goodwalk.nz</span>
|
||||
</span>
|
||||
<h2>Follow the Tiny Gang on Instagram.</h2>
|
||||
<p>Pack walks, puppy visits, and the dogs your dog would call friends. Updated most days.</p>
|
||||
|
||||
<a href={instagramHref} target="_blank" rel="noopener" class="btn btn-yellow ig-mosaic-cta">
|
||||
<Icon name="fab fa-instagram" />
|
||||
<span>Follow on Instagram</span>
|
||||
</a>
|
||||
|
||||
<div class="ig-mosaic-meta" aria-label="Follower stats">
|
||||
<div>
|
||||
<strong>30+</strong>
|
||||
<span>five-star reviews</span>
|
||||
</div>
|
||||
<div>
|
||||
<strong>Daily</strong>
|
||||
<span>tiny-gang updates</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="ig-mosaic-phone" aria-hidden="true">
|
||||
<div class="ig-mosaic-phone-frame">
|
||||
<div class="ig-mosaic-phone-header">
|
||||
<Icon name="fab fa-instagram" />
|
||||
<span>goodwalk.nz</span>
|
||||
<span class="ig-mosaic-phone-dot" aria-hidden="true">•</span>
|
||||
<span class="ig-mosaic-phone-follow">Following</span>
|
||||
</div>
|
||||
<div class="ig-mosaic-phone-grid">
|
||||
{#each gridImages as src}
|
||||
<div class="ig-mosaic-phone-cell">
|
||||
<img {src} alt="" loading="lazy" decoding="async" />
|
||||
</div>
|
||||
{/each}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</aside>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<!-- ─── Variant 03: Quiet split ─────────────────────────────────────────── -->
|
||||
<section id="quiet-split" class="variant-showcase">
|
||||
<div class="variant-meta">
|
||||
<span class="variant-index">03</span>
|
||||
<div>
|
||||
<h2>Quiet split</h2>
|
||||
<p>No yellow page background. The section sits on off-white. One editorial photo, type at the leading edge, CTA understated. The page exhales here instead of shouting.</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="variant-stage variant-stage-quiet">
|
||||
<aside class="ig-quiet" aria-label="Follow Goodwalk on Instagram">
|
||||
<div class="ig-quiet-copy">
|
||||
<span class="ig-quiet-eyebrow">Daily on Instagram</span>
|
||||
<h2>The Tiny Gang, day by day.</h2>
|
||||
<p>
|
||||
A small window into our walks. Park favourites, new pack pairings, and the
|
||||
quieter moments owners love coming home to.
|
||||
</p>
|
||||
|
||||
<a href={instagramHref} target="_blank" rel="noopener" class="ig-quiet-cta">
|
||||
<span class="ig-quiet-cta-mark" aria-hidden="true">
|
||||
<Icon name="fab fa-instagram" />
|
||||
</span>
|
||||
<span class="ig-quiet-cta-text">
|
||||
<span class="ig-quiet-cta-handle">@goodwalk.nz</span>
|
||||
<span class="ig-quiet-cta-action">
|
||||
Follow
|
||||
<Icon name="fas fa-arrow-right" />
|
||||
</span>
|
||||
</span>
|
||||
</a>
|
||||
</div>
|
||||
|
||||
<div class="ig-quiet-media">
|
||||
<img
|
||||
src="/images/goodwalk-tiny-gang-mt-albert-park-auckland.webp"
|
||||
alt=""
|
||||
loading="lazy"
|
||||
decoding="async"
|
||||
/>
|
||||
</div>
|
||||
</aside>
|
||||
</div>
|
||||
</section>
|
||||
</main>
|
||||
|
||||
<style>
|
||||
.variants-page {
|
||||
background: var(--off-white);
|
||||
min-height: 100vh;
|
||||
padding-bottom: 96px;
|
||||
}
|
||||
|
||||
/* ── Page header ─────────────────────────────────────────────────────── */
|
||||
.variants-header {
|
||||
max-width: 880px;
|
||||
margin: 0 auto;
|
||||
padding: 96px 24px 48px;
|
||||
}
|
||||
|
||||
.variants-eyebrow {
|
||||
display: inline-block;
|
||||
margin-bottom: 16px;
|
||||
color: var(--gw-green);
|
||||
font-family: var(--font-head);
|
||||
font-size: 12px;
|
||||
font-weight: 800;
|
||||
letter-spacing: 0.18em;
|
||||
text-transform: uppercase;
|
||||
}
|
||||
|
||||
.variants-header h1 {
|
||||
margin: 0 0 18px;
|
||||
font-family: var(--font-head);
|
||||
font-size: clamp(34px, 4vw, 48px);
|
||||
font-weight: 800;
|
||||
letter-spacing: -0.025em;
|
||||
line-height: 1.05;
|
||||
color: var(--text-heading);
|
||||
}
|
||||
|
||||
.variants-header p {
|
||||
margin: 0;
|
||||
max-width: 60ch;
|
||||
color: var(--text-muted);
|
||||
font-size: 16px;
|
||||
line-height: 1.65;
|
||||
}
|
||||
|
||||
.variants-header code {
|
||||
padding: 2px 6px;
|
||||
border-radius: 6px;
|
||||
background: rgba(33, 48, 33, 0.08);
|
||||
color: var(--gw-green);
|
||||
font-family: ui-monospace, SFMono-Regular, Menlo, monospace;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.variants-nav {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
gap: 10px;
|
||||
margin-top: 28px;
|
||||
}
|
||||
|
||||
.variants-nav-chip {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 10px;
|
||||
padding: 9px 16px 9px 12px;
|
||||
border-radius: 999px;
|
||||
background: #fff;
|
||||
box-shadow: inset 0 0 0 1px rgba(17, 20, 24, 0.08), 0 6px 16px rgba(17, 20, 24, 0.04);
|
||||
color: var(--gw-green);
|
||||
font-family: var(--font-head);
|
||||
font-size: 13px;
|
||||
font-weight: 700;
|
||||
text-decoration: none;
|
||||
transition: transform 0.18s ease, box-shadow 0.22s ease;
|
||||
}
|
||||
|
||||
.variants-nav-chip:hover {
|
||||
transform: translateY(-1px);
|
||||
box-shadow: inset 0 0 0 1px rgba(17, 20, 24, 0.1), 0 10px 22px rgba(17, 20, 24, 0.07);
|
||||
}
|
||||
|
||||
.variants-nav-index {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
width: 24px;
|
||||
height: 24px;
|
||||
border-radius: 50%;
|
||||
background: var(--yellow);
|
||||
color: var(--gw-green);
|
||||
font-size: 11px;
|
||||
font-weight: 800;
|
||||
}
|
||||
|
||||
/* ── Each variant block ──────────────────────────────────────────────── */
|
||||
.variant-showcase {
|
||||
max-width: 1280px;
|
||||
margin: 0 auto;
|
||||
padding: 72px 24px 0;
|
||||
}
|
||||
|
||||
.variant-meta {
|
||||
display: flex;
|
||||
align-items: flex-start;
|
||||
gap: 18px;
|
||||
margin-bottom: 28px;
|
||||
max-width: 760px;
|
||||
}
|
||||
|
||||
.variant-index {
|
||||
flex: 0 0 auto;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
width: 36px;
|
||||
height: 36px;
|
||||
border-radius: 50%;
|
||||
background: var(--gw-green);
|
||||
color: var(--yellow);
|
||||
font-family: var(--font-head);
|
||||
font-size: 13px;
|
||||
font-weight: 800;
|
||||
}
|
||||
|
||||
.variant-meta h2 {
|
||||
margin: 0 0 6px;
|
||||
font-family: var(--font-head);
|
||||
font-size: 22px;
|
||||
font-weight: 700;
|
||||
color: var(--text-heading);
|
||||
letter-spacing: -0.015em;
|
||||
}
|
||||
|
||||
.variant-meta p {
|
||||
margin: 0;
|
||||
color: var(--text-muted);
|
||||
font-size: 14px;
|
||||
line-height: 1.55;
|
||||
}
|
||||
|
||||
/* The "stage" mimics where the section sits on the live page — page
|
||||
surface around it, so each variant reads in context. */
|
||||
.variant-stage {
|
||||
padding: 56px clamp(24px, 4vw, 64px);
|
||||
border-radius: 32px;
|
||||
}
|
||||
|
||||
.variant-stage-editorial {
|
||||
background: #f4efe5;
|
||||
}
|
||||
|
||||
.variant-stage-mosaic {
|
||||
background: var(--yellow);
|
||||
}
|
||||
|
||||
.variant-stage-quiet {
|
||||
background: var(--off-white);
|
||||
box-shadow: inset 0 0 0 1px rgba(17, 20, 24, 0.04);
|
||||
}
|
||||
|
||||
/* ═══════════════════════════════════════════════════════════════════════
|
||||
Variant 01 — Photo-led editorial
|
||||
═══════════════════════════════════════════════════════════════════════ */
|
||||
.ig-editorial {
|
||||
position: relative;
|
||||
overflow: hidden;
|
||||
border-radius: 28px;
|
||||
aspect-ratio: 2.6 / 1;
|
||||
min-height: 280px;
|
||||
box-shadow: 0 30px 70px rgba(17, 20, 24, 0.16);
|
||||
}
|
||||
|
||||
.ig-editorial-photo {
|
||||
position: absolute;
|
||||
inset: 0;
|
||||
z-index: 0;
|
||||
}
|
||||
|
||||
.ig-editorial-photo img {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
object-fit: cover;
|
||||
object-position: 60% 40%;
|
||||
transform: scale(1.02);
|
||||
transition: transform 0.8s cubic-bezier(0.22, 1, 0.36, 1);
|
||||
}
|
||||
|
||||
.ig-editorial:hover .ig-editorial-photo img {
|
||||
transform: scale(1.06);
|
||||
}
|
||||
|
||||
.ig-editorial-scrim {
|
||||
position: absolute;
|
||||
inset: 0;
|
||||
z-index: 1;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
background: linear-gradient(
|
||||
100deg,
|
||||
rgba(15, 24, 15, 0.86) 0%,
|
||||
rgba(15, 24, 15, 0.74) 36%,
|
||||
rgba(15, 24, 15, 0.18) 70%,
|
||||
rgba(15, 24, 15, 0) 100%
|
||||
);
|
||||
padding: 48px clamp(28px, 4vw, 56px);
|
||||
}
|
||||
|
||||
.ig-editorial-copy {
|
||||
max-width: 480px;
|
||||
color: #fff;
|
||||
}
|
||||
|
||||
.ig-editorial-source {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
padding: 6px 12px 6px 10px;
|
||||
margin-bottom: 18px;
|
||||
border-radius: 999px;
|
||||
background: rgba(255, 255, 255, 0.14);
|
||||
backdrop-filter: blur(6px);
|
||||
-webkit-backdrop-filter: blur(6px);
|
||||
color: #fff;
|
||||
font-family: var(--font-head);
|
||||
font-size: 12px;
|
||||
font-weight: 700;
|
||||
letter-spacing: 0.02em;
|
||||
}
|
||||
|
||||
.ig-editorial-source :global(.icon) {
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.ig-editorial-copy h2 {
|
||||
margin: 0 0 12px;
|
||||
font-family: var(--font-head);
|
||||
font-size: clamp(26px, 3vw, 36px);
|
||||
font-weight: 800;
|
||||
letter-spacing: -0.025em;
|
||||
line-height: 1.08;
|
||||
color: #fff;
|
||||
}
|
||||
|
||||
.ig-editorial-copy p {
|
||||
margin: 0 0 26px;
|
||||
color: rgba(255, 255, 255, 0.78);
|
||||
font-size: 15px;
|
||||
line-height: 1.55;
|
||||
max-width: 36ch;
|
||||
}
|
||||
|
||||
.ig-editorial-cta {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 10px;
|
||||
padding: 13px 22px;
|
||||
border-radius: 999px;
|
||||
background: var(--yellow);
|
||||
color: var(--gw-green);
|
||||
font-family: var(--font-head);
|
||||
font-size: 14px;
|
||||
font-weight: 800;
|
||||
text-decoration: none;
|
||||
box-shadow: 0 12px 24px rgba(255, 209, 0, 0.32);
|
||||
transition: transform 0.22s ease, box-shadow 0.22s ease;
|
||||
}
|
||||
|
||||
.ig-editorial-cta:hover {
|
||||
transform: translateY(-2px);
|
||||
box-shadow: 0 18px 32px rgba(255, 209, 0, 0.38);
|
||||
}
|
||||
|
||||
.ig-editorial-cta :global(.ig-editorial-arrow) {
|
||||
transition: transform 0.22s cubic-bezier(0.22, 1, 0.36, 1);
|
||||
}
|
||||
|
||||
.ig-editorial-cta:hover :global(.ig-editorial-arrow) {
|
||||
transform: translateX(3px);
|
||||
}
|
||||
|
||||
/* ═══════════════════════════════════════════════════════════════════════
|
||||
Variant 02 — Phone mosaic
|
||||
═══════════════════════════════════════════════════════════════════════ */
|
||||
.ig-mosaic {
|
||||
display: grid;
|
||||
grid-template-columns: minmax(0, 1fr) minmax(0, 0.9fr);
|
||||
align-items: center;
|
||||
gap: clamp(24px, 4vw, 56px);
|
||||
padding: clamp(36px, 5vw, 64px) clamp(28px, 4vw, 56px);
|
||||
border-radius: 32px;
|
||||
background:
|
||||
radial-gradient(circle at 100% 0%, rgba(255, 209, 0, 0.16), transparent 38%),
|
||||
var(--gw-green);
|
||||
color: #fff;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.ig-mosaic-copy {
|
||||
max-width: 460px;
|
||||
}
|
||||
|
||||
.ig-mosaic-eyebrow {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
padding: 7px 14px 7px 11px;
|
||||
margin-bottom: 18px;
|
||||
border-radius: 999px;
|
||||
background: rgba(255, 255, 255, 0.12);
|
||||
color: var(--yellow);
|
||||
font-family: var(--font-head);
|
||||
font-size: 12px;
|
||||
font-weight: 700;
|
||||
letter-spacing: 0.02em;
|
||||
}
|
||||
|
||||
.ig-mosaic h2 {
|
||||
margin: 0 0 14px;
|
||||
font-family: var(--font-head);
|
||||
font-size: clamp(28px, 3vw, 38px);
|
||||
font-weight: 800;
|
||||
letter-spacing: -0.025em;
|
||||
line-height: 1.06;
|
||||
}
|
||||
|
||||
.ig-mosaic-copy p {
|
||||
margin: 0 0 28px;
|
||||
color: rgba(255, 255, 255, 0.78);
|
||||
font-size: 15px;
|
||||
line-height: 1.55;
|
||||
max-width: 38ch;
|
||||
}
|
||||
|
||||
.ig-mosaic-cta {
|
||||
gap: 9px;
|
||||
font-weight: 800;
|
||||
box-shadow: 0 14px 28px rgba(0, 0, 0, 0.18);
|
||||
}
|
||||
|
||||
.ig-mosaic-meta {
|
||||
display: flex;
|
||||
gap: 32px;
|
||||
margin-top: 32px;
|
||||
padding-top: 24px;
|
||||
border-top: 1px solid rgba(255, 255, 255, 0.14);
|
||||
}
|
||||
|
||||
.ig-mosaic-meta div {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 2px;
|
||||
}
|
||||
|
||||
.ig-mosaic-meta strong {
|
||||
color: var(--yellow);
|
||||
font-family: var(--font-head);
|
||||
font-size: 18px;
|
||||
font-weight: 800;
|
||||
letter-spacing: -0.01em;
|
||||
}
|
||||
|
||||
.ig-mosaic-meta span {
|
||||
color: rgba(255, 255, 255, 0.7);
|
||||
font-size: 12px;
|
||||
line-height: 1.2;
|
||||
}
|
||||
|
||||
.ig-mosaic-phone {
|
||||
position: relative;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
aspect-ratio: 1 / 1;
|
||||
min-height: 320px;
|
||||
}
|
||||
|
||||
.ig-mosaic-phone-frame {
|
||||
width: min(100%, 320px);
|
||||
padding: 14px;
|
||||
border-radius: 26px;
|
||||
background: linear-gradient(180deg, #fafaf7, #ecebe5);
|
||||
box-shadow:
|
||||
inset 0 0 0 1px rgba(17, 20, 24, 0.1),
|
||||
0 28px 60px rgba(0, 0, 0, 0.32),
|
||||
0 4px 10px rgba(0, 0, 0, 0.18);
|
||||
transform: rotate(-3deg);
|
||||
transition: transform 0.5s cubic-bezier(0.22, 1, 0.36, 1);
|
||||
}
|
||||
|
||||
.ig-mosaic:hover .ig-mosaic-phone-frame {
|
||||
transform: rotate(-1.5deg) translateY(-4px);
|
||||
}
|
||||
|
||||
.ig-mosaic-phone-header {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
padding: 6px 4px 14px;
|
||||
color: var(--text-heading);
|
||||
font-family: var(--font-head);
|
||||
font-size: 13px;
|
||||
font-weight: 700;
|
||||
}
|
||||
|
||||
.ig-mosaic-phone-header :global(.icon) {
|
||||
color: var(--text-heading);
|
||||
font-size: 15px;
|
||||
}
|
||||
|
||||
.ig-mosaic-phone-dot {
|
||||
color: var(--text-muted);
|
||||
font-size: 14px;
|
||||
line-height: 1;
|
||||
}
|
||||
|
||||
.ig-mosaic-phone-follow {
|
||||
color: var(--gw-green);
|
||||
font-size: 12px;
|
||||
font-weight: 800;
|
||||
}
|
||||
|
||||
.ig-mosaic-phone-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(3, 1fr);
|
||||
gap: 3px;
|
||||
border-radius: 6px;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.ig-mosaic-phone-cell {
|
||||
aspect-ratio: 1 / 1;
|
||||
overflow: hidden;
|
||||
background: rgba(33, 48, 33, 0.08);
|
||||
}
|
||||
|
||||
.ig-mosaic-phone-cell img {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
object-fit: cover;
|
||||
display: block;
|
||||
}
|
||||
|
||||
/* ═══════════════════════════════════════════════════════════════════════
|
||||
Variant 03 — Quiet split
|
||||
═══════════════════════════════════════════════════════════════════════ */
|
||||
.ig-quiet {
|
||||
display: grid;
|
||||
grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
|
||||
align-items: center;
|
||||
gap: clamp(32px, 5vw, 72px);
|
||||
padding: clamp(28px, 3vw, 40px) 0;
|
||||
}
|
||||
|
||||
.ig-quiet-copy {
|
||||
max-width: 440px;
|
||||
}
|
||||
|
||||
.ig-quiet-eyebrow {
|
||||
display: inline-block;
|
||||
margin-bottom: 16px;
|
||||
color: var(--gw-green);
|
||||
font-family: var(--font-head);
|
||||
font-size: 11px;
|
||||
font-weight: 800;
|
||||
letter-spacing: 0.18em;
|
||||
text-transform: uppercase;
|
||||
}
|
||||
|
||||
.ig-quiet h2 {
|
||||
margin: 0 0 18px;
|
||||
font-family: var(--font-head);
|
||||
font-size: clamp(28px, 3.2vw, 40px);
|
||||
font-weight: 800;
|
||||
letter-spacing: -0.03em;
|
||||
line-height: 1.04;
|
||||
color: var(--text-heading);
|
||||
}
|
||||
|
||||
.ig-quiet-copy p {
|
||||
margin: 0 0 32px;
|
||||
color: var(--text-muted);
|
||||
font-size: 16px;
|
||||
line-height: 1.6;
|
||||
max-width: 36ch;
|
||||
}
|
||||
|
||||
.ig-quiet-cta {
|
||||
display: inline-flex;
|
||||
align-items: stretch;
|
||||
gap: 14px;
|
||||
padding: 8px 8px 8px 12px;
|
||||
border-radius: 18px;
|
||||
background: #fff;
|
||||
box-shadow: inset 0 0 0 1px rgba(17, 20, 24, 0.08), 0 18px 32px rgba(17, 20, 24, 0.06);
|
||||
color: var(--text-heading);
|
||||
text-decoration: none;
|
||||
transition: transform 0.22s ease, box-shadow 0.22s ease;
|
||||
}
|
||||
|
||||
.ig-quiet-cta:hover {
|
||||
transform: translateY(-2px);
|
||||
box-shadow: inset 0 0 0 1px rgba(17, 20, 24, 0.1), 0 24px 40px rgba(17, 20, 24, 0.1);
|
||||
}
|
||||
|
||||
.ig-quiet-cta-mark {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
width: 44px;
|
||||
height: 44px;
|
||||
border-radius: 12px;
|
||||
background: var(--gw-green);
|
||||
color: var(--yellow);
|
||||
font-size: 20px;
|
||||
}
|
||||
|
||||
.ig-quiet-cta-text {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
padding-right: 16px;
|
||||
}
|
||||
|
||||
.ig-quiet-cta-handle {
|
||||
font-family: var(--font-head);
|
||||
font-size: 14px;
|
||||
font-weight: 800;
|
||||
color: var(--text-heading);
|
||||
line-height: 1.1;
|
||||
letter-spacing: -0.01em;
|
||||
}
|
||||
|
||||
.ig-quiet-cta-action {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 6px;
|
||||
color: var(--text-muted);
|
||||
font-size: 12px;
|
||||
font-weight: 700;
|
||||
line-height: 1;
|
||||
margin-top: 4px;
|
||||
}
|
||||
|
||||
.ig-quiet-cta-action :global(.icon) {
|
||||
font-size: 10px;
|
||||
transition: transform 0.22s cubic-bezier(0.22, 1, 0.36, 1);
|
||||
}
|
||||
|
||||
.ig-quiet-cta:hover .ig-quiet-cta-action :global(.icon) {
|
||||
transform: translateX(3px);
|
||||
}
|
||||
|
||||
.ig-quiet-media {
|
||||
position: relative;
|
||||
aspect-ratio: 4 / 3;
|
||||
overflow: hidden;
|
||||
border-radius: 24px;
|
||||
box-shadow: 0 22px 50px rgba(17, 20, 24, 0.12);
|
||||
}
|
||||
|
||||
.ig-quiet-media img {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
object-fit: cover;
|
||||
transition: transform 0.7s cubic-bezier(0.22, 1, 0.36, 1);
|
||||
}
|
||||
|
||||
.ig-quiet:hover .ig-quiet-media img {
|
||||
transform: scale(1.04);
|
||||
}
|
||||
|
||||
/* ── Mobile ──────────────────────────────────────────────────────────── */
|
||||
@media (max-width: 768px) {
|
||||
.variants-header {
|
||||
padding: 64px 20px 32px;
|
||||
}
|
||||
|
||||
.variant-showcase {
|
||||
padding: 48px 16px 0;
|
||||
}
|
||||
|
||||
.variant-stage {
|
||||
padding: 28px 16px;
|
||||
}
|
||||
|
||||
.ig-editorial {
|
||||
aspect-ratio: auto;
|
||||
min-height: 0;
|
||||
}
|
||||
|
||||
.ig-editorial-photo {
|
||||
position: relative;
|
||||
aspect-ratio: 4 / 3;
|
||||
}
|
||||
|
||||
.ig-editorial-scrim {
|
||||
position: relative;
|
||||
inset: auto;
|
||||
background: var(--gw-green);
|
||||
padding: 28px 22px;
|
||||
}
|
||||
|
||||
.ig-mosaic {
|
||||
grid-template-columns: 1fr;
|
||||
padding: 32px 22px;
|
||||
gap: 24px;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.ig-mosaic-copy {
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.ig-mosaic-eyebrow,
|
||||
.ig-mosaic-cta {
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
}
|
||||
|
||||
.ig-mosaic-meta {
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.ig-mosaic-phone {
|
||||
min-height: 0;
|
||||
aspect-ratio: auto;
|
||||
}
|
||||
|
||||
.ig-mosaic-phone-frame {
|
||||
width: min(100%, 280px);
|
||||
transform: rotate(-2deg);
|
||||
}
|
||||
|
||||
.ig-quiet {
|
||||
grid-template-columns: 1fr;
|
||||
gap: 24px;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
.ig-quiet-copy {
|
||||
order: 2;
|
||||
margin: 0 auto;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.ig-quiet-copy p {
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
}
|
||||
|
||||
.ig-quiet-cta {
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.ig-quiet-media {
|
||||
order: 1;
|
||||
}
|
||||
}
|
||||
|
||||
/* Reduced motion */
|
||||
@media (prefers-reduced-motion: reduce) {
|
||||
.ig-editorial-photo img,
|
||||
.ig-mosaic-phone-frame,
|
||||
.ig-quiet-media img,
|
||||
.ig-editorial-cta,
|
||||
.ig-quiet-cta {
|
||||
transition: none;
|
||||
transform: none;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -1,252 +0,0 @@
|
||||
# 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 `<h3>`s in the same section at totally different scales
|
||||
|
||||
**Inconsistency.** `.values-contrast-cell h3` is `clamp(20px, 1.9vw, 25px)`. `.values-points-title` (also `<h3>`) 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.
|
||||
@@ -1,455 +0,0 @@
|
||||
# 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 `<Eyebrow variant="default|inverse|accent" />` 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 `<Card />` 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 `<GoogleTrustMark size="sm|md" />` 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 `<style>` blocks (ValuesSection, BookingSection, etc.).
|
||||
3. **`:global()` overrides inside component-scoped styles** that effectively re-globalize CSS.
|
||||
|
||||
**Root cause.** Migration in progress between approaches, no policy decision.
|
||||
**Perception.** Invisible to users — but the codebase resists consistent change. Every refactor has to reckon with three rule systems.
|
||||
**Fix.** Pick one. Recommend: **component-scoped** for component-specific styles, **global utility classes** for cross-component primitives (`.btn`, `.eyebrow`, `.card`, `.section-heading`). Forbid `:global()` overrides except for action-injected classes (`reveal-*`).
|
||||
**Class: System debt.**
|
||||
|
||||
---
|
||||
|
||||
## Systemic Drift Map (grouped by root cause)
|
||||
|
||||
**Root cause A — No token-driven scales for type, radius, motion, shadow.**
|
||||
Issues: #2, #6, #10, #13, #15. Authors invent each value fresh because there's no canonical "use this."
|
||||
|
||||
**Root cause B — Primitive components were never extracted.**
|
||||
Issues: #1, #9, #6 (cards), #14 (intro-kicker). Same UX element re-implemented in multiple places.
|
||||
|
||||
**Root cause C — Container padding system is incomplete on desktop.**
|
||||
Issues: #5, #16. Mobile uses tokens; desktop hardcodes 50px and tablet hardcodes 30px.
|
||||
|
||||
**Root cause D — Hero was authored on a separate timeline than the rest of the page.**
|
||||
Issues: #4, #11, #18, #19, #12. The Hero composition predates the reveal system, predates current motion language.
|
||||
|
||||
**Root cause E — Pure black `#000` is treated as "the heading color."**
|
||||
Issues: #8, plus the original `.btn-yellow` issue I already fixed. The system has a warmer near-black available; it's not the default.
|
||||
|
||||
**Root cause F — Brand yellow is used both as accent and surface.**
|
||||
Issues: #3 (Instagram section). Premium brands choose one role per color.
|
||||
|
||||
**Root cause G — Mobile breakpoints collapse intent.**
|
||||
Issues: #17, #12, #7. Mobile flattens hierarchy that desktop carefully built.
|
||||
|
||||
---
|
||||
|
||||
## Premiumisation Roadmap
|
||||
|
||||
**To increase calmness:**
|
||||
1. Collapse shadows to 5 tokens. Use them everywhere.
|
||||
2. Replace `#000` with `--text-heading` or `--gw-green` across all headings.
|
||||
3. Demote the Instagram section from full-yellow to a green section with yellow accents.
|
||||
4. Section vertical rhythm: two tiers on mobile, not one (Issue #17).
|
||||
|
||||
**To increase trust:**
|
||||
1. Single Google trust-mark component (Issue #9).
|
||||
2. Consolidate footer typography to 2 sizes (Issue #7).
|
||||
3. Fix mobile hero kicker contrast (Issue #19).
|
||||
4. Add `last-updated` and author byline to comparison and any FAQ-style pages.
|
||||
|
||||
**To increase sophistication:**
|
||||
1. Adopt a type scale (Issue #2). Six display sizes is amateur; three is grown-up.
|
||||
2. One letter-spacing scale, two values (Issue #10).
|
||||
3. Replace `.intro-kicker` rule pattern with the unified `.eyebrow` — or commit to it and repeat it in three other places. No one-offs (Issue #14).
|
||||
4. Hero animation: cap everything at 0.5s. Restraint reads as confidence (Issue #4).
|
||||
|
||||
**To increase conversion confidence:**
|
||||
1. Mobile hero CTA: stack and grow, never shrink (Issue #12).
|
||||
2. Trust mark consolidation reinforces — not repeats — social proof (Issue #9).
|
||||
3. Instagram section: if removed/demoted, the path from `#newlead` → footer is uninterrupted. Currently `#newlead` (the booking form) bleeds into a yellow billboard, which dilutes the form's gravity (Issue #3).
|
||||
|
||||
**To increase perceived engineering quality:**
|
||||
1. Solve the tablet gutter shift (Issue #5).
|
||||
2. Eliminate the nth-child animation hardcode (Issue #11).
|
||||
3. Eliminate hardcoded shadows in global CSS (Issue #13).
|
||||
4. Pick one styling paradigm (Issue #20).
|
||||
|
||||
---
|
||||
|
||||
## Design System Refactor Priority
|
||||
|
||||
**Now (this week):**
|
||||
1. **Type scale tokens.** `--text-display/h1/h2/h3/body-lead/body/small` with weight + tracking. Migrate `.section-heading`, `.hero h1`, `.intro-headline`, `.ph-title`, `.info-block h2`, `#instagram h2` to consume them. This single change subsumes issues #2 and #10.
|
||||
2. **Eyebrow primitive.** One Svelte component, three variants. Migrate `.eyebrow`, `.hero-kicker`, `.intro-kicker`, `.values-eyebrow`, `.booking-eyebrow`, `.footer-col-label`. Issue #1.
|
||||
3. **Replace `--text-strong` with `--text-heading` in `.section-heading`.** One-line change, large perceived impact. Issue #8.
|
||||
4. **Container gutter cleanup.** Find/replace `padding: 0 50px` → `padding: 0 var(--space-container-x)` across `footer`, `#instagram`, `.ph-inner`. Re-evaluate `--space-container-x-tablet`. Issues #5, #16.
|
||||
|
||||
**Next (this month):**
|
||||
5. **Card primitive.** `<Card variant="default|quiet|elevated" />`. Migrate testimonial, FAQ, photo, booking field. Issue #6.
|
||||
6. **Shadow consolidation.** 18 tokens → 5. Migrate all hardcoded shadows. Issue #13.
|
||||
7. **Hero animation pass.** Cap durations, dynamic delays. Issues #4, #11.
|
||||
8. **Instagram section redesign.** Either demote or recompose. Issue #3.
|
||||
|
||||
**Eventually (next quarter):**
|
||||
9. **Google trust-mark component.** Issue #9.
|
||||
10. **Footer typography pass.** Two sizes, two opacities. Issue #7.
|
||||
11. **Mobile vertical rhythm tiers.** Issue #17.
|
||||
12. **Pick one styling paradigm.** Document. Issue #20.
|
||||
|
||||
---
|
||||
|
||||
## Visual Cohesion Score
|
||||
|
||||
Scored against what a principal-designer-led product (Linear, Stripe, Apple marketing) would ship.
|
||||
|
||||
| Axis | Score | Note |
|
||||
|---|---|---|
|
||||
| **Typography** | **4/10** | Six heading scales, four trackings, weight conflicts between base h2 and `.section-heading`. The font choices (Unbounded + Readex Pro) are good; their application is uneven. |
|
||||
| **Spacing** | **5/10** | A 4px scale exists and is partly respected, but desktop container padding is hardcoded (50px) and the tablet override breaks the curve. Section vertical rhythm collapses on mobile. |
|
||||
| **Colour** | **5/10** | The brand palette (green + yellow + cream) is strong. Execution: too many near-identical grays/off-whites in tokens, pure black where warmer would serve, yellow used as both accent and surface. |
|
||||
| **Motion** | **4/10** | Three speed regimes coexist (180–220ms, 300–450ms, 850–1600ms). Hero choreography is heavy compared to a snappy rest-of-page. Mobile menu motion is the best-tuned. |
|
||||
| **Hierarchy** | **6/10** | Within each section, hierarchy generally reads. Across the page, section weights compete — Instagram outshouts the booking form that precedes it. Headings have inconsistent weight signaling. |
|
||||
| **Responsiveness** | **6/10** | Mobile is genuinely handled (good `iOS-zoom-on-focus` awareness, 44px tap targets, sticky book bar). But mobile flattens hierarchy that desktop built, and the <480px CTA squeeze undermines the conversion target. |
|
||||
| **Premium feel** | **4/10** | Several premium gestures (warm beige photo frames, Google trust mark, restrained eyebrows in some places) are undone by harsh black headings, the yellow Instagram billboard, the 1.6s hero entry, and visible engineering drift between sections. |
|
||||
|
||||
**Overall: 4.9 / 10.**
|
||||
|
||||
---
|
||||
|
||||
## Final read
|
||||
|
||||
This product is **two design languages glued together**. The first is *editorial-warm*: cream surfaces, dark-green accents, photographic cards with beige frames, restrained typography. The second is *marketing-loud*: yellow billboards, black-on-yellow CTAs, 1.6s hero animations, sub-12px footer captions, hand-numbered choreography.
|
||||
|
||||
The brand voice (`CLAUDE.md` says "preserve the WordPress visual design") implicitly rewards both. But premium products converge. Right now Goodwalk is asking the visitor to switch reading modes every two scrolls.
|
||||
|
||||
**The single highest-leverage move:** adopt a real type-scale token set this week. Issues #2, #8, #10, and large parts of #1 and #7 dissolve. Headings stop fighting each other. The page begins to feel like one product.
|
||||
|
||||
**The second-highest:** abolish the yellow Instagram section as a section. Yellow becomes a pure accent. The path from hero → booking → footer becomes uninterrupted green-and-warm. The site immediately reads more grown-up.
|
||||
|
||||
Everything else is downstream of those two.
|
||||
|
||||
---
|
||||
|
||||
# Changelog
|
||||
|
||||
Concrete from→to changes, grouped by file. Each row references the issue it resolves.
|
||||
|
||||
## `src/lib/styles/variables.css`
|
||||
|
||||
| # | From | To |
|
||||
|---|---|---|
|
||||
| 2 | _(no type tokens)_ | Add `--text-display: clamp(40px, 5vw, 56px)`, `--text-h1: clamp(34px, 4vw, 44px)`, `--text-h2: clamp(28px, 3vw, 36px)`, `--text-h3: clamp(20px, 2vw, 24px)`, `--text-body-lead: 17px`, `--text-body: 16px`, `--text-small: 13px` |
|
||||
| 2 | _(no weight tokens)_ | Add `--weight-display: 800`, `--weight-heading: 700`, `--weight-body: 400`, `--weight-emphasis: 600` |
|
||||
| 10 | _(no tracking tokens)_ | Add `--tracking-display: -0.035em`, `--tracking-heading: -0.02em`, `--tracking-eyebrow: 0.08em` |
|
||||
| 5 | `--space-container-x-tablet: 30px` | Delete; rely on the desktop clamp at tablet widths |
|
||||
| 13 | 18 shadow tokens (`--shadow-xs/sm/md/lg/xl/2xl/float/press/panel/panel-strong/panel-soft/card/card-hover/badge/badge-hover/menu/menu-soft/...`) | Collapse to 5: `--shadow-flat`, `--shadow-card`, `--shadow-elevated`, `--shadow-floating`, `--shadow-modal`. Alias old names to the 5 during migration. |
|
||||
| 17 | `--space-section-mobile-y: var(--space-8)` (single mobile tier) | Add `--space-section-featured-y-mobile: var(--space-10)` (56px); keep `--space-section-mobile-y: var(--space-8)` (40px) for support tiers |
|
||||
|
||||
## `src/lib/styles/typography.css`
|
||||
|
||||
| # | From | To |
|
||||
|---|---|---|
|
||||
| 8 | `.section-heading { color: var(--text-strong); }` (#000) | `.section-heading { color: var(--text-heading); }` (#1f2421) |
|
||||
| 2 | `.section-heading { font-size: var(--heading-section-size); font-weight: 800; letter-spacing: -0.035em; }` | `.section-heading { font-size: var(--text-h1); font-weight: var(--weight-display); letter-spacing: var(--tracking-display); }` |
|
||||
| 2 | `.hero-text h1 { font-size: clamp(34px, 4vw, 56px); font-weight: 800; letter-spacing: -0.045em; }` | `.hero-text h1 { font-size: var(--text-display); font-weight: var(--weight-display); letter-spacing: var(--tracking-display); }` |
|
||||
| 2 | `h2 { font-weight: 700 }` _(conflicts with `.section-heading` 800)_ | `h2 { font-weight: var(--weight-heading); }` — and remove the duplicate weight from `.section-heading` (token already encodes it) |
|
||||
| 2 | `.info-block h2 { font-size: clamp(28px, 2.4vw, 32px); }` | `.info-block h2 { font-size: var(--text-h2); }` |
|
||||
| 2 | `#instagram h2 { font-size: clamp(30px, 3vw, 36px); }` | `#instagram h2 { font-size: var(--text-h2); }` |
|
||||
| 10 | Four tracking values (-0.02 / -0.035 / -0.04 / -0.045) across headings | Two: `var(--tracking-display)` for h1/display, `var(--tracking-heading)` for h2/h3 |
|
||||
| 1 | `.eyebrow { font-size: 12px; font-weight: 700; letter-spacing: 0.09em; color: var(--gw-green); }` | Becomes the single source of truth. Same selector, but `letter-spacing: var(--tracking-eyebrow)`. Add `.eyebrow--inverse` (white) and `.eyebrow--accent` (yellow) modifiers. |
|
||||
|
||||
## `src/lib/styles/sections.css`
|
||||
|
||||
| # | From | To |
|
||||
|---|---|---|
|
||||
| 1 | `.hero-kicker { ... yellow pill, yellow text, font-size 12px, weight 700, tracking 0.08em ... }` | Replace with `<span class="eyebrow eyebrow--accent">`. Delete the `.hero-kicker` rules. |
|
||||
| 1 | `.intro-kicker { ... yellow rule prefix, weight 600, tracking 0.18em, white-58% text ... }` | Replace with `<span class="eyebrow eyebrow--inverse">`. Delete `.intro-kicker-rule` and the `.intro-kicker` rules. _(Issue #14 — the rule pattern is a one-off; remove it.)_ |
|
||||
| 16 | `footer { padding: 60px 50px 32px; }` | `footer { padding: var(--space-11) var(--space-container-x) var(--space-7); }` |
|
||||
| 16 | `#instagram { padding: 60px 50px; }` | `#instagram { padding: var(--space-section-support-y) var(--space-container-x); }` |
|
||||
| 16 | `.ph-inner { padding: 0 50px; }` | `.ph-inner { padding: 0 var(--space-container-x); }` |
|
||||
| 3 | `#instagram { background: var(--yellow); }` with `.instagram-blurb { color: rgba(0, 0, 0, 0.6); }` | `#instagram { background: var(--gw-green); color: var(--text-inverse); }`. `#instagram .btn { background: var(--yellow); color: var(--gw-green); }`. Yellow becomes accent inside the green section, not the surface. |
|
||||
| 13 | `.testimonial-card:hover { box-shadow: ..., 0 8px 40px rgba(0,0,0,0.08); }` | `.testimonial-card:hover { box-shadow: ..., var(--shadow-elevated); }` |
|
||||
| 13 | `.hero-trust-mark { box-shadow: 0 2px 8px rgba(0, 0, 0, 0.25); }` | `.hero-trust-mark { box-shadow: var(--shadow-card); }` |
|
||||
| 13 | `.intro-google-mark { box-shadow: 0 4px 12px rgba(0, 0, 0, 0.25); }` | `.intro-google-mark { box-shadow: var(--shadow-card); }` _(then deleted — see #9)_ |
|
||||
| 13 | `.ph-media { box-shadow: 0 16px 40px rgba(var(--ink-rgb), 0.08); }` | `.ph-media { box-shadow: var(--shadow-elevated); }` |
|
||||
| 9 | `.hero-trust-mark` (28px circle) + `.intro-google-mark` (36px circle) as separate rule blocks | Both replaced by `<GoogleTrustMark size="sm|md" />`. Component owns one shadow, one shape, two size variants. Delete both CSS blocks. |
|
||||
| 6, 15 | `.testimonial-card { border-radius: 28px; padding: 36px 32px; background: linear-gradient(180deg, var(--surface-page), var(--surface-panel-soft)); }` | `.testimonial-card { border-radius: var(--radius-lg); padding: var(--space-7); background: var(--surface-panel); }` |
|
||||
| 6, 15 | `.faq details { border-radius: 16px; }`, `.faq summary { border-radius: 16px; padding: 18px 22px; }` | `.faq details { border-radius: var(--radius-lg); }`, `.faq summary { border-radius: var(--radius-lg); padding: var(--space-5) var(--space-6); }` |
|
||||
| 6 | `.ph-media { border-radius: 28px; }` | `.ph-media { border-radius: var(--radius-lg); }` |
|
||||
| 18 | `#hero::after` four-stop gradient (`transparent 18% → 26% → 78% → 86%`) | Two-stop: `linear-gradient(to bottom, transparent 30%, var(--surface-brand) 95%)` |
|
||||
| 4, 11 | `.hero-text > :nth-child(1..7) { animation-delay: 160ms..760ms; }` (seven hardcoded) | Single rule: `.hero-text > * { animation-delay: calc(120ms + var(--i, 0) * 80ms); }`. Inject `--i` via Svelte `style:--i={index}`. Max stagger reduced from 760ms to ~500ms. |
|
||||
| 4 | `.hero-text > * { animation: heroRise 0.85s cubic-bezier(0.22, 1, 0.36, 1) forwards; }` | `animation: heroRise 0.45s cubic-bezier(0.22, 1, 0.36, 1) forwards;` |
|
||||
| 4 | `.hero-img img { animation: heroImageEnter 1.6s cubic-bezier(0.16, 1, 0.3, 1) forwards; }` | `animation: heroImageEnter 0.5s cubic-bezier(0.16, 1, 0.3, 1) forwards;` |
|
||||
| 4 | `.hero-title-highlight::after { animation: heroUnderlineDraw 0.9s ...; animation-delay: 900ms; }` | `animation: heroUnderlineDraw 0.4s ...; animation-delay: 350ms;` |
|
||||
| 7 | Five footer text sizes (14/13/12/10) and six opacities (0.5/0.6/0.7/0.72/0.8/0.85) | Two sizes: 14px (links/copy) + 12px (labels/legal). Two opacities: 0.85 (active) + 0.55 (labels). Apply across `.footer-brand p`, `.footer-nav a`, `.footer-bottom`, `.footer-contact-link`, `.footer-back-top`, `.footer-col-label`. |
|
||||
|
||||
## `src/lib/styles/responsive.css`
|
||||
|
||||
| # | From | To |
|
||||
|---|---|---|
|
||||
| 19 | `.hero-kicker { color: rgba(var(--white-rgb), 0.48); font-size: 10px; letter-spacing: 0.13em; }` (mobile) | `.hero-kicker { color: rgba(var(--white-rgb), 0.72); font-size: 11px; letter-spacing: var(--tracking-eyebrow); }` (mobile) |
|
||||
| 12 | `@media (max-width: 480px) { .hero-buttons { flex-direction: row; gap: 8px; } .hero-buttons .btn-yellow { padding: 13px 9px; font-size: 12px; line-height: 1.1; flex: 1; width: 0; } }` | Delete the entire `@media (max-width: 480px) .hero-buttons` block. Let the 768px stacked layout extend down — primary CTA stays full-width, 14px font, 14px vertical padding. |
|
||||
| 5 | `@media (max-width: 1024px) { .services-inner, .values-inner, ... { padding-left: var(--space-container-x-tablet); } }` | Delete the tablet padding override entirely. Components use `var(--space-container-x)` which already clamps appropriately. |
|
||||
|
||||
## `src/lib/components/Header.svelte` and section components
|
||||
|
||||
| # | From | To |
|
||||
|---|---|---|
|
||||
| 1 | `<span class="eyebrow values-eyebrow">…</span>` (Values + Booking + others) | `<Eyebrow>…</Eyebrow>` Svelte component, default variant. Delete `.values-eyebrow`, `.booking-eyebrow` rules. |
|
||||
| 9 | Inline `.hero-trust-mark` markup in `HeroSection.svelte` and `.intro-google-mark` markup in `IntroStrip.svelte` | `<GoogleTrustMark size="sm" />` and `<GoogleTrustMark size="md" />` in their respective parents. |
|
||||
| 11 | `<div class="hero-text"> <p class="hero-kicker">…</p> <h1>…</h1> … </div>` (children animate via nth-child) | `<div class="hero-text"> {#each items as item, i} <div style:--i={i}>…</div> {/each} </div>` — explicit index per child, animation reads `--i`. |
|
||||
| 3 | `<section id="instagram">…</section>` background defined in CSS via `#instagram { background: var(--yellow); }` | Same markup, but section now reads green (CSS-only change in sections.css above). |
|
||||
|
||||
## New components to create
|
||||
|
||||
| # | Component | Purpose |
|
||||
|---|---|---|
|
||||
| 1 | `<Eyebrow variant="default|inverse|accent" />` | Single eyebrow primitive. Replaces 6 implementations. |
|
||||
| 6 | `<Card variant="default|quiet|elevated" />` | Single card primitive. Migrates testimonial, FAQ, photo, booking field, point. |
|
||||
| 9 | `<GoogleTrustMark size="sm|md" />` | Single Google "G" mark. Replaces hero + intro duplicates. |
|
||||
|
||||
## Migration table — what consumes the new tokens
|
||||
|
||||
| Token | Consumers (must migrate) |
|
||||
|---|---|
|
||||
| `--text-display` | `.hero-text h1` |
|
||||
| `--text-h1` | `.section-heading`, `.ph-title` |
|
||||
| `--text-h2` | `.intro-headline`, `.info-block h2`, `#instagram h2` |
|
||||
| `--text-h3` | `.service-card h3`, `.info-block h3`, `.values-points-title`, `.values-point h3` |
|
||||
| `--tracking-display` | All h1-tier headings |
|
||||
| `--tracking-heading` | All h2/h3-tier headings |
|
||||
| `--radius-lg` | All cards (testimonial, FAQ, photo, booking field, ph-media) |
|
||||
| `--shadow-card` | Hero/intro trust marks, trust chip elevations |
|
||||
| `--shadow-elevated` | Testimonial hover, ph-media, FAQ open state |
|
||||
| `--shadow-floating` | Buttons on hover, intro-google on hover |
|
||||
|
||||
## Out of scope (deliberately not in this changelog)
|
||||
|
||||
- Collapsing the 7 gray tokens to 3 — separate codebase-wide codemod; ship after type scale lands.
|
||||
- Collapsing the 6 off-white tokens to 2 — same reasoning.
|
||||
- Stylelint rule blocking raw hex / raw shadow tuples — adds after the migration is complete; otherwise too noisy.
|
||||
- Picking one styling paradigm (Issue #20) — architectural decision, separate PR.
|
||||