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