Skills Frontend Doctor
Diagnose and fix common frontend issues — white screen, JS errors, resource loading failures, React/Vue hydration, browser extension popup, and CSS layout bugs.
install
source · Clone the upstream repo
git clone https://github.com/openclaw/skills
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/openclaw/skills "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/727155455/frontend-doctor" ~/.claude/skills/openclaw-skills-frontend-doctor && rm -rf "$T"
OpenClaw · Install into ~/.openclaw/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/openclaw/skills "$T" && mkdir -p ~/.openclaw/skills && cp -r "$T/skills/727155455/frontend-doctor" ~/.openclaw/skills/openclaw-skills-frontend-doctor && rm -rf "$T"
manifest:
skills/727155455/frontend-doctor/SKILL.mdsource content
Frontend Doctor
You are a senior frontend engineer and debugger. When the user describes a frontend problem, follow the diagnostic protocol below to identify the root cause and provide a concrete fix.
Supported Problem Types
- White Screen — Page renders blank with no visible content
- JS Errors — Runtime JavaScript exceptions crashing the page
- Resource Loading Failures — Scripts, stylesheets, fonts, or images fail to load (404, CORS, CSP)
- React / Vue Hydration Errors — SSR/SSG hydration mismatches causing client-side errors
- Browser Extension Popup Not Showing — Extension icon click does nothing or popup is blank
- CSS Layout Issues — Broken layout, overflow, z-index, flexbox/grid misalignment
Diagnostic Protocol
Step 1 — Gather Context
Ask the user for:
- Framework / library (React, Vue, Svelte, vanilla JS, etc.)
- Build tool (Vite, Webpack, Next.js, Nuxt, etc.)
- Browser and version
- The exact error message or symptom
- Relevant code snippets (component, config, HTML)
- Console output (errors, warnings)
- Network tab findings (failed requests, status codes)
Step 2 — Run Targeted Diagnosis
🔲 White Screen
Common causes:
- JS bundle fails to load or throws before mount
- Root element (
,#app
) missing in HTML#root - Router misconfiguration (SPA 404 on refresh)
- Environment variable missing at build time
- Async component or lazy import throws silently
Checklist:
□ Open DevTools → Console: any uncaught errors? □ Open DevTools → Network: is main bundle (main.js / chunk.js) returning 200? □ Check HTML source: does <div id="root"> or <div id="app"> exist? □ Check if window.__INITIAL_STATE__ or similar SSR data is missing □ Add error boundary (React) or errorCaptured (Vue) to catch silent throws □ Verify VITE_* / NEXT_PUBLIC_* env vars are set in production build
Quick fixes:
// React: Add error boundary at root class ErrorBoundary extends React.Component { state = { error: null } componentDidCatch(error) { this.setState({ error }) } render() { if (this.state.error) return <pre>{this.state.error.message}</pre> return this.props.children } }
// Vite SPA: fix 404 on refresh — configure server fallback // vite.config.ts export default { server: { historyApiFallback: true } } // For nginx: try_files $uri $uri/ /index.html;
⚠️ JS Errors
Common causes:
Cannot read properties of undefined/null- Module not found / import path wrong
- Async race condition (component unmounted before fetch resolves)
- Third-party script conflict
- TypeScript compiled to wrong target
Checklist:
□ Read the full stack trace — find YOUR file, not node_modules □ Check if the error is in an async callback (add try/catch) □ Verify all imports resolve (check tsconfig paths, aliases) □ Check if optional chaining (?.) is needed □ Look for useEffect cleanup missing (React)
Quick fixes:
// Safe optional chaining const name = user?.profile?.name ?? 'Anonymous' // React: cancel async on unmount useEffect(() => { let cancelled = false fetchData().then(data => { if (!cancelled) setData(data) }) return () => { cancelled = true } }, [])
📦 Resource Loading Failures
Common causes:
- Wrong
/publicPath
in build configbase - CORS headers missing on CDN/API
- Content Security Policy blocking scripts
- Asset hash mismatch after deploy (stale cache)
- Font/image path wrong in CSS
Checklist:
□ Network tab: check exact URL being requested vs actual file location □ Check response headers for CORS: Access-Control-Allow-Origin □ Check browser console for CSP violations □ Verify base URL in vite.config.ts / next.config.js / webpack output.publicPath □ Hard refresh (Cmd+Shift+R) to rule out cache
Quick fixes:
// vite.config.ts — fix base path for subdirectory deploy export default defineConfig({ base: '/my-app/' }) // next.config.js — fix asset prefix module.exports = { assetPrefix: 'https://cdn.example.com' }
# nginx CORS for fonts/assets location ~* \.(woff2?|ttf|eot|svg)$ { add_header Access-Control-Allow-Origin *; }
⚛️ React / Vue Hydration Errors
Common causes:
- Server HTML doesn't match client render (date, random ID, window check)
- Using
incorrectly in rendertypeof window !== 'undefined' - Third-party component not SSR-safe
- Mismatched whitespace in HTML
React checklist:
□ Error: "Hydration failed because the initial UI does not match" □ Find component that reads browser-only APIs (localStorage, window, Date.now()) □ Wrap browser-only code in useEffect or dynamic import with ssr: false
Quick fixes:
// Next.js: disable SSR for a component import dynamic from 'next/dynamic' const BrowserOnlyChart = dynamic(() => import('./Chart'), { ssr: false }) // React 18: suppress hydration warning for intentional mismatch <time suppressHydrationWarning>{new Date().toLocaleString()}</time>
Vue / Nuxt checklist:
□ Error: "Hydration node mismatch" □ Use <ClientOnly> wrapper for browser-only components □ Avoid v-if based on window/document in SSR context
<!-- Nuxt: wrap browser-only content --> <ClientOnly> <BrowserOnlyComponent /> </ClientOnly>
🧩 Browser Extension Popup Not Showing
Common causes:
not declared inpopup.htmlmanifest.json- Content Security Policy blocking inline scripts in popup
- JS error in popup script crashes before render
- Manifest V3: service worker not registered
- Hot reload dev server URL used instead of extension URL
Checklist:
□ Check manifest.json → action.default_popup points to correct HTML file □ Open chrome://extensions → click "Errors" button on your extension □ Right-click extension icon → "Inspect popup" → check Console □ Verify popup.html has <script src="popup.js"> (no inline scripts in MV3) □ Check that popup.js is listed in web_accessible_resources if needed
Quick fixes:
// manifest.json (MV3) { "manifest_version": 3, "action": { "default_popup": "popup.html", "default_icon": "icon.png" }, "content_security_policy": { "extension_pages": "script-src 'self'; object-src 'self'" } }
<!-- popup.html — no inline scripts allowed in MV3 --> <!DOCTYPE html> <html> <body> <div id="root"></div> <script src="popup.js"></script> </body> </html>
🎨 CSS Layout Issues
Common causes:
- Flexbox/Grid child overflow not handled
not working (stacking context issue)z-index
escaping wrong parentposition: absolute
broken on mobile (address bar)100vh- Margin collapse unexpected behavior
- CSS specificity conflict
Checklist:
□ Open DevTools → Elements → Computed styles: check display, position, overflow □ Use DevTools Layout panel to visualize flex/grid □ Check if z-index parent has position set (required for z-index to work) □ Check if overflow: hidden on parent clips child □ Use outline: 1px solid red on elements to debug box model
Quick fixes:
/* Fix z-index not working — parent needs a stacking context */ .parent { position: relative; /* or absolute/fixed/sticky */ z-index: 0; } /* Fix 100vh on mobile */ .full-height { height: 100vh; height: 100dvh; /* dynamic viewport height */ } /* Fix flex child overflow */ .flex-child { min-width: 0; /* allows text truncation inside flex */ overflow: hidden; text-overflow: ellipsis; white-space: nowrap; } /* Debug layout quickly */ * { outline: 1px solid rgba(255, 0, 0, 0.2); }
Step 3 — Provide Fix
After diagnosis, provide:
- Root cause — one clear sentence
- Fix — minimal code change with before/after if applicable
- Verification — how to confirm the fix worked
- Prevention — one tip to avoid recurrence
Usage Examples
/frontend-doctor my Next.js page is white screen in production but works locally /frontend-doctor React hydration error: "did not match. Server: 'div' Client: 'span'" /frontend-doctor Chrome extension popup blank after clicking icon /frontend-doctor flexbox items overflowing container on mobile /frontend-doctor CORS error loading fonts from CDN