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.
Files changed (160) hide show
  1. package/README.md +99 -0
  2. package/assets/data/_sync_all.py +414 -0
  3. package/assets/data/app-interface.csv +31 -0
  4. package/assets/data/charts.csv +26 -0
  5. package/assets/data/colors.csv +162 -0
  6. package/assets/data/design.csv +1776 -0
  7. package/assets/data/draft.csv +1779 -0
  8. package/assets/data/google-fonts.csv +1924 -0
  9. package/assets/data/icons.csv +106 -0
  10. package/assets/data/landing.csv +35 -0
  11. package/assets/data/products.csv +162 -0
  12. package/assets/data/react-performance.csv +45 -0
  13. package/assets/data/stacks/angular.csv +51 -0
  14. package/assets/data/stacks/astro.csv +54 -0
  15. package/assets/data/stacks/flutter.csv +53 -0
  16. package/assets/data/stacks/html-tailwind.csv +56 -0
  17. package/assets/data/stacks/javafx.csv +76 -0
  18. package/assets/data/stacks/jetpack-compose.csv +53 -0
  19. package/assets/data/stacks/laravel.csv +51 -0
  20. package/assets/data/stacks/nextjs.csv +53 -0
  21. package/assets/data/stacks/nuxt-ui.csv +71 -0
  22. package/assets/data/stacks/nuxtjs.csv +59 -0
  23. package/assets/data/stacks/react-native.csv +52 -0
  24. package/assets/data/stacks/react.csv +54 -0
  25. package/assets/data/stacks/shadcn.csv +61 -0
  26. package/assets/data/stacks/svelte.csv +54 -0
  27. package/assets/data/stacks/swiftui.csv +51 -0
  28. package/assets/data/stacks/threejs.csv +54 -0
  29. package/assets/data/stacks/vue.csv +50 -0
  30. package/assets/data/styles.csv +85 -0
  31. package/assets/data/typography.csv +74 -0
  32. package/assets/data/ui-reasoning.csv +162 -0
  33. package/assets/data/ux-guidelines.csv +100 -0
  34. package/assets/scripts/core.py +263 -0
  35. package/assets/scripts/design_system.py +1157 -0
  36. package/assets/scripts/search.py +114 -0
  37. package/assets/skills/banner-design/SKILL.md +196 -0
  38. package/assets/skills/banner-design/references/banner-sizes-and-styles.md +118 -0
  39. package/assets/skills/brand/SKILL.md +97 -0
  40. package/assets/skills/brand/references/approval-checklist.md +169 -0
  41. package/assets/skills/brand/references/asset-organization.md +157 -0
  42. package/assets/skills/brand/references/brand-guideline-template.md +140 -0
  43. package/assets/skills/brand/references/color-palette-management.md +186 -0
  44. package/assets/skills/brand/references/consistency-checklist.md +94 -0
  45. package/assets/skills/brand/references/logo-usage-rules.md +185 -0
  46. package/assets/skills/brand/references/messaging-framework.md +85 -0
  47. package/assets/skills/brand/references/typography-specifications.md +214 -0
  48. package/assets/skills/brand/references/update.md +118 -0
  49. package/assets/skills/brand/references/visual-identity.md +96 -0
  50. package/assets/skills/brand/references/voice-framework.md +88 -0
  51. package/assets/skills/brand/scripts/extract-colors.cjs +341 -0
  52. package/assets/skills/brand/scripts/inject-brand-context.cjs +349 -0
  53. package/assets/skills/brand/scripts/sync-brand-to-tokens.cjs +266 -0
  54. package/assets/skills/brand/scripts/validate-asset.cjs +387 -0
  55. package/assets/skills/brand/templates/brand-guidelines-starter.md +275 -0
  56. package/assets/skills/design/SKILL.md +313 -0
  57. package/assets/skills/design/data/cip/deliverables.csv +51 -0
  58. package/assets/skills/design/data/cip/industries.csv +21 -0
  59. package/assets/skills/design/data/cip/mockup-contexts.csv +21 -0
  60. package/assets/skills/design/data/cip/styles.csv +21 -0
  61. package/assets/skills/design/data/icon/styles.csv +16 -0
  62. package/assets/skills/design/data/logo/colors.csv +56 -0
  63. package/assets/skills/design/data/logo/industries.csv +56 -0
  64. package/assets/skills/design/data/logo/styles.csv +56 -0
  65. package/assets/skills/design/references/banner-sizes-and-styles.md +118 -0
  66. package/assets/skills/design/references/cip-deliverable-guide.md +95 -0
  67. package/assets/skills/design/references/cip-design.md +121 -0
  68. package/assets/skills/design/references/cip-prompt-engineering.md +84 -0
  69. package/assets/skills/design/references/cip-style-guide.md +68 -0
  70. package/assets/skills/design/references/design-routing.md +207 -0
  71. package/assets/skills/design/references/icon-design.md +122 -0
  72. package/assets/skills/design/references/logo-color-psychology.md +101 -0
  73. package/assets/skills/design/references/logo-design.md +92 -0
  74. package/assets/skills/design/references/logo-prompt-engineering.md +158 -0
  75. package/assets/skills/design/references/logo-style-guide.md +109 -0
  76. package/assets/skills/design/references/slides-copywriting-formulas.md +84 -0
  77. package/assets/skills/design/references/slides-create.md +4 -0
  78. package/assets/skills/design/references/slides-html-template.md +295 -0
  79. package/assets/skills/design/references/slides-layout-patterns.md +137 -0
  80. package/assets/skills/design/references/slides-strategies.md +94 -0
  81. package/assets/skills/design/references/slides.md +42 -0
  82. package/assets/skills/design/references/social-photos-design.md +329 -0
  83. package/assets/skills/design/scripts/cip/core.py +215 -0
  84. package/assets/skills/design/scripts/cip/generate.py +484 -0
  85. package/assets/skills/design/scripts/cip/render-html.py +424 -0
  86. package/assets/skills/design/scripts/cip/search.py +127 -0
  87. package/assets/skills/design/scripts/icon/generate.py +487 -0
  88. package/assets/skills/design/scripts/logo/core.py +175 -0
  89. package/assets/skills/design/scripts/logo/generate.py +362 -0
  90. package/assets/skills/design/scripts/logo/search.py +114 -0
  91. package/assets/skills/design-system/SKILL.md +244 -0
  92. package/assets/skills/design-system/data/slide-backgrounds.csv +11 -0
  93. package/assets/skills/design-system/data/slide-charts.csv +26 -0
  94. package/assets/skills/design-system/data/slide-color-logic.csv +14 -0
  95. package/assets/skills/design-system/data/slide-copy.csv +26 -0
  96. package/assets/skills/design-system/data/slide-layout-logic.csv +16 -0
  97. package/assets/skills/design-system/data/slide-layouts.csv +26 -0
  98. package/assets/skills/design-system/data/slide-strategies.csv +16 -0
  99. package/assets/skills/design-system/data/slide-typography.csv +15 -0
  100. package/assets/skills/design-system/references/component-specs.md +236 -0
  101. package/assets/skills/design-system/references/component-tokens.md +214 -0
  102. package/assets/skills/design-system/references/primitive-tokens.md +203 -0
  103. package/assets/skills/design-system/references/semantic-tokens.md +215 -0
  104. package/assets/skills/design-system/references/states-and-variants.md +241 -0
  105. package/assets/skills/design-system/references/tailwind-integration.md +251 -0
  106. package/assets/skills/design-system/references/token-architecture.md +224 -0
  107. package/assets/skills/design-system/scripts/embed-tokens.cjs +99 -0
  108. package/assets/skills/design-system/scripts/fetch-background.py +317 -0
  109. package/assets/skills/design-system/scripts/generate-slide.py +770 -0
  110. package/assets/skills/design-system/scripts/generate-tokens.cjs +205 -0
  111. package/assets/skills/design-system/scripts/html-token-validator.py +327 -0
  112. package/assets/skills/design-system/scripts/search-slides.py +218 -0
  113. package/assets/skills/design-system/scripts/slide-token-validator.py +35 -0
  114. package/assets/skills/design-system/scripts/slide_search_core.py +453 -0
  115. package/assets/skills/design-system/scripts/validate-tokens.cjs +251 -0
  116. package/assets/skills/design-system/templates/design-tokens-starter.json +143 -0
  117. package/assets/skills/slides/SKILL.md +40 -0
  118. package/assets/skills/slides/references/copywriting-formulas.md +84 -0
  119. package/assets/skills/slides/references/create.md +4 -0
  120. package/assets/skills/slides/references/html-template.md +295 -0
  121. package/assets/skills/slides/references/layout-patterns.md +137 -0
  122. package/assets/skills/slides/references/slide-strategies.md +94 -0
  123. package/assets/skills/ui-styling/LICENSE.txt +202 -0
  124. package/assets/skills/ui-styling/SKILL.md +324 -0
  125. package/assets/skills/ui-styling/references/canvas-design-system.md +320 -0
  126. package/assets/skills/ui-styling/references/shadcn-accessibility.md +471 -0
  127. package/assets/skills/ui-styling/references/shadcn-components.md +424 -0
  128. package/assets/skills/ui-styling/references/shadcn-theming.md +373 -0
  129. package/assets/skills/ui-styling/references/tailwind-customization.md +483 -0
  130. package/assets/skills/ui-styling/references/tailwind-responsive.md +382 -0
  131. package/assets/skills/ui-styling/references/tailwind-utilities.md +455 -0
  132. package/assets/skills/ui-styling/scripts/requirements.txt +17 -0
  133. package/assets/skills/ui-styling/scripts/shadcn_add.py +308 -0
  134. package/assets/skills/ui-styling/scripts/tailwind_config_gen.py +473 -0
  135. package/assets/skills/ui-styling/scripts/tests/coverage-ui.json +1 -0
  136. package/assets/skills/ui-styling/scripts/tests/requirements.txt +3 -0
  137. package/assets/skills/ui-styling/scripts/tests/test_shadcn_add.py +266 -0
  138. package/assets/skills/ui-styling/scripts/tests/test_tailwind_config_gen.py +336 -0
  139. package/assets/templates/base/quick-reference.md +297 -0
  140. package/assets/templates/base/skill-content.md +368 -0
  141. package/assets/templates/platforms/agent.json +21 -0
  142. package/assets/templates/platforms/augment.json +18 -0
  143. package/assets/templates/platforms/claude.json +21 -0
  144. package/assets/templates/platforms/codebuddy.json +21 -0
  145. package/assets/templates/platforms/codex.json +21 -0
  146. package/assets/templates/platforms/continue.json +21 -0
  147. package/assets/templates/platforms/copilot.json +21 -0
  148. package/assets/templates/platforms/cursor.json +21 -0
  149. package/assets/templates/platforms/droid.json +21 -0
  150. package/assets/templates/platforms/gemini.json +21 -0
  151. package/assets/templates/platforms/kilocode.json +21 -0
  152. package/assets/templates/platforms/kiro.json +21 -0
  153. package/assets/templates/platforms/opencode.json +21 -0
  154. package/assets/templates/platforms/qoder.json +21 -0
  155. package/assets/templates/platforms/roocode.json +21 -0
  156. package/assets/templates/platforms/trae.json +21 -0
  157. package/assets/templates/platforms/warp.json +18 -0
  158. package/assets/templates/platforms/windsurf.json +21 -0
  159. package/dist/index.js +10630 -0
  160. 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()