ui-ux-pro-max-cli 2.8.8
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +99 -0
- package/assets/data/_sync_all.py +414 -0
- package/assets/data/app-interface.csv +31 -0
- package/assets/data/charts.csv +26 -0
- package/assets/data/colors.csv +162 -0
- package/assets/data/design.csv +1776 -0
- package/assets/data/draft.csv +1779 -0
- package/assets/data/google-fonts.csv +1924 -0
- package/assets/data/icons.csv +106 -0
- package/assets/data/landing.csv +35 -0
- package/assets/data/products.csv +162 -0
- package/assets/data/react-performance.csv +45 -0
- package/assets/data/stacks/angular.csv +51 -0
- package/assets/data/stacks/astro.csv +54 -0
- package/assets/data/stacks/flutter.csv +53 -0
- package/assets/data/stacks/html-tailwind.csv +56 -0
- package/assets/data/stacks/javafx.csv +76 -0
- package/assets/data/stacks/jetpack-compose.csv +53 -0
- package/assets/data/stacks/laravel.csv +51 -0
- package/assets/data/stacks/nextjs.csv +53 -0
- package/assets/data/stacks/nuxt-ui.csv +71 -0
- package/assets/data/stacks/nuxtjs.csv +59 -0
- package/assets/data/stacks/react-native.csv +52 -0
- package/assets/data/stacks/react.csv +54 -0
- package/assets/data/stacks/shadcn.csv +61 -0
- package/assets/data/stacks/svelte.csv +54 -0
- package/assets/data/stacks/swiftui.csv +51 -0
- package/assets/data/stacks/threejs.csv +54 -0
- package/assets/data/stacks/vue.csv +50 -0
- package/assets/data/styles.csv +85 -0
- package/assets/data/typography.csv +74 -0
- package/assets/data/ui-reasoning.csv +162 -0
- package/assets/data/ux-guidelines.csv +100 -0
- package/assets/scripts/core.py +263 -0
- package/assets/scripts/design_system.py +1157 -0
- package/assets/scripts/search.py +114 -0
- package/assets/skills/banner-design/SKILL.md +196 -0
- package/assets/skills/banner-design/references/banner-sizes-and-styles.md +118 -0
- package/assets/skills/brand/SKILL.md +97 -0
- package/assets/skills/brand/references/approval-checklist.md +169 -0
- package/assets/skills/brand/references/asset-organization.md +157 -0
- package/assets/skills/brand/references/brand-guideline-template.md +140 -0
- package/assets/skills/brand/references/color-palette-management.md +186 -0
- package/assets/skills/brand/references/consistency-checklist.md +94 -0
- package/assets/skills/brand/references/logo-usage-rules.md +185 -0
- package/assets/skills/brand/references/messaging-framework.md +85 -0
- package/assets/skills/brand/references/typography-specifications.md +214 -0
- package/assets/skills/brand/references/update.md +118 -0
- package/assets/skills/brand/references/visual-identity.md +96 -0
- package/assets/skills/brand/references/voice-framework.md +88 -0
- package/assets/skills/brand/scripts/extract-colors.cjs +341 -0
- package/assets/skills/brand/scripts/inject-brand-context.cjs +349 -0
- package/assets/skills/brand/scripts/sync-brand-to-tokens.cjs +266 -0
- package/assets/skills/brand/scripts/validate-asset.cjs +387 -0
- package/assets/skills/brand/templates/brand-guidelines-starter.md +275 -0
- package/assets/skills/design/SKILL.md +313 -0
- package/assets/skills/design/data/cip/deliverables.csv +51 -0
- package/assets/skills/design/data/cip/industries.csv +21 -0
- package/assets/skills/design/data/cip/mockup-contexts.csv +21 -0
- package/assets/skills/design/data/cip/styles.csv +21 -0
- package/assets/skills/design/data/icon/styles.csv +16 -0
- package/assets/skills/design/data/logo/colors.csv +56 -0
- package/assets/skills/design/data/logo/industries.csv +56 -0
- package/assets/skills/design/data/logo/styles.csv +56 -0
- package/assets/skills/design/references/banner-sizes-and-styles.md +118 -0
- package/assets/skills/design/references/cip-deliverable-guide.md +95 -0
- package/assets/skills/design/references/cip-design.md +121 -0
- package/assets/skills/design/references/cip-prompt-engineering.md +84 -0
- package/assets/skills/design/references/cip-style-guide.md +68 -0
- package/assets/skills/design/references/design-routing.md +207 -0
- package/assets/skills/design/references/icon-design.md +122 -0
- package/assets/skills/design/references/logo-color-psychology.md +101 -0
- package/assets/skills/design/references/logo-design.md +92 -0
- package/assets/skills/design/references/logo-prompt-engineering.md +158 -0
- package/assets/skills/design/references/logo-style-guide.md +109 -0
- package/assets/skills/design/references/slides-copywriting-formulas.md +84 -0
- package/assets/skills/design/references/slides-create.md +4 -0
- package/assets/skills/design/references/slides-html-template.md +295 -0
- package/assets/skills/design/references/slides-layout-patterns.md +137 -0
- package/assets/skills/design/references/slides-strategies.md +94 -0
- package/assets/skills/design/references/slides.md +42 -0
- package/assets/skills/design/references/social-photos-design.md +329 -0
- package/assets/skills/design/scripts/cip/core.py +215 -0
- package/assets/skills/design/scripts/cip/generate.py +484 -0
- package/assets/skills/design/scripts/cip/render-html.py +424 -0
- package/assets/skills/design/scripts/cip/search.py +127 -0
- package/assets/skills/design/scripts/icon/generate.py +487 -0
- package/assets/skills/design/scripts/logo/core.py +175 -0
- package/assets/skills/design/scripts/logo/generate.py +362 -0
- package/assets/skills/design/scripts/logo/search.py +114 -0
- package/assets/skills/design-system/SKILL.md +244 -0
- package/assets/skills/design-system/data/slide-backgrounds.csv +11 -0
- package/assets/skills/design-system/data/slide-charts.csv +26 -0
- package/assets/skills/design-system/data/slide-color-logic.csv +14 -0
- package/assets/skills/design-system/data/slide-copy.csv +26 -0
- package/assets/skills/design-system/data/slide-layout-logic.csv +16 -0
- package/assets/skills/design-system/data/slide-layouts.csv +26 -0
- package/assets/skills/design-system/data/slide-strategies.csv +16 -0
- package/assets/skills/design-system/data/slide-typography.csv +15 -0
- package/assets/skills/design-system/references/component-specs.md +236 -0
- package/assets/skills/design-system/references/component-tokens.md +214 -0
- package/assets/skills/design-system/references/primitive-tokens.md +203 -0
- package/assets/skills/design-system/references/semantic-tokens.md +215 -0
- package/assets/skills/design-system/references/states-and-variants.md +241 -0
- package/assets/skills/design-system/references/tailwind-integration.md +251 -0
- package/assets/skills/design-system/references/token-architecture.md +224 -0
- package/assets/skills/design-system/scripts/embed-tokens.cjs +99 -0
- package/assets/skills/design-system/scripts/fetch-background.py +317 -0
- package/assets/skills/design-system/scripts/generate-slide.py +770 -0
- package/assets/skills/design-system/scripts/generate-tokens.cjs +205 -0
- package/assets/skills/design-system/scripts/html-token-validator.py +327 -0
- package/assets/skills/design-system/scripts/search-slides.py +218 -0
- package/assets/skills/design-system/scripts/slide-token-validator.py +35 -0
- package/assets/skills/design-system/scripts/slide_search_core.py +453 -0
- package/assets/skills/design-system/scripts/validate-tokens.cjs +251 -0
- package/assets/skills/design-system/templates/design-tokens-starter.json +143 -0
- package/assets/skills/slides/SKILL.md +40 -0
- package/assets/skills/slides/references/copywriting-formulas.md +84 -0
- package/assets/skills/slides/references/create.md +4 -0
- package/assets/skills/slides/references/html-template.md +295 -0
- package/assets/skills/slides/references/layout-patterns.md +137 -0
- package/assets/skills/slides/references/slide-strategies.md +94 -0
- package/assets/skills/ui-styling/LICENSE.txt +202 -0
- package/assets/skills/ui-styling/SKILL.md +324 -0
- package/assets/skills/ui-styling/references/canvas-design-system.md +320 -0
- package/assets/skills/ui-styling/references/shadcn-accessibility.md +471 -0
- package/assets/skills/ui-styling/references/shadcn-components.md +424 -0
- package/assets/skills/ui-styling/references/shadcn-theming.md +373 -0
- package/assets/skills/ui-styling/references/tailwind-customization.md +483 -0
- package/assets/skills/ui-styling/references/tailwind-responsive.md +382 -0
- package/assets/skills/ui-styling/references/tailwind-utilities.md +455 -0
- package/assets/skills/ui-styling/scripts/requirements.txt +17 -0
- package/assets/skills/ui-styling/scripts/shadcn_add.py +308 -0
- package/assets/skills/ui-styling/scripts/tailwind_config_gen.py +473 -0
- package/assets/skills/ui-styling/scripts/tests/coverage-ui.json +1 -0
- package/assets/skills/ui-styling/scripts/tests/requirements.txt +3 -0
- package/assets/skills/ui-styling/scripts/tests/test_shadcn_add.py +266 -0
- package/assets/skills/ui-styling/scripts/tests/test_tailwind_config_gen.py +336 -0
- package/assets/templates/base/quick-reference.md +297 -0
- package/assets/templates/base/skill-content.md +368 -0
- package/assets/templates/platforms/agent.json +21 -0
- package/assets/templates/platforms/augment.json +18 -0
- package/assets/templates/platforms/claude.json +21 -0
- package/assets/templates/platforms/codebuddy.json +21 -0
- package/assets/templates/platforms/codex.json +21 -0
- package/assets/templates/platforms/continue.json +21 -0
- package/assets/templates/platforms/copilot.json +21 -0
- package/assets/templates/platforms/cursor.json +21 -0
- package/assets/templates/platforms/droid.json +21 -0
- package/assets/templates/platforms/gemini.json +21 -0
- package/assets/templates/platforms/kilocode.json +21 -0
- package/assets/templates/platforms/kiro.json +21 -0
- package/assets/templates/platforms/opencode.json +21 -0
- package/assets/templates/platforms/qoder.json +21 -0
- package/assets/templates/platforms/roocode.json +21 -0
- package/assets/templates/platforms/trae.json +21 -0
- package/assets/templates/platforms/warp.json +18 -0
- package/assets/templates/platforms/windsurf.json +21 -0
- package/dist/index.js +10630 -0
- package/package.json +51 -0
|
@@ -0,0 +1,224 @@
|
|
|
1
|
+
# Token Architecture
|
|
2
|
+
|
|
3
|
+
Three-layer token system for scalable, themeable design systems.
|
|
4
|
+
|
|
5
|
+
## Layer Overview
|
|
6
|
+
|
|
7
|
+
```
|
|
8
|
+
┌─────────────────────────────────────────┐
|
|
9
|
+
│ Component Tokens │ Per-component overrides
|
|
10
|
+
│ --button-bg, --card-padding │
|
|
11
|
+
├─────────────────────────────────────────┤
|
|
12
|
+
│ Semantic Tokens │ Purpose-based aliases
|
|
13
|
+
│ --color-primary, --spacing-section │
|
|
14
|
+
├─────────────────────────────────────────┤
|
|
15
|
+
│ Primitive Tokens │ Raw design values
|
|
16
|
+
│ --color-blue-600, --space-4 │
|
|
17
|
+
└─────────────────────────────────────────┘
|
|
18
|
+
```
|
|
19
|
+
|
|
20
|
+
## Why Three Layers?
|
|
21
|
+
|
|
22
|
+
| Layer | Purpose | When to Change |
|
|
23
|
+
|-------|---------|----------------|
|
|
24
|
+
| Primitive | Base values (colors, sizes) | Rarely - foundational |
|
|
25
|
+
| Semantic | Meaning assignment | Theme switching |
|
|
26
|
+
| Component | Component customization | Per-component needs |
|
|
27
|
+
|
|
28
|
+
## Layer 1: Primitive Tokens
|
|
29
|
+
|
|
30
|
+
Raw design values without semantic meaning.
|
|
31
|
+
|
|
32
|
+
```css
|
|
33
|
+
:root {
|
|
34
|
+
/* Colors */
|
|
35
|
+
--color-gray-50: #F9FAFB;
|
|
36
|
+
--color-gray-900: #111827;
|
|
37
|
+
--color-blue-500: #3B82F6;
|
|
38
|
+
--color-blue-600: #2563EB;
|
|
39
|
+
|
|
40
|
+
/* Spacing (4px base) */
|
|
41
|
+
--space-1: 0.25rem; /* 4px */
|
|
42
|
+
--space-2: 0.5rem; /* 8px */
|
|
43
|
+
--space-4: 1rem; /* 16px */
|
|
44
|
+
--space-6: 1.5rem; /* 24px */
|
|
45
|
+
|
|
46
|
+
/* Typography */
|
|
47
|
+
--font-size-sm: 0.875rem;
|
|
48
|
+
--font-size-base: 1rem;
|
|
49
|
+
--font-size-lg: 1.125rem;
|
|
50
|
+
|
|
51
|
+
/* Radius */
|
|
52
|
+
--radius-sm: 0.25rem;
|
|
53
|
+
--radius-default: 0.5rem;
|
|
54
|
+
--radius-lg: 0.75rem;
|
|
55
|
+
|
|
56
|
+
/* Shadows */
|
|
57
|
+
--shadow-sm: 0 1px 2px rgb(0 0 0 / 0.05);
|
|
58
|
+
--shadow-default: 0 1px 3px rgb(0 0 0 / 0.1);
|
|
59
|
+
}
|
|
60
|
+
```
|
|
61
|
+
|
|
62
|
+
## Layer 2: Semantic Tokens
|
|
63
|
+
|
|
64
|
+
Purpose-based aliases that reference primitives.
|
|
65
|
+
|
|
66
|
+
```css
|
|
67
|
+
:root {
|
|
68
|
+
/* Background */
|
|
69
|
+
--color-background: var(--color-gray-50);
|
|
70
|
+
--color-foreground: var(--color-gray-900);
|
|
71
|
+
|
|
72
|
+
/* Primary */
|
|
73
|
+
--color-primary: var(--color-blue-600);
|
|
74
|
+
--color-primary-hover: var(--color-blue-700);
|
|
75
|
+
|
|
76
|
+
/* Secondary */
|
|
77
|
+
--color-secondary: var(--color-gray-100);
|
|
78
|
+
--color-secondary-foreground: var(--color-gray-900);
|
|
79
|
+
|
|
80
|
+
/* Muted */
|
|
81
|
+
--color-muted: var(--color-gray-100);
|
|
82
|
+
--color-muted-foreground: var(--color-gray-500);
|
|
83
|
+
|
|
84
|
+
/* Destructive */
|
|
85
|
+
--color-destructive: var(--color-red-600);
|
|
86
|
+
--color-destructive-foreground: white;
|
|
87
|
+
|
|
88
|
+
/* Spacing */
|
|
89
|
+
--spacing-component: var(--space-4);
|
|
90
|
+
--spacing-section: var(--space-6);
|
|
91
|
+
}
|
|
92
|
+
```
|
|
93
|
+
|
|
94
|
+
## Layer 3: Component Tokens
|
|
95
|
+
|
|
96
|
+
Component-specific tokens referencing semantic layer.
|
|
97
|
+
|
|
98
|
+
```css
|
|
99
|
+
:root {
|
|
100
|
+
/* Button */
|
|
101
|
+
--button-bg: var(--color-primary);
|
|
102
|
+
--button-fg: white;
|
|
103
|
+
--button-hover-bg: var(--color-primary-hover);
|
|
104
|
+
--button-padding-x: var(--space-4);
|
|
105
|
+
--button-padding-y: var(--space-2);
|
|
106
|
+
--button-radius: var(--radius-default);
|
|
107
|
+
|
|
108
|
+
/* Input */
|
|
109
|
+
--input-bg: var(--color-background);
|
|
110
|
+
--input-border: var(--color-gray-300);
|
|
111
|
+
--input-focus-ring: var(--color-primary);
|
|
112
|
+
--input-padding: var(--space-2) var(--space-3);
|
|
113
|
+
|
|
114
|
+
/* Card */
|
|
115
|
+
--card-bg: var(--color-background);
|
|
116
|
+
--card-border: var(--color-gray-200);
|
|
117
|
+
--card-padding: var(--space-4);
|
|
118
|
+
--card-radius: var(--radius-lg);
|
|
119
|
+
--card-shadow: var(--shadow-default);
|
|
120
|
+
}
|
|
121
|
+
```
|
|
122
|
+
|
|
123
|
+
## Dark Mode
|
|
124
|
+
|
|
125
|
+
Override semantic tokens for dark theme:
|
|
126
|
+
|
|
127
|
+
```css
|
|
128
|
+
.dark {
|
|
129
|
+
--color-background: var(--color-gray-900);
|
|
130
|
+
--color-foreground: var(--color-gray-50);
|
|
131
|
+
--color-muted: var(--color-gray-800);
|
|
132
|
+
--color-muted-foreground: var(--color-gray-400);
|
|
133
|
+
--color-secondary: var(--color-gray-800);
|
|
134
|
+
}
|
|
135
|
+
```
|
|
136
|
+
|
|
137
|
+
## Naming Convention
|
|
138
|
+
|
|
139
|
+
```
|
|
140
|
+
--{category}-{item}-{variant}-{state}
|
|
141
|
+
|
|
142
|
+
Examples:
|
|
143
|
+
--color-primary # category-item
|
|
144
|
+
--color-primary-hover # category-item-state
|
|
145
|
+
--button-bg-hover # component-property-state
|
|
146
|
+
--space-section-sm # category-semantic-variant
|
|
147
|
+
```
|
|
148
|
+
|
|
149
|
+
## Categories
|
|
150
|
+
|
|
151
|
+
| Category | Examples |
|
|
152
|
+
|----------|----------|
|
|
153
|
+
| color | primary, secondary, muted, destructive |
|
|
154
|
+
| space | 1, 2, 4, 8, section, component |
|
|
155
|
+
| font-size | xs, sm, base, lg, xl |
|
|
156
|
+
| radius | sm, default, lg, full |
|
|
157
|
+
| shadow | sm, default, lg |
|
|
158
|
+
| duration | fast, normal, slow |
|
|
159
|
+
|
|
160
|
+
## File Organization
|
|
161
|
+
|
|
162
|
+
```
|
|
163
|
+
tokens/
|
|
164
|
+
├── primitives.css # Raw values
|
|
165
|
+
├── semantic.css # Purpose aliases
|
|
166
|
+
├── components.css # Component tokens
|
|
167
|
+
└── index.css # Imports all
|
|
168
|
+
```
|
|
169
|
+
|
|
170
|
+
Or single file with layer comments:
|
|
171
|
+
|
|
172
|
+
```css
|
|
173
|
+
/* === PRIMITIVES === */
|
|
174
|
+
:root { ... }
|
|
175
|
+
|
|
176
|
+
/* === SEMANTIC === */
|
|
177
|
+
:root { ... }
|
|
178
|
+
|
|
179
|
+
/* === COMPONENTS === */
|
|
180
|
+
:root { ... }
|
|
181
|
+
|
|
182
|
+
/* === DARK MODE === */
|
|
183
|
+
.dark { ... }
|
|
184
|
+
```
|
|
185
|
+
|
|
186
|
+
## Migration from Flat Tokens
|
|
187
|
+
|
|
188
|
+
Before (flat):
|
|
189
|
+
```css
|
|
190
|
+
--button-primary-bg: #2563EB;
|
|
191
|
+
--button-secondary-bg: #F3F4F6;
|
|
192
|
+
```
|
|
193
|
+
|
|
194
|
+
After (three-layer):
|
|
195
|
+
```css
|
|
196
|
+
/* Primitive */
|
|
197
|
+
--color-blue-600: #2563EB;
|
|
198
|
+
--color-gray-100: #F3F4F6;
|
|
199
|
+
|
|
200
|
+
/* Semantic */
|
|
201
|
+
--color-primary: var(--color-blue-600);
|
|
202
|
+
--color-secondary: var(--color-gray-100);
|
|
203
|
+
|
|
204
|
+
/* Component */
|
|
205
|
+
--button-bg: var(--color-primary);
|
|
206
|
+
--button-secondary-bg: var(--color-secondary);
|
|
207
|
+
```
|
|
208
|
+
|
|
209
|
+
## W3C DTCG Alignment
|
|
210
|
+
|
|
211
|
+
Token JSON format (W3C Design Tokens Community Group):
|
|
212
|
+
|
|
213
|
+
```json
|
|
214
|
+
{
|
|
215
|
+
"color": {
|
|
216
|
+
"blue": {
|
|
217
|
+
"600": {
|
|
218
|
+
"$value": "#2563EB",
|
|
219
|
+
"$type": "color"
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
```
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* embed-tokens.cjs
|
|
4
|
+
* Reads design-tokens.css and outputs embeddable inline CSS.
|
|
5
|
+
* Use when generating standalone HTML files (infographics, slides, etc.)
|
|
6
|
+
*
|
|
7
|
+
* Usage:
|
|
8
|
+
* node embed-tokens.cjs # Output full CSS
|
|
9
|
+
* node embed-tokens.cjs --minimal # Output only commonly used tokens
|
|
10
|
+
* node embed-tokens.cjs --style # Wrap in <style> tags
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
const fs = require('fs');
|
|
14
|
+
const path = require('path');
|
|
15
|
+
|
|
16
|
+
// Find project root (look for assets/design-tokens.css)
|
|
17
|
+
function findProjectRoot(startDir) {
|
|
18
|
+
let dir = startDir;
|
|
19
|
+
while (dir !== '/') {
|
|
20
|
+
if (fs.existsSync(path.join(dir, 'assets', 'design-tokens.css'))) {
|
|
21
|
+
return dir;
|
|
22
|
+
}
|
|
23
|
+
dir = path.dirname(dir);
|
|
24
|
+
}
|
|
25
|
+
return null;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
const projectRoot = findProjectRoot(process.cwd());
|
|
29
|
+
if (!projectRoot) {
|
|
30
|
+
console.error('Error: Could not find assets/design-tokens.css');
|
|
31
|
+
process.exit(1);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
const tokensPath = path.join(projectRoot, 'assets', 'design-tokens.css');
|
|
35
|
+
|
|
36
|
+
// Minimal tokens commonly used in infographics/slides
|
|
37
|
+
const MINIMAL_TOKENS = [
|
|
38
|
+
'--primitive-spacing-',
|
|
39
|
+
'--primitive-fontSize-',
|
|
40
|
+
'--primitive-fontWeight-',
|
|
41
|
+
'--primitive-lineHeight-',
|
|
42
|
+
'--primitive-radius-',
|
|
43
|
+
'--primitive-shadow-glow-',
|
|
44
|
+
'--primitive-gradient-',
|
|
45
|
+
'--primitive-duration-',
|
|
46
|
+
'--color-primary',
|
|
47
|
+
'--color-secondary',
|
|
48
|
+
'--color-accent',
|
|
49
|
+
'--color-background',
|
|
50
|
+
'--color-surface',
|
|
51
|
+
'--color-foreground',
|
|
52
|
+
'--color-border',
|
|
53
|
+
'--typography-font-',
|
|
54
|
+
'--card-',
|
|
55
|
+
];
|
|
56
|
+
|
|
57
|
+
function extractTokens(css, minimal = false) {
|
|
58
|
+
// Extract :root block
|
|
59
|
+
const rootMatch = css.match(/:root\s*\{([^}]+)\}/g);
|
|
60
|
+
if (!rootMatch) return '';
|
|
61
|
+
|
|
62
|
+
let allVars = [];
|
|
63
|
+
for (const block of rootMatch) {
|
|
64
|
+
const vars = block.match(/--[\w-]+:\s*[^;]+;/g) || [];
|
|
65
|
+
allVars = allVars.concat(vars);
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
if (minimal) {
|
|
69
|
+
allVars = allVars.filter(v =>
|
|
70
|
+
MINIMAL_TOKENS.some(token => v.includes(token))
|
|
71
|
+
);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
// Dedupe
|
|
75
|
+
allVars = [...new Set(allVars)];
|
|
76
|
+
|
|
77
|
+
return `:root {\n ${allVars.join('\n ')}\n}`;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// Parse args
|
|
81
|
+
const args = process.argv.slice(2);
|
|
82
|
+
const minimal = args.includes('--minimal');
|
|
83
|
+
const wrapStyle = args.includes('--style');
|
|
84
|
+
|
|
85
|
+
try {
|
|
86
|
+
const css = fs.readFileSync(tokensPath, 'utf-8');
|
|
87
|
+
let output = extractTokens(css, minimal);
|
|
88
|
+
|
|
89
|
+
if (wrapStyle) {
|
|
90
|
+
output = `<style>\n/* Design Tokens (embedded for standalone HTML) */\n${output}\n</style>`;
|
|
91
|
+
} else {
|
|
92
|
+
output = `/* Design Tokens (embedded for standalone HTML) */\n${output}`;
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
console.log(output);
|
|
96
|
+
} catch (err) {
|
|
97
|
+
console.error(`Error reading tokens: ${err.message}`);
|
|
98
|
+
process.exit(1);
|
|
99
|
+
}
|
|
@@ -0,0 +1,317 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""
|
|
3
|
+
Background Image Fetcher
|
|
4
|
+
Fetches real images from Pexels for slide backgrounds.
|
|
5
|
+
Uses web scraping (no API key required) or WebFetch tool integration.
|
|
6
|
+
"""
|
|
7
|
+
|
|
8
|
+
import json
|
|
9
|
+
import csv
|
|
10
|
+
import re
|
|
11
|
+
import sys
|
|
12
|
+
from pathlib import Path
|
|
13
|
+
|
|
14
|
+
# Project root relative to this script
|
|
15
|
+
PROJECT_ROOT = Path(__file__).parent.parent.parent.parent.parent
|
|
16
|
+
TOKENS_PATH = PROJECT_ROOT / 'assets' / 'design-tokens.json'
|
|
17
|
+
BACKGROUNDS_CSV = Path(__file__).parent.parent / 'data' / 'slide-backgrounds.csv'
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
def resolve_token_reference(ref: str, tokens: dict) -> str:
|
|
21
|
+
"""Resolve token reference like {primitive.color.ocean-blue.500} to hex value."""
|
|
22
|
+
if not ref or not ref.startswith('{') or not ref.endswith('}'):
|
|
23
|
+
return ref # Already a value, not a reference
|
|
24
|
+
|
|
25
|
+
# Parse reference: {primitive.color.ocean-blue.500}
|
|
26
|
+
path = ref[1:-1].split('.') # ['primitive', 'color', 'ocean-blue', '500']
|
|
27
|
+
current = tokens
|
|
28
|
+
for key in path:
|
|
29
|
+
if isinstance(current, dict):
|
|
30
|
+
current = current.get(key)
|
|
31
|
+
else:
|
|
32
|
+
return None # Invalid path
|
|
33
|
+
# Return $value if it's a token object
|
|
34
|
+
if isinstance(current, dict) and '$value' in current:
|
|
35
|
+
return current['$value']
|
|
36
|
+
return current
|
|
37
|
+
|
|
38
|
+
|
|
39
|
+
def load_brand_colors():
|
|
40
|
+
"""Load colors from assets/design-tokens.json for overlay gradients.
|
|
41
|
+
|
|
42
|
+
Resolves semantic token references to actual hex values.
|
|
43
|
+
"""
|
|
44
|
+
try:
|
|
45
|
+
with open(TOKENS_PATH) as f:
|
|
46
|
+
tokens = json.load(f)
|
|
47
|
+
|
|
48
|
+
colors = tokens.get('primitive', {}).get('color', {})
|
|
49
|
+
semantic = tokens.get('semantic', {}).get('color', {})
|
|
50
|
+
|
|
51
|
+
# Try semantic tokens first (preferred) - resolve references
|
|
52
|
+
if semantic:
|
|
53
|
+
primary_ref = semantic.get('primary', {}).get('$value')
|
|
54
|
+
secondary_ref = semantic.get('secondary', {}).get('$value')
|
|
55
|
+
accent_ref = semantic.get('accent', {}).get('$value')
|
|
56
|
+
background_ref = semantic.get('background', {}).get('$value')
|
|
57
|
+
|
|
58
|
+
primary = resolve_token_reference(primary_ref, tokens)
|
|
59
|
+
secondary = resolve_token_reference(secondary_ref, tokens)
|
|
60
|
+
accent = resolve_token_reference(accent_ref, tokens)
|
|
61
|
+
background = resolve_token_reference(background_ref, tokens)
|
|
62
|
+
|
|
63
|
+
if primary and secondary:
|
|
64
|
+
return {
|
|
65
|
+
'primary': primary,
|
|
66
|
+
'secondary': secondary,
|
|
67
|
+
'accent': accent or primary,
|
|
68
|
+
'background': background or '#0D0D0D',
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
# Fallback: find first color palette with 500 value (primary)
|
|
72
|
+
primary_keys = ['ocean-blue', 'coral', 'blue', 'primary']
|
|
73
|
+
secondary_keys = ['golden-amber', 'purple', 'amber', 'secondary']
|
|
74
|
+
accent_keys = ['emerald', 'mint', 'green', 'accent']
|
|
75
|
+
|
|
76
|
+
primary_color = None
|
|
77
|
+
secondary_color = None
|
|
78
|
+
accent_color = None
|
|
79
|
+
|
|
80
|
+
for key in primary_keys:
|
|
81
|
+
if key in colors and isinstance(colors[key], dict):
|
|
82
|
+
primary_color = colors[key].get('500', {}).get('$value')
|
|
83
|
+
if primary_color:
|
|
84
|
+
break
|
|
85
|
+
|
|
86
|
+
for key in secondary_keys:
|
|
87
|
+
if key in colors and isinstance(colors[key], dict):
|
|
88
|
+
secondary_color = colors[key].get('500', {}).get('$value')
|
|
89
|
+
if secondary_color:
|
|
90
|
+
break
|
|
91
|
+
|
|
92
|
+
for key in accent_keys:
|
|
93
|
+
if key in colors and isinstance(colors[key], dict):
|
|
94
|
+
accent_color = colors[key].get('500', {}).get('$value')
|
|
95
|
+
if accent_color:
|
|
96
|
+
break
|
|
97
|
+
|
|
98
|
+
background = colors.get('dark', {}).get('800', {}).get('$value', '#0D0D0D')
|
|
99
|
+
|
|
100
|
+
return {
|
|
101
|
+
'primary': primary_color or '#3B82F6',
|
|
102
|
+
'secondary': secondary_color or '#F59E0B',
|
|
103
|
+
'accent': accent_color or '#10B981',
|
|
104
|
+
'background': background,
|
|
105
|
+
}
|
|
106
|
+
except (FileNotFoundError, KeyError, TypeError):
|
|
107
|
+
# Fallback defaults
|
|
108
|
+
return {
|
|
109
|
+
'primary': '#3B82F6',
|
|
110
|
+
'secondary': '#F59E0B',
|
|
111
|
+
'accent': '#10B981',
|
|
112
|
+
'background': '#0D0D0D',
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
|
|
116
|
+
def load_backgrounds_config():
|
|
117
|
+
"""Load background configuration from CSV."""
|
|
118
|
+
config = {}
|
|
119
|
+
try:
|
|
120
|
+
with open(BACKGROUNDS_CSV, newline='') as f:
|
|
121
|
+
reader = csv.DictReader(f)
|
|
122
|
+
for row in reader:
|
|
123
|
+
config[row['slide_type']] = row
|
|
124
|
+
except FileNotFoundError:
|
|
125
|
+
print(f"Warning: {BACKGROUNDS_CSV} not found")
|
|
126
|
+
return config
|
|
127
|
+
|
|
128
|
+
|
|
129
|
+
def get_overlay_css(style: str, brand_colors: dict) -> str:
|
|
130
|
+
"""Generate overlay CSS using brand colors from design-tokens.json."""
|
|
131
|
+
overlays = {
|
|
132
|
+
'gradient-dark': f"linear-gradient(135deg, {brand_colors['background']}E6, {brand_colors['background']}B3)",
|
|
133
|
+
'gradient-brand': f"linear-gradient(135deg, {brand_colors['primary']}CC, {brand_colors['secondary']}99)",
|
|
134
|
+
'gradient-accent': f"linear-gradient(135deg, {brand_colors['accent']}99, transparent)",
|
|
135
|
+
'blur-dark': f"rgba(13,13,13,0.8)",
|
|
136
|
+
'desaturate-dark': f"rgba(13,13,13,0.7)",
|
|
137
|
+
}
|
|
138
|
+
return overlays.get(style, overlays['gradient-dark'])
|
|
139
|
+
|
|
140
|
+
|
|
141
|
+
# Curated high-quality images from Pexels (free to use, pre-selected for brand aesthetic)
|
|
142
|
+
CURATED_IMAGES = {
|
|
143
|
+
'hero': [
|
|
144
|
+
'https://images.pexels.com/photos/3861969/pexels-photo-3861969.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
145
|
+
'https://images.pexels.com/photos/2582937/pexels-photo-2582937.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
146
|
+
'https://images.pexels.com/photos/1089438/pexels-photo-1089438.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
147
|
+
],
|
|
148
|
+
'vision': [
|
|
149
|
+
'https://images.pexels.com/photos/3183150/pexels-photo-3183150.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
150
|
+
'https://images.pexels.com/photos/3182812/pexels-photo-3182812.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
151
|
+
'https://images.pexels.com/photos/3184291/pexels-photo-3184291.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
152
|
+
],
|
|
153
|
+
'team': [
|
|
154
|
+
'https://images.pexels.com/photos/3184418/pexels-photo-3184418.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
155
|
+
'https://images.pexels.com/photos/3184338/pexels-photo-3184338.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
156
|
+
'https://images.pexels.com/photos/3182773/pexels-photo-3182773.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
157
|
+
],
|
|
158
|
+
'testimonial': [
|
|
159
|
+
'https://images.pexels.com/photos/3184465/pexels-photo-3184465.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
160
|
+
'https://images.pexels.com/photos/1181622/pexels-photo-1181622.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
161
|
+
],
|
|
162
|
+
'cta': [
|
|
163
|
+
'https://images.pexels.com/photos/3184339/pexels-photo-3184339.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
164
|
+
'https://images.pexels.com/photos/3184298/pexels-photo-3184298.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
165
|
+
],
|
|
166
|
+
'problem': [
|
|
167
|
+
'https://images.pexels.com/photos/3760529/pexels-photo-3760529.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
168
|
+
'https://images.pexels.com/photos/897817/pexels-photo-897817.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
169
|
+
],
|
|
170
|
+
'solution': [
|
|
171
|
+
'https://images.pexels.com/photos/3184292/pexels-photo-3184292.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
172
|
+
'https://images.pexels.com/photos/3184644/pexels-photo-3184644.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
173
|
+
],
|
|
174
|
+
'hook': [
|
|
175
|
+
'https://images.pexels.com/photos/2582937/pexels-photo-2582937.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
176
|
+
'https://images.pexels.com/photos/1089438/pexels-photo-1089438.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
177
|
+
],
|
|
178
|
+
'social': [
|
|
179
|
+
'https://images.pexels.com/photos/3184360/pexels-photo-3184360.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
180
|
+
'https://images.pexels.com/photos/3184287/pexels-photo-3184287.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
181
|
+
],
|
|
182
|
+
'demo': [
|
|
183
|
+
'https://images.pexels.com/photos/1181675/pexels-photo-1181675.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
184
|
+
'https://images.pexels.com/photos/3861958/pexels-photo-3861958.jpeg?auto=compress&cs=tinysrgb&w=1920',
|
|
185
|
+
],
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
|
|
189
|
+
def get_curated_images(slide_type: str) -> list:
|
|
190
|
+
"""Get curated images for slide type."""
|
|
191
|
+
return CURATED_IMAGES.get(slide_type, CURATED_IMAGES.get('hero', []))
|
|
192
|
+
|
|
193
|
+
|
|
194
|
+
def get_pexels_search_url(keywords: str) -> str:
|
|
195
|
+
"""Generate Pexels search URL for manual lookup."""
|
|
196
|
+
import urllib.parse
|
|
197
|
+
return f"https://www.pexels.com/search/{urllib.parse.quote(keywords)}/"
|
|
198
|
+
|
|
199
|
+
|
|
200
|
+
def get_background_image(slide_type: str) -> dict:
|
|
201
|
+
"""
|
|
202
|
+
Get curated image matching slide type and brand aesthetic.
|
|
203
|
+
Uses pre-selected Pexels images (no API/scraping needed).
|
|
204
|
+
"""
|
|
205
|
+
brand_colors = load_brand_colors()
|
|
206
|
+
config = load_backgrounds_config()
|
|
207
|
+
|
|
208
|
+
slide_config = config.get(slide_type)
|
|
209
|
+
overlay_style = 'gradient-dark'
|
|
210
|
+
keywords = slide_type
|
|
211
|
+
|
|
212
|
+
if slide_config:
|
|
213
|
+
keywords = slide_config.get('search_keywords', slide_config.get('image_category', slide_type))
|
|
214
|
+
overlay_style = slide_config.get('overlay_style', 'gradient-dark')
|
|
215
|
+
|
|
216
|
+
# Get curated images
|
|
217
|
+
urls = get_curated_images(slide_type)
|
|
218
|
+
if urls:
|
|
219
|
+
return {
|
|
220
|
+
'url': urls[0],
|
|
221
|
+
'all_urls': urls,
|
|
222
|
+
'overlay': get_overlay_css(overlay_style, brand_colors),
|
|
223
|
+
'attribution': 'Photo from Pexels (free to use)',
|
|
224
|
+
'source': 'pexels-curated',
|
|
225
|
+
'search_url': get_pexels_search_url(keywords),
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
# Fallback: provide search URL for manual selection
|
|
229
|
+
return {
|
|
230
|
+
'url': None,
|
|
231
|
+
'overlay': get_overlay_css(overlay_style, brand_colors),
|
|
232
|
+
'keywords': keywords,
|
|
233
|
+
'search_url': get_pexels_search_url(keywords),
|
|
234
|
+
'available_types': list(CURATED_IMAGES.keys()),
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
|
|
238
|
+
def generate_css_for_background(result: dict, slide_class: str = '.slide-with-bg') -> str:
|
|
239
|
+
"""Generate CSS for a background slide."""
|
|
240
|
+
if not result.get('url'):
|
|
241
|
+
search_url = result.get('search_url', '')
|
|
242
|
+
return f"""/* No image scraped. Search manually: {search_url} */
|
|
243
|
+
/* Overlay ready: {result.get('overlay', 'gradient-dark')} */
|
|
244
|
+
"""
|
|
245
|
+
|
|
246
|
+
return f"""{slide_class} {{
|
|
247
|
+
background-image: url('{result['url']}');
|
|
248
|
+
background-size: cover;
|
|
249
|
+
background-position: center;
|
|
250
|
+
position: relative;
|
|
251
|
+
}}
|
|
252
|
+
|
|
253
|
+
{slide_class}::before {{
|
|
254
|
+
content: '';
|
|
255
|
+
position: absolute;
|
|
256
|
+
inset: 0;
|
|
257
|
+
background: {result['overlay']};
|
|
258
|
+
}}
|
|
259
|
+
|
|
260
|
+
{slide_class} .content {{
|
|
261
|
+
position: relative;
|
|
262
|
+
z-index: 1;
|
|
263
|
+
}}
|
|
264
|
+
|
|
265
|
+
/* {result.get('attribution', 'Pexels')} - {result.get('search_url', '')} */
|
|
266
|
+
"""
|
|
267
|
+
|
|
268
|
+
|
|
269
|
+
def main():
|
|
270
|
+
"""CLI entry point."""
|
|
271
|
+
import argparse
|
|
272
|
+
|
|
273
|
+
parser = argparse.ArgumentParser(description='Get background images for slides')
|
|
274
|
+
parser.add_argument('slide_type', nargs='?', help='Slide type (hero, vision, team, etc.)')
|
|
275
|
+
parser.add_argument('--list', action='store_true', help='List available slide types')
|
|
276
|
+
parser.add_argument('--css', action='store_true', help='Output CSS for the background')
|
|
277
|
+
parser.add_argument('--json', action='store_true', help='Output JSON')
|
|
278
|
+
parser.add_argument('--colors', action='store_true', help='Show brand colors')
|
|
279
|
+
parser.add_argument('--all', action='store_true', help='Show all curated URLs')
|
|
280
|
+
|
|
281
|
+
args = parser.parse_args()
|
|
282
|
+
|
|
283
|
+
if args.colors:
|
|
284
|
+
colors = load_brand_colors()
|
|
285
|
+
print("\nBrand Colors (from design-tokens.json):")
|
|
286
|
+
for name, value in colors.items():
|
|
287
|
+
print(f" {name}: {value}")
|
|
288
|
+
return
|
|
289
|
+
|
|
290
|
+
if args.list:
|
|
291
|
+
print("\nAvailable slide types (curated images):")
|
|
292
|
+
for slide_type, urls in CURATED_IMAGES.items():
|
|
293
|
+
print(f" {slide_type}: {len(urls)} images")
|
|
294
|
+
return
|
|
295
|
+
|
|
296
|
+
if not args.slide_type:
|
|
297
|
+
parser.print_help()
|
|
298
|
+
return
|
|
299
|
+
|
|
300
|
+
result = get_background_image(args.slide_type)
|
|
301
|
+
|
|
302
|
+
if args.json:
|
|
303
|
+
print(json.dumps(result, indent=2))
|
|
304
|
+
elif args.css:
|
|
305
|
+
print(generate_css_for_background(result))
|
|
306
|
+
elif args.all:
|
|
307
|
+
print(f"\nAll images for '{args.slide_type}':")
|
|
308
|
+
for i, url in enumerate(result.get('all_urls', []), 1):
|
|
309
|
+
print(f" {i}. {url}")
|
|
310
|
+
else:
|
|
311
|
+
print(f"\nImage URL: {result['url']}")
|
|
312
|
+
print(f"Alternatives: {len(result.get('all_urls', []))} available (use --all)")
|
|
313
|
+
print(f"Overlay: {result['overlay']}")
|
|
314
|
+
|
|
315
|
+
|
|
316
|
+
if __name__ == '__main__':
|
|
317
|
+
main()
|