@mandujs/mcp 0.13.0 → 0.16.0
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 +6 -7
- package/package.json +3 -2
- package/src/adapters/index.ts +20 -20
- package/src/adapters/monitor-adapter.ts +100 -100
- package/src/adapters/tool-adapter.ts +88 -88
- package/src/executor/error-handler.ts +250 -250
- package/src/executor/index.ts +22 -22
- package/src/executor/tool-executor.ts +148 -148
- package/src/hooks/config-watcher.ts +174 -174
- package/src/hooks/index.ts +23 -23
- package/src/hooks/mcp-hooks.ts +227 -227
- package/src/logging/index.ts +15 -15
- package/src/logging/mcp-transport.ts +134 -134
- package/src/registry/index.ts +13 -13
- package/src/registry/mcp-tool-registry.ts +298 -298
- package/src/resources/skills/guides.ts +1136 -1136
- package/src/resources/skills/index.ts +12 -12
- package/src/resources/skills/loader.ts +218 -218
- package/src/resources/skills/mandu-composition/SKILL.md +91 -91
- package/src/resources/skills/mandu-composition/metadata.json +13 -13
- package/src/resources/skills/mandu-composition/rules/_sections.md +26 -26
- package/src/resources/skills/mandu-composition/rules/_template.md +77 -77
- package/src/resources/skills/mandu-composition/rules/comp-arch-avoid-boolean-props.md +146 -146
- package/src/resources/skills/mandu-composition/rules/comp-arch-compound-components.md +164 -164
- package/src/resources/skills/mandu-composition/rules/comp-island-event.md +161 -161
- package/src/resources/skills/mandu-composition/rules/comp-island-slot-split.md +167 -167
- package/src/resources/skills/mandu-composition/rules/comp-pattern-children.md +149 -149
- package/src/resources/skills/mandu-composition/rules/comp-state-context-interface.md +148 -148
- package/src/resources/skills/mandu-composition/rules/comp-state-lift-state.md +150 -150
- package/src/resources/skills/mandu-deployment/SKILL.md +92 -92
- package/src/resources/skills/mandu-deployment/_sections.md +41 -41
- package/src/resources/skills/mandu-deployment/_template.md +38 -38
- package/src/resources/skills/mandu-deployment/metadata.json +13 -13
- package/src/resources/skills/mandu-deployment/rules/deploy-build-bun.md +109 -109
- package/src/resources/skills/mandu-deployment/rules/deploy-build-output.md +115 -115
- package/src/resources/skills/mandu-deployment/rules/deploy-cicd-github.md +219 -219
- package/src/resources/skills/mandu-deployment/rules/deploy-docker-bun.md +150 -150
- package/src/resources/skills/mandu-deployment/rules/deploy-docker-compose.md +223 -223
- package/src/resources/skills/mandu-deployment/rules/deploy-platform-fly.md +152 -152
- package/src/resources/skills/mandu-deployment/rules/deploy-platform-render.md +179 -179
- package/src/resources/skills/mandu-deployment/rules/deploy-platform-supabase.md +323 -323
- package/src/resources/skills/mandu-deployment/rules/deploy-platform-vercel.md +140 -140
- package/src/resources/skills/mandu-fs-routes/SKILL.md +82 -82
- package/src/resources/skills/mandu-fs-routes/metadata.json +12 -12
- package/src/resources/skills/mandu-fs-routes/rules/_sections.md +36 -36
- package/src/resources/skills/mandu-fs-routes/rules/_template.md +69 -69
- package/src/resources/skills/mandu-fs-routes/rules/routes-api-methods.md +65 -65
- package/src/resources/skills/mandu-fs-routes/rules/routes-dynamic-param.md +93 -93
- package/src/resources/skills/mandu-fs-routes/rules/routes-naming-page.md +55 -55
- package/src/resources/skills/mandu-guard/SKILL.md +129 -129
- package/src/resources/skills/mandu-guard/metadata.json +12 -12
- package/src/resources/skills/mandu-guard/rules/_sections.md +36 -36
- package/src/resources/skills/mandu-guard/rules/_template.md +82 -82
- package/src/resources/skills/mandu-guard/rules/guard-config-rules.md +100 -100
- package/src/resources/skills/mandu-guard/rules/guard-layer-direction.md +76 -76
- package/src/resources/skills/mandu-guard/rules/guard-preset-mandu.md +81 -81
- package/src/resources/skills/mandu-guard/rules/guard-validate-import.md +80 -80
- package/src/resources/skills/mandu-hydration/SKILL.md +91 -91
- package/src/resources/skills/mandu-hydration/metadata.json +12 -12
- package/src/resources/skills/mandu-hydration/rules/_sections.md +31 -31
- package/src/resources/skills/mandu-hydration/rules/_template.md +72 -72
- package/src/resources/skills/mandu-hydration/rules/hydration-data-event.md +109 -109
- package/src/resources/skills/mandu-hydration/rules/hydration-directive-use-client.md +55 -55
- package/src/resources/skills/mandu-hydration/rules/hydration-island-setup.md +113 -113
- package/src/resources/skills/mandu-hydration/rules/hydration-priority-visible.md +68 -68
- package/src/resources/skills/mandu-performance/SKILL.md +85 -85
- package/src/resources/skills/mandu-performance/metadata.json +14 -14
- package/src/resources/skills/mandu-performance/rules/_sections.md +31 -31
- package/src/resources/skills/mandu-performance/rules/_template.md +64 -64
- package/src/resources/skills/mandu-performance/rules/perf-async-defer-await.md +103 -103
- package/src/resources/skills/mandu-performance/rules/perf-async-parallel.md +95 -95
- package/src/resources/skills/mandu-performance/rules/perf-bun-file.md +124 -124
- package/src/resources/skills/mandu-performance/rules/perf-bun-serve.md +125 -125
- package/src/resources/skills/mandu-performance/rules/perf-bundle-imports.md +80 -80
- package/src/resources/skills/mandu-performance/rules/perf-bundle-island-lazy.md +145 -145
- package/src/resources/skills/mandu-performance/rules/perf-cache-react.md +98 -98
- package/src/resources/skills/mandu-performance/rules/perf-render-transitions.md +154 -154
- package/src/resources/skills/mandu-security/SKILL.md +87 -87
- package/src/resources/skills/mandu-security/metadata.json +13 -13
- package/src/resources/skills/mandu-security/rules/_sections.md +31 -31
- package/src/resources/skills/mandu-security/rules/_template.md +74 -74
- package/src/resources/skills/mandu-security/rules/sec-auth-guard.md +127 -127
- package/src/resources/skills/mandu-security/rules/sec-env-management.md +133 -133
- package/src/resources/skills/mandu-security/rules/sec-input-validate.md +148 -148
- package/src/resources/skills/mandu-security/rules/sec-protect-csrf.md +146 -146
- package/src/resources/skills/mandu-security/rules/sec-protect-headers.md +138 -138
- package/src/resources/skills/mandu-slot/SKILL.md +85 -85
- package/src/resources/skills/mandu-slot/metadata.json +12 -12
- package/src/resources/skills/mandu-slot/rules/_sections.md +36 -36
- package/src/resources/skills/mandu-slot/rules/_template.md +63 -63
- package/src/resources/skills/mandu-slot/rules/slot-basic-structure.md +38 -38
- package/src/resources/skills/mandu-slot/rules/slot-ctx-response.md +56 -56
- package/src/resources/skills/mandu-slot/rules/slot-guard-auth.md +59 -59
- package/src/resources/skills/mandu-slot/rules/slot-http-methods.md +64 -64
- package/src/resources/skills/mandu-styling/SKILL.md +154 -154
- package/src/resources/skills/mandu-styling/_sections.md +43 -43
- package/src/resources/skills/mandu-styling/_template.md +32 -32
- package/src/resources/skills/mandu-styling/metadata.json +15 -15
- package/src/resources/skills/mandu-styling/rules/style-component-compound.md +235 -235
- package/src/resources/skills/mandu-styling/rules/style-component-slots.md +255 -255
- package/src/resources/skills/mandu-styling/rules/style-component-tokens.md +205 -205
- package/src/resources/skills/mandu-styling/rules/style-island-animations.md +272 -272
- package/src/resources/skills/mandu-styling/rules/style-island-scoping.md +167 -167
- package/src/resources/skills/mandu-styling/rules/style-island-variants.md +221 -221
- package/src/resources/skills/mandu-styling/rules/style-perf-critical.md +209 -209
- package/src/resources/skills/mandu-styling/rules/style-perf-purge.md +192 -192
- package/src/resources/skills/mandu-styling/rules/style-setup-modules.md +162 -162
- package/src/resources/skills/mandu-styling/rules/style-setup-panda.md +164 -164
- package/src/resources/skills/mandu-styling/rules/style-setup-tailwind.md +170 -170
- package/src/resources/skills/mandu-styling/rules/style-tailwind-v4-gotchas.md +179 -179
- package/src/resources/skills/mandu-styling/rules/style-theme-darkmode.md +229 -229
- package/src/resources/skills/mandu-testing/SKILL.md +99 -99
- package/src/resources/skills/mandu-testing/metadata.json +13 -13
- package/src/resources/skills/mandu-testing/rules/_sections.md +26 -26
- package/src/resources/skills/mandu-testing/rules/_template.md +65 -65
- package/src/resources/skills/mandu-testing/rules/test-component-island.md +195 -195
- package/src/resources/skills/mandu-testing/rules/test-e2e-playwright.md +196 -196
- package/src/resources/skills/mandu-testing/rules/test-mock-fetch.md +219 -219
- package/src/resources/skills/mandu-testing/rules/test-slot-unit.md +192 -192
- package/src/resources/skills/mandu-ui/SKILL.md +117 -117
- package/src/resources/skills/mandu-ui/_sections.md +23 -23
- package/src/resources/skills/mandu-ui/_template.md +32 -32
- package/src/resources/skills/mandu-ui/metadata.json +13 -13
- package/src/resources/skills/mandu-ui/rules/ui-accessibility-aria.md +232 -232
- package/src/resources/skills/mandu-ui/rules/ui-accessibility-focus.md +238 -238
- package/src/resources/skills/mandu-ui/rules/ui-composition-patterns.md +259 -259
- package/src/resources/skills/mandu-ui/rules/ui-island-integration.md +258 -258
- package/src/resources/skills/mandu-ui/rules/ui-radix-patterns.md +213 -213
- package/src/resources/skills/mandu-ui/rules/ui-shadcn-setup.md +209 -209
- package/src/resources/skills/recipes.ts +932 -932
- package/src/tools/ate.ts +129 -0
- package/src/tools/index.ts +4 -1
- package/src/tools/project.ts +334 -334
- package/src/tools/runtime.ts +497 -497
- package/src/tools/seo.ts +417 -417
- package/src/utils/withWarnings.ts +83 -83
|
@@ -1,164 +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)
|
|
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)
|
|
@@ -1,161 +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 |
|
|
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 |
|