fe-kit-cli 0.0.1

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 (61) hide show
  1. package/README.md +89 -0
  2. package/dist/cli.mjs +1738 -0
  3. package/dist/cli.mjs.map +1 -0
  4. package/dist/rules/common/typescript.mdc +21 -0
  5. package/dist/rules/react/component-conventions.mdc +24 -0
  6. package/dist/rules/react/hooks.mdc +20 -0
  7. package/dist/rules/react/react-router.mdc +18 -0
  8. package/dist/rules/react/state-management.mdc +21 -0
  9. package/dist/rules/vue/component-conventions.mdc +23 -0
  10. package/dist/rules/vue/composition-api.mdc +24 -0
  11. package/dist/rules/vue/state-management.mdc +16 -0
  12. package/dist/rules/vue/vue-router.mdc +18 -0
  13. package/dist/skills/app-ui-design/SKILL.md +62 -0
  14. package/dist/skills/app-ui-design/references/rules.md +127 -0
  15. package/dist/skills/e2e-testing/SKILL.md +327 -0
  16. package/dist/skills/eval-harness/SKILL.md +271 -0
  17. package/dist/skills/frontend-design/SKILL.md +43 -0
  18. package/dist/skills/frontend-patterns/SKILL.md +643 -0
  19. package/dist/skills/security-review/SKILL.md +496 -0
  20. package/dist/skills/tailwindcss-advanced-layouts/SKILL.md +595 -0
  21. package/dist/skills/tdd-workflow/SKILL.md +464 -0
  22. package/dist/skills/verification-loop/SKILL.md +127 -0
  23. package/dist/skills/wechat-ui-design/SKILL.md +64 -0
  24. package/dist/skills/wechat-ui-design/references/rules.md +121 -0
  25. package/dist/templates/react-rspack-ts/index.html +11 -0
  26. package/dist/templates/react-rspack-ts/package.json +20 -0
  27. package/dist/templates/react-rspack-ts/rspack.config.ts +23 -0
  28. package/dist/templates/react-rspack-ts/src/App.tsx +7 -0
  29. package/dist/templates/react-rspack-ts/src/main.tsx +9 -0
  30. package/dist/templates/react-rspack-ts/tsconfig.json +17 -0
  31. package/dist/templates/react-vite-ts/index.html +12 -0
  32. package/dist/templates/react-vite-ts/package.json +22 -0
  33. package/dist/templates/react-vite-ts/src/App.tsx +7 -0
  34. package/dist/templates/react-vite-ts/src/main.tsx +9 -0
  35. package/dist/templates/react-vite-ts/tsconfig.json +19 -0
  36. package/dist/templates/react-vite-ts/vite.config.ts +9 -0
  37. package/dist/templates/react-webpack-ts/index.html +11 -0
  38. package/dist/templates/react-webpack-ts/package.json +25 -0
  39. package/dist/templates/react-webpack-ts/src/App.tsx +7 -0
  40. package/dist/templates/react-webpack-ts/src/main.tsx +9 -0
  41. package/dist/templates/react-webpack-ts/tsconfig.json +17 -0
  42. package/dist/templates/react-webpack-ts/webpack.config.ts +29 -0
  43. package/dist/templates/vue-rspack-ts/index.html +11 -0
  44. package/dist/templates/vue-rspack-ts/package.json +18 -0
  45. package/dist/templates/vue-rspack-ts/rspack.config.ts +16 -0
  46. package/dist/templates/vue-rspack-ts/src/App.vue +7 -0
  47. package/dist/templates/vue-rspack-ts/src/main.ts +4 -0
  48. package/dist/templates/vue-rspack-ts/tsconfig.json +17 -0
  49. package/dist/templates/vue-vite-ts/index.html +12 -0
  50. package/dist/templates/vue-vite-ts/package.json +19 -0
  51. package/dist/templates/vue-vite-ts/src/App.vue +7 -0
  52. package/dist/templates/vue-vite-ts/src/main.ts +4 -0
  53. package/dist/templates/vue-vite-ts/tsconfig.json +19 -0
  54. package/dist/templates/vue-vite-ts/vite.config.ts +9 -0
  55. package/dist/templates/vue-webpack-ts/index.html +11 -0
  56. package/dist/templates/vue-webpack-ts/package.json +24 -0
  57. package/dist/templates/vue-webpack-ts/src/App.vue +7 -0
  58. package/dist/templates/vue-webpack-ts/src/main.ts +4 -0
  59. package/dist/templates/vue-webpack-ts/tsconfig.json +17 -0
  60. package/dist/templates/vue-webpack-ts/webpack.config.ts +32 -0
  61. package/package.json +63 -0
