@mandujs/mcp 0.9.19 → 0.9.21

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 (122) hide show
  1. package/README.md +320 -0
  2. package/package.json +1 -1
  3. package/src/activity-monitor.ts +847 -231
  4. package/src/resources/handlers.ts +244 -0
  5. package/src/resources/skills/guides.ts +1136 -0
  6. package/src/resources/skills/index.ts +12 -0
  7. package/src/resources/skills/loader.ts +218 -0
  8. package/src/resources/skills/mandu-composition/SKILL.md +91 -0
  9. package/src/resources/skills/mandu-composition/metadata.json +13 -0
  10. package/src/resources/skills/mandu-composition/rules/_sections.md +26 -0
  11. package/src/resources/skills/mandu-composition/rules/_template.md +77 -0
  12. package/src/resources/skills/mandu-composition/rules/comp-arch-avoid-boolean-props.md +146 -0
  13. package/src/resources/skills/mandu-composition/rules/comp-arch-compound-components.md +164 -0
  14. package/src/resources/skills/mandu-composition/rules/comp-island-event.md +161 -0
  15. package/src/resources/skills/mandu-composition/rules/comp-island-slot-split.md +167 -0
  16. package/src/resources/skills/mandu-composition/rules/comp-pattern-children.md +149 -0
  17. package/src/resources/skills/mandu-composition/rules/comp-state-context-interface.md +148 -0
  18. package/src/resources/skills/mandu-composition/rules/comp-state-lift-state.md +150 -0
  19. package/src/resources/skills/mandu-deployment/SKILL.md +92 -0
  20. package/src/resources/skills/mandu-deployment/_sections.md +41 -0
  21. package/src/resources/skills/mandu-deployment/_template.md +38 -0
  22. package/src/resources/skills/mandu-deployment/metadata.json +13 -0
  23. package/src/resources/skills/mandu-deployment/rules/deploy-build-bun.md +109 -0
  24. package/src/resources/skills/mandu-deployment/rules/deploy-build-output.md +115 -0
  25. package/src/resources/skills/mandu-deployment/rules/deploy-cicd-github.md +219 -0
  26. package/src/resources/skills/mandu-deployment/rules/deploy-docker-bun.md +150 -0
  27. package/src/resources/skills/mandu-deployment/rules/deploy-docker-compose.md +223 -0
  28. package/src/resources/skills/mandu-deployment/rules/deploy-platform-fly.md +152 -0
  29. package/src/resources/skills/mandu-deployment/rules/deploy-platform-render.md +179 -0
  30. package/src/resources/skills/mandu-deployment/rules/deploy-platform-supabase.md +323 -0
  31. package/src/resources/skills/mandu-deployment/rules/deploy-platform-vercel.md +140 -0
  32. package/src/resources/skills/mandu-fs-routes/SKILL.md +82 -0
  33. package/src/resources/skills/mandu-fs-routes/metadata.json +12 -0
  34. package/src/resources/skills/mandu-fs-routes/rules/_sections.md +36 -0
  35. package/src/resources/skills/mandu-fs-routes/rules/_template.md +69 -0
  36. package/src/resources/skills/mandu-fs-routes/rules/routes-api-methods.md +65 -0
  37. package/src/resources/skills/mandu-fs-routes/rules/routes-dynamic-param.md +93 -0
  38. package/src/resources/skills/mandu-fs-routes/rules/routes-naming-page.md +55 -0
  39. package/src/resources/skills/mandu-guard/SKILL.md +129 -0
  40. package/src/resources/skills/mandu-guard/metadata.json +12 -0
  41. package/src/resources/skills/mandu-guard/rules/_sections.md +36 -0
  42. package/src/resources/skills/mandu-guard/rules/_template.md +82 -0
  43. package/src/resources/skills/mandu-guard/rules/guard-config-rules.md +100 -0
  44. package/src/resources/skills/mandu-guard/rules/guard-layer-direction.md +76 -0
  45. package/src/resources/skills/mandu-guard/rules/guard-preset-mandu.md +81 -0
  46. package/src/resources/skills/mandu-guard/rules/guard-validate-import.md +80 -0
  47. package/src/resources/skills/mandu-hydration/SKILL.md +91 -0
  48. package/src/resources/skills/mandu-hydration/metadata.json +12 -0
  49. package/src/resources/skills/mandu-hydration/rules/_sections.md +31 -0
  50. package/src/resources/skills/mandu-hydration/rules/_template.md +72 -0
  51. package/src/resources/skills/mandu-hydration/rules/hydration-data-event.md +109 -0
  52. package/src/resources/skills/mandu-hydration/rules/hydration-directive-use-client.md +55 -0
  53. package/src/resources/skills/mandu-hydration/rules/hydration-island-setup.md +113 -0
  54. package/src/resources/skills/mandu-hydration/rules/hydration-priority-visible.md +68 -0
  55. package/src/resources/skills/mandu-performance/SKILL.md +85 -0
  56. package/src/resources/skills/mandu-performance/metadata.json +14 -0
  57. package/src/resources/skills/mandu-performance/rules/_sections.md +31 -0
  58. package/src/resources/skills/mandu-performance/rules/_template.md +64 -0
  59. package/src/resources/skills/mandu-performance/rules/perf-async-defer-await.md +103 -0
  60. package/src/resources/skills/mandu-performance/rules/perf-async-parallel.md +95 -0
  61. package/src/resources/skills/mandu-performance/rules/perf-bun-file.md +124 -0
  62. package/src/resources/skills/mandu-performance/rules/perf-bun-serve.md +125 -0
  63. package/src/resources/skills/mandu-performance/rules/perf-bundle-imports.md +80 -0
  64. package/src/resources/skills/mandu-performance/rules/perf-bundle-island-lazy.md +145 -0
  65. package/src/resources/skills/mandu-performance/rules/perf-cache-react.md +98 -0
  66. package/src/resources/skills/mandu-performance/rules/perf-render-transitions.md +154 -0
  67. package/src/resources/skills/mandu-security/SKILL.md +87 -0
  68. package/src/resources/skills/mandu-security/metadata.json +13 -0
  69. package/src/resources/skills/mandu-security/rules/_sections.md +31 -0
  70. package/src/resources/skills/mandu-security/rules/_template.md +74 -0
  71. package/src/resources/skills/mandu-security/rules/sec-auth-guard.md +127 -0
  72. package/src/resources/skills/mandu-security/rules/sec-env-management.md +133 -0
  73. package/src/resources/skills/mandu-security/rules/sec-input-validate.md +148 -0
  74. package/src/resources/skills/mandu-security/rules/sec-protect-csrf.md +146 -0
  75. package/src/resources/skills/mandu-security/rules/sec-protect-headers.md +138 -0
  76. package/src/resources/skills/mandu-slot/SKILL.md +85 -0
  77. package/src/resources/skills/mandu-slot/metadata.json +12 -0
  78. package/src/resources/skills/mandu-slot/rules/_sections.md +36 -0
  79. package/src/resources/skills/mandu-slot/rules/_template.md +63 -0
  80. package/src/resources/skills/mandu-slot/rules/slot-basic-structure.md +38 -0
  81. package/src/resources/skills/mandu-slot/rules/slot-ctx-response.md +56 -0
  82. package/src/resources/skills/mandu-slot/rules/slot-guard-auth.md +59 -0
  83. package/src/resources/skills/mandu-slot/rules/slot-http-methods.md +64 -0
  84. package/src/resources/skills/mandu-styling/SKILL.md +118 -0
  85. package/src/resources/skills/mandu-styling/_sections.md +36 -0
  86. package/src/resources/skills/mandu-styling/_template.md +32 -0
  87. package/src/resources/skills/mandu-styling/metadata.json +13 -0
  88. package/src/resources/skills/mandu-styling/rules/style-component-compound.md +235 -0
  89. package/src/resources/skills/mandu-styling/rules/style-component-slots.md +255 -0
  90. package/src/resources/skills/mandu-styling/rules/style-component-tokens.md +205 -0
  91. package/src/resources/skills/mandu-styling/rules/style-island-animations.md +272 -0
  92. package/src/resources/skills/mandu-styling/rules/style-island-scoping.md +167 -0
  93. package/src/resources/skills/mandu-styling/rules/style-island-variants.md +221 -0
  94. package/src/resources/skills/mandu-styling/rules/style-perf-critical.md +209 -0
  95. package/src/resources/skills/mandu-styling/rules/style-perf-purge.md +192 -0
  96. package/src/resources/skills/mandu-styling/rules/style-setup-modules.md +162 -0
  97. package/src/resources/skills/mandu-styling/rules/style-setup-panda.md +164 -0
  98. package/src/resources/skills/mandu-styling/rules/style-setup-tailwind.md +161 -0
  99. package/src/resources/skills/mandu-styling/rules/style-theme-darkmode.md +229 -0
  100. package/src/resources/skills/mandu-testing/SKILL.md +99 -0
  101. package/src/resources/skills/mandu-testing/metadata.json +13 -0
  102. package/src/resources/skills/mandu-testing/rules/_sections.md +26 -0
  103. package/src/resources/skills/mandu-testing/rules/_template.md +65 -0
  104. package/src/resources/skills/mandu-testing/rules/test-component-island.md +195 -0
  105. package/src/resources/skills/mandu-testing/rules/test-e2e-playwright.md +196 -0
  106. package/src/resources/skills/mandu-testing/rules/test-mock-fetch.md +219 -0
  107. package/src/resources/skills/mandu-testing/rules/test-slot-unit.md +192 -0
  108. package/src/resources/skills/mandu-ui/SKILL.md +117 -0
  109. package/src/resources/skills/mandu-ui/_sections.md +23 -0
  110. package/src/resources/skills/mandu-ui/_template.md +32 -0
  111. package/src/resources/skills/mandu-ui/metadata.json +13 -0
  112. package/src/resources/skills/mandu-ui/rules/ui-accessibility-aria.md +232 -0
  113. package/src/resources/skills/mandu-ui/rules/ui-accessibility-focus.md +238 -0
  114. package/src/resources/skills/mandu-ui/rules/ui-composition-patterns.md +259 -0
  115. package/src/resources/skills/mandu-ui/rules/ui-island-integration.md +258 -0
  116. package/src/resources/skills/mandu-ui/rules/ui-radix-patterns.md +213 -0
  117. package/src/resources/skills/mandu-ui/rules/ui-shadcn-setup.md +209 -0
  118. package/src/resources/skills/recipes.ts +932 -0
  119. package/src/server.ts +3 -0
  120. package/src/tools/hydration.ts +8 -8
  121. package/src/tools/index.ts +1 -0
  122. package/src/tools/seo.ts +417 -0
