@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,164 @@
1
+ ---
2
+ title: Structure Islands as Compound Components
3
+ impact: HIGH
4
+ impactDescription: Enables flexible composition without prop drilling
5
+ tags: composition, compound, architecture, island
6
+ ---
7
+
8
+ ## Structure Islands as Compound Components
9
+
10
+ **Impact: HIGH (Enables flexible composition without prop drilling)**
11
+
12
+ 복잡한 Island를 컴파운드 컴포넌트로 구조화하세요. 각 서브컴포넌트는 props가 아닌 context로 공유 상태에 접근합니다.
13
+
14
+ **Incorrect (모놀리식 컴포넌트):**
15
+
16
+ ```tsx
17
+ // app/composer/client.tsx
18
+ "use client";
19
+
20
+ function ComposerIsland({
21
+ renderHeader,
22
+ renderFooter,
23
+ renderActions,
24
+ showAttachments,
25
+ showFormatting,
26
+ showEmojis,
27
+ }: Props) {
28
+ const [input, setInput] = useState("");
29
+
30
+ return (
31
+ <form>
32
+ {renderHeader?.()}
33
+ <Input value={input} onChange={setInput} />
34
+ {showAttachments && <Attachments />}
35
+ {renderFooter ? (
36
+ renderFooter()
37
+ ) : (
38
+ <Footer>
39
+ {showFormatting && <Formatting />}
40
+ {showEmojis && <Emojis />}
41
+ {renderActions?.()}
42
+ </Footer>
43
+ )}
44
+ </form>
45
+ );
46
+ }
47
+ ```
48
+
49
+ **Correct (컴파운드 Island):**
50
+
51
+ ```tsx
52
+ // app/composer/client.tsx
53
+ "use client";
54
+
55
+ import { createContext, use, useState, useCallback } from "react";
56
+
57
+ // Context 정의
58
+ interface ComposerContextValue {
59
+ state: { input: string; attachments: File[] };
60
+ actions: {
61
+ updateInput: (text: string) => void;
62
+ submit: () => void;
63
+ };
64
+ }
65
+
66
+ const ComposerContext = createContext<ComposerContextValue | null>(null);
67
+
68
+ // Provider
69
+ function ComposerProvider({ children, onSubmit }: ProviderProps) {
70
+ const [input, setInput] = useState("");
71
+ const [attachments, setAttachments] = useState<File[]>([]);
72
+
73
+ const submit = useCallback(() => {
74
+ onSubmit?.({ input, attachments });
75
+ setInput("");
76
+ setAttachments([]);
77
+ }, [input, attachments, onSubmit]);
78
+
79
+ return (
80
+ <ComposerContext value={{
81
+ state: { input, attachments },
82
+ actions: { updateInput: setInput, submit },
83
+ }}>
84
+ {children}
85
+ </ComposerContext>
86
+ );
87
+ }
88
+
89
+ // 서브컴포넌트들
90
+ function ComposerFrame({ children }: { children: React.ReactNode }) {
91
+ return <form onSubmit={(e) => e.preventDefault()}>{children}</form>;
92
+ }
93
+
94
+ function ComposerInput() {
95
+ const { state, actions } = use(ComposerContext)!;
96
+ return (
97
+ <textarea
98
+ value={state.input}
99
+ onChange={(e) => actions.updateInput(e.target.value)}
100
+ placeholder="Type a message..."
101
+ />
102
+ );
103
+ }
104
+
105
+ function ComposerSubmit() {
106
+ const { actions } = use(ComposerContext)!;
107
+ return <button onClick={actions.submit}>Send</button>;
108
+ }
109
+
110
+ function ComposerEmojis() {
111
+ const { actions } = use(ComposerContext)!;
112
+ return (
113
+ <EmojiPicker onSelect={(emoji) => {
114
+ actions.updateInput((prev) => prev + emoji);
115
+ }} />
116
+ );
117
+ }
118
+
119
+ // 컴파운드로 export
120
+ export const Composer = {
121
+ Provider: ComposerProvider,
122
+ Frame: ComposerFrame,
123
+ Input: ComposerInput,
124
+ Submit: ComposerSubmit,
125
+ Emojis: ComposerEmojis,
126
+ Attachments: ComposerAttachments,
127
+ Formatting: ComposerFormatting,
128
+ };
129
+ ```
130
+
131
+ **사용법:**
132
+
133
+ ```tsx
134
+ // app/chat/page.tsx
135
+ import { Composer } from "../composer/client";
136
+
137
+ export default function ChatPage() {
138
+ return (
139
+ <div>
140
+ <h1>Chat</h1>
141
+
142
+ {/* 필요한 조각만 조합 */}
143
+ <Composer.Provider onSubmit={handleSubmit}>
144
+ <Composer.Frame>
145
+ <Composer.Input />
146
+ <footer>
147
+ <Composer.Emojis />
148
+ <Composer.Submit />
149
+ </footer>
150
+ </Composer.Frame>
151
+ </Composer.Provider>
152
+ </div>
153
+ );
154
+ }
155
+ ```
156
+
157
+ ## 장점
158
+
159
+ - 소비자가 필요한 것만 명시적으로 조합
160
+ - 숨겨진 조건문 없음
161
+ - state/actions가 Provider에 의해 주입됨
162
+ - 같은 컴포넌트 구조를 다양한 구현과 재사용 가능
163
+
164
+ Reference: [Compound Components Pattern](https://www.patterns.dev/react/compound-pattern)
@@ -0,0 +1,161 @@
1
+ ---
2
+ title: Communicate Between Islands with useIslandEvent
3
+ impact: MEDIUM
4
+ impactDescription: Enables Island-to-Island data flow
5
+ tags: composition, island, event, communication
6
+ ---
7
+
8
+ ## Communicate Between Islands with useIslandEvent
9
+
10
+ **Impact: MEDIUM (Enables Island-to-Island data flow)**
11
+
12
+ Island는 기본적으로 격리되어 있습니다. `useIslandEvent`를 사용하여 Island 간 통신하세요.
13
+
14
+ **Incorrect (전역 상태 공유):**
15
+
16
+ ```tsx
17
+ // ❌ 전역 변수로 Island 간 통신
18
+ let globalCount = 0;
19
+
20
+ // Island A
21
+ export function CounterIsland() {
22
+ const [count, setCount] = useState(globalCount);
23
+ // 다른 Island와 동기화되지 않음
24
+ }
25
+ ```
26
+
27
+ **Correct (useIslandEvent):**
28
+
29
+ ```tsx
30
+ // Island A: Counter (이벤트 발송)
31
+ "use client";
32
+
33
+ import { useState } from "react";
34
+ import { useIslandEvent } from "@mandujs/core/client";
35
+
36
+ export function CounterIsland() {
37
+ const [count, setCount] = useState(0);
38
+ const { emit } = useIslandEvent<{ count: number }>("counter-update");
39
+
40
+ const increment = () => {
41
+ const newCount = count + 1;
42
+ setCount(newCount);
43
+ emit({ count: newCount }); // 다른 Island에 알림
44
+ };
45
+
46
+ return <button onClick={increment}>Count: {count}</button>;
47
+ }
48
+ ```
49
+
50
+ ```tsx
51
+ // Island B: Display (이벤트 수신)
52
+ "use client";
53
+
54
+ import { useState } from "react";
55
+ import { useIslandEvent } from "@mandujs/core/client";
56
+
57
+ export function DisplayIsland() {
58
+ const [lastCount, setLastCount] = useState(0);
59
+
60
+ useIslandEvent<{ count: number }>("counter-update", (data) => {
61
+ setLastCount(data.count); // 카운터 업데이트에 반응
62
+ });
63
+
64
+ return <p>Last count received: {lastCount}</p>;
65
+ }
66
+ ```
67
+
68
+ ## 실용적인 패턴
69
+
70
+ ### 장바구니 업데이트
71
+
72
+ ```tsx
73
+ // Product Island
74
+ function ProductIsland({ product }) {
75
+ const { emit } = useIslandEvent("cart-update");
76
+
77
+ const addToCart = () => {
78
+ emit({ action: "add", productId: product.id, quantity: 1 });
79
+ };
80
+
81
+ return <button onClick={addToCart}>Add to Cart</button>;
82
+ }
83
+
84
+ // Cart Island
85
+ function CartIsland() {
86
+ const [items, setItems] = useState([]);
87
+
88
+ useIslandEvent<CartEvent>("cart-update", ({ action, productId, quantity }) => {
89
+ if (action === "add") {
90
+ setItems(prev => [...prev, { productId, quantity }]);
91
+ }
92
+ });
93
+
94
+ return <CartSummary items={items} />;
95
+ }
96
+
97
+ // Header Cart Badge Island
98
+ function CartBadgeIsland() {
99
+ const [count, setCount] = useState(0);
100
+
101
+ useIslandEvent<CartEvent>("cart-update", ({ action }) => {
102
+ if (action === "add") setCount(c => c + 1);
103
+ if (action === "remove") setCount(c => c - 1);
104
+ });
105
+
106
+ return <span className="badge">{count}</span>;
107
+ }
108
+ ```
109
+
110
+ ### 필터 동기화
111
+
112
+ ```tsx
113
+ // Filter Island
114
+ function FilterIsland() {
115
+ const { emit } = useIslandEvent("filter-change");
116
+ const [filters, setFilters] = useState({});
117
+
118
+ const updateFilter = (key, value) => {
119
+ const newFilters = { ...filters, [key]: value };
120
+ setFilters(newFilters);
121
+ emit(newFilters);
122
+ };
123
+
124
+ return <FilterControls filters={filters} onChange={updateFilter} />;
125
+ }
126
+
127
+ // Product List Island
128
+ function ProductListIsland({ initialProducts }) {
129
+ const [products, setProducts] = useState(initialProducts);
130
+
131
+ useIslandEvent<Filters>("filter-change", async (filters) => {
132
+ const filtered = await fetchProducts(filters);
133
+ setProducts(filtered);
134
+ });
135
+
136
+ return <ProductGrid products={products} />;
137
+ }
138
+ ```
139
+
140
+ ## API 요약
141
+
142
+ ```typescript
143
+ // 이벤트 발송
144
+ const { emit } = useIslandEvent<T>(eventName);
145
+ emit(data);
146
+
147
+ // 이벤트 수신
148
+ useIslandEvent<T>(eventName, (data) => { ... });
149
+
150
+ // 발송 + 수신
151
+ const { emit } = useIslandEvent<T>(eventName, (data) => { ... });
152
+ ```
153
+
154
+ ## 언제 useIslandEvent를 사용하나?
155
+
156
+ | 상황 | 권장 방법 |
157
+ |------|-----------|
158
+ | 같은 Island 내 상태 공유 | useState/useContext |
159
+ | 부모-자식 Island | props |
160
+ | 형제 Island 간 통신 | **useIslandEvent** |
161
+ | 페이지 전체 상태 | slot + server state |
@@ -0,0 +1,167 @@
1
+ ---
2
+ title: Separate Server Logic (Slot) from Client (Island)
3
+ impact: MEDIUM
4
+ impactDescription: Clear server-client boundary
5
+ tags: composition, island, slot, separation
6
+ ---
7
+
8
+ ## Separate Server Logic (Slot) from Client (Island)
9
+
10
+ **Impact: MEDIUM (Clear server-client boundary)**
11
+
12
+ 서버 로직(slot)과 클라이언트 로직(Island)을 명확히 분리하세요. slot은 데이터 페칭과 비즈니스 로직, Island는 인터랙션을 담당합니다.
13
+
14
+ **Incorrect (혼합된 관심사):**
15
+
16
+ ```tsx
17
+ // ❌ 클라이언트 컴포넌트에서 데이터 페칭
18
+ "use client";
19
+
20
+ export function TodosIsland() {
21
+ const [todos, setTodos] = useState([]);
22
+
23
+ useEffect(() => {
24
+ // 클라이언트에서 데이터 페칭 → 워터폴 발생
25
+ fetch("/api/todos")
26
+ .then(res => res.json())
27
+ .then(setTodos);
28
+ }, []);
29
+
30
+ return <TodoList todos={todos} />;
31
+ }
32
+ ```
33
+
34
+ **Correct (slot-client 분리):**
35
+
36
+ ```typescript
37
+ // spec/slots/todos.slot.ts - 서버 로직
38
+ import { Mandu } from "@mandujs/core";
39
+ import { db } from "@/lib/db";
40
+
41
+ export default Mandu.filling()
42
+ .guard((ctx) => {
43
+ // 인증 체크 (서버에서)
44
+ if (!ctx.get("user")) {
45
+ return ctx.unauthorized("Login required");
46
+ }
47
+ })
48
+ .get(async (ctx) => {
49
+ // 데이터 페칭 (서버에서)
50
+ const todos = await db.todo.findMany({
51
+ where: { userId: ctx.get("user").id },
52
+ orderBy: { createdAt: "desc" },
53
+ });
54
+
55
+ return ctx.ok({ todos });
56
+ })
57
+ .post(async (ctx) => {
58
+ // 생성 로직 (서버에서)
59
+ const body = await ctx.body<{ text: string }>();
60
+ const todo = await db.todo.create({
61
+ data: { text: body.text, userId: ctx.get("user").id },
62
+ });
63
+
64
+ return ctx.created({ todo });
65
+ });
66
+ ```
67
+
68
+ ```tsx
69
+ // app/todos/client.tsx - 클라이언트 인터랙션
70
+ "use client";
71
+
72
+ import { useState, useCallback } from "react";
73
+
74
+ interface TodosIslandProps {
75
+ initialTodos: Todo[]; // 서버에서 받은 초기 데이터
76
+ }
77
+
78
+ export function TodosIsland({ initialTodos }: TodosIslandProps) {
79
+ const [todos, setTodos] = useState(initialTodos);
80
+ const [input, setInput] = useState("");
81
+
82
+ const addTodo = useCallback(async () => {
83
+ // 낙관적 업데이트
84
+ const optimisticTodo = { id: Date.now(), text: input, done: false };
85
+ setTodos(prev => [optimisticTodo, ...prev]);
86
+ setInput("");
87
+
88
+ // 서버에 요청
89
+ const res = await fetch("/api/todos", {
90
+ method: "POST",
91
+ body: JSON.stringify({ text: input }),
92
+ });
93
+ const { todo } = await res.json();
94
+
95
+ // 실제 데이터로 교체
96
+ setTodos(prev => prev.map(t =>
97
+ t.id === optimisticTodo.id ? todo : t
98
+ ));
99
+ }, [input]);
100
+
101
+ return (
102
+ <div>
103
+ <input
104
+ value={input}
105
+ onChange={(e) => setInput(e.target.value)}
106
+ placeholder="New todo..."
107
+ />
108
+ <button onClick={addTodo}>Add</button>
109
+ <ul>
110
+ {todos.map(todo => (
111
+ <TodoItem key={todo.id} todo={todo} />
112
+ ))}
113
+ </ul>
114
+ </div>
115
+ );
116
+ }
117
+ ```
118
+
119
+ ```tsx
120
+ // app/todos/page.tsx - 페이지 (서버 컴포넌트)
121
+ import { TodosIsland } from "./client";
122
+ import { loadTodos } from "./slot";
123
+
124
+ export default async function TodosPage() {
125
+ // 서버에서 데이터 로드 (워터폴 없음)
126
+ const { todos } = await loadTodos();
127
+
128
+ return (
129
+ <div>
130
+ <h1>My Todos</h1>
131
+ {/* 초기 데이터를 Island에 전달 */}
132
+ <TodosIsland initialTodos={todos} />
133
+ </div>
134
+ );
135
+ }
136
+ ```
137
+
138
+ ## 분리 원칙
139
+
140
+ | 관심사 | 위치 | 예시 |
141
+ |--------|------|------|
142
+ | 데이터 페칭 | Slot | DB 쿼리, API 호출 |
143
+ | 인증/인가 | Slot Guard | 권한 체크 |
144
+ | 비즈니스 로직 | Slot | 유효성 검사, 계산 |
145
+ | 인터랙션 | Island | 클릭, 입력, 애니메이션 |
146
+ | 클라이언트 상태 | Island | useState, useReducer |
147
+ | 낙관적 업데이트 | Island | 즉시 UI 반영 |
148
+
149
+ ## 데이터 흐름
150
+
151
+ ```
152
+ ┌─────────────────────────────────────────┐
153
+ │ Page (Server Component) │
154
+ │ └─ loadTodos() from slot │
155
+ │ └─ DB Query │
156
+ │ │
157
+ │ ↓ initialTodos prop │
158
+ │ │
159
+ │ ┌───────────────────────────────────┐ │
160
+ │ │ TodosIsland (Client Component) │ │
161
+ │ │ └─ useState(initialTodos) │ │
162
+ │ │ └─ User interactions │ │
163
+ │ │ └─ Optimistic updates │ │
164
+ │ │ └─ POST to /api/todos │ │
165
+ │ └───────────────────────────────────┘ │
166
+ └─────────────────────────────────────────┘
167
+ ```
@@ -0,0 +1,149 @@
1
+ ---
2
+ title: Use Children for Composition Over Render Props
3
+ impact: MEDIUM
4
+ impactDescription: Simpler API, better composition
5
+ tags: composition, children, render-props, pattern
6
+ ---
7
+
8
+ ## Use Children for Composition Over Render Props
9
+
10
+ **Impact: MEDIUM (Simpler API, better composition)**
11
+
12
+ `renderX` props 대신 `children`을 사용하여 컴포지션하세요. 더 선언적이고 유연합니다.
13
+
14
+ **Incorrect (render props):**
15
+
16
+ ```tsx
17
+ // ❌ renderX props 패턴
18
+ function Modal({
19
+ renderHeader,
20
+ renderBody,
21
+ renderFooter,
22
+ renderCloseButton,
23
+ }: ModalProps) {
24
+ return (
25
+ <div className="modal">
26
+ <div className="modal-header">
27
+ {renderCloseButton?.()}
28
+ {renderHeader?.()}
29
+ </div>
30
+ <div className="modal-body">
31
+ {renderBody?.()}
32
+ </div>
33
+ <div className="modal-footer">
34
+ {renderFooter?.()}
35
+ </div>
36
+ </div>
37
+ );
38
+ }
39
+
40
+ // 사용 시 복잡함
41
+ <Modal
42
+ renderHeader={() => <h2>Title</h2>}
43
+ renderBody={() => <p>Content</p>}
44
+ renderFooter={() => (
45
+ <>
46
+ <Button>Cancel</Button>
47
+ <Button>Save</Button>
48
+ </>
49
+ )}
50
+ renderCloseButton={() => <CloseButton />}
51
+ />
52
+ ```
53
+
54
+ **Correct (children + compound):**
55
+
56
+ ```tsx
57
+ // ✅ children과 컴파운드 패턴
58
+ function Modal({ children }: { children: React.ReactNode }) {
59
+ return <div className="modal">{children}</div>;
60
+ }
61
+
62
+ function ModalHeader({ children }: { children: React.ReactNode }) {
63
+ return <div className="modal-header">{children}</div>;
64
+ }
65
+
66
+ function ModalBody({ children }: { children: React.ReactNode }) {
67
+ return <div className="modal-body">{children}</div>;
68
+ }
69
+
70
+ function ModalFooter({ children }: { children: React.ReactNode }) {
71
+ return <div className="modal-footer">{children}</div>;
72
+ }
73
+
74
+ function ModalClose({ onClose }: { onClose: () => void }) {
75
+ return <button onClick={onClose} className="modal-close">×</button>;
76
+ }
77
+
78
+ export { Modal, ModalHeader, ModalBody, ModalFooter, ModalClose };
79
+ ```
80
+
81
+ **사용:**
82
+
83
+ ```tsx
84
+ // 선언적이고 명확함
85
+ <Modal>
86
+ <ModalHeader>
87
+ <ModalClose onClose={handleClose} />
88
+ <h2>Title</h2>
89
+ </ModalHeader>
90
+
91
+ <ModalBody>
92
+ <p>Content goes here</p>
93
+ </ModalBody>
94
+
95
+ <ModalFooter>
96
+ <Button onClick={handleClose}>Cancel</Button>
97
+ <Button onClick={handleSave}>Save</Button>
98
+ </ModalFooter>
99
+ </Modal>
100
+ ```
101
+
102
+ ## 장점
103
+
104
+ | render props | children |
105
+ |--------------|----------|
106
+ | 숨겨진 구조 | 명시적 구조 |
107
+ | 함수 호출 문법 | JSX 문법 |
108
+ | 순서가 props에 의존 | 순서를 소비자가 제어 |
109
+ | 어떤 props가 있는지 봐야 함 | 자동완성 지원 |
110
+
111
+ ## Mandu Island에서의 적용
112
+
113
+ ```tsx
114
+ // ❌ 피해야 할 패턴
115
+ <FormIsland
116
+ renderInput={(value, onChange) => <Input value={value} onChange={onChange} />}
117
+ renderSubmit={(onSubmit) => <Button onClick={onSubmit}>Submit</Button>}
118
+ renderError={(error) => <ErrorMessage error={error} />}
119
+ />
120
+
121
+ // ✅ 권장 패턴
122
+ <Form.Provider>
123
+ <Form.Frame>
124
+ <Form.Input name="email" />
125
+ <Form.Input name="password" type="password" />
126
+ <Form.Error />
127
+ <Form.Submit>Sign In</Form.Submit>
128
+ </Form.Frame>
129
+ </Form.Provider>
130
+ ```
131
+
132
+ ## 언제 Render Props를 사용하나?
133
+
134
+ render props가 여전히 유용한 경우:
135
+ - 부모가 데이터를 제공하고 자식이 렌더링 방법을 결정 (예: virtualized list)
136
+ - 자식에게 상태를 노출해야 하는 headless 컴포넌트
137
+
138
+ ```tsx
139
+ // Render props가 적합한 예: Virtualized List
140
+ <VirtualList
141
+ items={items}
142
+ itemHeight={50}
143
+ renderItem={(item, index) => (
144
+ <div key={item.id}>{item.name}</div>
145
+ )}
146
+ />
147
+ ```
148
+
149
+ Reference: [Compound Components](https://kentcdodds.com/blog/compound-components-with-react-hooks)