@@ -0,0 +1,643 @@
1
+ ---
2
+ name: frontend-patterns
3
+ description: Frontend development patterns for React, Next.js, state management, performance optimization, and UI best practices.
4
+ origin: ECC
5
+ ---
6
+
7
+ # Frontend Development Patterns
8
+
9
+ Modern frontend patterns for React, Next.js, and performant user interfaces.
10
+
11
+ ## When to Activate
12
+
13
+ - Building React components (composition, props, rendering)
14
+ - Managing state (useState, useReducer, Zustand, Context)
15
+ - Implementing data fetching (SWR, React Query, server components)
16
+ - Optimizing performance (memoization, virtualization, code splitting)
17
+ - Working with forms (validation, controlled inputs, Zod schemas)
18
+ - Handling client-side routing and navigation
19
+ - Building accessible, responsive UI patterns
20
+
21
+ ## Component Patterns
22
+
23
+ ### Composition Over Inheritance
24
+
25
+ ```typescript
26
+ // PASS: GOOD: Component composition
27
+ interface CardProps {
28
+ children: React.ReactNode
29
+ variant?: 'default' | 'outlined'
30
+ }
31
+
32
+ export function Card({ children, variant = 'default' }: CardProps) {
33
+ return <div className={`card card-${variant}`}>{children}</div>
34
+ }
35
+
36
+ export function CardHeader({ children }: { children: React.ReactNode }) {
37
+ return <div className="card-header">{children}</div>
38
+ }
39
+
40
+ export function CardBody({ children }: { children: React.ReactNode }) {
41
+ return <div className="card-body">{children}</div>
42
+ }
43
+
44
+ // Usage
45
+ <Card>
46
+ <CardHeader>Title</CardHeader>
47
+ <CardBody>Content</CardBody>
48
+ </Card>
49
+ ```
50
+
51
+ ### Compound Components
52
+
53
+ ```typescript
54
+ interface TabsContextValue {
55
+ activeTab: string
56
+ setActiveTab: (tab: string) => void
57
+ }
58
+
59
+ const TabsContext = createContext<TabsContextValue | undefined>(undefined)
60
+
61
+ export function Tabs({ children, defaultTab }: {
62
+ children: React.ReactNode
63
+ defaultTab: string
64
+ }) {
65
+ const [activeTab, setActiveTab] = useState(defaultTab)
66
+
67
+ return (
68
+ <TabsContext.Provider value={{ activeTab, setActiveTab }}>
69
+ {children}
70
+ </TabsContext.Provider>
71
+ )
72
+ }
73
+
74
+ export function TabList({ children }: { children: React.ReactNode }) {
75
+ return <div className="tab-list">{children}</div>
76
+ }
77
+
78
+ export function Tab({ id, children }: { id: string, children: React.ReactNode }) {
79
+ const context = useContext(TabsContext)
80
+ if (!context) throw new Error('Tab must be used within Tabs')
81
+
82
+ return (
83
+ <button
84
+ className={context.activeTab === id ? 'active' : ''}
85
+ onClick={() => context.setActiveTab(id)}
86
+ >
87
+ {children}
88
+ </button>
89
+ )
90
+ }
91
+
92
+ // Usage
93
+ <Tabs defaultTab="overview">
94
+ <TabList>
95
+ <Tab id="overview">Overview</Tab>
96
+ <Tab id="details">Details</Tab>
97
+ </TabList>
98
+ </Tabs>
99
+ ```
100
+
101
+ ### Render Props Pattern
102
+
103
+ ```typescript
104
+ interface DataLoaderProps<T> {
105
+ url: string
106
+ children: (data: T | null, loading: boolean, error: Error | null) => React.ReactNode
107
+ }
108
+
109
+ export function DataLoader<T>({ url, children }: DataLoaderProps<T>) {
110
+ const [data, setData] = useState<T | null>(null)
111
+ const [loading, setLoading] = useState(true)
112
+ const [error, setError] = useState<Error | null>(null)
113
+
114
+ useEffect(() => {
115
+ fetch(url)
116
+ .then(res => res.json())
117
+ .then(setData)
118
+ .catch(setError)
119
+ .finally(() => setLoading(false))
120
+ }, [url])
121
+
122
+ return <>{children(data, loading, error)}</>
123
+ }
124
+
125
+ // Usage
126
+ <DataLoader<Market[]> url="/api/markets">
127
+ {(markets, loading, error) => {
128
+ if (loading) return <Spinner />
129
+ if (error) return <Error error={error} />
130
+ return <MarketList markets={markets!} />
131
+ }}
132
+ </DataLoader>
133
+ ```
134
+
135
+ ## Custom Hooks Patterns
136
+
137
+ ### State Management Hook
138
+
139
+ ```typescript
140
+ export function useToggle(initialValue = false): [boolean, () => void] {
141
+ const [value, setValue] = useState(initialValue)
142
+
143
+ const toggle = useCallback(() => {
144
+ setValue(v => !v)
145
+ }, [])
146
+
147
+ return [value, toggle]
148
+ }
149
+
150
+ // Usage
151
+ const [isOpen, toggleOpen] = useToggle()
152
+ ```
153
+
154
+ ### Async Data Fetching Hook
155
+
156
+ ```typescript
157
+ interface UseQueryOptions<T> {
158
+ onSuccess?: (data: T) => void
159
+ onError?: (error: Error) => void
160
+ enabled?: boolean
161
+ }
162
+
163
+ export function useQuery<T>(
164
+ key: string,
165
+ fetcher: () => Promise<T>,
166
+ options?: UseQueryOptions<T>
167
+ ) {
168
+ const [data, setData] = useState<T | null>(null)
169
+ const [error, setError] = useState<Error | null>(null)
170
+ const [loading, setLoading] = useState(false)
171
+
172
+ const refetch = useCallback(async () => {
173
+ setLoading(true)
174
+ setError(null)
175
+
176
+ try {
177
+ const result = await fetcher()
178
+ setData(result)
179
+ options?.onSuccess?.(result)
180
+ } catch (err) {
181
+ const error = err as Error
182
+ setError(error)
183
+ options?.onError?.(error)
184
+ } finally {
185
+ setLoading(false)
186
+ }
187
+ }, [fetcher, options])
188
+
189
+ useEffect(() => {
190
+ if (options?.enabled !== false) {
191
+ refetch()
192
+ }
193
+ }, [key, refetch, options?.enabled])
194
+
195
+ return { data, error, loading, refetch }
196
+ }
197
+
198
+ // Usage
199
+ const { data: markets, loading, error, refetch } = useQuery(
200
+ 'markets',
201
+ () => fetch('/api/markets').then(r => r.json()),
202
+ {
203
+ onSuccess: data => console.log('Fetched', data.length, 'markets'),
204
+ onError: err => console.error('Failed:', err)
205
+ }
206
+ )
207
+ ```
208
+
209
+ ### Debounce Hook
210
+
211
+ ```typescript
212
+ export function useDebounce<T>(value: T, delay: number): T {
213
+ const [debouncedValue, setDebouncedValue] = useState<T>(value)
214
+
215
+ useEffect(() => {
216
+ const handler = setTimeout(() => {
217
+ setDebouncedValue(value)
218
+ }, delay)
219
+
220
+ return () => clearTimeout(handler)
221
+ }, [value, delay])
222
+
223
+ return debouncedValue
224
+ }
225
+
226
+ // Usage
227
+ const [searchQuery, setSearchQuery] = useState('')
228
+ const debouncedQuery = useDebounce(searchQuery, 500)
229
+
230
+ useEffect(() => {
231
+ if (debouncedQuery) {
232
+ performSearch(debouncedQuery)
233
+ }
234
+ }, [debouncedQuery])
235
+ ```
236
+
237
+ ## State Management Patterns
238
+
239
+ ### Context + Reducer Pattern
240
+
241
+ ```typescript
242
+ interface State {
243
+ markets: Market[]
244
+ selectedMarket: Market | null
245
+ loading: boolean
246
+ }
247
+
248
+ type Action =
249
+ | { type: 'SET_MARKETS'; payload: Market[] }
250
+ | { type: 'SELECT_MARKET'; payload: Market }
251
+ | { type: 'SET_LOADING'; payload: boolean }
252
+
253
+ function reducer(state: State, action: Action): State {
254
+ switch (action.type) {
255
+ case 'SET_MARKETS':
256
+ return { ...state, markets: action.payload }
257
+ case 'SELECT_MARKET':
258
+ return { ...state, selectedMarket: action.payload }
259
+ case 'SET_LOADING':
260
+ return { ...state, loading: action.payload }
261
+ default:
262
+ return state
263
+ }
264
+ }
265
+
266
+ const MarketContext = createContext<{
267
+ state: State
268
+ dispatch: Dispatch<Action>
269
+ } | undefined>(undefined)
270
+
271
+ export function MarketProvider({ children }: { children: React.ReactNode }) {
272
+ const [state, dispatch] = useReducer(reducer, {
273
+ markets: [],
274
+ selectedMarket: null,
275
+ loading: false
276
+ })
277
+
278
+ return (
279
+ <MarketContext.Provider value={{ state, dispatch }}>
280
+ {children}
281
+ </MarketContext.Provider>
282
+ )
283
+ }
284
+
285
+ export function useMarkets() {
286
+ const context = useContext(MarketContext)
287
+ if (!context) throw new Error('useMarkets must be used within MarketProvider')
288
+ return context
289
+ }
290
+ ```
291
+
292
+ ## Performance Optimization
293
+
294
+ ### Memoization
295
+
296
+ ```typescript
297
+ // PASS: useMemo for expensive computations
298
+ const sortedMarkets = useMemo(() => {
299
+ return markets.sort((a, b) => b.volume - a.volume)
300
+ }, [markets])
301
+
302
+ // PASS: useCallback for functions passed to children
303
+ const handleSearch = useCallback((query: string) => {
304
+ setSearchQuery(query)
305
+ }, [])
306
+
307
+ // PASS: React.memo for pure components
308
+ export const MarketCard = React.memo<MarketCardProps>(({ market }) => {
309
+ return (
310
+ <div className="market-card">
311
+ <h3>{market.name}</h3>
312
+ <p>{market.description}</p>
313
+ </div>
314
+ )
315
+ })
316
+ ```
317
+
318
+ ### Code Splitting & Lazy Loading
319
+
320
+ ```typescript
321
+ import { lazy, Suspense } from 'react'
322
+
323
+ // PASS: Lazy load heavy components
324
+ const HeavyChart = lazy(() => import('./HeavyChart'))
325
+ const ThreeJsBackground = lazy(() => import('./ThreeJsBackground'))
326
+
327
+ export function Dashboard() {
328
+ return (
329
+ <div>
330
+ <Suspense fallback={<ChartSkeleton />}>
331
+ <HeavyChart data={data} />
332
+ </Suspense>
333
+
334
+ <Suspense fallback={null}>
335
+ <ThreeJsBackground />
336
+ </Suspense>
337
+ </div>
338
+ )
339
+ }
340
+ ```
341
+
342
+ ### Virtualization for Long Lists
343
+
344
+ ```typescript
345
+ import { useVirtualizer } from '@tanstack/react-virtual'
346
+
347
+ export function VirtualMarketList({ markets }: { markets: Market[] }) {
348
+ const parentRef = useRef<HTMLDivElement>(null)
349
+
350
+ const virtualizer = useVirtualizer({
351
+ count: markets.length,
352
+ getScrollElement: () => parentRef.current,
353
+ estimateSize: () => 100, // Estimated row height
354
+ overscan: 5 // Extra items to render
355
+ })
356
+
357
+ return (
358
+ <div ref={parentRef} style={{ height: '600px', overflow: 'auto' }}>
359
+ <div
360
+ style={{
361
+ height: `${virtualizer.getTotalSize()}px`,
362
+ position: 'relative'
363
+ }}
364
+ >
365
+ {virtualizer.getVirtualItems().map(virtualRow => (
366
+ <div
367
+ key={virtualRow.index}
368
+ style={{
369
+ position: 'absolute',
370
+ top: 0,
371
+ left: 0,
372
+ width: '100%',
373
+ height: `${virtualRow.size}px`,
374
+ transform: `translateY(${virtualRow.start}px)`
375
+ }}
376
+ >
377
+ <MarketCard market={markets[virtualRow.index]} />
378
+ </div>
379
+ ))}
380
+ </div>
381
+ </div>
382
+ )
383
+ }
384
+ ```
385
+
386
+ ## Form Handling Patterns
387
+
388
+ ### Controlled Form with Validation
389
+
390
+ ```typescript
391
+ interface FormData {
392
+ name: string
393
+ description: string
394
+ endDate: string
395
+ }
396
+
397
+ interface FormErrors {
398
+ name?: string
399
+ description?: string
400
+ endDate?: string
401
+ }
402
+
403
+ export function CreateMarketForm() {
404
+ const [formData, setFormData] = useState<FormData>({
405
+ name: '',
406
+ description: '',
407
+ endDate: ''
408
+ })
409
+
410
+ const [errors, setErrors] = useState<FormErrors>({})
411
+
412
+ const validate = (): boolean => {
413
+ const newErrors: FormErrors = {}
414
+
415
+ if (!formData.name.trim()) {
416
+ newErrors.name = 'Name is required'
417
+ } else if (formData.name.length > 200) {
418
+ newErrors.name = 'Name must be under 200 characters'
419
+ }
420
+
421
+ if (!formData.description.trim()) {
422
+ newErrors.description = 'Description is required'
423
+ }
424
+
425
+ if (!formData.endDate) {
426
+ newErrors.endDate = 'End date is required'
427
+ }
428
+
429
+ setErrors(newErrors)
430
+ return Object.keys(newErrors).length === 0
431
+ }
432
+
433
+ const handleSubmit = async (e: React.FormEvent) => {
434
+ e.preventDefault()
435
+
436
+ if (!validate()) return
437
+
438
+ try {
439
+ await createMarket(formData)
440
+ // Success handling
441
+ } catch (error) {
442
+ // Error handling
443
+ }
444
+ }
445
+
446
+ return (
447
+ <form onSubmit={handleSubmit}>
448
+ <input
449
+ value={formData.name}
450
+ onChange={e => setFormData(prev => ({ ...prev, name: e.target.value }))}
451
+ placeholder="Market name"
452
+ />
453
+ {errors.name && <span className="error">{errors.name}</span>}
454
+
455
+ {/* Other fields */}
456
+
457
+ <button type="submit">Create Market</button>
458
+ </form>
459
+ )
460
+ }
461
+ ```
462
+
463
+ ## Error Boundary Pattern
464
+
465
+ ```typescript
466
+ interface ErrorBoundaryState {
467
+ hasError: boolean
468
+ error: Error | null
469
+ }
470
+
471
+ export class ErrorBoundary extends React.Component<
472
+ { children: React.ReactNode },
473
+ ErrorBoundaryState
474
+ > {
475
+ state: ErrorBoundaryState = {
476
+ hasError: false,
477
+ error: null
478
+ }
479
+
480
+ static getDerivedStateFromError(error: Error): ErrorBoundaryState {
481
+ return { hasError: true, error }
482
+ }
483
+
484
+ componentDidCatch(error: Error, errorInfo: React.ErrorInfo) {
485
+ console.error('Error boundary caught:', error, errorInfo)
486
+ }
487
+
488
+ render() {
489
+ if (this.state.hasError) {
490
+ return (
491
+ <div className="error-fallback">
492
+ <h2>Something went wrong</h2>
493
+ <p>{this.state.error?.message}</p>
494
+ <button onClick={() => this.setState({ hasError: false })}>
495
+ Try again
496
+ </button>
497
+ </div>
498
+ )
499
+ }
500
+
501
+ return this.props.children
502
+ }
503
+ }
504
+
505
+ // Usage
506
+ <ErrorBoundary>
507
+ <App />
508
+ </ErrorBoundary>
509
+ ```
510
+
511
+ ## Animation Patterns
512
+
513
+ ### Framer Motion Animations
514
+
515
+ ```typescript
516
+ import { motion, AnimatePresence } from 'framer-motion'
517
+
518
+ // PASS: List animations
519
+ export function AnimatedMarketList({ markets }: { markets: Market[] }) {
520
+ return (
521
+ <AnimatePresence>
522
+ {markets.map(market => (
523
+ <motion.div
524
+ key={market.id}
525
+ initial={{ opacity: 0, y: 20 }}
526
+ animate={{ opacity: 1, y: 0 }}
527
+ exit={{ opacity: 0, y: -20 }}
528
+ transition={{ duration: 0.3 }}
529
+ >
530
+ <MarketCard market={market} />
531
+ </motion.div>
532
+ ))}
533
+ </AnimatePresence>
534
+ )
535
+ }
536
+
537
+ // PASS: Modal animations
538
+ export function Modal({ isOpen, onClose, children }: ModalProps) {
539
+ return (
540
+ <AnimatePresence>
541
+ {isOpen && (
542
+ <>
543
+ <motion.div
544
+ className="modal-overlay"
545
+ initial={{ opacity: 0 }}
546
+ animate={{ opacity: 1 }}
547
+ exit={{ opacity: 0 }}
548
+ onClick={onClose}
549
+ />
550
+ <motion.div
551
+ className="modal-content"
552
+ initial={{ opacity: 0, scale: 0.9, y: 20 }}
553
+ animate={{ opacity: 1, scale: 1, y: 0 }}
554
+ exit={{ opacity: 0, scale: 0.9, y: 20 }}
555
+ >
556
+ {children}
557
+ </motion.div>
558
+ </>
559
+ )}
560
+ </AnimatePresence>
561
+ )
562
+ }
563
+ ```
564
+
565
+ ## Accessibility Patterns
566
+
567
+ ### Keyboard Navigation
568
+
569
+ ```typescript
570
+ export function Dropdown({ options, onSelect }: DropdownProps) {
571
+ const [isOpen, setIsOpen] = useState(false)
572
+ const [activeIndex, setActiveIndex] = useState(0)
573
+
574
+ const handleKeyDown = (e: React.KeyboardEvent) => {
575
+ switch (e.key) {
576
+ case 'ArrowDown':
577
+ e.preventDefault()
578
+ setActiveIndex(i => Math.min(i + 1, options.length - 1))
579
+ break
580
+ case 'ArrowUp':
581
+ e.preventDefault()
582
+ setActiveIndex(i => Math.max(i - 1, 0))
583
+ break
584
+ case 'Enter':
585
+ e.preventDefault()
586
+ onSelect(options[activeIndex])
587
+ setIsOpen(false)
588
+ break
589
+ case 'Escape':
590
+ setIsOpen(false)
591
+ break
592
+ }
593
+ }
594
+
595
+ return (
596
+ <div
597
+ role="combobox"
598
+ aria-expanded={isOpen}
599
+ aria-haspopup="listbox"
600
+ onKeyDown={handleKeyDown}
601
+ >
602
+ {/* Dropdown implementation */}
603
+ </div>
604
+ )
605
+ }
606
+ ```
607
+
608
+ ### Focus Management
609
+
610
+ ```typescript
611
+ export function Modal({ isOpen, onClose, children }: ModalProps) {
612
+ const modalRef = useRef<HTMLDivElement>(null)
613
+ const previousFocusRef = useRef<HTMLElement | null>(null)
614
+
615
+ useEffect(() => {
616
+ if (isOpen) {
617
+ // Save currently focused element
618
+ previousFocusRef.current = document.activeElement as HTMLElement
619
+
620
+ // Focus modal
621
+ modalRef.current?.focus()
622
+ } else {
623
+ // Restore focus when closing
624
+ previousFocusRef.current?.focus()
625
+ }
626
+ }, [isOpen])
627
+
628
+ return isOpen ? (
629
+ <div
630
+ ref={modalRef}
631
+ role="dialog"
632
+ aria-modal="true"
633
+ tabIndex={-1}
634
+ onKeyDown={e => e.key === 'Escape' && onClose()}
635
+ >
636
+ {children}
637
+ </div>
638
+ ) : null
639
+ }
640
+ ```
641
+
642
+ **Remember**: Modern frontend patterns enable maintainable, performant user interfaces. Choose patterns that fit your project complexity.
643
+