@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,148 +1,148 @@
1
- ---
2
- title: Define Generic Context Interface for Dependency Injection
3
- impact: HIGH
4
- impactDescription: Enables dependency-injectable state across use-cases
5
- tags: composition, context, state, typescript, dependency-injection
6
- ---
7
-
8
- ## Define Generic Context Interface for Dependency Injection
9
-
10
- **Impact: HIGH (Enables dependency-injectable state across use-cases)**
11
-
12
- 컴포넌트 context를 위한 제네릭 인터페이스를 `state`, `actions`, `meta` 세 부분으로 정의하세요. 어떤 Provider든 이 인터페이스를 구현할 수 있어, 같은 UI 컴포넌트가 완전히 다른 상태 구현과 작동할 수 있습니다.
13
-
14
- **핵심 원칙:** Lift state, compose internals, make state dependency-injectable.
15
-
16
- **Incorrect (특정 상태 구현에 결합):**
17
-
18
- ```tsx
19
- // ❌ 특정 훅에 강하게 결합
20
- function ComposerInput() {
21
- const { input, setInput } = useChannelComposerState();
22
- return <TextInput value={input} onChangeText={setInput} />;
23
- }
24
- ```
25
-
26
- **Correct (제네릭 인터페이스로 의존성 주입):**
27
-
28
- ```tsx
29
- // ✅ 어떤 Provider든 구현할 수 있는 제네릭 인터페이스
30
- interface ComposerState {
31
- input: string;
32
- attachments: Attachment[];
33
- isSubmitting: boolean;
34
- }
35
-
36
- interface ComposerActions {
37
- update: (updater: (state: ComposerState) => ComposerState) => void;
38
- submit: () => void;
39
- addAttachment: (file: File) => void;
40
- }
41
-
42
- interface ComposerMeta {
43
- inputRef: React.RefObject<HTMLTextAreaElement>;
44
- maxLength?: number;
45
- }
46
-
47
- interface ComposerContextValue {
48
- state: ComposerState;
49
- actions: ComposerActions;
50
- meta: ComposerMeta;
51
- }
52
-
53
- const ComposerContext = createContext<ComposerContextValue | null>(null);
54
- ```
55
-
56
- **UI 컴포넌트는 인터페이스만 소비:**
57
-
58
- ```tsx
59
- function ComposerInput() {
60
- const {
61
- state,
62
- actions: { update },
63
- meta,
64
- } = use(ComposerContext)!;
65
-
66
- // 이 컴포넌트는 인터페이스를 구현한 어떤 Provider와도 작동
67
- return (
68
- <textarea
69
- ref={meta.inputRef}
70
- value={state.input}
71
- maxLength={meta.maxLength}
72
- onChange={(e) => update((s) => ({ ...s, input: e.target.value }))}
73
- />
74
- );
75
- }
76
- ```
77
-
78
- **서로 다른 Provider가 같은 인터페이스 구현:**
79
-
80
- ```tsx
81
- // Provider A: 임시 폼을 위한 로컬 상태
82
- function ForwardMessageProvider({ children }: { children: React.ReactNode }) {
83
- const [state, setState] = useState(initialState);
84
- const inputRef = useRef<HTMLTextAreaElement>(null);
85
- const submit = useForwardMessage();
86
-
87
- return (
88
- <ComposerContext value={{
89
- state,
90
- actions: { update: setState, submit, addAttachment: ... },
91
- meta: { inputRef },
92
- }}>
93
- {children}
94
- </ComposerContext>
95
- );
96
- }
97
-
98
- // Provider B: 채널을 위한 전역 동기화 상태
99
- function ChannelProvider({ channelId, children }: Props) {
100
- const { state, update, submit } = useGlobalChannel(channelId);
101
- const inputRef = useRef<HTMLTextAreaElement>(null);
102
-
103
- return (
104
- <ComposerContext value={{
105
- state,
106
- actions: { update, submit, addAttachment: ... },
107
- meta: { inputRef, maxLength: 2000 },
108
- }}>
109
- {children}
110
- </ComposerContext>
111
- );
112
- }
113
- ```
114
-
115
- **같은 UI가 두 Provider와 모두 작동:**
116
-
117
- ```tsx
118
- // ForwardMessageProvider와 작동 (로컬 상태)
119
- <ForwardMessageProvider>
120
- <Composer.Frame>
121
- <Composer.Input />
122
- <Composer.Submit />
123
- </Composer.Frame>
124
- </ForwardMessageProvider>
125
-
126
- // ChannelProvider와 작동 (전역 동기화 상태)
127
- <ChannelProvider channelId="abc">
128
- <Composer.Frame>
129
- <Composer.Input />
130
- <Composer.Submit />
131
- </Composer.Frame>
132
- </ChannelProvider>
133
- ```
134
-
135
- ## Mandu Island에서의 활용
136
-
137
- ```tsx
138
- // Island 컴포넌트는 인터페이스만 알면 됨
139
- // slot에서 어떤 Provider를 사용하든 같은 Island 재사용 가능
140
-
141
- // spec/slots/forward.slot.ts → ForwardMessageProvider 사용
142
- // spec/slots/channel.slot.ts → ChannelProvider 사용
143
- // 둘 다 같은 Composer Island 컴포넌트 사용
144
- ```
145
-
146
- UI는 재사용 가능한 조각들이고, 상태는 Provider가 주입합니다. Provider를 바꾸면 UI는 그대로!
147
-
148
- Reference: [Context Interface Pattern](https://kentcdodds.com/blog/how-to-use-react-context-effectively)
1
+ ---
2
+ title: Define Generic Context Interface for Dependency Injection
3
+ impact: HIGH
4
+ impactDescription: Enables dependency-injectable state across use-cases
5
+ tags: composition, context, state, typescript, dependency-injection
6
+ ---
7
+
8
+ ## Define Generic Context Interface for Dependency Injection
9
+
10
+ **Impact: HIGH (Enables dependency-injectable state across use-cases)**
11
+
12
+ 컴포넌트 context를 위한 제네릭 인터페이스를 `state`, `actions`, `meta` 세 부분으로 정의하세요. 어떤 Provider든 이 인터페이스를 구현할 수 있어, 같은 UI 컴포넌트가 완전히 다른 상태 구현과 작동할 수 있습니다.
13
+
14
+ **핵심 원칙:** Lift state, compose internals, make state dependency-injectable.
15
+
16
+ **Incorrect (특정 상태 구현에 결합):**
17
+
18
+ ```tsx
19
+ // ❌ 특정 훅에 강하게 결합
20
+ function ComposerInput() {
21
+ const { input, setInput } = useChannelComposerState();
22
+ return <TextInput value={input} onChangeText={setInput} />;
23
+ }
24
+ ```
25
+
26
+ **Correct (제네릭 인터페이스로 의존성 주입):**
27
+
28
+ ```tsx
29
+ // ✅ 어떤 Provider든 구현할 수 있는 제네릭 인터페이스
30
+ interface ComposerState {
31
+ input: string;
32
+ attachments: Attachment[];
33
+ isSubmitting: boolean;
34
+ }
35
+
36
+ interface ComposerActions {
37
+ update: (updater: (state: ComposerState) => ComposerState) => void;
38
+ submit: () => void;
39
+ addAttachment: (file: File) => void;
40
+ }
41
+
42
+ interface ComposerMeta {
43
+ inputRef: React.RefObject<HTMLTextAreaElement>;
44
+ maxLength?: number;
45
+ }
46
+
47
+ interface ComposerContextValue {
48
+ state: ComposerState;
49
+ actions: ComposerActions;
50
+ meta: ComposerMeta;
51
+ }
52
+
53
+ const ComposerContext = createContext<ComposerContextValue | null>(null);
54
+ ```
55
+
56
+ **UI 컴포넌트는 인터페이스만 소비:**
57
+
58
+ ```tsx
59
+ function ComposerInput() {
60
+ const {
61
+ state,
62
+ actions: { update },
63
+ meta,
64
+ } = use(ComposerContext)!;
65
+
66
+ // 이 컴포넌트는 인터페이스를 구현한 어떤 Provider와도 작동
67
+ return (
68
+ <textarea
69
+ ref={meta.inputRef}
70
+ value={state.input}
71
+ maxLength={meta.maxLength}
72
+ onChange={(e) => update((s) => ({ ...s, input: e.target.value }))}
73
+ />
74
+ );
75
+ }
76
+ ```
77
+
78
+ **서로 다른 Provider가 같은 인터페이스 구현:**
79
+
80
+ ```tsx
81
+ // Provider A: 임시 폼을 위한 로컬 상태
82
+ function ForwardMessageProvider({ children }: { children: React.ReactNode }) {
83
+ const [state, setState] = useState(initialState);
84
+ const inputRef = useRef<HTMLTextAreaElement>(null);
85
+ const submit = useForwardMessage();
86
+
87
+ return (
88
+ <ComposerContext value={{
89
+ state,
90
+ actions: { update: setState, submit, addAttachment: ... },
91
+ meta: { inputRef },
92
+ }}>
93
+ {children}
94
+ </ComposerContext>
95
+ );
96
+ }
97
+
98
+ // Provider B: 채널을 위한 전역 동기화 상태
99
+ function ChannelProvider({ channelId, children }: Props) {
100
+ const { state, update, submit } = useGlobalChannel(channelId);
101
+ const inputRef = useRef<HTMLTextAreaElement>(null);
102
+
103
+ return (
104
+ <ComposerContext value={{
105
+ state,
106
+ actions: { update, submit, addAttachment: ... },
107
+ meta: { inputRef, maxLength: 2000 },
108
+ }}>
109
+ {children}
110
+ </ComposerContext>
111
+ );
112
+ }
113
+ ```
114
+
115
+ **같은 UI가 두 Provider와 모두 작동:**
116
+
117
+ ```tsx
118
+ // ForwardMessageProvider와 작동 (로컬 상태)
119
+ <ForwardMessageProvider>
120
+ <Composer.Frame>
121
+ <Composer.Input />
122
+ <Composer.Submit />
123
+ </Composer.Frame>
124
+ </ForwardMessageProvider>
125
+
126
+ // ChannelProvider와 작동 (전역 동기화 상태)
127
+ <ChannelProvider channelId="abc">
128
+ <Composer.Frame>
129
+ <Composer.Input />
130
+ <Composer.Submit />
131
+ </Composer.Frame>
132
+ </ChannelProvider>
133
+ ```
134
+
135
+ ## Mandu Island에서의 활용
136
+
137
+ ```tsx
138
+ // Island 컴포넌트는 인터페이스만 알면 됨
139
+ // slot에서 어떤 Provider를 사용하든 같은 Island 재사용 가능
140
+
141
+ // spec/slots/forward.slot.ts → ForwardMessageProvider 사용
142
+ // spec/slots/channel.slot.ts → ChannelProvider 사용
143
+ // 둘 다 같은 Composer Island 컴포넌트 사용
144
+ ```
145
+
146
+ UI는 재사용 가능한 조각들이고, 상태는 Provider가 주입합니다. Provider를 바꾸면 UI는 그대로!
147
+
148
+ Reference: [Context Interface Pattern](https://kentcdodds.com/blog/how-to-use-react-context-effectively)
@@ -1,150 +1,150 @@
1
- ---
2
- title: Lift State into Provider for Sibling Access
3
- impact: HIGH
4
- impactDescription: Enables state sharing without prop drilling
5
- tags: composition, state, provider, lift-state
6
- ---
7
-
8
- ## Lift State into Provider for Sibling Access
9
-
10
- **Impact: HIGH (Enables state sharing without prop drilling)**
11
-
12
- 형제 컴포넌트가 상태를 공유해야 할 때, 상태를 Provider로 끌어올리세요. Provider 경계 내의 모든 컴포넌트가 상태에 접근할 수 있습니다.
13
-
14
- **Incorrect (prop drilling):**
15
-
16
- ```tsx
17
- // ❌ 상태를 여러 단계로 전달
18
- function ChatPage() {
19
- const [message, setMessage] = useState("");
20
- const [isTyping, setIsTyping] = useState(false);
21
-
22
- return (
23
- <div>
24
- <MessageList
25
- isTyping={isTyping}
26
- typingMessage={message} // prop drilling
27
- />
28
- <Composer
29
- message={message}
30
- setMessage={setMessage}
31
- setIsTyping={setIsTyping}
32
- />
33
- <TypingIndicator isTyping={isTyping} /> // prop drilling
34
- </div>
35
- );
36
- }
37
- ```
38
-
39
- **Correct (Provider로 끌어올리기):**
40
-
41
- ```tsx
42
- // ✅ Provider가 상태 관리
43
- interface ChatContextValue {
44
- state: { message: string; isTyping: boolean };
45
- actions: { setMessage: (msg: string) => void; setIsTyping: (val: boolean) => void };
46
- }
47
-
48
- const ChatContext = createContext<ChatContextValue | null>(null);
49
-
50
- function ChatProvider({ children }: { children: React.ReactNode }) {
51
- const [message, setMessage] = useState("");
52
- const [isTyping, setIsTyping] = useState(false);
53
-
54
- return (
55
- <ChatContext value={{
56
- state: { message, isTyping },
57
- actions: { setMessage, setIsTyping },
58
- }}>
59
- {children}
60
- </ChatContext>
61
- );
62
- }
63
-
64
- // 각 컴포넌트가 필요한 것만 가져감
65
- function MessageList() {
66
- const { state } = use(ChatContext)!;
67
- return (
68
- <div>
69
- {messages.map(msg => <Message key={msg.id} {...msg} />)}
70
- {state.isTyping && <TypingPreview text={state.message} />}
71
- </div>
72
- );
73
- }
74
-
75
- function Composer() {
76
- const { state, actions } = use(ChatContext)!;
77
- return (
78
- <textarea
79
- value={state.message}
80
- onChange={(e) => {
81
- actions.setMessage(e.target.value);
82
- actions.setIsTyping(e.target.value.length > 0);
83
- }}
84
- />
85
- );
86
- }
87
-
88
- function TypingIndicator() {
89
- const { state } = use(ChatContext)!;
90
- if (!state.isTyping) return null;
91
- return <span>Someone is typing...</span>;
92
- }
93
- ```
94
-
95
- **사용:**
96
-
97
- ```tsx
98
- function ChatPage() {
99
- return (
100
- <ChatProvider>
101
- <div>
102
- <MessageList /> {/* context에서 상태 읽음 */}
103
- <Composer /> {/* context에서 상태 수정 */}
104
- <TypingIndicator /> {/* context에서 상태 읽음 */}
105
- </div>
106
- </ChatProvider>
107
- );
108
- }
109
- ```
110
-
111
- ## Provider 경계 이해하기
112
-
113
- Provider 경계 내라면 시각적 위치와 무관하게 상태 접근 가능:
114
-
115
- ```tsx
116
- <ChatProvider>
117
- <Dialog>
118
- {/* Dialog 내부 */}
119
- <Composer.Frame>
120
- <Composer.Input />
121
- </Composer.Frame>
122
-
123
- {/* Frame 외부지만 Provider 내부! */}
124
- <MessagePreview /> {/* ✅ context 접근 가능 */}
125
-
126
- <DialogActions>
127
- <SendButton /> {/* ✅ context 접근 가능 */}
128
- </DialogActions>
129
- </Dialog>
130
- </ChatProvider>
131
- ```
132
-
133
- ## Mandu Island에서의 적용
134
-
135
- ```tsx
136
- // app/chat/page.tsx
137
- import { ChatIsland } from "./client";
138
-
139
- export default function ChatPage({ data }) {
140
- return (
141
- <div>
142
- <h1>Chat</h1>
143
- {/* Island 내부에서 Provider로 상태 관리 */}
144
- <ChatIsland initialMessages={data.messages} />
145
- </div>
146
- );
147
- }
148
- ```
149
-
150
- Reference: [Lifting State Up](https://react.dev/learn/sharing-state-between-components)
1
+ ---
2
+ title: Lift State into Provider for Sibling Access
3
+ impact: HIGH
4
+ impactDescription: Enables state sharing without prop drilling
5
+ tags: composition, state, provider, lift-state
6
+ ---
7
+
8
+ ## Lift State into Provider for Sibling Access
9
+
10
+ **Impact: HIGH (Enables state sharing without prop drilling)**
11
+
12
+ 형제 컴포넌트가 상태를 공유해야 할 때, 상태를 Provider로 끌어올리세요. Provider 경계 내의 모든 컴포넌트가 상태에 접근할 수 있습니다.
13
+
14
+ **Incorrect (prop drilling):**
15
+
16
+ ```tsx
17
+ // ❌ 상태를 여러 단계로 전달
18
+ function ChatPage() {
19
+ const [message, setMessage] = useState("");
20
+ const [isTyping, setIsTyping] = useState(false);
21
+
22
+ return (
23
+ <div>
24
+ <MessageList
25
+ isTyping={isTyping}
26
+ typingMessage={message} // prop drilling
27
+ />
28
+ <Composer
29
+ message={message}
30
+ setMessage={setMessage}
31
+ setIsTyping={setIsTyping}
32
+ />
33
+ <TypingIndicator isTyping={isTyping} /> // prop drilling
34
+ </div>
35
+ );
36
+ }
37
+ ```
38
+
39
+ **Correct (Provider로 끌어올리기):**
40
+
41
+ ```tsx
42
+ // ✅ Provider가 상태 관리
43
+ interface ChatContextValue {
44
+ state: { message: string; isTyping: boolean };
45
+ actions: { setMessage: (msg: string) => void; setIsTyping: (val: boolean) => void };
46
+ }
47
+
48
+ const ChatContext = createContext<ChatContextValue | null>(null);
49
+
50
+ function ChatProvider({ children }: { children: React.ReactNode }) {
51
+ const [message, setMessage] = useState("");
52
+ const [isTyping, setIsTyping] = useState(false);
53
+
54
+ return (
55
+ <ChatContext value={{
56
+ state: { message, isTyping },
57
+ actions: { setMessage, setIsTyping },
58
+ }}>
59
+ {children}
60
+ </ChatContext>
61
+ );
62
+ }
63
+
64
+ // 각 컴포넌트가 필요한 것만 가져감
65
+ function MessageList() {
66
+ const { state } = use(ChatContext)!;
67
+ return (
68
+ <div>
69
+ {messages.map(msg => <Message key={msg.id} {...msg} />)}
70
+ {state.isTyping && <TypingPreview text={state.message} />}
71
+ </div>
72
+ );
73
+ }
74
+
75
+ function Composer() {
76
+ const { state, actions } = use(ChatContext)!;
77
+ return (
78
+ <textarea
79
+ value={state.message}
80
+ onChange={(e) => {
81
+ actions.setMessage(e.target.value);
82
+ actions.setIsTyping(e.target.value.length > 0);
83
+ }}
84
+ />
85
+ );
86
+ }
87
+
88
+ function TypingIndicator() {
89
+ const { state } = use(ChatContext)!;
90
+ if (!state.isTyping) return null;
91
+ return <span>Someone is typing...</span>;
92
+ }
93
+ ```
94
+
95
+ **사용:**
96
+
97
+ ```tsx
98
+ function ChatPage() {
99
+ return (
100
+ <ChatProvider>
101
+ <div>
102
+ <MessageList /> {/* context에서 상태 읽음 */}
103
+ <Composer /> {/* context에서 상태 수정 */}
104
+ <TypingIndicator /> {/* context에서 상태 읽음 */}
105
+ </div>
106
+ </ChatProvider>
107
+ );
108
+ }
109
+ ```
110
+
111
+ ## Provider 경계 이해하기
112
+
113
+ Provider 경계 내라면 시각적 위치와 무관하게 상태 접근 가능:
114
+
115
+ ```tsx
116
+ <ChatProvider>
117
+ <Dialog>
118
+ {/* Dialog 내부 */}
119
+ <Composer.Frame>
120
+ <Composer.Input />
121
+ </Composer.Frame>
122
+
123
+ {/* Frame 외부지만 Provider 내부! */}
124
+ <MessagePreview /> {/* ✅ context 접근 가능 */}
125
+
126
+ <DialogActions>
127
+ <SendButton /> {/* ✅ context 접근 가능 */}
128
+ </DialogActions>
129
+ </Dialog>
130
+ </ChatProvider>
131
+ ```
132
+
133
+ ## Mandu Island에서의 적용
134
+
135
+ ```tsx
136
+ // app/chat/page.tsx
137
+ import { ChatIsland } from "./client";
138
+
139
+ export default function ChatPage({ data }) {
140
+ return (
141
+ <div>
142
+ <h1>Chat</h1>
143
+ {/* Island 내부에서 Provider로 상태 관리 */}
144
+ <ChatIsland initialMessages={data.messages} />
145
+ </div>
146
+ );
147
+ }
148
+ ```
149
+
150
+ Reference: [Lifting State Up](https://react.dev/learn/sharing-state-between-components)