Loading...
Loading...
Chrome extension that extracts design systems from websites and generates DESIGN.md or SKILL.md files for AI coding agents using TypeUI format
npx skill4agent add aradotso/design-skills design-md-chrome-extractorSkill by ara.so — Design Skills collection.
DESIGN.mdSKILL.mdgit clone https://github.com/bergside/design-md-chrome.git
cd design-md-chromechrome://extensionsdesign-md-chromehttps://stripe.comcontent.js// Core extraction happens via message passing
chrome.runtime.sendMessage({
action: 'extract',
url: window.location.href
}, (response) => {
console.log('Extracted tokens:', response.tokens);
});{
typography: {
fontFamilies: ['Inter', 'system-ui', 'sans-serif'],
fontSizes: ['12px', '14px', '16px', '20px', '24px', '32px'],
fontWeights: ['400', '500', '600', '700'],
lineHeights: ['1.2', '1.5', '1.6']
},
colors: {
primary: ['#635BFF', '#0A2540'],
neutral: ['#FFFFFF', '#F6F9FC', '#E3E8EE'],
semantic: {
success: '#00D924',
error: '#DF1B41',
warning: '#FFC043'
}
},
spacing: ['4px', '8px', '12px', '16px', '24px', '32px', '48px', '64px'],
radius: ['0px', '4px', '8px', '12px', '16px', '9999px'],
shadows: [
'0 1px 3px rgba(0,0,0,0.1)',
'0 4px 6px rgba(0,0,0,0.1)',
'0 10px 40px rgba(0,0,0,0.15)'
],
motion: {
durations: ['150ms', '200ms', '300ms', '500ms'],
easings: ['ease', 'ease-in-out', 'cubic-bezier(0.4,0,0.2,1)']
}
}# [Site Name] Design System
## Mission
Define and maintain consistent visual language for [product].
## Brand
- **URL**: https://example.com
- **Audience**: [Developers/Designers/General]
- **Surface**: Web application
## Style Foundations
### Typography
- Font families: Inter, system-ui
- Sizes: 12px, 14px, 16px, 20px, 24px, 32px
- Weights: 400 (regular), 500 (medium), 600 (semibold), 700 (bold)
- Line heights: 1.2 (tight), 1.5 (base), 1.6 (relaxed)
### Colors
**Primary**
- `#635BFF` - Brand primary
- `#0A2540` - Dark navy
**Neutral**
- `#FFFFFF` - White
- `#F6F9FC` - Light gray
- `#E3E8EE` - Medium gray
### Spacing
Scale: 4px, 8px, 12px, 16px, 24px, 32px, 48px, 64px
### Radius
Values: 0px, 4px, 8px, 12px, 16px, 9999px (pill)
## Accessibility
- WCAG 2.2 AA compliance required
- Minimum contrast ratio 4.5:1 for text
- Focus indicators required on all interactive elements{
"manifest_version": 3,
"name": "DESIGN.MD Style Extractor",
"version": "1.0.0",
"permissions": ["activeTab", "scripting"],
"action": {
"default_popup": "popup.html"
},
"content_scripts": [{
"matches": ["<all_urls>"],
"js": ["content.js"]
}]
}// popup.js
chrome.tabs.query({active: true, currentWindow: true}, (tabs) => {
chrome.tabs.sendMessage(tabs[0].id, {
action: 'extractStyles'
}, (response) => {
displayTokens(response.tokens);
});
});// content.js
chrome.runtime.onMessage.addListener((request, sender, sendResponse) => {
if (request.action === 'extractStyles') {
const tokens = extractDesignTokens();
sendResponse({tokens});
}
});| Action | Description | Response |
|---|---|---|
| Scan page for design tokens | |
| Create DESIGN.md content | |
| Create SKILL.md content | |
| Re-run extraction | |
| Trigger file download | |
// Extract all computed styles from page
function extractDesignTokens() {
const allElements = document.querySelectorAll('*');
const tokens = {
typography: new Set(),
colors: new Set(),
spacing: new Set(),
radius: new Set(),
shadows: new Set()
};
allElements.forEach(el => {
const computed = window.getComputedStyle(el);
// Extract typography
tokens.typography.add(computed.fontFamily);
tokens.typography.add(computed.fontSize);
tokens.typography.add(computed.fontWeight);
// Extract colors
if (computed.color !== 'rgba(0, 0, 0, 0)') {
tokens.colors.add(computed.color);
}
if (computed.backgroundColor !== 'rgba(0, 0, 0, 0)') {
tokens.colors.add(computed.backgroundColor);
}
// Extract spacing
['margin', 'padding'].forEach(prop => {
['Top', 'Right', 'Bottom', 'Left'].forEach(side => {
const value = computed[prop + side];
if (value !== '0px') tokens.spacing.add(value);
});
});
// Extract border radius
if (computed.borderRadius !== '0px') {
tokens.radius.add(computed.borderRadius);
}
// Extract box shadows
if (computed.boxShadow !== 'none') {
tokens.shadows.add(computed.boxShadow);
}
});
return normalizeTokens(tokens);
}
// Normalize and deduplicate tokens
function normalizeTokens(rawTokens) {
return {
typography: {
fontFamilies: Array.from(rawTokens.typography).filter(isFontFamily),
fontSizes: Array.from(rawTokens.typography).filter(isFontSize).sort(),
fontWeights: Array.from(rawTokens.typography).filter(isFontWeight).sort()
},
colors: Array.from(rawTokens.colors).map(normalizeColor),
spacing: Array.from(rawTokens.spacing).sort(numericalSort),
radius: Array.from(rawTokens.radius).sort(numericalSort),
shadows: Array.from(rawTokens.shadows)
};
}node tests/run-tests.mjsasync function extractAndSave(format = 'DESIGN') {
// 1. Extract tokens
const tokens = await extractDesignTokens();
// 2. Generate markdown
const markdown = format === 'DESIGN'
? generateDesignMD(tokens)
: generateSkillMD(tokens);
// 3. Download file
const blob = new Blob([markdown], {type: 'text/markdown'});
const url = URL.createObjectURL(blob);
const a = document.createElement('a');
a.href = url;
a.download = `${format}.md`;
a.click();
}// Filter extracted colors by frequency
function filterFrequentColors(colors, minOccurrences = 5) {
const colorMap = new Map();
colors.forEach(color => {
const normalized = normalizeColor(color);
colorMap.set(normalized, (colorMap.get(normalized) || 0) + 1);
});
return Array.from(colorMap.entries())
.filter(([_, count]) => count >= minOccurrences)
.map(([color, _]) => color);
}// Combine tokens from multiple pages
function mergeTokens(tokensArray) {
return {
typography: {
fontFamilies: [...new Set(tokensArray.flatMap(t => t.typography.fontFamilies))],
fontSizes: [...new Set(tokensArray.flatMap(t => t.typography.fontSizes))].sort(),
fontWeights: [...new Set(tokensArray.flatMap(t => t.typography.fontWeights))].sort()
},
colors: [...new Set(tokensArray.flatMap(t => t.colors))],
spacing: [...new Set(tokensArray.flatMap(t => t.spacing))].sort(numericalSort),
radius: [...new Set(tokensArray.flatMap(t => t.radius))].sort(numericalSort),
shadows: [...new Set(tokensArray.flatMap(t => t.shadows))]
};
}chrome://extensions// Ensure proper blob MIME type
const blob = new Blob([content], {type: 'text/markdown;charset=utf-8'});
// Add fallback for different browsers
if (navigator.msSaveBlob) {
navigator.msSaveBlob(blob, filename);
} else {
const url = URL.createObjectURL(blob);
const a = document.createElement('a');
a.href = url;
a.download = filename;
document.body.appendChild(a);
a.click();
document.body.removeChild(a);
URL.revokeObjectURL(url);
}16px16.5px// Round to nearest whole pixel
function roundPixels(value) {
const num = parseFloat(value);
return Math.round(num) + 'px';
}
// Apply tolerance-based deduplication
function deduplicateWithTolerance(values, tolerance = 1) {
return values.reduce((acc, val) => {
const num = parseFloat(val);
if (!acc.some(existing => Math.abs(parseFloat(existing) - num) < tolerance)) {
acc.push(val);
}
return acc;
}, []);
}<all_urls>function extractFromIframes() {
const iframes = document.querySelectorAll('iframe');
iframes.forEach(iframe => {
try {
const iframeDoc = iframe.contentDocument || iframe.contentWindow.document;
// Extract from iframe document
extractDesignTokens(iframeDoc);
} catch (e) {
console.warn('Cannot access iframe:', e);
}
});
}# Add to project context
cp DESIGN.md .claude/
# Reference in prompts
"Build a button component following DESIGN.md spacing and color tokens"# Add to .cursorrules
cp DESIGN.md .cursorrules/design-system.md
# Reference in code
// Uses spacing-4 (16px) from DESIGN.md
<div className="p-4">SKILL.md