@@ -0,0 +1,255 @@
1
+ ---
2
+ title: Slot-based Style Customization
3
+ impact: HIGH
4
+ impactDescription: Flexible style overrides for reusable components
5
+ tags: styling, slots, customization, override
6
+ ---
7
+
8
+ ## Slot-based Style Customization
9
+
10
+ **Impact: HIGH (Flexible style overrides for reusable components)**
11
+
12
+ 컴포넌트의 특정 부분만 스타일을 오버라이드할 수 있는 slot 패턴입니다.
13
+
14
+ ## classNames prop 패턴
15
+
16
+ ```tsx
17
+ // app/input/client.tsx
18
+ "use client";
19
+
20
+ import { cn } from "@/lib/utils";
21
+
22
+ interface InputClassNames {
23
+ root?: string;
24
+ label?: string;
25
+ input?: string;
26
+ helper?: string;
27
+ error?: string;
28
+ }
29
+
30
+ interface InputProps extends React.InputHTMLAttributes<HTMLInputElement> {
31
+ label?: string;
32
+ helper?: string;
33
+ error?: string;
34
+ classNames?: InputClassNames;
35
+ }
36
+
37
+ export function InputIsland({
38
+ label,
39
+ helper,
40
+ error,
41
+ classNames,
42
+ className,
43
+ ...props
44
+ }: InputProps) {
45
+ return (
46
+ <div className={cn("space-y-2", classNames?.root)}>
47
+ {label && (
48
+ <label
49
+ className={cn(
50
+ "text-sm font-medium leading-none",
51
+ classNames?.label
52
+ )}
53
+ >
54
+ {label}
55
+ </label>
56
+ )}
57
+
58
+ <input
59
+ className={cn(
60
+ "flex h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-sm",
61
+ "focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring",
62
+ error && "border-destructive focus-visible:ring-destructive",
63
+ classNames?.input,
64
+ className
65
+ )}
66
+ {...props}
67
+ />
68
+
69
+ {helper && !error && (
70
+ <p className={cn("text-sm text-muted-foreground", classNames?.helper)}>
71
+ {helper}
72
+ </p>
73
+ )}
74
+
75
+ {error && (
76
+ <p className={cn("text-sm text-destructive", classNames?.error)}>
77
+ {error}
78
+ </p>
79
+ )}
80
+ </div>
81
+ );
82
+ }
83
+ ```
84
+
85
+ ## 사용 예시
86
+
87
+ ```tsx
88
+ <InputIsland
89
+ label="Email"
90
+ placeholder="you@example.com"
91
+ classNames={{
92
+ root: "max-w-md",
93
+ label: "text-blue-600",
94
+ input: "border-2",
95
+ helper: "italic",
96
+ }}
97
+ />
98
+ ```
99
+
100
+ ## styles prop 패턴
101
+
102
+ ```tsx
103
+ // app/card/client.tsx
104
+ "use client";
105
+
106
+ interface CardStyles {
107
+ root?: React.CSSProperties;
108
+ header?: React.CSSProperties;
109
+ content?: React.CSSProperties;
110
+ footer?: React.CSSProperties;
111
+ }
112
+
113
+ interface CardProps {
114
+ styles?: CardStyles;
115
+ // ...
116
+ }
117
+
118
+ export function CardIsland({ styles, children }: CardProps) {
119
+ return (
120
+ <div className="rounded-lg border" style={styles?.root}>
121
+ {children}
122
+ </div>
123
+ );
124
+ }
125
+ ```
126
+
127
+ ## slotProps 패턴 (고급)
128
+
129
+ ```tsx
130
+ // app/modal/client.tsx
131
+ "use client";
132
+
133
+ interface SlotProps<T extends React.ElementType = "div"> {
134
+ component?: T;
135
+ className?: string;
136
+ style?: React.CSSProperties;
137
+ }
138
+
139
+ interface ModalSlotProps {
140
+ root?: SlotProps;
141
+ overlay?: SlotProps;
142
+ content?: SlotProps<"div" | "form">;
143
+ header?: SlotProps;
144
+ body?: SlotProps;
145
+ footer?: SlotProps;
146
+ }
147
+
148
+ interface ModalProps {
149
+ open: boolean;
150
+ onClose: () => void;
151
+ slotProps?: ModalSlotProps;
152
+ children: React.ReactNode;
153
+ }
154
+
155
+ export function ModalIsland({ open, onClose, slotProps, children }: ModalProps) {
156
+ if (!open) return null;
157
+
158
+ const OverlayComponent = slotProps?.overlay?.component || "div";
159
+ const ContentComponent = slotProps?.content?.component || "div";
160
+
161
+ return (
162
+ <div
163
+ className={cn("fixed inset-0 z-50", slotProps?.root?.className)}
164
+ style={slotProps?.root?.style}
165
+ >
166
+ <OverlayComponent
167
+ className={cn(
168
+ "fixed inset-0 bg-black/50",
169
+ slotProps?.overlay?.className
170
+ )}
171
+ style={slotProps?.overlay?.style}
172
+ onClick={onClose}
173
+ />
174
+
175
+ <ContentComponent
176
+ className={cn(
177
+ "fixed left-1/2 top-1/2 -translate-x-1/2 -translate-y-1/2",
178
+ "w-full max-w-lg rounded-lg bg-background p-6 shadow-lg",
179
+ slotProps?.content?.className
180
+ )}
181
+ style={slotProps?.content?.style}
182
+ >
183
+ {children}
184
+ </ContentComponent>
185
+ </div>
186
+ );
187
+ }
188
+ ```
189
+
190
+ ## 사용 예시 (고급)
191
+
192
+ ```tsx
193
+ <ModalIsland
194
+ open={isOpen}
195
+ onClose={() => setIsOpen(false)}
196
+ slotProps={{
197
+ overlay: {
198
+ className: "backdrop-blur-sm",
199
+ },
200
+ content: {
201
+ component: "form",
202
+ className: "max-w-2xl",
203
+ style: { maxHeight: "80vh" },
204
+ },
205
+ }}
206
+ >
207
+ <form onSubmit={handleSubmit}>...</form>
208
+ </ModalIsland>
209
+ ```
210
+
211
+ ## Tailwind merge로 안전한 오버라이드
212
+
213
+ ```tsx
214
+ // cn 함수가 충돌을 해결
215
+ <button
216
+ className={cn(
217
+ "px-4 py-2 bg-blue-500", // 기본
218
+ className // 오버라이드: "px-6" → px-6 적용
219
+ )}
220
+ />
221
+ ```
222
+
223
+ ## 우선순위 명확화
224
+
225
+ ```tsx
226
+ // 1. 기본 스타일 (lowest)
227
+ // 2. variant 스타일
228
+ // 3. classNames prop
229
+ // 4. className prop (highest)
230
+
231
+ export function Button({
232
+ variant,
233
+ classNames,
234
+ className,
235
+ ...props
236
+ }) {
237
+ return (
238
+ <button
239
+ className={cn(
240
+ // 1. Base
241
+ "inline-flex items-center",
242
+ // 2. Variant
243
+ buttonVariants({ variant }),
244
+ // 3. classNames slot
245
+ classNames?.button,
246
+ // 4. className (최우선)
247
+ className
248
+ )}
249
+ {...props}
250
+ />
251
+ );
252
+ }
253
+ ```
254
+
255
+ Reference: [Material UI sx prop](https://mui.com/system/getting-started/the-sx-prop/)
@@ -0,0 +1,205 @@
1
+ ---
2
+ title: Design Tokens
3
+ impact: HIGH
4
+ impactDescription: Consistent design system foundation
5
+ tags: styling, tokens, design-system, variables
6
+ ---
7
+
8
+ ## Design Tokens
9
+
10
+ **Impact: HIGH (Consistent design system foundation)**
11
+
12
+ 디자인 토큰을 사용하여 일관된 스타일 시스템을 구축하세요.
13
+
14
+ ## CSS Variables 기반 토큰
15
+
16
+ ```css
17
+ /* app/globals.css */
18
+ @layer base {
19
+ :root {
20
+ /* Colors - HSL format for opacity support */
21
+ --mandu-background: 0 0% 100%;
22
+ --mandu-foreground: 222.2 84% 4.9%;
23
+
24
+ --mandu-card: 0 0% 100%;
25
+ --mandu-card-foreground: 222.2 84% 4.9%;
26
+
27
+ --mandu-primary: 221.2 83.2% 53.3%;
28
+ --mandu-primary-foreground: 210 40% 98%;
29
+
30
+ --mandu-secondary: 210 40% 96.1%;
31
+ --mandu-secondary-foreground: 222.2 47.4% 11.2%;
32
+
33
+ --mandu-muted: 210 40% 96.1%;
34
+ --mandu-muted-foreground: 215.4 16.3% 46.9%;
35
+
36
+ --mandu-accent: 210 40% 96.1%;
37
+ --mandu-accent-foreground: 222.2 47.4% 11.2%;
38
+
39
+ --mandu-destructive: 0 84.2% 60.2%;
40
+ --mandu-destructive-foreground: 210 40% 98%;
41
+
42
+ --mandu-border: 214.3 31.8% 91.4%;
43
+ --mandu-input: 214.3 31.8% 91.4%;
44
+ --mandu-ring: 221.2 83.2% 53.3%;
45
+
46
+ /* Spacing */
47
+ --mandu-spacing-xs: 0.25rem;
48
+ --mandu-spacing-sm: 0.5rem;
49
+ --mandu-spacing-md: 1rem;
50
+ --mandu-spacing-lg: 1.5rem;
51
+ --mandu-spacing-xl: 2rem;
52
+
53
+ /* Typography */
54
+ --mandu-font-sans: ui-sans-serif, system-ui, sans-serif;
55
+ --mandu-font-mono: ui-monospace, monospace;
56
+
57
+ --mandu-text-xs: 0.75rem;
58
+ --mandu-text-sm: 0.875rem;
59
+ --mandu-text-base: 1rem;
60
+ --mandu-text-lg: 1.125rem;
61
+ --mandu-text-xl: 1.25rem;
62
+
63
+ /* Borders */
64
+ --mandu-radius-sm: 0.25rem;
65
+ --mandu-radius-md: 0.5rem;
66
+ --mandu-radius-lg: 0.75rem;
67
+ --mandu-radius-full: 9999px;
68
+
69
+ /* Shadows */
70
+ --mandu-shadow-sm: 0 1px 2px 0 rgb(0 0 0 / 0.05);
71
+ --mandu-shadow-md: 0 4px 6px -1px rgb(0 0 0 / 0.1);
72
+ --mandu-shadow-lg: 0 10px 15px -3px rgb(0 0 0 / 0.1);
73
+
74
+ /* Transitions */
75
+ --mandu-transition-fast: 150ms;
76
+ --mandu-transition-normal: 200ms;
77
+ --mandu-transition-slow: 300ms;
78
+ }
79
+
80
+ .dark {
81
+ --mandu-background: 222.2 84% 4.9%;
82
+ --mandu-foreground: 210 40% 98%;
83
+
84
+ --mandu-card: 222.2 84% 4.9%;
85
+ --mandu-card-foreground: 210 40% 98%;
86
+
87
+ --mandu-primary: 217.2 91.2% 59.8%;
88
+ --mandu-primary-foreground: 222.2 47.4% 11.2%;
89
+
90
+ --mandu-secondary: 217.2 32.6% 17.5%;
91
+ --mandu-secondary-foreground: 210 40% 98%;
92
+
93
+ --mandu-muted: 217.2 32.6% 17.5%;
94
+ --mandu-muted-foreground: 215 20.2% 65.1%;
95
+
96
+ --mandu-accent: 217.2 32.6% 17.5%;
97
+ --mandu-accent-foreground: 210 40% 98%;
98
+
99
+ --mandu-destructive: 0 62.8% 30.6%;
100
+ --mandu-destructive-foreground: 210 40% 98%;
101
+
102
+ --mandu-border: 217.2 32.6% 17.5%;
103
+ --mandu-input: 217.2 32.6% 17.5%;
104
+ --mandu-ring: 224.3 76.3% 48%;
105
+ }
106
+ }
107
+ ```
108
+
109
+ ## Tailwind 토큰 연결
110
+
111
+ ```typescript
112
+ // tailwind.config.ts
113
+ export default {
114
+ theme: {
115
+ extend: {
116
+ colors: {
117
+ background: "hsl(var(--mandu-background))",
118
+ foreground: "hsl(var(--mandu-foreground))",
119
+ card: {
120
+ DEFAULT: "hsl(var(--mandu-card))",
121
+ foreground: "hsl(var(--mandu-card-foreground))",
122
+ },
123
+ primary: {
124
+ DEFAULT: "hsl(var(--mandu-primary))",
125
+ foreground: "hsl(var(--mandu-primary-foreground))",
126
+ },
127
+ secondary: {
128
+ DEFAULT: "hsl(var(--mandu-secondary))",
129
+ foreground: "hsl(var(--mandu-secondary-foreground))",
130
+ },
131
+ muted: {
132
+ DEFAULT: "hsl(var(--mandu-muted))",
133
+ foreground: "hsl(var(--mandu-muted-foreground))",
134
+ },
135
+ accent: {
136
+ DEFAULT: "hsl(var(--mandu-accent))",
137
+ foreground: "hsl(var(--mandu-accent-foreground))",
138
+ },
139
+ destructive: {
140
+ DEFAULT: "hsl(var(--mandu-destructive))",
141
+ foreground: "hsl(var(--mandu-destructive-foreground))",
142
+ },
143
+ border: "hsl(var(--mandu-border))",
144
+ input: "hsl(var(--mandu-input))",
145
+ ring: "hsl(var(--mandu-ring))",
146
+ },
147
+ borderRadius: {
148
+ sm: "var(--mandu-radius-sm)",
149
+ md: "var(--mandu-radius-md)",
150
+ lg: "var(--mandu-radius-lg)",
151
+ },
152
+ fontFamily: {
153
+ sans: ["var(--mandu-font-sans)"],
154
+ mono: ["var(--mandu-font-mono)"],
155
+ },
156
+ },
157
+ },
158
+ };
159
+ ```
160
+
161
+ ## 토큰 사용 예시
162
+
163
+ ```tsx
164
+ // 자동으로 다크모드 지원
165
+ <div className="bg-background text-foreground">
166
+ <button className="bg-primary text-primary-foreground rounded-md">
167
+ Button
168
+ </button>
169
+ </div>
170
+
171
+ // HSL로 opacity 조절 가능
172
+ <div className="bg-primary/50">
173
+ 50% opacity primary
174
+ </div>
175
+ ```
176
+
177
+ ## 토큰 타입 정의 (TypeScript)
178
+
179
+ ```typescript
180
+ // types/tokens.ts
181
+ export interface DesignTokens {
182
+ colors: {
183
+ background: string;
184
+ foreground: string;
185
+ primary: string;
186
+ secondary: string;
187
+ // ...
188
+ };
189
+ spacing: {
190
+ xs: string;
191
+ sm: string;
192
+ md: string;
193
+ lg: string;
194
+ xl: string;
195
+ };
196
+ radii: {
197
+ sm: string;
198
+ md: string;
199
+ lg: string;
200
+ full: string;
201
+ };
202
+ }
203
+ ```
204
+
205
+ Reference: [Design Tokens W3C](https://www.w3.org/community/design-tokens/)