skills/zeke/faster-chrome-devtools-skill/faster-chrome-devtools-skill

faster-chrome-devtools-skill

Installation
SKILL.md

Chrome DevTools MCP: faster patterns

Tool speed reference

Measured from real session data (medians, at default viewport):

Tool Avg Notes
take_snapshot 80ms Fastest page inspection. Prefer over screenshot.
list_console_messages 3ms Cheap
evaluate_script 227ms Fast; use as escape hatch for React components
fill 358ms
click 696ms
take_screenshot 1,118ms Slow; only use when visual appearance matters
navigate_page 2,672ms Highly variable; always set timeout
list_pages 2,737ms High variance; avoid in tight loops
new_page 3,506ms Expensive; reuse existing tabs when possible

Screenshot safety

PNG is lossless and uncompressed — a full-page PNG of a typical 1280px-wide page can easily reach 3–7MB. JPEG and WebP use lossy compression; at quality 75 a JPEG is typically 90%+ smaller than the equivalent PNG with no perceptible quality loss for the purposes of page inspection.

This matters because of two size thresholds in the pipeline:

  • 2MB (MCP threshold): screenshots >= 2MB are saved to a temp file and the model receives only a file path. The model never sees the image. This happens silently with no warning.
  • 5MB (Claude API limit): if an inline screenshot exceeds 5MB as base64, the API rejects the entire request and the session becomes permanently unrecoverable — compaction doesn't help because it replays the same images.

Always use JPEG or WebP with a quality setting when the screenshot will be shown to the model:

// Safe
take_screenshot({ format: "jpeg", quality: 75 })

// Dangerous — PNG has no compression, fullPage compounds it
take_screenshot({ fullPage: true })

Only use fullPage: true when you genuinely need the full page, and never without format: "jpeg", quality: 75.

If you get back a file path instead of an image, the screenshot exceeded 2MB. Retry with JPEG at quality 60.

Snapshot over screenshot

take_snapshot returns the page's accessibility tree — element roles, names, and UIDs you can pass to other tools. take_screenshot renders a pixel image via Puppeteer.

Use take_snapshot when you need to know what's on the page. Use take_screenshot only when visual appearance (images, CSS rendering, canvas) matters.

// Check page state — fast
take_snapshot()

// Verify a chart rendered correctly — screenshot warranted
take_screenshot({ format: "jpeg", quality: 75 })

After navigate_page, take_snapshot resolves in ~15ms. wait_for followed by take_screenshot averages 3,800ms for the same information.

Always set a timeout on navigate_page

With no timeout, navigate_page can block indefinitely. A localhost server with no timeout was observed hanging for 43 seconds.

// Always include a timeout
navigate_page({ type: "url", url: "https://example.com", timeout: 15000 })

// Dangerous — no timeout
navigate_page({ type: "url", url: "http://localhost:3000" })

Recommended timeouts by context:

Context Timeout
Local dev server 10,000ms
Normal web page 15,000ms
Slow or resource-heavy page 30,000ms
OAuth / external redirect flow 60,000ms

Reuse tabs

new_page averages 3,500ms. If a relevant tab is already open, use it.

// Check first
list_pages()
select_page({ pageId: <id> })

// Only open a new tab if the URL isn't already open
new_page({ url: "https://example.com" })

How wait_for works

wait_for is MutationObserver-based, not a polling loop. It resolves the moment matching text appears in the DOM. When the content is already present or appears quickly, it resolves in 40–100ms.

The cost only comes when the expected content never appears and the timeout elapses. Set timeouts that reflect how long the operation could realistically take:

// After a click that triggers a UI update — short timeout is fine
wait_for({ text: ["Success", "Done"], timeout: 5000 })

// After submitting a form that hits a slow API
wait_for({ text: ["Order confirmed"], timeout: 15000 })

// After starting an OAuth flow — needs time for external redirect
wait_for({ text: ["refresh_token"], timeout: 60000 })

Do not use wait_for for things that will never appear in the accessibility tree (background processes, DNS propagation, external service completion). Use evaluate_script to poll a JS condition instead.

evaluate_script for hard cases

The accessibility tree is insufficient for React custom components, headless dropdowns, and synthetic event inputs. Use evaluate_script as the escape hatch.

Programmatic click (React-select and similar):

evaluate_script({
  function: () => {
    const option = document.querySelector('[class*="option"]');
    option?.click();
  }
})

Range slider with React synthetic events:

evaluate_script({
  function: () => {
    const input = document.querySelector('input[type="range"]');
    const setter = Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, 'value').set;
    setter.call(input, '75');
    input.dispatchEvent(new Event('input', { bubbles: true }));
  }
})

Read state that isn't in the a11y tree:

evaluate_script({
  function: () => document.querySelector('.status')?.dataset.state
})

Canonical interaction pattern

The sub-100ms loop observed across automated sessions — every state transition confirmed with wait_for before the next action, no arbitrary sleeps:

click({ uid: "..." })                                     // ~105ms
wait_for({ text: ["Enter symbol"], timeout: 3000 })       // ~60ms
fill({ uid: "...", value: "AAPL" })                       // ~105ms
press_key({ key: "Enter" })                               // ~105ms
wait_for({ text: ["Sell All", "Action"], timeout: 3000 }) // ~65ms
fill({ uid: "...", value: "Sell" })                       // ~105ms
click({ uid: "..." })                                     // ~105ms
wait_for({ text: ["Order confirmed"], timeout: 5000 })    // ~55ms

Anti-patterns

Anti-pattern Instead
take_screenshot() to check DOM state take_snapshot()
take_screenshot({ fullPage: true }) take_screenshot({ fullPage: true, format: "jpeg", quality: 75 })
navigate_page({ url }) with no timeout Always include timeout
new_page() when tab is already open list_pages() then select_page()
Long wait_for for async external events evaluate_script polling a JS condition
Clicking into React components via a11y evaluate_script with direct DOM manipulation
Weekly Installs
8
GitHub Stars
1
First Seen
Today