pix
npx skills add https://github.com/skobak/pix --skill pix
Agent 安装分布
Skill 文档
/pix: The Pixel-Perfect Autonomous Loop
User provides a Figma link. You implement it pixel-perfect. That’s it.
Tool names may vary based on your MCP configuration (e.g.,
figma__get_screenshotormcp__figma__get_screenshot). Run/mcpto see available tools.
Resource Strategy
Tool Costs
| Tool | Cost | Use For |
|---|---|---|
get_metadata |
Cheap | Node tree with child IDs, positions, sizes. No styling. |
get_variable_defs |
Cheap | All design tokens (colors, spacing, radii) as nameâvalue. |
get_code_connect_map |
Cheap | Check which Figma nodes map to existing code components. |
get_screenshot |
Medium | Visual image of a specific node. Target a nodeId to crop/zoom. |
get_design_context |
Expensive | Full code + styling + assets. NEVER call on a large parent â call on individual sections. |
Core rule: Cheap calls first to build a map, then expensive calls only on the smallest necessary nodes.
Image Budget
Claude API limits images to 2000px per dimension when >20 images are in a conversation. Old images cannot be individually removed.
- ~20 screenshots per conversation before the resolution limit kicks in
- 3-image working set: Figma layout (overview), Figma detail (current section), Chrome state (rendered result)
- Never re-screenshot a static Figma node â designs don’t change mid-session
- Use
getComputedStylefor numerical properties â zero image cost - Recovery:
/compactdrops old images from context (see Recovery section)
Phase 1: Reconnaissance
When the user provides a Figma link, extract fileKey and nodeId from the URL and immediately start working. If any MCP call or Chrome interaction fails, run the Doctor Check to diagnose and fix, then resume.
Step 1: Get the Node Tree
Call get_metadata(nodeId, fileKey):
<Frame id="45:1" name="Header" type="FRAME" x="0" y="0" width="1440" height="80">
<Component id="45:10" name="Logo" x="20" y="16" width="120" height="48"/>
<Frame id="45:15" name="NavLinks" x="400" y="20" width="600" height="40">
<Text id="45:16" name="Home" x="0" y="0" width="60" height="40"/>
</Frame>
</Frame>
Build a mental map:
- Identify every major section and note each
nodeId - Note positions and sizes to understand the layout grid
- Identify components vs frames vs text vs icons
Step 2: Extract All Design Tokens (Once)
Call get_variable_defs(nodeId, fileKey). Save the result â do NOT call this again.
Step 3: Check Existing Components
Call get_code_connect_map(nodeId, fileKey). For matched components, follow this priority:
- Reuse as-is â if it covers the Figma design exactly
- Extend minimally â add a prop or variant if close but not exact
- Compose â combine existing components
- Create new â only if nothing existing fits
Step 4: Visual Overview
Call get_screenshot(nodeId, fileKey) on the root selection. This is Image 1 of your budget â your layout reference. Do NOT retake it.
At this point you have NOT called get_design_context at all. You have a complete structural map, all tokens, reusable component info, and a visual reference â all from cheap calls.
Phase 2: Study & Implement (Code in the Dark)
Study the design deeply, memorize every detail, then code from memory.
Step 1: Layout Shell
Using the metadata tree, implement the outer layout:
- Container dimensions and positioning (flex, grid)
- Major section placement
- Background colors from tokens
- Borders and dividers
The metadata + visual overview are usually sufficient. Only call get_design_context if you need specific properties you can’t infer.
Step 2: Study Every Section
For each major section:
-
Figma Detail:
get_screenshot(sectionNodeId, fileKey)â zoomed-in visual reference. -
Design Context:
get_design_context(sectionNodeId, fileKey)â text only, no image cost. If truncated, use child node IDs fromget_metadataand fetch children individually. -
Absorb every detail: fonts, sizes, weights, colors, spacing, borders, shadows, icon shapes. Burn it into memory.
-
Color sanity check: Compare
get_design_contextcolors against the Figma screenshot. If the screenshot reveals opacity layering, overlapping fills, or gradients â the raw token values will be wrong. Use the visual truth, not the raw token.
No Chrome screenshots during this phase. You are studying, not checking.
Key rule: NEVER call get_design_context on the root selection. Always target the smallest meaningful node.
Step 3: Code from Memory
Implement everything using what you memorized:
- Design context output for exact properties
- Tokens from Phase 1 (do not re-extract)
- Reusable components from Phase 1
- Respect project rules: Check for
.claude/rules,CLAUDE.md, and project instruction files. Follow established patterns for component structure, file placement, naming, and styling. Figma MCP output is a design representation â translate it into your project’s conventions, don’t paste it verbatim.
Frontend only. Don’t touch backend, API routes, or database unless the user explicitly asks. Use mock/placeholder data if APIs don’t exist yet.
Minimize screenshots during coding. You studied the design â use what you memorized. But if you’re missing crucial data for a specific element (exact icon shape, a nested layout you didn’t drill into, a subtle gradient), take a targeted get_screenshot on that Figma node rather than guessing. Getting it right the first time is cheaper than a refinement round.
Step 4: Design System Sync
NEVER hardcode values. Sync to the project’s design system:
- Tailwind: Update
tailwind.config.*if a token is missing. Nevertext-[#f3f3f3]. - CSS-in-JS: Add tokens to the theme object. Never inline hex.
- Component Library: Map to existing theme tokens. Never bypass the theme.
- CSS/SCSS: Add custom properties to
:root. Never scatter magic values.
Step 5: Icons & Assets
Icons: Find a match in the project’s existing icon library first â by name, then by visual shape. Match stroke-width and size exactly.
Fallback: If the layer name doesn’t map, get_screenshot(iconNodeId) and identify visually. Match by shape, not name.
Images/illustrations: Download from the Figma MCP assets endpoint and save to the project’s public/static folder. Don’t create inline SVG blobs for complex illustrations. Don’t import new icon packages without asking.
Property Checklist
Before writing code for ANY element, verify ALL applicable properties:
- Text: font-family, font-size, font-weight, line-height, letter-spacing, color, opacity, text-align, text-decoration, text-transform
- Container: width, height, min/max-width, padding (all 4), margin, background-color, border-radius, border-width/color/style, box-shadow, opacity, overflow
- Icon: size, fill, stroke, stroke-width, color (independent from parent text)
- Button/Link: All text + container props + cursor, hover/active/disabled states
- Image: width, height, object-fit, border-radius, border, aspect-ratio
- Spacing: gap, row-gap, column-gap
Layout Principle: Avoid hardcoded sizes. With correct font-size, line-height, padding, and parent width, elements render correctly. Hardcoded dimensions are a symptom of wrong upstream layout.
Responsive: Consider how the component adapts to different screen sizes. Use the project’s responsive approach (Tailwind breakpoints, media queries, container queries).
Never use approximate Tailwind classes (like text-zinc-500) when exact hex values are available from tokens.
Phase 3: Refinement Loop
You think you’re done. Now prove it. Keep cycling until the result is perfect.
The Loop
Step 1: Chrome screenshot (1 image)
First look at what you built. Compare against Figma screenshots in context. Be extremely picky:
- Visual alignment issues
- Missing elements or wrong proportions
- Layout shifts, spacing that looks off
- Wrong icon shapes
- Color or weight mismatches
- Color comparison: Sample dominant colors from Figma vs Chrome. Flag anything that looks off â opacity/layering can cause perceived differences even when hex values match.
Step 2: Numerical audit (zero images)
Run getComputedStyle on every element and compare against Figma values:
const el = document.querySelector('.target-element');
const s = getComputedStyle(el);
JSON.stringify({
padding: s.padding, margin: s.margin, gap: s.gap,
backgroundColor: s.backgroundColor, color: s.color,
fontSize: s.fontSize, fontWeight: s.fontWeight,
lineHeight: s.lineHeight, letterSpacing: s.letterSpacing,
borderRadius: s.borderRadius, borderWidth: s.borderWidth,
borderColor: s.borderColor, boxShadow: s.boxShadow
});
12px is not 10px. #f97316 is not #ff611c.
CSS specificity check: Icons inside wrapper components (Button, Link) often get wrong colors from parent overrides. Run getComputedStyle on the SVG itself. If the computed color doesn’t match, use the Tailwind important modifier (prefix with !) to force it.
Step 3: Picky mismatch list
Combine visual + numerical issues into one list. No issue is too small. 2px off? List it.
Step 4: Fix everything
Batch all fixes. No screenshots between individual fixes.
Step 5: Repeat from Step 1
Exit condition: Retake get_screenshot on the Figma root node. Place side-by-side with your latest Chrome screenshot. Compare colors directly. Only move on when both screenshots match AND numerical audit shows zero mismatches.
Icon & Shape Verification
If an icon looks wrong during any loop iteration:
get_screenshot(iconNodeId, fileKey)on the Figma icon- Screenshot Chrome icon at 3x zoom
- Compare the SILHOUETTE â stroke count, shape, proportions
Common mismatches:
- “filter” â often lines-with-circles, NOT a funnel
- “calendar” â many variants (with/without dots)
- “mail” â open vs closed envelope
If it doesn’t match: try another icon from the library, or use the Figma SVG inline.
Anti-Pattern: “Looks Good Enough”
This ALWAYS misses: 2-4px spacing differences, wrong icon variant, slightly wrong color, missing shadow, font weight mismatch.
RULE: Not done until zero visual mismatches AND zero numerical mismatches.
Recovery: Compact & Resume
If you hit the image limit (or proactively after ~15 screenshots):
- Run
/compactâ summarizes conversation, drops old images. Notes and progress are preserved. - Retake 3 reference images: Figma overview, Figma detail (current section), Chrome state.
- Continue the refinement loop â you still know everything from the compacted history. ~17 more image slots available.
Repeatable: compact â retake 3 â continue â compact again if needed.
Phase 4: User Review
Stop and ask:
- “Here’s the final result. Are you happy with it?”
- “If something looks off, paste a Figma ‘Link to Selection’ for the specific area.”
If the user provides a new link, re-run Phases 1-3 scoped to that selection.
Doctor Check
Run this ONLY when something fails. Not at startup.
Figma MCP not working?
- Call
whoamito check authentication - Not connected â alert user to configure Figma MCP
- Not authenticated â guide through Figma OAuth
Chrome not responding?
- Ensure Claude Chrome extension is active
- Navigate to the correct localhost page
Dev server not running?
- Check lockfile:
package-lock.json(npm),yarn.lock(yarn),pnpm-lock.yaml(pnpm),bun.lockb(bun) - Read
package.jsonfor dev command and port - Default ports: 5173 (Vite), 3000 (Next/CRA), 8080 (Vue CLI)
lsof -i :<PORT>â if not running, start in background
Unknown design system or icon library?
- Scan
package.json:tailwindcss,styled-components,@emotion/*,@chakra-ui/*,@mui/*,@mantine/* - Icon libraries:
lucide-react,@heroicons/react,react-icons,@radix-ui/react-icons,@fortawesome/*,@phosphor-icons/react,@tabler/icons-react
Examples
Invocation:
/pix
> Paste Figma link: https://figma.com/design/abc123/MyApp?node-id=42-100
What happens:
- Recon:
get_metadata+get_variable_defs+get_code_connect_map(0 images) - Figma overview:
get_screenshoton root (image 1) - Study section 1:
get_screenshot(image 2) +get_design_context(text). Memorize. - Study section 2:
get_screenshot(image 3) +get_design_context(text). Memorize. - Code everything from memory (0 images)
- Loop round 1: Chrome screenshot (image 4) + audit â 4 mismatches + 1 bad icon
- Fix mismatches. Icon check:
get_screenshoton Figma icon (image 5) â swap it. - Loop round 2: Chrome screenshot (image 6) + audit â 1 spacing issue
- Fix spacing.
- Loop round 3: Chrome screenshot (image 7) + audit â zero mismatches.
- Final check: Retake Figma
get_screenshot(image 8) side-by-side with Chrome â match. - Done. Total: 8 images.
Bad patterns:
- Screenshotting every element individually (budget killer)
- Re-screenshotting the same Figma node (it’s static)
- Chrome screenshots after every small CSS tweak
- Using screenshots to verify what
getComputedStylegives for free - Not running
/compactwhen approaching image limit get_design_contexton root (token waste)get_variable_defsmore than once (redundant)- Hardcoded hex in Tailwind (
text-[#f3f3f3]) - Magic width numbers (
w-[247px]) - Saying “looks close” without numerical verification