skills/pwa-install/SKILL.md
--- name: pwa-install description: Turn a web app into an installable Progressive Web App. Wires a web manifest, minimal service worker (network-first for HTML, cache-first for assets), icons, and an install-prompt component. Use for daily-return utilities, dashboards, and any app where "add to home screen" replaces the need for an email list. category: quality-review argument-hint: [--runtime cf|vercel|node] [--name <app>] [--theme <hex>] [--no-sw] [--offline] allowed-tools: Bash(*) Read Write
npx skillsauth add RonanCodes/ronan-skills skills/pwa-installInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Add the three ingredients that make a web app installable: a manifest, a service worker, and a prompt UI. For daily-return apps, this is the single biggest re-engagement lever — installing the app to the home screen turns casual visits into a habit loop without needing auth or email capture.
/ro:pwa-install # auto-detect, default template
/ro:pwa-install --name "Connections" # custom display name
/ro:pwa-install --theme "#14b8a6" # theme colour
/ro:pwa-install --no-sw # manifest only (no offline)
/ro:pwa-install --offline # add offline fallback page + richer caching
Before scaffolding anything, check what's already there. If all four core pieces exist, stop and report "PWA already wired" with the file list, instead of dumping the recipe. The user can pass --force to re-scaffold.
test -f "$REPO/public/manifest.webmanifest" -o -f "$REPO/public/manifest.json" && echo "✓ manifest"
test -f "$REPO/public/sw.js" -o -f "$REPO/public/service-worker.js" -o -d "$REPO/public/workbox-*" && echo "✓ service worker"
ls "$REPO"/public/icons/icon-{192,512}.png 2>/dev/null && echo "✓ icons"
grep -rqE "navigator\.serviceWorker\.register|workbox-window" "$REPO/src" && echo "✓ SW registration"
grep -qE "rel=\"manifest\"|rel: 'manifest'" "$REPO"/src/routes/__root.* 2>/dev/null && echo "✓ manifest link in root"
Report each as ✓ / ✗. If 4+ of 5 are ✓, say "PWA already wired; nothing to scaffold. Audit: [list]. Use --force to re-scaffold or pick the missing piece by hand." Skip the rest of the steps.
If partially wired (1-3 of 5 ✓), tell the user which pieces exist and ask whether to fill the gaps or re-scaffold from scratch.
public/manifest.webmanifest — app metadata, icons, display mode, theme colour.public/sw.js — minimal service worker with sensible defaults.public/icons/icon-192.png + icon-512.png — generated from an existing favicon or prompted.src/routes/__root.tsx — link rel="manifest", theme-color, apple-touch-icon.src/components/InstallPrompt.tsx) — detects beforeinstallprompt, shows an unobtrusive toast.public/manifest.webmanifest{
"name": "Connections Helper",
"short_name": "Connections",
"description": "NYT Connections puzzle sidekick.",
"start_url": "/?source=pwa",
"display": "standalone",
"display_override": ["window-controls-overlay", "standalone", "browser"],
"background_color": "#ffffff",
"theme_color": "#14b8a6",
"orientation": "portrait-primary",
"categories": ["games", "puzzles", "utilities"],
"icons": [
{ "src": "/icons/icon-192.png", "sizes": "192x192", "type": "image/png", "purpose": "any" },
{ "src": "/icons/icon-512.png", "sizes": "512x512", "type": "image/png", "purpose": "any" },
{ "src": "/icons/icon-maskable-512.png", "sizes": "512x512", "type": "image/png", "purpose": "maskable" }
],
"screenshots": [
{ "src": "/screenshots/desktop.png", "sizes": "1280x720", "type": "image/png", "form_factor": "wide" },
{ "src": "/screenshots/mobile.png", "sizes": "390x844", "type": "image/png", "form_factor": "narrow" }
]
}
start_url with ?source=pwa lets analytics distinguish standalone opens from browser opens (useful to measure install conversion).
Maskable icon: Android adaptive icons mask to the platform shape. Without a maskable icon, Android crops your icon badly. Generate via npx pwa-asset-generator or a manual mask-safe design (content inside 60% centre).
public/sw.js — default service workerTwo strategies depending on app shape:
Network-first for HTML (content apps, dashboards, daily utilities). Fresh content is the priority; offline is a fallback.
const CACHE = 'app-v1'
const ASSETS = ['/', '/offline']
self.addEventListener('install', (e) => {
e.waitUntil(caches.open(CACHE).then((c) => c.addAll(ASSETS)))
self.skipWaiting()
})
self.addEventListener('activate', (e) => {
e.waitUntil(
caches.keys().then((keys) =>
Promise.all(keys.filter((k) => k !== CACHE).map((k) => caches.delete(k))),
),
)
self.clients.claim()
})
self.addEventListener('fetch', (e) => {
const { request } = e
if (request.method !== 'GET') return
// Network-first for HTML documents
if (request.mode === 'navigate' || request.headers.get('accept')?.includes('text/html')) {
e.respondWith(
fetch(request)
.then((r) => {
const clone = r.clone()
caches.open(CACHE).then((c) => c.put(request, clone))
return r
})
.catch(() => caches.match(request).then((r) => r ?? caches.match('/offline'))),
)
return
}
// Cache-first for static assets (immutable hashed filenames from Vite)
if (/\.(js|css|woff2?|png|svg|webp|avif|jpg|jpeg)$/.test(new URL(request.url).pathname)) {
e.respondWith(
caches.match(request).then((cached) => {
if (cached) return cached
return fetch(request).then((r) => {
const clone = r.clone()
caches.open(CACHE).then((c) => c.put(request, clone))
return r
})
}),
)
return
}
// Everything else (API calls): network-only, no caching.
})
Cache-first for everything (fully offline-first apps, e.g. a note-taking PWA) — flip the defaults. Use Workbox (workbox-window) if the caching logic is getting complex; handwritten SW is fine up to about 50 lines.
Never cache API responses generically — it'll serve stale data after a deploy. If you need API caching, do it in the server with Cache-Control: s-maxage=..., not in the SW.
In src/routes/__root.tsx (or equivalent <head> location):
head: () => ({
links: [
{ rel: 'manifest', href: '/manifest.webmanifest' },
{ rel: 'icon', href: '/icons/icon-192.png', sizes: '192x192' },
{ rel: 'apple-touch-icon', href: '/icons/icon-192.png' },
],
meta: [
{ name: 'theme-color', content: '#14b8a6' },
{ name: 'mobile-web-app-capable', content: 'yes' },
{ name: 'apple-mobile-web-app-capable', content: 'yes' },
{ name: 'apple-mobile-web-app-status-bar-style', content: 'black-translucent' },
],
}),
In the client entry (src/client.tsx or src/app.tsx):
if (import.meta.env.PROD && 'serviceWorker' in navigator) {
window.addEventListener('load', () => {
navigator.serviceWorker.register('/sw.js').catch((err) => {
console.warn('SW registration failed:', err)
})
})
}
Prod only. A registered SW in dev caches your HMR bundles and turns every hot-reload into a confused mess.
// src/components/InstallPrompt.tsx
import { useEffect, useState } from 'react'
import { track } from '@/lib/posthog' // if using /ro:posthog
interface BeforeInstallPromptEvent extends Event {
prompt: () => Promise<void>
userChoice: Promise<{ outcome: 'accepted' | 'dismissed' }>
}
export function InstallPrompt() {
const [event, setEvent] = useState<BeforeInstallPromptEvent | null>(null)
const [dismissed, setDismissed] = useState(
() => typeof window !== 'undefined' && localStorage.getItem('pwa-install-dismissed') === 'true',
)
useEffect(() => {
const onBeforeInstall = (e: Event) => {
e.preventDefault()
setEvent(e as BeforeInstallPromptEvent)
}
window.addEventListener('beforeinstallprompt', onBeforeInstall)
return () => window.removeEventListener('beforeinstallprompt', onBeforeInstall)
}, [])
if (!event || dismissed) return null
const install = async () => {
await event.prompt()
const { outcome } = await event.userChoice
track?.('cta_clicked', { cta: 'pwa_install', location: `prompt:${outcome}` })
setEvent(null)
}
const dismiss = () => {
localStorage.setItem('pwa-install-dismissed', 'true')
setDismissed(true)
track?.('cta_clicked', { cta: 'pwa_install_dismiss', location: 'prompt' })
}
return (
<div role="dialog" aria-label="Install this app" className="...">
<p>Install for one-tap access from your home screen.</p>
<button onClick={install}>Install</button>
<button onClick={dismiss} aria-label="Dismiss install prompt">Not now</button>
</div>
)
}
Important UX rules:
localStorage flag is the lightest solution.beforeinstallprompt. Use a UA check to show a small "Add to Home Screen via Share" hint instead.⚠️ The single most important step. Browsers pick the maskable icon for the install prompt. If icon-maskable-512.png is blank or a flat colour, the install dialog renders a solid square with no glyph — and users dismiss without installing. Verified-on-the-wild bug: connections-helper shipped with a flat green-500 maskable tile and silently looked broken to every new install for 24h.
Always view the rendered PNGs before declaring done:
file public/icons/*.png # confirm bit depth > 1 and reasonable file sizes (≥ 5 KB)
open public/icons/icon-maskable-512.png # macOS — eyeball it
A 1-bit / sub-1KB icon is a placeholder, not real artwork. Same for any solid-colour tile.
The cleanest pattern is one master SVG that renders to all three sizes (so future tweaks don't drift). Drop it at public/icons/icon.svg and render with ImageMagick:
magick -background none public/icons/icon.svg -resize 192x192 public/icons/icon-192.png
magick -background none public/icons/icon.svg -resize 512x512 public/icons/icon-512.png
magick -background none public/icons/icon.svg -resize 512x512 public/icons/icon-maskable-512.png
Maskable safe zone. Android crops to a circle of radius 40% (diameter 80%) of the canvas centre. Keep the visual content within an inner square of ~78% of the canvas (~12% margin per side on a 512×512). Background must be full-bleed with no transparent corners — the OS does the rounding, your file shouldn't.
If no logo exists yet, lean into the app's domain. For a colour-themed app (puzzle helper, calendar tool, palette utility) a thematic abstract pattern works fine — connections-helper uses a 4-row grid in the NYT Connections palette and reads instantly. Plain "first letter on coloured background" is a fallback only.
npx pwa-asset-generator ./src/assets/logo.svg ./public/icons \
--manifest ./public/manifest.webmanifest \
--background "#ffffff" \
--padding "10%" \
--icon-only
Useful when there's already a designed logo. Still verify the output PNGs by eye before shipping — the tool is happy to produce blank tiles if the source SVG has no fill.
# Local build (SW only runs in prod):
pnpm build && pnpm preview
# In the browser:
# 1. DevTools → Application → Manifest. Verify no warnings.
# 2. DevTools → Application → Service Workers. Verify registered + running.
# 3. DevTools → Lighthouse → PWA audit. Should be green.
# 4. Network tab → throttle to "Offline" → reload. Offline fallback should appear.
Real-device test: the beforeinstallprompt Add-to-Home flow only fires on mobile Chrome and desktop Chrome/Edge. Test on an actual Android device via USB debugging before shipping the install component.
/sw.js controls the whole origin. /foo/sw.js only controls /foo/*. Always serve from the origin root unless you want to scope.CACHE in sw.js is the shipping mechanism. Old clients hold the previous SW until the tab closes — add a "reload for latest version" toast when a new SW activates.http:// (localhost is the exception).Cache-Control: no-cache, max-age=0, must-revalidate for /sw.js./ro:app-polish — umbrella; this is check #4/ro:seo-launch-ready — complementary meta-tag setup/ro:posthog — track install prompt acceptance via cta_clicked/ro:accessibility-ci — the install prompt needs role="dialog" + proper focusdevelopment
Close the loop on a Linear ticket when its work ships - move the status and post a deploy comment with the PR link, what shipped, and a try-it link, mentioning the collaborator. Used as the tail of /ro:linear-nightshift for every merged mirror, or manually after an ad-hoc build. Triggers on "linear update", "update the linear ticket", "mark NUT-x done", "tell eoin it shipped", "/ro:linear-update".
devops
Run a night-shift against a collaborator's Linear board. Pulls the team's Grilled tickets (/ro:linear-grill moves a ticket to Grilled once its questions are answered), VERIFIES the questions were actually answered (unanswered → bounce the ticket to the "Question for <name>" state), mirrors verified tickets to ephemeral GitHub issues with ready-for-agent, then runs the standard /ro:night-shift machinery on GitHub. Tail-calls /ro:linear-update for everything that merged + deployed. Triggers on "linear nightshift", "nightshift linear", "drain the linear board", "run the shift off linear", "/ro:linear-nightshift".
development
Grill a collaborator's Linear tickets and move every processed ticket to where it belongs. Resolves the board from the repo's .ro-linear.json, reads the collaborator's Backlog / Ready-for-agent issues, then per ticket either posts 3-5 decision-extracting questions (state moves to "Question for <name>") or confirms it build-ready (state moves to "Grilled", the gate /ro:linear-nightshift consumes); shipped-and-confirmed tickets close as Done. The async-collaborator counterpart of /ro:day-shift for people who never touch GitHub. Triggers on "grill linear", "grill eoin's tickets", "linear grill", "add questions to the linear tickets", "/ro:linear-grill".
development
--- name: about-page description: Add a standard About page to any web app, what it is, the tech stack, and an FAQ, wired into a footer link with a sticky footer. Built with Spartan + Tailwind (the canonical component layer) and falls back to semantic HTML so it ships reliably. Use whenever building, polishing, or shipping an app, every app should have one. Triggers on "add an about page", "about page", "footer about link", or as a standard step in app build/polish. category: frontend argument-h