@su-record/vibe 2.4.17 โ†’ 2.4.19

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 (81) hide show
  1. package/.claude/settings.json +48 -48
  2. package/.claude/settings.local.json +28 -28
  3. package/.claude/vibe/constitution.md +184 -184
  4. package/.claude/vibe/rules/core/communication-guide.md +104 -104
  5. package/.claude/vibe/rules/core/development-philosophy.md +52 -52
  6. package/.claude/vibe/rules/core/quick-start.md +120 -120
  7. package/.claude/vibe/rules/languages/dart-flutter.md +509 -509
  8. package/.claude/vibe/rules/languages/go.md +396 -396
  9. package/.claude/vibe/rules/languages/java-spring.md +586 -586
  10. package/.claude/vibe/rules/languages/kotlin-android.md +491 -491
  11. package/.claude/vibe/rules/languages/python-django.md +371 -371
  12. package/.claude/vibe/rules/languages/python-fastapi.md +386 -386
  13. package/.claude/vibe/rules/languages/rust.md +425 -425
  14. package/.claude/vibe/rules/languages/swift-ios.md +516 -516
  15. package/.claude/vibe/rules/languages/typescript-nextjs.md +441 -441
  16. package/.claude/vibe/rules/languages/typescript-node.md +375 -375
  17. package/.claude/vibe/rules/languages/typescript-nuxt.md +521 -521
  18. package/.claude/vibe/rules/languages/typescript-react-native.md +446 -446
  19. package/.claude/vibe/rules/languages/typescript-react.md +525 -525
  20. package/.claude/vibe/rules/languages/typescript-vue.md +353 -353
  21. package/.claude/vibe/rules/quality/bdd-contract-testing.md +388 -388
  22. package/.claude/vibe/rules/quality/checklist.md +276 -276
  23. package/.claude/vibe/rules/quality/testing-strategy.md +437 -437
  24. package/.claude/vibe/rules/standards/anti-patterns.md +369 -369
  25. package/.claude/vibe/rules/standards/code-structure.md +291 -291
  26. package/.claude/vibe/rules/standards/complexity-metrics.md +312 -312
  27. package/.claude/vibe/rules/standards/naming-conventions.md +198 -198
  28. package/.claude/vibe/setup.sh +31 -31
  29. package/.claude/vibe/templates/constitution-template.md +184 -184
  30. package/.claude/vibe/templates/contract-backend-template.md +517 -517
  31. package/.claude/vibe/templates/contract-frontend-template.md +594 -594
  32. package/.claude/vibe/templates/feature-template.md +96 -96
  33. package/.claude/vibe/templates/spec-template.md +199 -199
  34. package/CLAUDE.md +333 -333
  35. package/LICENSE +21 -21
  36. package/README.md +205 -205
  37. package/agents/compounder.md +261 -261
  38. package/agents/diagrammer.md +178 -178
  39. package/agents/e2e-tester.md +266 -266
  40. package/agents/explorer.md +48 -48
  41. package/agents/implementer.md +53 -53
  42. package/agents/research/best-practices-agent.md +139 -139
  43. package/agents/research/codebase-patterns-agent.md +147 -147
  44. package/agents/research/framework-docs-agent.md +181 -181
  45. package/agents/research/security-advisory-agent.md +167 -167
  46. package/agents/review/architecture-reviewer.md +107 -107
  47. package/agents/review/complexity-reviewer.md +116 -116
  48. package/agents/review/data-integrity-reviewer.md +88 -88
  49. package/agents/review/git-history-reviewer.md +103 -103
  50. package/agents/review/performance-reviewer.md +86 -86
  51. package/agents/review/python-reviewer.md +152 -152
  52. package/agents/review/rails-reviewer.md +139 -139
  53. package/agents/review/react-reviewer.md +144 -144
  54. package/agents/review/security-reviewer.md +80 -80
  55. package/agents/review/simplicity-reviewer.md +140 -140
  56. package/agents/review/test-coverage-reviewer.md +116 -116
  57. package/agents/review/typescript-reviewer.md +127 -127
  58. package/agents/searcher.md +54 -54
  59. package/agents/simplifier.md +119 -119
  60. package/agents/tester.md +49 -49
  61. package/agents/ui-previewer.md +137 -137
  62. package/commands/vibe.analyze.md +260 -260
  63. package/commands/vibe.reason.md +223 -223
  64. package/commands/vibe.review.md +213 -213
  65. package/commands/vibe.run.md +935 -935
  66. package/commands/vibe.spec.md +442 -442
  67. package/commands/vibe.utils.md +101 -101
  68. package/commands/vibe.verify.md +282 -282
  69. package/dist/cli/collaborator.js +52 -52
  70. package/dist/cli/detect.js +32 -32
  71. package/dist/cli/index.js +0 -0
  72. package/dist/cli/llm.js +144 -144
  73. package/hooks/hooks.json +195 -195
  74. package/package.json +87 -87
  75. package/skills/context7-usage.md +82 -82
  76. package/skills/git-worktree.md +181 -181
  77. package/skills/multi-llm-orchestration.md +97 -97
  78. package/skills/parallel-research.md +77 -77
  79. package/skills/priority-todos.md +239 -239
  80. package/skills/tool-fallback.md +126 -126
  81. package/skills/vibe-capabilities.md +127 -127
