Use this skill when the user asks to "generate dark mode", "create dark theme", "add dark mode support", "convert to dark mode colors", "generate dark color palette", or wants to automatically generate dark mode variants for their components with intelligent color inversion and accessibility preservation.
/plugin marketplace add flight505/storybook-assistant-plugin/plugin install flight505-storybook-assistant@flight505/storybook-assistant-pluginThis skill inherits all available tools. When active, it can use any tool Claude has access to.
Automatically generate dark mode color schemes from light mode designs with intelligent color inversion, contrast preservation, and accessibility compliance (WCAG 2.2).
Input: Light mode component Output: Dark mode CSS variables, color mappings, theme toggle, updated stories
Extract all colors from component:
// Detected colors
Background: #FFFFFF
Text: #1F2937 (gray-800)
Primary: #2196F3 (blue-500)
Border: #E5E7EB (gray-200)
Shadow: rgba(0, 0, 0, 0.1)
Apply intelligent transformations:
/* Light Mode */
--bg-primary: #FFFFFF; /* white */
--text-primary: #1F2937; /* dark gray */
--color-primary: #2196F3; /* blue */
--border: #E5E7EB; /* light gray */
/* Dark Mode (Generated) */
--bg-primary: #1F2937; /* dark gray (inverted) */
--text-primary: #F9FAFB; /* off-white (inverted) */
--color-primary: #60A5FA; /* lighter blue (adjusted for contrast) */
--border: #374151; /* medium gray (adjusted) */
Rules:
Ensure WCAG compliance maintained:
Light Mode: #1F2937 on #FFFFFF = 16.1:1 (AAA) ✓
Dark Mode: #F9FAFB on #1F2937 = 15.8:1 (AAA) ✓
Light Mode: #2196F3 on #FFFFFF = 3.1:1 (AA Large) ✓
Dark Mode: #60A5FA on #1F2937 = 4.7:1 (AA Normal) ✓ (Improved!)
Create complete dark mode implementation:
CSS Variables:
/* themes/colors.css */
:root {
--bg-primary: #FFFFFF;
--bg-secondary: #F9FAFB;
--text-primary: #1F2937;
--text-secondary: #6B7280;
--color-primary: #2196F3;
--border: #E5E7EB;
--shadow: rgba(0, 0, 0, 0.1);
}
[data-theme="dark"] {
--bg-primary: #1F2937;
--bg-secondary: #111827;
--text-primary: #F9FAFB;
--text-secondary: #D1D5DB;
--color-primary: #60A5FA;
--border: #374151;
--shadow: rgba(0, 0, 0, 0.3);
}
Theme Toggle Component:
// components/ThemeToggle.tsx
'use client';
import { useEffect, useState } from 'react';
import { Moon, Sun } from 'lucide-react';
export function ThemeToggle() {
const [theme, setTheme] = useState<'light' | 'dark'>('light');
useEffect(() => {
const saved = localStorage.getItem('theme') || 'light';
setTheme(saved as 'light' | 'dark');
document.documentElement.setAttribute('data-theme', saved);
}, []);
const toggleTheme = () => {
const newTheme = theme === 'light' ? 'dark' : 'light';
setTheme(newTheme);
document.documentElement.setAttribute('data-theme', newTheme);
localStorage.setItem('theme', newTheme);
};
return (
<button
onClick={toggleTheme}
aria-label={`Switch to ${theme === 'light' ? 'dark' : 'light'} mode`}
className="p-2 rounded-lg hover:bg-gray-100 dark:hover:bg-gray-800"
>
{theme === 'light' ? <Moon size={20} /> : <Sun size={20} />}
</button>
);
}
Add dark mode variants:
// Button.stories.tsx
export const LightMode: Story = {
parameters: {
backgrounds: { default: 'light' },
theme: 'light',
},
};
export const DarkMode: Story = {
parameters: {
backgrounds: { default: 'dark' },
theme: 'dark',
},
decorators: [
(Story) => (
<div data-theme="dark">
<Story />
</div>
),
],
};
export const AllThemes: Story = {
render: () => (
<div className="grid grid-cols-2 gap-4">
<div data-theme="light" className="p-4 bg-white">
<h3>Light Mode</h3>
<Button variant="primary">Click me</Button>
</div>
<div data-theme="dark" className="p-4 bg-gray-900">
<h3>Dark Mode</h3>
<Button variant="primary">Click me</Button>
</div>
</div>
),
};
Light → Dark
#FFFFFF → #1F2937 (primary background)
#F9FAFB → #111827 (secondary background)
#F3F4F6 → #0F172A (tertiary background)
Light → Dark
#111827 → #F9FAFB (primary text)
#374151 → #E5E7EB (secondary text)
#6B7280 → #9CA3AF (tertiary text)
Adjust for contrast, not just invert:
Primary Blue:
#2196F3 → #60A5FA (lighter for dark BG)
Success Green:
#10B981 → #34D399 (lighter)
Error Red:
#EF4444 → #F87171 (lighter)
Warning Yellow:
#F59E0B → #FBBF24 (lighter)
#E5E7EB → #374151 (visible on dark)
#D1D5DB → #4B5563 (medium)
#9CA3AF → #6B7280 (subtle)
Light Mode: rgba(0, 0, 0, 0.1)
Dark Mode: rgba(0, 0, 0, 0.3) (darker, more pronounced)
Maintain semantic meaning across themes:
/* Semantic tokens */
:root {
--color-success: #10B981;
--color-warning: #F59E0B;
--color-error: #EF4444;
--color-info: #3B82F6;
}
[data-theme="dark"] {
--color-success: #34D399; /* Lighter green */
--color-warning: #FBBF24; /* Lighter yellow */
--color-error: #F87171; /* Lighter red */
--color-info: #60A5FA; /* Lighter blue */
}
// tailwind.config.js
module.exports = {
darkMode: 'class', // or 'media'
theme: {
extend: {
colors: {
primary: {
light: '#2196F3',
dark: '#60A5FA',
},
},
},
},
};
// Usage
<div className="bg-white dark:bg-gray-900">
<h1 className="text-gray-900 dark:text-gray-100">
/* Component.module.css */
.container {
background: var(--bg-primary);
color: var(--text-primary);
}
[data-theme="dark"] .container {
/* Inherits CSS variables */
}
Automatic dark mode generation:
Result: Production-ready dark mode in minutes, not hours.