Agent skill
webflow-code-components
Expert guidance for developing React code components for Webflow. This skill should be used when users are building, troubleshooting, or optimizing React components for import into Webflow Designer, working with Shadow DOM styling, managing component communication, or configuring the Webflow CLI.
Install this agent skill to your Project
npx add-skill https://github.com/majiayu000/claude-skill-registry/tree/main/skills/development/webflow-code-components
SKILL.md
Webflow Code Components Developer
This skill provides expert guidance for building React code components that integrate with Webflow Designer. It covers component architecture, prop configuration, styling within Shadow DOM, state management across isolated components, and CLI workflows.
When to Use This Skill
Trigger this skill when users are:
- Setting up a new Webflow code component project
- Creating or declaring React components for Webflow
- Configuring props with Webflow data types
- Styling components within Shadow DOM constraints
- Implementing component communication patterns
- Fetching external data in components
- Troubleshooting CLI import issues or component rendering problems
- Optimizing component performance or bundle size
- Working with wrapper components for complex prop types
- Using Webflow hooks like
useWebflowContext - Managing the Webflow site using Designer MCP tools (if available)
- Adding or editing content on Webflow pages programmatically
Core Concepts
Architecture Overview
Code components run as isolated React applications in Shadow DOM:
- Each component instance has its own React root
- Shadow DOM creates style and DOM boundaries
- Components cannot share state through React Context
- SSR is enabled by default but can be disabled
Key constraint: React Context, Redux, Zustand, and similar state management libraries do NOT work across separate component instances. Use alternative communication patterns instead.
Project Structure
project/
├── .env # Workspace API token (gitignored)
├── webflow.json # CLI configuration
├── webpack.webflow.js # Optional webpack overrides
└── src/
└── components/
└── ComponentName/
├── ComponentName.tsx # React component
├── ComponentName.module.css # Styles
└── ComponentName.webflow.tsx # Webflow declaration
Setup Workflow
Initial Setup
-
Install dependencies:
bashnpm i --save-dev @webflow/webflow-cli @webflow/react @webflow/data-types -
Create
webflow.json:json{ "library": { "name": "Component Library Name", "components": ["./src/**/*.webflow.@(js|jsx|mjs|ts|tsx)"] } } -
Authenticate:
- Run
npx webflow library sharefor interactive auth, OR - Get workspace token from Webflow → Apps & Integrations → Workspace API Access
- Add to
.env:WEBFLOW_WORKSPACE_API_TOKEN=your_token
- Run
Component Creation Workflow
- Create React component (
ComponentName.tsx) - Add styles (
ComponentName.module.css) - Create Webflow declaration (
ComponentName.webflow.tsx) - Import to Webflow:
npx webflow library share
Component Declaration
Always use declareComponent in .webflow.tsx files:
import { declareComponent } from '@webflow/react';
import { props } from '@webflow/data-types';
import { Component } from './Component';
import './Component.module.css'; // ← Import styles HERE
export default declareComponent(Component, {
name: 'Component Name',
description: 'Brief description',
group: 'Category', // Optional: 'Interactive', 'Content', 'Layout', etc.
props: {
propName: props.Text({
name: 'Display Name',
defaultValue: 'default value',
group: 'Prop Group', // Optional
tooltip: 'Help text' // Optional
}),
},
options: {
applyTagSelectors: false, // Default: false
ssr: true, // Default: true
},
});
Critical: Always import styles in .webflow.tsx, not just in .tsx.
Prop Types Reference
Quick reference for available prop types:
| Type | Returns | Use For |
|---|---|---|
props.Text() |
string |
Single-line text |
props.RichText() |
string |
Multi-line text |
props.TextNode() |
string |
Canvas-editable text |
props.Link() |
{ href, target, preload } |
URLs (requires wrapper) |
props.Image() |
string |
Image URLs |
props.Number() |
number |
Numeric values |
props.Boolean() |
boolean |
Toggles |
props.Variant() |
string |
Dropdown options |
props.Visibility() |
boolean |
Show/hide controls |
props.Slot() |
ReactNode |
Flexible content areas |
props.Color() |
string |
Color picker ⚠️ NOT YET AVAILABLE |
props.ID() |
string |
HTML element IDs ⚠️ NOT YET AVAILABLE |
For detailed configuration options for each prop type, refer to references/prop-types.md.
Note: Some prop types are documented but not yet available in the current Webflow release. Do not use prop types marked as "NOT YET AVAILABLE".
Wrapper Components for Link Props
props.Link() returns an object, but React components often expect separate props. Use wrapper components:
// Component.webflow.tsx
import { props, PropType, PropValues } from "@webflow/data-types";
import { Component, ComponentProps } from "./Component";
type WebflowProps = {
link: PropValues[PropType.Link];
} & Omit<ComponentProps, "href" | "target">;
const WebflowComponent = ({ link: { href, target }, ...rest }: WebflowProps) => {
return <Component href={href} target={target} {...rest} />;
};
export default declareComponent(WebflowComponent, {
// ... configuration
});
For complete wrapper patterns and other complex transformations, refer to references/wrapper-components.md.
Styling in Shadow DOM
Key Principles
- Import in
.webflow.tsx- Styles must be imported in declaration file - Use CSS Modules - Prevent global conflicts
- Use site variables - Connect to Webflow's design system:
var(--variable-name, fallback) - Always provide fallbacks -
var(--colors-primary, #007bff) - Site classes don't work - Shadow DOM isolation prevents access
- Use inheritance -
font-family: inheritworks across Shadow boundary
Site Variables
Get variable names from Webflow Designer:
- Variables panel → Three dots → Copy CSS
- Use in styles:
background-color: var(--colors-primary, #007bff);
Tag Selectors
Enable site-wide tag styles (h1, p, etc.) with:
declareComponent(Article, {
options: {
applyTagSelectors: true,
},
});
For comprehensive styling strategies, Shadow DOM workarounds, and CSS-in-JS configuration, refer to references/styling.md.
Component Communication
Since React Context doesn't work across component instances, use these patterns:
| Method | Best For | Persistence |
|---|---|---|
| URL Parameters | Filters, search state, navigation | Browser history |
| localStorage | User preferences, settings | Across sessions |
| sessionStorage | Temporary session data | Until tab closes |
| Nano Stores | Complex shared state, reactive updates | In-memory only |
| Custom Events | One-way notifications, broadcasts | Event-based |
Quick Examples
URL Parameters:
const url = new URL(window.location.href);
url.searchParams.set('filter', 'active');
window.history.pushState({}, '', url);
window.dispatchEvent(new Event('urlchange'));
localStorage + Events:
localStorage.setItem('theme', 'dark');
window.dispatchEvent(new CustomEvent('theme-changed', { detail: { theme: 'dark' } }));
Nano Stores:
npm install nanostores @nanostores/react
import { atom } from 'nanostores';
export const $cart = atom<CartItem[]>([]);
For complete communication patterns, type-safe implementations, and complex state sharing examples, refer to references/component-communication.md.
Data Fetching
Key Constraints
- Client-side only - No server-side data fetching
- Public APIs only - Never include API keys (visible to users)
- CORS required - API must allow cross-origin requests
- No environment variables -
.envfiles not supported
Basic Pattern
const [data, setData] = useState(null);
const [loading, setLoading] = useState(true);
const [error, setError] = useState(null);
useEffect(() => {
const controller = new AbortController();
fetch(apiUrl, { signal: controller.signal })
.then(res => res.json())
.then(setData)
.catch(err => {
if (err.name !== 'AbortError') setError(err.message);
})
.finally(() => setLoading(false));
return () => controller.abort();
}, [apiUrl]);
For complete data fetching patterns, caching strategies, POST requests, and error handling, refer to references/data-fetching.md.
Webflow Hooks
useWebflowContext
Access current Webflow environment information:
import { useWebflowContext } from '@webflow/react';
const { mode, interactive, locale } = useWebflowContext();
Returns:
mode- Current Webflow mode (design,publish, etc.)interactive- Whether component should be interactivelocale- User's locale string or null
Common use cases:
- Show expanded states in designer:
defaultExpanded={!interactive} - Disable interactions in designer:
draggable={interactive} - Locale-aware content:
translations[locale] || translations.en - Designer helpers:
if (mode === 'design') { /* show helpful message */ }
For detailed hook usage, mode types, and advanced patterns, refer to references/hooks.md.
SSR Configuration
When to Disable SSR
Disable SSR (ssr: false) when component uses:
- Browser APIs (
window,document,localStorage) - Client-specific data or user context
- Heavy interactive UI (WebGL, Canvas)
- Non-deterministic rendering (random, time-based)
When to Keep SSR Enabled
Keep SSR enabled (default) for:
- Static content (text, images, layouts)
- SEO-critical content
- Performance optimization
- Deterministic rendering
Guarding Browser APIs
If SSR is needed but some browser APIs are used:
useEffect(() => {
if (typeof window !== 'undefined') {
// Browser-only code
setWidth(window.innerWidth);
}
}, []);
For SSR architecture details, hydration, and React Server Components limitations, refer to references/architecture.md.
CLI Commands
Primary Commands
Import components:
npx webflow library share
Bundle locally:
npx webflow library bundle --public-path http://localhost:4000/
View import logs:
npx webflow library log
Common Flags
--verbose- Detailed output for debugging--dev- Development mode (no minification)--no-input- Skip prompts (for CI/CD)--api-token <token>- Pass token directly
For complete CLI reference, webpack configuration, CSS Modules setup, and CI/CD integration, refer to references/cli-reference.md.
Webflow Designer MCP (Optional)
Note: Webflow Designer MCP tools may not be available in all environments. Always check for tool availability before attempting to use them.
Availability Check
// Check if Webflow MCP tools are available
const hasWebflowMCP = typeof mcp__webflow__sites_list === 'function';
if (!hasWebflowMCP) {
// MCP tools not available - use alternative approach
console.log('Webflow Designer MCP not available');
}
What Are Designer MCP Tools?
Model Context Protocol (MCP) tools for Webflow Designer allow you to programmatically:
- Access and navigate Webflow sites and pages
- Create and modify page elements
- Manage CMS collections and content
- Publish site changes
- Add content to component showcase pages
Use cases:
- Adding descriptive content to component demo pages
- Updating page content programmatically
- Managing CMS data for components
- Bulk content operations across pages
Connection Workflow
-
Check tool availability (see above)
-
Get site information:
typescriptconst { sites } = await mcp__webflow__sites_list(); const site = sites.find(s => s.displayName === 'My Site'); -
Provide connection link to user:
markdownClick this link to connect: [Launch Webflow Designer](https://{shortName}.design.webflow.com?app={appId}) -
Wait for user confirmation that Designer is connected
-
Navigate and make changes:
typescript// Switch to page await mcp__webflow__de_page_tool({ siteId: site.id, actions: [{ switch_page: { page_id: pageId } }] }); // Ask user to select target element // User MUST manually select in Designer // Create content await mcp__webflow__element_builder({ siteId: site.id, actions: [{ parent_element_id: selectedElementId, creation_position: "append", element_schema: { type: "Paragraph", set_text: { text: "Content here" } } }] });
Critical Requirements
User must manually select elements:
- For operations that target specific elements, user must select them in Designer first
- Always ask user to select element, then confirm before proceeding
- Example: "Please select the
section-inner--aboutsection, then let me know when ready"
Connection can drop:
- Long operations may lose connection
- Ask user to reconnect if errors occur
- Provide fresh connection link when needed
Element creation is sequential:
- Each element operation is separate
- Creating multiple elements takes time
- Batch operations in logical groups
Quick Reference
Common tasks:
- List sites:
mcp__webflow__sites_list() - List pages:
mcp__webflow__pages_list({ site_id }) - Switch page:
mcp__webflow__de_page_tool({ siteId, actions: [{ switch_page }] }) - Get selected:
mcp__webflow__element_tool({ siteId, actions: [{ get_selected_element }] }) - Create element:
mcp__webflow__element_builder({ siteId, actions: [...] }) - Publish:
mcp__webflow__sites_publish({ site_id })
For complete Designer MCP documentation, patterns, troubleshooting, and API reference, refer to references/webflow-designer-mcp.md.
Troubleshooting Quick Reference
Components Don't Appear
- Check glob pattern in
webflow.jsonmatches files - Verify
.webflow.tsxextension - Look for compilation errors in terminal
- Refresh Webflow Designer
Styles Not Showing
- Import styles in
.webflow.tsx(not just.tsx) - Verify CSS Module import:
import styles from './Component.module.css' - Check Shadow DOM in browser DevTools
Authentication Fails
- Verify token in
.env - Ensure you're Workspace Admin
- Try manual auth:
--api-tokenflag
State Not Shared
- React Context doesn't work across components
- Use URL params, localStorage, Nano Stores, or Custom Events
- Refer to component communication patterns
API Requests Fail
- Check CORS headers on API
- Verify public endpoint (no authentication required)
- Test API in browser console
For comprehensive troubleshooting, error messages, and solutions, refer to references/troubleshooting.md.
Best Practices Summary
Component Design
- Keep components focused on single responsibility
- Use TypeScript for type safety
- Provide sensible defaults for all props
- Handle edge cases (empty states, errors, loading)
Prop Configuration
- Use descriptive prop names
- Group related props together
- Add helpful tooltips for complex props
- Use appropriate prop types for data
Styling
- Use site variables with fallbacks
- Use CSS Modules for scoped styles
- Respect
prefers-reduced-motion - Mobile-first responsive design
Performance
- Minimize bundle size (target < 10MB)
- Use
React.memofor expensive components - Clean up side effects in
useEffect - Lazy load heavy features
Data & State
- Handle all states (loading, error, empty, success)
- Use AbortController for fetch cleanup
- Validate API responses
- Never hardcode secrets or API keys
For complete best practices, code examples, and detailed patterns, refer to references/best-practices.md.
Documentation Reference Map
Use these references for detailed information:
references/getting-started.md- Setup, installation, first componentreferences/component-declaration.md- CompletedeclareComponentAPIreferences/prop-types.md- All prop types with full configurationreferences/wrapper-components.md- Transform complex prop typesreferences/hooks.md-useWebflowContextpatterns and examplesreferences/architecture.md- Shadow DOM, React roots, SSR detailsreferences/styling.md- CSS strategies, Shadow DOM, site variablesreferences/component-communication.md- State sharing patternsreferences/data-fetching.md- API integration, caching, requestsreferences/cli-reference.md- All CLI commands and optionsreferences/webflow-designer-mcp.md- Webflow Designer MCP tools (if available)references/best-practices.md- Comprehensive recommendationsreferences/troubleshooting.md- Common issues and solutions
Implementation Workflow
When helping users develop components:
- Understand requirements - Clarify component purpose and props needed
- Choose appropriate patterns - Select prop types, styling approach, communication method
- Implement incrementally - Start with basic structure, add features progressively
- Reference documentation - Load relevant reference files as needed for detailed guidance
- Test and troubleshoot - Use CLI to bundle and import, fix issues iteratively
- Optimize - Review bundle size, performance, accessibility
Context Management
Progressive disclosure: Load reference files only when needed for specific questions:
- Basic setup → Load
references/getting-started.md - Prop configuration → Load
references/prop-types.md - Styling issues → Load
references/styling.md - Component communication → Load
references/component-communication.md - Troubleshooting → Load
references/troubleshooting.md
This keeps context efficient while providing comprehensive guidance when required.
Didn't find tool you were looking for?