@@ -1,291 +1,291 @@
1
- # ๐Ÿ—๏ธ ์ฝ”๋“œ ๊ตฌ์กฐ ์ž๋™ํ™” ๊ทœ์น™
2
-
3
- ## ์ปดํฌ๋„ŒํŠธ ๊ตฌ์กฐ (์—„๊ฒฉํ•œ ์ˆœ์„œ)
4
-
5
- ```typescript
6
- // 1. Import ๋ฌธ
7
- import React, { useState, useEffect } from 'react';
8
-
9
- // 2. ํƒ€์ž…/์ธํ„ฐํŽ˜์ด์Šค ์ •์˜
10
- interface Props {
11
- userId: string;
12
- }
13
-
14
- // 3. ์ปดํฌ๋„ŒํŠธ ์ •์˜
15
- function UserProfile({ userId }: Props) {
16
- // 4. State & Refs
17
- const [user, setUser] = useState<User | null>(null);
18
- const inputRef = useRef<HTMLInputElement>(null);
19
-
20
- // 5. Custom Hooks
21
- const { isAuthenticated } = useAuth();
22
- const { data, loading } = useUserData(userId);
23
-
24
- // 6. Event Handlers
25
- const handleSubmit = (e: FormEvent) => {
26
- e.preventDefault();
27
- // ...
28
- };
29
-
30
- // 7. Effects
31
- useEffect(() => {
32
- // ...
33
- }, [userId]);
34
-
35
- // 8. Early returns
36
- if (loading) return <Spinner />;
37
- if (!user) return <NotFound />;
38
-
39
- // 9. Main return JSX
40
- return (
41
- <div>
42
- {/* ... */}
43
- </div>
44
- );
45
- }
46
- ```
47
-
48
- ## ํ•จ์ˆ˜ ๋ถ„๋ฆฌ ๊ธฐ์ค€
49
-
50
- ### 1. ํ•จ์ˆ˜ ๊ธธ์ด ๊ธฐ์ค€
51
-
52
- ```typescript
53
- // โŒ 20์ค„ ์ดˆ๊ณผ - ๋ถ„๋ฆฌ ํ•„์š”
54
- function processUserData(user: User) {
55
- // 30์ค„์˜ ๋ณต์žกํ•œ ๋กœ์ง
56
- }
57
-
58
- // โœ… ๋‹จ์ผ ์ฑ…์ž„์œผ๋กœ ๋ถ„๋ฆฌ
59
- function processUserData(user: User) {
60
- const validated = validateUser(user);
61
- const transformed = transformUserData(validated);
62
- return saveUserData(transformed);
63
- }
64
-
65
- function validateUser(user: User) { /* ... */ }
66
- function transformUserData(user: User) { /* ... */ }
67
- function saveUserData(user: User) { /* ... */ }
68
- ```
69
-
70
- ### 2. ์ปดํฌ๋„ŒํŠธ JSX ๊ธธ์ด ๊ธฐ์ค€
71
-
72
- ```typescript
73
- // โŒ JSX 50์ค„ ์ดˆ๊ณผ - ๋ถ„๋ฆฌ ํ•„์š”
74
- function Dashboard() {
75
- return (
76
- <div>
77
- {/* 60์ค„์˜ ๋ณต์žกํ•œ JSX */}
78
- </div>
79
- );
80
- }
81
-
82
- // โœ… ์„œ๋ธŒ ์ปดํฌ๋„ŒํŠธ ์ถ”์ถœ
83
- function Dashboard() {
84
- return (
85
- <div>
86
- <DashboardHeader />
87
- <DashboardContent />
88
- <DashboardFooter />
89
- </div>
90
- );
91
- }
92
-
93
- function DashboardHeader() { /* ... */ }
94
- function DashboardContent() { /* ... */ }
95
- function DashboardFooter() { /* ... */ }
96
- ```
97
-
98
- ### 3. ์ค‘์ฒฉ ๊นŠ์ด ๊ธฐ์ค€
99
-
100
- ```typescript
101
- // โŒ ์ค‘์ฒฉ 3๋‹จ๊ณ„ ์ดˆ๊ณผ
102
- function processData(data: Data) {
103
- if (data) {
104
- if (data.isValid) {
105
- if (data.user) {
106
- if (data.user.isActive) {
107
- // ๋„ˆ๋ฌด ๊นŠ์€ ์ค‘์ฒฉ
108
- }
109
- }
110
- }
111
- }
112
- }
113
-
114
- // โœ… Early return์œผ๋กœ ํ‰ํƒ„ํ™”
115
- function processData(data: Data) {
116
- if (!data) return null;
117
- if (!data.isValid) return null;
118
- if (!data.user) return null;
119
- if (!data.user.isActive) return null;
120
-
121
- // ๋กœ์ง ์‹คํ–‰
122
- }
123
- ```
124
-
125
- ### 4. Cyclomatic Complexity > 10
126
-
127
- ```typescript
128
- // โŒ ๋ณต์žก๋„ ๋†’์Œ (15)
129
- function calculatePrice(item: Item) {
130
- let price = item.basePrice;
131
- if (item.discount) price *= 0.9;
132
- if (item.bulk) price *= 0.8;
133
- if (item.seasonal) price *= 0.95;
134
- if (item.member) price *= 0.85;
135
- if (item.firstTime) price *= 0.9;
136
- // ... ๋” ๋งŽ์€ ์กฐ๊ฑด
137
- return price;
138
- }
139
-
140
- // โœ… ๋ณต์žก๋„ ๊ฐ์†Œ (3)
141
- function calculatePrice(item: Item) {
142
- const basePrice = item.basePrice;
143
- const discounts = getApplicableDiscounts(item);
144
- return applyDiscounts(basePrice, discounts);
145
- }
146
- ```
147
-
148
- ### 5. Cognitive Complexity > 15
149
-
150
- ```typescript
151
- // โŒ ์ธ์ง€ ๋ณต์žก๋„ ๋†’์Œ
152
- function processOrder(order: Order) {
153
- if (order.isPremium) {
154
- for (let item of order.items) {
155
- if (item.category === 'electronics') {
156
- if (item.price > 1000) {
157
- // ์ค‘์ฒฉ๋œ ๋ณต์žกํ•œ ๋กœ์ง
158
- }
159
- }
160
- }
161
- }
162
- }
163
-
164
- // โœ… ์ธ์ง€ ๋ณต์žก๋„ ๊ฐ์†Œ
165
- function processOrder(order: Order) {
166
- if (!order.isPremium) return;
167
-
168
- const electronics = filterElectronics(order.items);
169
- const expensive = filterExpensive(electronics);
170
-
171
- processItems(expensive);
172
- }
173
- ```
174
-
175
- ## ํŒŒ์ผ ๊ตฌ์กฐ ํ‘œ์ค€
176
-
177
- ```typescript
178
- // ๐Ÿ“ user-profile.component.tsx
179
-
180
- // 1. Imports
181
- import { ... } from 'react';
182
- import { ... } from '@/lib';
183
-
184
- // 2. Types
185
- interface UserProfileProps { }
186
- type UserRole = 'admin' | 'user';
187
-
188
- // 3. Constants
189
- const MAX_BIO_LENGTH = 500;
190
- const DEFAULT_AVATAR = '/avatar.png';
191
-
192
- // 4. Helper Functions (๋‚ด๋ถ€ ์ „์šฉ)
193
- function formatUserName(name: string) { }
194
-
195
- // 5. Main Component
196
- export function UserProfile() { }
197
-
198
- // 6. Sub Components (exportํ•˜์ง€ ์•Š์Œ)
199
- function ProfileHeader() { }
200
- function ProfileContent() { }
201
- ```
202
-
203
- ## ๋ชจ๋“ˆ ๊ตฌ์„ฑ ์›์น™
204
-
205
- ### 1. ์‘์ง‘๋„ (Cohesion)
206
-
207
- ```typescript
208
- // โœ… ๋†’์€ ์‘์ง‘๋„ - ๊ด€๋ จ ๊ธฐ๋Šฅ๋งŒ ๋ชจ์Œ
209
- // ๐Ÿ“ user.service.ts
210
- export class UserService {
211
- getUser(id: string) { }
212
- updateUser(id: string, data: User) { }
213
- deleteUser(id: string) { }
214
- }
215
-
216
- // โŒ ๋‚ฎ์€ ์‘์ง‘๋„ - ๊ด€๋ จ ์—†๋Š” ๊ธฐ๋Šฅ ํ˜ผ์žฌ
217
- // ๐Ÿ“ utils.ts (์•ˆํ‹ฐํŒจํ„ด)
218
- export class Utils {
219
- validateEmail(email: string) { }
220
- formatCurrency(amount: number) { }
221
- uploadFile(file: File) { }
222
- }
223
- ```
224
-
225
- ### 2. ๊ฒฐํ•ฉ๋„ (Coupling)
226
-
227
- ```typescript
228
- // โœ… ๋А์Šจํ•œ ๊ฒฐํ•ฉ - ์ธํ„ฐํŽ˜์ด์Šค ์˜์กด
229
- interface Storage {
230
- save(key: string, value: unknown): void;
231
- load(key: string): unknown;
232
- }
233
-
234
- class UserService {
235
- constructor(private storage: Storage) { }
236
- }
237
-
238
- // โŒ ๊ฐ•ํ•œ ๊ฒฐํ•ฉ - ๊ตฌํ˜„์ฒด ์ง์ ‘ ์˜์กด
239
- class UserService {
240
- private storage = new LocalStorage(); // ์ง์ ‘ ์˜์กด
241
- }
242
- ```
243
-
244
- ## ํ•จ์ˆ˜ ๋งค๊ฐœ๋ณ€์ˆ˜ ์ œํ•œ
245
-
246
- ```typescript
247
- // โŒ ๋งค๊ฐœ๋ณ€์ˆ˜ 5๊ฐœ ์ดˆ๊ณผ
248
- function createUser(
249
- name: string,
250
- email: string,
251
- age: number,
252
- address: string,
253
- phone: string,
254
- role: string
255
- ) { }
256
-
257
- // โœ… ๊ฐ์ฒด๋กœ ๊ทธ๋ฃนํ™”
258
- interface CreateUserParams {
259
- name: string;
260
- email: string;
261
- age: number;
262
- address: string;
263
- phone: string;
264
- role: string;
265
- }
266
-
267
- function createUser(params: CreateUserParams) { }
268
- ```
269
-
270
- ## ์ˆœํ™˜ ์˜์กด์„ฑ ๋ฐฉ์ง€
271
-
272
- ```typescript
273
- // โŒ ์ˆœํ™˜ ์˜์กด์„ฑ
274
- // fileA.ts
275
- import { funcB } from './fileB';
276
- export function funcA() { funcB(); }
277
-
278
- // fileB.ts
279
- import { funcA } from './fileA'; // ์ˆœํ™˜!
280
- export function funcB() { funcA(); }
281
-
282
- // โœ… ๊ณตํ†ต ๋ชจ๋“ˆ ๋ถ„๋ฆฌ
283
- // shared.ts
284
- export function sharedFunc() { }
285
-
286
- // fileA.ts
287
- import { sharedFunc } from './shared';
288
-
289
- // fileB.ts
290
- import { sharedFunc } from './shared';
291
- ```
1
+ # ๐Ÿ—๏ธ ์ฝ”๋“œ ๊ตฌ์กฐ ์ž๋™ํ™” ๊ทœ์น™
2
+
3
+ ## ์ปดํฌ๋„ŒํŠธ ๊ตฌ์กฐ (์—„๊ฒฉํ•œ ์ˆœ์„œ)
4
+
5
+ ```typescript
6
+ // 1. Import ๋ฌธ
7
+ import React, { useState, useEffect } from 'react';
8
+
9
+ // 2. ํƒ€์ž…/์ธํ„ฐํŽ˜์ด์Šค ์ •์˜
10
+ interface Props {
11
+ userId: string;
12
+ }
13
+
14
+ // 3. ์ปดํฌ๋„ŒํŠธ ์ •์˜
15
+ function UserProfile({ userId }: Props) {
16
+ // 4. State & Refs
17
+ const [user, setUser] = useState<User | null>(null);
18
+ const inputRef = useRef<HTMLInputElement>(null);
19
+
20
+ // 5. Custom Hooks
21
+ const { isAuthenticated } = useAuth();
22
+ const { data, loading } = useUserData(userId);
23
+
24
+ // 6. Event Handlers
25
+ const handleSubmit = (e: FormEvent) => {
26
+ e.preventDefault();
27
+ // ...
28
+ };
29
+
30
+ // 7. Effects
31
+ useEffect(() => {
32
+ // ...
33
+ }, [userId]);
34
+
35
+ // 8. Early returns
36
+ if (loading) return <Spinner />;
37
+ if (!user) return <NotFound />;
38
+
39
+ // 9. Main return JSX
40
+ return (
41
+ <div>
42
+ {/* ... */}
43
+ </div>
44
+ );
45
+ }
46
+ ```
47
+
48
+ ## ํ•จ์ˆ˜ ๋ถ„๋ฆฌ ๊ธฐ์ค€
49
+
50
+ ### 1. ํ•จ์ˆ˜ ๊ธธ์ด ๊ธฐ์ค€
51
+
52
+ ```typescript
53
+ // โŒ 20์ค„ ์ดˆ๊ณผ - ๋ถ„๋ฆฌ ํ•„์š”
54
+ function processUserData(user: User) {
55
+ // 30์ค„์˜ ๋ณต์žกํ•œ ๋กœ์ง
56
+ }
57
+
58
+ // โœ… ๋‹จ์ผ ์ฑ…์ž„์œผ๋กœ ๋ถ„๋ฆฌ
59
+ function processUserData(user: User) {
60
+ const validated = validateUser(user);
61
+ const transformed = transformUserData(validated);
62
+ return saveUserData(transformed);
63
+ }
64
+
65
+ function validateUser(user: User) { /* ... */ }
66
+ function transformUserData(user: User) { /* ... */ }
67
+ function saveUserData(user: User) { /* ... */ }
68
+ ```
69
+
70
+ ### 2. ์ปดํฌ๋„ŒํŠธ JSX ๊ธธ์ด ๊ธฐ์ค€
71
+
72
+ ```typescript
73
+ // โŒ JSX 50์ค„ ์ดˆ๊ณผ - ๋ถ„๋ฆฌ ํ•„์š”
74
+ function Dashboard() {
75
+ return (
76
+ <div>
77
+ {/* 60์ค„์˜ ๋ณต์žกํ•œ JSX */}
78
+ </div>
79
+ );
80
+ }
81
+
82
+ // โœ… ์„œ๋ธŒ ์ปดํฌ๋„ŒํŠธ ์ถ”์ถœ
83
+ function Dashboard() {
84
+ return (
85
+ <div>
86
+ <DashboardHeader />
87
+ <DashboardContent />
88
+ <DashboardFooter />
89
+ </div>
90
+ );
91
+ }
92
+
93
+ function DashboardHeader() { /* ... */ }
94
+ function DashboardContent() { /* ... */ }
95
+ function DashboardFooter() { /* ... */ }
96
+ ```
97
+
98
+ ### 3. ์ค‘์ฒฉ ๊นŠ์ด ๊ธฐ์ค€
99
+
100
+ ```typescript
101
+ // โŒ ์ค‘์ฒฉ 3๋‹จ๊ณ„ ์ดˆ๊ณผ
102
+ function processData(data: Data) {
103
+ if (data) {
104
+ if (data.isValid) {
105
+ if (data.user) {
106
+ if (data.user.isActive) {
107
+ // ๋„ˆ๋ฌด ๊นŠ์€ ์ค‘์ฒฉ
108
+ }
109
+ }
110
+ }
111
+ }
112
+ }
113
+
114
+ // โœ… Early return์œผ๋กœ ํ‰ํƒ„ํ™”
115
+ function processData(data: Data) {
116
+ if (!data) return null;
117
+ if (!data.isValid) return null;
118
+ if (!data.user) return null;
119
+ if (!data.user.isActive) return null;
120
+
121
+ // ๋กœ์ง ์‹คํ–‰
122
+ }
123
+ ```
124
+
125
+ ### 4. Cyclomatic Complexity > 10
126
+
127
+ ```typescript
128
+ // โŒ ๋ณต์žก๋„ ๋†’์Œ (15)
129
+ function calculatePrice(item: Item) {
130
+ let price = item.basePrice;
131
+ if (item.discount) price *= 0.9;
132
+ if (item.bulk) price *= 0.8;
133
+ if (item.seasonal) price *= 0.95;
134
+ if (item.member) price *= 0.85;
135
+ if (item.firstTime) price *= 0.9;
136
+ // ... ๋” ๋งŽ์€ ์กฐ๊ฑด
137
+ return price;
138
+ }
139
+
140
+ // โœ… ๋ณต์žก๋„ ๊ฐ์†Œ (3)
141
+ function calculatePrice(item: Item) {
142
+ const basePrice = item.basePrice;
143
+ const discounts = getApplicableDiscounts(item);
144
+ return applyDiscounts(basePrice, discounts);
145
+ }
146
+ ```
147
+
148
+ ### 5. Cognitive Complexity > 15
149
+
150
+ ```typescript
151
+ // โŒ ์ธ์ง€ ๋ณต์žก๋„ ๋†’์Œ
152
+ function processOrder(order: Order) {
153
+ if (order.isPremium) {
154
+ for (let item of order.items) {
155
+ if (item.category === 'electronics') {
156
+ if (item.price > 1000) {
157
+ // ์ค‘์ฒฉ๋œ ๋ณต์žกํ•œ ๋กœ์ง
158
+ }
159
+ }
160
+ }
161
+ }
162
+ }
163
+
164
+ // โœ… ์ธ์ง€ ๋ณต์žก๋„ ๊ฐ์†Œ
165
+ function processOrder(order: Order) {
166
+ if (!order.isPremium) return;
167
+
168
+ const electronics = filterElectronics(order.items);
169
+ const expensive = filterExpensive(electronics);
170
+
171
+ processItems(expensive);
172
+ }
173
+ ```
174
+
175
+ ## ํŒŒ์ผ ๊ตฌ์กฐ ํ‘œ์ค€
176
+
177
+ ```typescript
178
+ // ๐Ÿ“ user-profile.component.tsx
179
+
180
+ // 1. Imports
181
+ import { ... } from 'react';
182
+ import { ... } from '@/lib';
183
+
184
+ // 2. Types
185
+ interface UserProfileProps { }
186
+ type UserRole = 'admin' | 'user';
187
+
188
+ // 3. Constants
189
+ const MAX_BIO_LENGTH = 500;
190
+ const DEFAULT_AVATAR = '/avatar.png';
191
+
192
+ // 4. Helper Functions (๋‚ด๋ถ€ ์ „์šฉ)
193
+ function formatUserName(name: string) { }
194
+
195
+ // 5. Main Component
196
+ export function UserProfile() { }
197
+
198
+ // 6. Sub Components (exportํ•˜์ง€ ์•Š์Œ)
199
+ function ProfileHeader() { }
200
+ function ProfileContent() { }
201
+ ```
202
+
203
+ ## ๋ชจ๋“ˆ ๊ตฌ์„ฑ ์›์น™
204
+
205
+ ### 1. ์‘์ง‘๋„ (Cohesion)
206
+
207
+ ```typescript
208
+ // โœ… ๋†’์€ ์‘์ง‘๋„ - ๊ด€๋ จ ๊ธฐ๋Šฅ๋งŒ ๋ชจ์Œ
209
+ // ๐Ÿ“ user.service.ts
210
+ export class UserService {
211
+ getUser(id: string) { }
212
+ updateUser(id: string, data: User) { }
213
+ deleteUser(id: string) { }
214
+ }
215
+
216
+ // โŒ ๋‚ฎ์€ ์‘์ง‘๋„ - ๊ด€๋ จ ์—†๋Š” ๊ธฐ๋Šฅ ํ˜ผ์žฌ
217
+ // ๐Ÿ“ utils.ts (์•ˆํ‹ฐํŒจํ„ด)
218
+ export class Utils {
219
+ validateEmail(email: string) { }
220
+ formatCurrency(amount: number) { }
221
+ uploadFile(file: File) { }
222
+ }
223
+ ```
224
+
225
+ ### 2. ๊ฒฐํ•ฉ๋„ (Coupling)
226
+
227
+ ```typescript
228
+ // โœ… ๋А์Šจํ•œ ๊ฒฐํ•ฉ - ์ธํ„ฐํŽ˜์ด์Šค ์˜์กด
229
+ interface Storage {
230
+ save(key: string, value: unknown): void;
231
+ load(key: string): unknown;
232
+ }
233
+
234
+ class UserService {
235
+ constructor(private storage: Storage) { }
236
+ }
237
+
238
+ // โŒ ๊ฐ•ํ•œ ๊ฒฐํ•ฉ - ๊ตฌํ˜„์ฒด ์ง์ ‘ ์˜์กด
239
+ class UserService {
240
+ private storage = new LocalStorage(); // ์ง์ ‘ ์˜์กด
241
+ }
242
+ ```
243
+
244
+ ## ํ•จ์ˆ˜ ๋งค๊ฐœ๋ณ€์ˆ˜ ์ œํ•œ
245
+
246
+ ```typescript
247
+ // โŒ ๋งค๊ฐœ๋ณ€์ˆ˜ 5๊ฐœ ์ดˆ๊ณผ
248
+ function createUser(
249
+ name: string,
250
+ email: string,
251
+ age: number,
252
+ address: string,
253
+ phone: string,
254
+ role: string
255
+ ) { }
256
+
257
+ // โœ… ๊ฐ์ฒด๋กœ ๊ทธ๋ฃนํ™”
258
+ interface CreateUserParams {
259
+ name: string;
260
+ email: string;
261
+ age: number;
262
+ address: string;
263
+ phone: string;
264
+ role: string;
265
+ }
266
+
267
+ function createUser(params: CreateUserParams) { }
268
+ ```
269
+
270
+ ## ์ˆœํ™˜ ์˜์กด์„ฑ ๋ฐฉ์ง€
271
+
272
+ ```typescript
273
+ // โŒ ์ˆœํ™˜ ์˜์กด์„ฑ
274
+ // fileA.ts
275
+ import { funcB } from './fileB';
276
+ export function funcA() { funcB(); }
277
+
278
+ // fileB.ts
279
+ import { funcA } from './fileA'; // ์ˆœํ™˜!
280
+ export function funcB() { funcA(); }
281
+
282
+ // โœ… ๊ณตํ†ต ๋ชจ๋“ˆ ๋ถ„๋ฆฌ
283
+ // shared.ts
284
+ export function sharedFunc() { }
285
+
286
+ // fileA.ts
287
+ import { sharedFunc } from './shared';
288
+
289
+ // fileB.ts
290
+ import { sharedFunc } from './shared';
291
+ ```