@mandujs/mcp 0.12.2 → 0.13.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.
Files changed (141) hide show
  1. package/README.md +367 -367
  2. package/package.json +2 -2
  3. package/src/activity-monitor.ts +847 -847
  4. package/src/adapters/index.ts +20 -20
  5. package/src/adapters/monitor-adapter.ts +100 -100
  6. package/src/adapters/tool-adapter.ts +88 -88
  7. package/src/executor/error-handler.ts +250 -250
  8. package/src/executor/index.ts +22 -22
  9. package/src/executor/tool-executor.ts +148 -148
  10. package/src/hooks/config-watcher.ts +174 -174
  11. package/src/hooks/index.ts +23 -23
  12. package/src/hooks/mcp-hooks.ts +227 -227
  13. package/src/index.ts +106 -106
  14. package/src/logging/index.ts +15 -15
  15. package/src/logging/mcp-transport.ts +134 -134
  16. package/src/registry/index.ts +13 -13
  17. package/src/registry/mcp-tool-registry.ts +298 -298
  18. package/src/resources/skills/guides.ts +1136 -1136
  19. package/src/resources/skills/index.ts +12 -12
  20. package/src/resources/skills/loader.ts +218 -218
  21. package/src/resources/skills/mandu-composition/SKILL.md +91 -91
  22. package/src/resources/skills/mandu-composition/metadata.json +13 -13
  23. package/src/resources/skills/mandu-composition/rules/_sections.md +26 -26
  24. package/src/resources/skills/mandu-composition/rules/_template.md +77 -77
  25. package/src/resources/skills/mandu-composition/rules/comp-arch-avoid-boolean-props.md +146 -146
  26. package/src/resources/skills/mandu-composition/rules/comp-arch-compound-components.md +164 -164
  27. package/src/resources/skills/mandu-composition/rules/comp-island-event.md +161 -161
  28. package/src/resources/skills/mandu-composition/rules/comp-island-slot-split.md +167 -167
  29. package/src/resources/skills/mandu-composition/rules/comp-pattern-children.md +149 -149
  30. package/src/resources/skills/mandu-composition/rules/comp-state-context-interface.md +148 -148
  31. package/src/resources/skills/mandu-composition/rules/comp-state-lift-state.md +150 -150
  32. package/src/resources/skills/mandu-deployment/SKILL.md +92 -92
  33. package/src/resources/skills/mandu-deployment/_sections.md +41 -41
  34. package/src/resources/skills/mandu-deployment/_template.md +38 -38
  35. package/src/resources/skills/mandu-deployment/metadata.json +13 -13
  36. package/src/resources/skills/mandu-deployment/rules/deploy-build-bun.md +109 -109
  37. package/src/resources/skills/mandu-deployment/rules/deploy-build-output.md +115 -115
  38. package/src/resources/skills/mandu-deployment/rules/deploy-cicd-github.md +219 -219
  39. package/src/resources/skills/mandu-deployment/rules/deploy-docker-bun.md +150 -150
  40. package/src/resources/skills/mandu-deployment/rules/deploy-docker-compose.md +223 -223
  41. package/src/resources/skills/mandu-deployment/rules/deploy-platform-fly.md +152 -152
  42. package/src/resources/skills/mandu-deployment/rules/deploy-platform-render.md +179 -179
  43. package/src/resources/skills/mandu-deployment/rules/deploy-platform-supabase.md +323 -323
  44. package/src/resources/skills/mandu-deployment/rules/deploy-platform-vercel.md +140 -140
  45. package/src/resources/skills/mandu-fs-routes/SKILL.md +82 -82
  46. package/src/resources/skills/mandu-fs-routes/metadata.json +12 -12
  47. package/src/resources/skills/mandu-fs-routes/rules/_sections.md +36 -36
  48. package/src/resources/skills/mandu-fs-routes/rules/_template.md +69 -69
  49. package/src/resources/skills/mandu-fs-routes/rules/routes-api-methods.md +65 -65
  50. package/src/resources/skills/mandu-fs-routes/rules/routes-dynamic-param.md +93 -93
  51. package/src/resources/skills/mandu-fs-routes/rules/routes-naming-page.md +55 -55
  52. package/src/resources/skills/mandu-guard/SKILL.md +129 -129
  53. package/src/resources/skills/mandu-guard/metadata.json +12 -12
  54. package/src/resources/skills/mandu-guard/rules/_sections.md +36 -36
  55. package/src/resources/skills/mandu-guard/rules/_template.md +82 -82
  56. package/src/resources/skills/mandu-guard/rules/guard-config-rules.md +100 -100
  57. package/src/resources/skills/mandu-guard/rules/guard-layer-direction.md +76 -76
  58. package/src/resources/skills/mandu-guard/rules/guard-preset-mandu.md +81 -81
  59. package/src/resources/skills/mandu-guard/rules/guard-validate-import.md +80 -80
  60. package/src/resources/skills/mandu-hydration/SKILL.md +91 -91
  61. package/src/resources/skills/mandu-hydration/metadata.json +12 -12
  62. package/src/resources/skills/mandu-hydration/rules/_sections.md +31 -31
  63. package/src/resources/skills/mandu-hydration/rules/_template.md +72 -72
  64. package/src/resources/skills/mandu-hydration/rules/hydration-data-event.md +109 -109
  65. package/src/resources/skills/mandu-hydration/rules/hydration-directive-use-client.md +55 -55
  66. package/src/resources/skills/mandu-hydration/rules/hydration-island-setup.md +113 -113
  67. package/src/resources/skills/mandu-hydration/rules/hydration-priority-visible.md +68 -68
  68. package/src/resources/skills/mandu-performance/SKILL.md +85 -85
  69. package/src/resources/skills/mandu-performance/metadata.json +14 -14
  70. package/src/resources/skills/mandu-performance/rules/_sections.md +31 -31
  71. package/src/resources/skills/mandu-performance/rules/_template.md +64 -64
  72. package/src/resources/skills/mandu-performance/rules/perf-async-defer-await.md +103 -103
  73. package/src/resources/skills/mandu-performance/rules/perf-async-parallel.md +95 -95
  74. package/src/resources/skills/mandu-performance/rules/perf-bun-file.md +124 -124
  75. package/src/resources/skills/mandu-performance/rules/perf-bun-serve.md +125 -125
  76. package/src/resources/skills/mandu-performance/rules/perf-bundle-imports.md +80 -80
  77. package/src/resources/skills/mandu-performance/rules/perf-bundle-island-lazy.md +145 -145
  78. package/src/resources/skills/mandu-performance/rules/perf-cache-react.md +98 -98
  79. package/src/resources/skills/mandu-performance/rules/perf-render-transitions.md +154 -154
  80. package/src/resources/skills/mandu-security/SKILL.md +87 -87
  81. package/src/resources/skills/mandu-security/metadata.json +13 -13
  82. package/src/resources/skills/mandu-security/rules/_sections.md +31 -31
  83. package/src/resources/skills/mandu-security/rules/_template.md +74 -74
  84. package/src/resources/skills/mandu-security/rules/sec-auth-guard.md +127 -127
  85. package/src/resources/skills/mandu-security/rules/sec-env-management.md +133 -133
  86. package/src/resources/skills/mandu-security/rules/sec-input-validate.md +148 -148
  87. package/src/resources/skills/mandu-security/rules/sec-protect-csrf.md +146 -146
  88. package/src/resources/skills/mandu-security/rules/sec-protect-headers.md +138 -138
  89. package/src/resources/skills/mandu-slot/SKILL.md +85 -85
  90. package/src/resources/skills/mandu-slot/metadata.json +12 -12
  91. package/src/resources/skills/mandu-slot/rules/_sections.md +36 -36
  92. package/src/resources/skills/mandu-slot/rules/_template.md +63 -63
  93. package/src/resources/skills/mandu-slot/rules/slot-basic-structure.md +38 -38
  94. package/src/resources/skills/mandu-slot/rules/slot-ctx-response.md +56 -56
  95. package/src/resources/skills/mandu-slot/rules/slot-guard-auth.md +59 -59
  96. package/src/resources/skills/mandu-slot/rules/slot-http-methods.md +64 -64
  97. package/src/resources/skills/mandu-styling/SKILL.md +154 -154
  98. package/src/resources/skills/mandu-styling/_sections.md +43 -43
  99. package/src/resources/skills/mandu-styling/_template.md +32 -32
  100. package/src/resources/skills/mandu-styling/metadata.json +15 -15
  101. package/src/resources/skills/mandu-styling/rules/style-component-compound.md +235 -235
  102. package/src/resources/skills/mandu-styling/rules/style-component-slots.md +255 -255
  103. package/src/resources/skills/mandu-styling/rules/style-component-tokens.md +205 -205
  104. package/src/resources/skills/mandu-styling/rules/style-island-animations.md +272 -272
  105. package/src/resources/skills/mandu-styling/rules/style-island-scoping.md +167 -167
  106. package/src/resources/skills/mandu-styling/rules/style-island-variants.md +221 -221
  107. package/src/resources/skills/mandu-styling/rules/style-perf-critical.md +209 -209
  108. package/src/resources/skills/mandu-styling/rules/style-perf-purge.md +192 -192
  109. package/src/resources/skills/mandu-styling/rules/style-setup-modules.md +162 -162
  110. package/src/resources/skills/mandu-styling/rules/style-setup-panda.md +164 -164
  111. package/src/resources/skills/mandu-styling/rules/style-setup-tailwind.md +170 -170
  112. package/src/resources/skills/mandu-styling/rules/style-tailwind-v4-gotchas.md +179 -179
  113. package/src/resources/skills/mandu-styling/rules/style-theme-darkmode.md +229 -229
  114. package/src/resources/skills/mandu-testing/SKILL.md +99 -99
  115. package/src/resources/skills/mandu-testing/metadata.json +13 -13
  116. package/src/resources/skills/mandu-testing/rules/_sections.md +26 -26
  117. package/src/resources/skills/mandu-testing/rules/_template.md +65 -65
  118. package/src/resources/skills/mandu-testing/rules/test-component-island.md +195 -195
  119. package/src/resources/skills/mandu-testing/rules/test-e2e-playwright.md +196 -196
  120. package/src/resources/skills/mandu-testing/rules/test-mock-fetch.md +219 -219
  121. package/src/resources/skills/mandu-testing/rules/test-slot-unit.md +192 -192
  122. package/src/resources/skills/mandu-ui/SKILL.md +117 -117
  123. package/src/resources/skills/mandu-ui/_sections.md +23 -23
  124. package/src/resources/skills/mandu-ui/_template.md +32 -32
  125. package/src/resources/skills/mandu-ui/metadata.json +13 -13
  126. package/src/resources/skills/mandu-ui/rules/ui-accessibility-aria.md +232 -232
  127. package/src/resources/skills/mandu-ui/rules/ui-accessibility-focus.md +238 -238
  128. package/src/resources/skills/mandu-ui/rules/ui-composition-patterns.md +259 -259
  129. package/src/resources/skills/mandu-ui/rules/ui-island-integration.md +258 -258
  130. package/src/resources/skills/mandu-ui/rules/ui-radix-patterns.md +213 -213
  131. package/src/resources/skills/mandu-ui/rules/ui-shadcn-setup.md +209 -209
  132. package/src/resources/skills/recipes.ts +932 -932
  133. package/src/tools/generate.ts +7 -4
  134. package/src/tools/guard.ts +17 -4
  135. package/src/tools/hydration.ts +10 -10
  136. package/src/tools/project.ts +334 -334
  137. package/src/tools/runtime.ts +497 -497
  138. package/src/tools/seo.ts +417 -417
  139. package/src/tools/spec.ts +80 -159
  140. package/src/utils/project.ts +22 -12
  141. 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 |