ccjk 14.1.10 → 14.2.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 (119) hide show
  1. package/dist/chunks/agent-teams.mjs +1 -1
  2. package/dist/chunks/api-cli.mjs +1 -1
  3. package/dist/chunks/api-config-selector.mjs +3 -3
  4. package/dist/chunks/ccjk-all.mjs +1 -1
  5. package/dist/chunks/ccjk-mcp.mjs +1 -1
  6. package/dist/chunks/ccjk-setup.mjs +1 -1
  7. package/dist/chunks/ccr.mjs +8 -8
  8. package/dist/chunks/check-updates.mjs +1 -1
  9. package/dist/chunks/claude-code-incremental-manager.mjs +3 -3
  10. package/dist/chunks/claude-config.mjs +1 -1
  11. package/dist/chunks/codex-config-switch.mjs +1 -1
  12. package/dist/chunks/codex-provider-manager.mjs +1 -1
  13. package/dist/chunks/config-switch.mjs +1 -1
  14. package/dist/chunks/config.mjs +19 -3
  15. package/dist/chunks/config2.mjs +1 -1
  16. package/dist/chunks/config3.mjs +1 -1
  17. package/dist/chunks/doctor.mjs +175 -6
  18. package/dist/chunks/features.mjs +3 -3
  19. package/dist/chunks/index10.mjs +19 -5
  20. package/dist/chunks/init.mjs +5 -5
  21. package/dist/chunks/mcp-cli.mjs +3 -3
  22. package/dist/chunks/mcp.mjs +3 -3
  23. package/dist/chunks/package.mjs +1 -1
  24. package/dist/chunks/quick-provider.mjs +1 -1
  25. package/dist/chunks/quick-setup.mjs +5 -5
  26. package/dist/chunks/simple-config.mjs +1 -1
  27. package/dist/chunks/smart-guide.mjs +1 -1
  28. package/dist/chunks/update.mjs +6 -6
  29. package/dist/chunks/zero-config.mjs +7 -4
  30. package/dist/cli.mjs +0 -0
  31. package/dist/index.d.mts +3 -0
  32. package/dist/index.d.ts +3 -0
  33. package/dist/index.mjs +1 -1
  34. package/dist/shared/{ccjk.BJ3Zpjo5.mjs → ccjk.BCzOWT1L.mjs} +3 -2
  35. package/dist/shared/{ccjk.B8oqkakg.mjs → ccjk.BLsIiTqO.mjs} +1 -1
  36. package/dist/shared/{ccjk.Hwoicrh8.mjs → ccjk.BXv8aYs1.mjs} +1 -1
  37. package/dist/shared/{ccjk.B9OuS4xZ.mjs → ccjk.CfKJnpbB.mjs} +1 -1
  38. package/dist/shared/{ccjk.BzxpiEPF.mjs → ccjk.Cgv_cFVX.mjs} +2 -2
  39. package/dist/shared/{ccjk.Di1IYU3u.mjs → ccjk.DDL-4C-k.mjs} +47 -10
  40. package/dist/shared/{ccjk.BEiR3L4C.mjs → ccjk.DOw7Fawt.mjs} +3 -3
  41. package/dist/shared/{ccjk.tI_s2uSh.mjs → ccjk.f3TBLJSt.mjs} +1 -1
  42. package/dist/templates/agents/README.md +78 -0
  43. package/dist/templates/common/error-prevention.md +267 -0
  44. package/dist/templates/common/karpathy-baseline.md +83 -0
  45. package/dist/templates/common/output-styles/zh-CN/carmack-mode.md +381 -0
  46. package/dist/templates/common/output-styles/zh-CN/dhh-mode.md +265 -0
  47. package/dist/templates/common/output-styles/zh-CN/evan-you-mode.md +539 -0
  48. package/dist/templates/common/output-styles/zh-CN/jobs-mode.md +369 -0
  49. package/dist/templates/common/output-styles/zh-CN/linus-mode.md +135 -0
  50. package/dist/templates/common/output-styles/zh-CN/uncle-bob-mode.md +221 -0
  51. package/dist/templates/common/workflow/continuousDelivery/en/continuous-delivery.md +628 -0
  52. package/dist/templates/common/workflow/continuousDelivery/zh-CN/continuous-delivery.md +628 -0
  53. package/dist/templates/common/workflow/essential/en/agents/ccjk-config-agent.md +187 -0
  54. package/dist/templates/common/workflow/essential/en/agents/ccjk-mcp-agent.md +191 -0
  55. package/dist/templates/common/workflow/essential/en/agents/ccjk-skill-agent.md +249 -0
  56. package/dist/templates/common/workflow/essential/en/agents/ccjk-workflow-agent.md +277 -0
  57. package/dist/templates/common/workflow/essential/en/agents/get-current-datetime.md +29 -0
  58. package/dist/templates/common/workflow/essential/en/agents/init-architect.md +115 -0
  59. package/dist/templates/common/workflow/essential/en/agents/ui-ux-designer.md +91 -0
  60. package/dist/templates/common/workflow/essential/en/feat.md +92 -0
  61. package/dist/templates/common/workflow/essential/en/goal.md +147 -0
  62. package/dist/templates/common/workflow/essential/en/init-project.md +53 -0
  63. package/dist/templates/common/workflow/essential/zh-CN/agents/get-current-datetime.md +29 -0
  64. package/dist/templates/common/workflow/essential/zh-CN/agents/init-architect.md +115 -0
  65. package/dist/templates/common/workflow/essential/zh-CN/agents/ui-ux-designer.md +91 -0
  66. package/dist/templates/common/workflow/essential/zh-CN/feat.md +315 -0
  67. package/dist/templates/common/workflow/essential/zh-CN/goal.md +146 -0
  68. package/dist/templates/common/workflow/essential/zh-CN/init-project.md +53 -0
  69. package/dist/templates/common/workflow/git/en/git-cleanBranches.md +102 -0
  70. package/dist/templates/common/workflow/git/en/git-commit.md +205 -0
  71. package/dist/templates/common/workflow/git/en/git-rollback.md +90 -0
  72. package/dist/templates/common/workflow/git/en/git-worktree.md +276 -0
  73. package/dist/templates/common/workflow/git/zh-CN/git-cleanBranches.md +102 -0
  74. package/dist/templates/common/workflow/git/zh-CN/git-commit.md +205 -0
  75. package/dist/templates/common/workflow/git/zh-CN/git-rollback.md +90 -0
  76. package/dist/templates/common/workflow/git/zh-CN/git-worktree.md +276 -0
  77. package/dist/templates/common/workflow/interview/en/interview.md +67 -0
  78. package/dist/templates/common/workflow/interview/zh-CN/interview.md +67 -0
  79. package/dist/templates/common/workflow/linearMethod/en/linear-method.md +651 -0
  80. package/dist/templates/common/workflow/linearMethod/zh-CN/linear-method.md +752 -0
  81. package/dist/templates/common/workflow/refactoringMaster/en/refactoring-master.md +516 -0
  82. package/dist/templates/common/workflow/refactoringMaster/zh-CN/refactoring-master.md +812 -0
  83. package/dist/templates/common/workflow/sixStep/en/workflow.md +83 -0
  84. package/dist/templates/common/workflow/sixStep/zh-CN/workflow.md +359 -0
  85. package/dist/templates/common/workflow/specFirstTDD/en/spec-first-tdd.md +364 -0
  86. package/dist/templates/common/workflow/specFirstTDD/zh-CN/spec-first-tdd.md +366 -0
  87. package/dist/templates/hooks/README.md +212 -0
  88. package/dist/templates/hooks/git-workflow-hooks.md +551 -0
  89. package/dist/templates/hooks/post-test-coverage.md +434 -0
  90. package/dist/templates/hooks/pre-commit-black.md +274 -0
  91. package/dist/templates/hooks/pre-commit-eslint.md +153 -0
  92. package/dist/templates/hooks/pre-commit-gofmt.md +284 -0
  93. package/dist/templates/hooks/pre-commit-prettier.md +212 -0
  94. package/dist/templates/hooks/pre-commit-type-check.md +377 -0
  95. package/dist/templates/skills/ccjk-init.md +154 -0
  96. package/dist/templates/skills/ccjk-mcp-setup.md +205 -0
  97. package/dist/templates/skills/ccjk-troubleshoot.md +228 -0
  98. package/dist/templates/skills/django-patterns.md +1016 -0
  99. package/dist/templates/skills/git-workflow.md +748 -0
  100. package/dist/templates/skills/go-idioms.md +963 -0
  101. package/dist/templates/skills/nextjs-optimization.md +694 -0
  102. package/dist/templates/skills/python-pep8.md +852 -0
  103. package/dist/templates/skills/react-patterns.md +686 -0
  104. package/dist/templates/skills/rust-patterns.md +1057 -0
  105. package/dist/templates/skills/security-best-practices.md +1413 -0
  106. package/dist/templates/skills/testing-best-practices.md +1315 -0
  107. package/dist/templates/skills/ts-best-practices.md +354 -0
  108. package/package.json +40 -43
  109. package/templates/common/karpathy-baseline.md +83 -0
  110. package/templates/common/output-styles/zh-CN/carmack-mode.md +14 -0
  111. package/templates/common/output-styles/zh-CN/dhh-mode.md +14 -0
  112. package/templates/common/output-styles/zh-CN/evan-you-mode.md +14 -0
  113. package/templates/common/output-styles/zh-CN/jobs-mode.md +14 -0
  114. package/templates/common/output-styles/zh-CN/linus-mode.md +14 -0
  115. package/templates/common/output-styles/zh-CN/uncle-bob-mode.md +14 -0
  116. package/templates/common/workflow/linearMethod/zh-CN/linear-method.md +2 -0
  117. package/templates/common/workflow/refactoringMaster/zh-CN/refactoring-master.md +2 -0
  118. package/templates/common/workflow/sixStep/zh-CN/workflow.md +2 -0
  119. package/templates/common/workflow/specFirstTDD/zh-CN/spec-first-tdd.md +2 -0
@@ -0,0 +1,694 @@
1
+ ---
2
+ name: nextjs-optimization
3
+ description: Next.js App Router patterns, server components, and performance optimization
4
+ description_zh: Next.js App Router 模式、服务器组件和性能优化
5
+ version: 1.0.0
6
+ category: frontend
7
+ triggers: ['/nextjs-optimization', '/nextjs', '/app-router', '/server-components']
8
+ use_when:
9
+ - Building Next.js 13+ applications with App Router
10
+ - Implementing server components and data fetching strategies
11
+ - Optimizing Next.js performance and SEO
12
+ - Code review for Next.js applications
13
+ use_when_zh:
14
+ - 使用 App Router 构建 Next.js 13+ 应用程序
15
+ - 实现服务器组件和数据获取策略
16
+ - 优化 Next.js 性能和 SEO
17
+ - Next.js 应用程序代码审查
18
+ auto_activate: true
19
+ priority: 8
20
+ agents: [nextjs-expert, fullstack-architect]
21
+ tags: [nextjs, app-router, server-components, performance, seo]
22
+ ---
23
+
24
+ # Next.js Optimization | Next.js 优化
25
+
26
+ ## Context | 上下文
27
+
28
+ Use this skill when developing Next.js 13+ applications with App Router that require optimal performance, SEO, and modern React patterns. Essential for production-ready Next.js applications.
29
+
30
+ 在开发需要最佳性能、SEO 和现代 React 模式的 Next.js 13+ App Router 应用程序时使用此技能。对于生产就绪的 Next.js 应用程序至关重要。
31
+
32
+ ## App Router Patterns | App Router 模式
33
+
34
+ ### 1. File-based Routing Structure | 基于文件的路由结构
35
+
36
+ ```
37
+ app/
38
+ ├── layout.tsx # Root layout
39
+ ├── page.tsx # Home page
40
+ ├── loading.tsx # Loading UI
41
+ ├── error.tsx # Error UI
42
+ ├── not-found.tsx # 404 page
43
+ ├── global-error.tsx # Global error boundary
44
+ ├── (auth)/ # Route groups
45
+ │ ├── login/
46
+ │ │ └── page.tsx
47
+ │ └── register/
48
+ │ └── page.tsx
49
+ ├── dashboard/
50
+ │ ├── layout.tsx # Nested layout
51
+ │ ├── page.tsx
52
+ │ ├── settings/
53
+ │ │ └── page.tsx
54
+ │ └── [userId]/ # Dynamic routes
55
+ │ ├── page.tsx
56
+ │ └── edit/
57
+ │ └── page.tsx
58
+ └── api/ # API routes
59
+ ├── auth/
60
+ │ └── route.ts
61
+ └── users/
62
+ └── route.ts
63
+ ```
64
+
65
+ ### 2. Server Components Best Practices | 服务器组件最佳实践
66
+
67
+ ```tsx
68
+ // ✅ Good: Server component with data fetching
69
+ import { Suspense } from 'react';
70
+ import { getUserPosts, getUser } from '@/lib/api';
71
+
72
+ // Server component - runs on server
73
+ async function UserProfile({ userId }: { userId: string }) {
74
+ // Fetch data directly in server component
75
+ const user = await getUser(userId);
76
+
77
+ if (!user) {
78
+ return <div>User not found</div>;
79
+ }
80
+
81
+ return (
82
+ <div>
83
+ <h1>{user.name}</h1>
84
+ <p>{user.email}</p>
85
+ <Suspense fallback={<div>Loading posts...</div>}>
86
+ <UserPosts userId={userId} />
87
+ </Suspense>
88
+ </div>
89
+ );
90
+ }
91
+
92
+ // Separate server component for posts
93
+ async function UserPosts({ userId }: { userId: string }) {
94
+ const posts = await getUserPosts(userId);
95
+
96
+ return (
97
+ <div>
98
+ {posts.map(post => (
99
+ <article key={post.id}>
100
+ <h2>{post.title}</h2>
101
+ <p>{post.excerpt}</p>
102
+ </article>
103
+ ))}
104
+ </div>
105
+ );
106
+ }
107
+
108
+ // Page component
109
+ export default function UserPage({ params }: { params: { userId: string } }) {
110
+ return (
111
+ <main>
112
+ <Suspense fallback={<div>Loading user...</div>}>
113
+ <UserProfile userId={params.userId} />
114
+ </Suspense>
115
+ </main>
116
+ );
117
+ }
118
+ ```
119
+
120
+ ### 3. Client Components Pattern | 客户端组件模式
121
+
122
+ ```tsx
123
+ 'use client'; // Mark as client component
124
+
125
+ import { useState, useEffect } from 'react';
126
+ import { useRouter } from 'next/navigation';
127
+
128
+ // ✅ Good: Client component for interactivity
129
+ export function InteractiveCounter({ initialCount = 0 }: { initialCount?: number }) {
130
+ const [count, setCount] = useState(initialCount);
131
+ const router = useRouter();
132
+
133
+ useEffect(() => {
134
+ // Client-side only code
135
+ const interval = setInterval(() => {
136
+ setCount(c => c + 1);
137
+ }, 1000);
138
+
139
+ return () => clearInterval(interval);
140
+ }, []);
141
+
142
+ const handleReset = () => {
143
+ setCount(0);
144
+ // Client-side navigation
145
+ router.refresh();
146
+ };
147
+
148
+ return (
149
+ <div>
150
+ <p>Count: {count}</p>
151
+ <button onClick={() => setCount(c => c + 1)}>
152
+ Increment
153
+ </button>
154
+ <button onClick={handleReset}>
155
+ Reset
156
+ </button>
157
+ </div>
158
+ );
159
+ }
160
+
161
+ // ✅ Good: Hybrid approach - server component wrapping client component
162
+ import { InteractiveCounter } from './InteractiveCounter';
163
+
164
+ async function CounterPage() {
165
+ // Server-side data fetching
166
+ const initialData = await fetch('https://api.example.com/counter').then(r => r.json());
167
+
168
+ return (
169
+ <div>
170
+ <h1>Counter Page</h1>
171
+ <InteractiveCounter initialCount={initialData.count} />
172
+ </div>
173
+ );
174
+ }
175
+ ```
176
+
177
+ ## Data Fetching Strategies | 数据获取策略
178
+
179
+ ### 1. Server-Side Data Fetching | 服务器端数据获取
180
+
181
+ ```tsx
182
+ // ✅ Good: Parallel data fetching
183
+ async function ProductPage({ params }: { params: { id: string } }) {
184
+ // Fetch data in parallel
185
+ const [product, reviews, relatedProducts] = await Promise.all([
186
+ getProduct(params.id),
187
+ getProductReviews(params.id),
188
+ getRelatedProducts(params.id)
189
+ ]);
190
+
191
+ return (
192
+ <div>
193
+ <ProductDetails product={product} />
194
+ <ProductReviews reviews={reviews} />
195
+ <RelatedProducts products={relatedProducts} />
196
+ </div>
197
+ );
198
+ }
199
+
200
+ // ✅ Good: Streaming with Suspense
201
+ function ProductPageWithStreaming({ params }: { params: { id: string } }) {
202
+ return (
203
+ <div>
204
+ <Suspense fallback={<ProductSkeleton />}>
205
+ <ProductDetails productId={params.id} />
206
+ </Suspense>
207
+
208
+ <Suspense fallback={<ReviewsSkeleton />}>
209
+ <ProductReviews productId={params.id} />
210
+ </Suspense>
211
+
212
+ <Suspense fallback={<RelatedSkeleton />}>
213
+ <RelatedProducts productId={params.id} />
214
+ </Suspense>
215
+ </div>
216
+ );
217
+ }
218
+
219
+ async function ProductDetails({ productId }: { productId: string }) {
220
+ const product = await getProduct(productId);
221
+ return <div>{/* Product details */}</div>;
222
+ }
223
+ ```
224
+
225
+ ### 2. Client-Side Data Fetching | 客户端数据获取
226
+
227
+ ```tsx
228
+ 'use client';
229
+
230
+ import { useState, useEffect } from 'react';
231
+ import useSWR from 'swr';
232
+
233
+ // ✅ Good: SWR for client-side data fetching
234
+ function UserDashboard() {
235
+ const { data: user, error, isLoading } = useSWR('/api/user', fetcher);
236
+ const { data: notifications } = useSWR('/api/notifications', fetcher, {
237
+ refreshInterval: 30000 // Refresh every 30 seconds
238
+ });
239
+
240
+ if (error) return <div>Failed to load user data</div>;
241
+ if (isLoading) return <div>Loading...</div>;
242
+
243
+ return (
244
+ <div>
245
+ <h1>Welcome, {user.name}</h1>
246
+ <NotificationList notifications={notifications} />
247
+ </div>
248
+ );
249
+ }
250
+
251
+ // ✅ Good: Custom hook for data fetching
252
+ function useApi<T>(url: string) {
253
+ const [data, setData] = useState<T | null>(null);
254
+ const [loading, setLoading] = useState(true);
255
+ const [error, setError] = useState<Error | null>(null);
256
+
257
+ useEffect(() => {
258
+ async function fetchData() {
259
+ try {
260
+ setLoading(true);
261
+ const response = await fetch(url);
262
+ if (!response.ok) throw new Error('Failed to fetch');
263
+ const result = await response.json();
264
+ setData(result);
265
+ } catch (err) {
266
+ setError(err instanceof Error ? err : new Error('Unknown error'));
267
+ } finally {
268
+ setLoading(false);
269
+ }
270
+ }
271
+
272
+ fetchData();
273
+ }, [url]);
274
+
275
+ return { data, loading, error };
276
+ }
277
+ ```
278
+
279
+ ### 3. Caching and Revalidation | 缓存和重新验证
280
+
281
+ ```tsx
282
+ // ✅ Good: Static generation with revalidation
283
+ export const revalidate = 3600; // Revalidate every hour
284
+
285
+ async function BlogPost({ params }: { params: { slug: string } }) {
286
+ const post = await getBlogPost(params.slug);
287
+
288
+ return (
289
+ <article>
290
+ <h1>{post.title}</h1>
291
+ <div dangerouslySetInnerHTML={{ __html: post.content }} />
292
+ </article>
293
+ );
294
+ }
295
+
296
+ // ✅ Good: Dynamic with custom cache
297
+ async function getProductWithCache(id: string) {
298
+ const response = await fetch(`https://api.example.com/products/${id}`, {
299
+ next: {
300
+ revalidate: 300, // Cache for 5 minutes
301
+ tags: [`product-${id}`] // For on-demand revalidation
302
+ }
303
+ });
304
+
305
+ if (!response.ok) {
306
+ throw new Error('Failed to fetch product');
307
+ }
308
+
309
+ return response.json();
310
+ }
311
+
312
+ // API route for cache revalidation
313
+ // app/api/revalidate/route.ts
314
+ import { revalidateTag } from 'next/cache';
315
+ import { NextRequest } from 'next/server';
316
+
317
+ export async function POST(request: NextRequest) {
318
+ const { tag } = await request.json();
319
+
320
+ revalidateTag(tag);
321
+
322
+ return Response.json({ revalidated: true });
323
+ }
324
+ ```
325
+
326
+ ## Performance Optimization | 性能优化
327
+
328
+ ### 1. Image Optimization | 图像优化
329
+
330
+ ```tsx
331
+ import Image from 'next/image';
332
+
333
+ // ✅ Good: Optimized images with Next.js Image component
334
+ function ProductGallery({ product }: { product: Product }) {
335
+ return (
336
+ <div className="grid grid-cols-2 gap-4">
337
+ {/* Hero image with priority loading */}
338
+ <Image
339
+ src={product.images[0]}
340
+ alt={product.name}
341
+ width={800}
342
+ height={600}
343
+ priority
344
+ className="rounded-lg"
345
+ placeholder="blur"
346
+ blurDataURL="data:image/jpeg;base64,/9j/4AAQSkZJRgABAQAAAQABAAD/2wBDAAYEBQYFBAYGBQYHBwYIChAKCgkJChQODwwQFxQYGBcUFhYaHSUfGhsjHBYWICwgIyYnKSopGR8tMC0oMCUoKSj/2wBDAQcHBwoIChMKChMoGhYaKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCgoKCj/wAARCAABAAEDASIAAhEBAxEB/8QAFQABAQAAAAAAAAAAAAAAAAAAAAv/xAAUEAEAAAAAAAAAAAAAAAAAAAAA/8QAFQEBAQAAAAAAAAAAAAAAAAAAAAX/xAAUEQEAAAAAAAAAAAAAAAAAAAAA/9oADAMBAAIRAxEAPwCdABmX/9k="
347
+ />
348
+
349
+ {/* Lazy loaded images */}
350
+ {product.images.slice(1).map((image, index) => (
351
+ <Image
352
+ key={index}
353
+ src={image}
354
+ alt={`${product.name} ${index + 2}`}
355
+ width={400}
356
+ height={300}
357
+ className="rounded-lg"
358
+ loading="lazy"
359
+ />
360
+ ))}
361
+ </div>
362
+ );
363
+ }
364
+
365
+ // ✅ Good: Responsive images
366
+ function ResponsiveHero({ image }: { image: string }) {
367
+ return (
368
+ <Image
369
+ src={image}
370
+ alt="Hero image"
371
+ fill
372
+ className="object-cover"
373
+ sizes="(max-width: 768px) 100vw, (max-width: 1200px) 50vw, 33vw"
374
+ priority
375
+ />
376
+ );
377
+ }
378
+ ```
379
+
380
+ ### 2. Bundle Optimization | 包优化
381
+
382
+ ```tsx
383
+ // ✅ Good: Dynamic imports for code splitting
384
+ import dynamic from 'next/dynamic';
385
+
386
+ // Lazy load heavy components
387
+ const Chart = dynamic(() => import('./Chart'), {
388
+ loading: () => <div>Loading chart...</div>,
389
+ ssr: false // Disable SSR for client-only components
390
+ });
391
+
392
+ const Modal = dynamic(() => import('./Modal'), {
393
+ loading: () => <div>Loading...</div>
394
+ });
395
+
396
+ function Dashboard() {
397
+ const [showChart, setShowChart] = useState(false);
398
+ const [showModal, setShowModal] = useState(false);
399
+
400
+ return (
401
+ <div>
402
+ <button onClick={() => setShowChart(true)}>
403
+ Show Chart
404
+ </button>
405
+
406
+ {showChart && <Chart />}
407
+ {showModal && <Modal onClose={() => setShowModal(false)} />}
408
+ </div>
409
+ );
410
+ }
411
+
412
+ // ✅ Good: Tree shaking with named imports
413
+ import { debounce } from 'lodash-es'; // Better than import _ from 'lodash'
414
+ import { format } from 'date-fns'; // Better than import * as dateFns
415
+ ```
416
+
417
+ ### 3. Streaming and Suspense | 流式传输和 Suspense
418
+
419
+ ```tsx
420
+ import { Suspense } from 'react';
421
+
422
+ // ✅ Good: Streaming layout
423
+ export default function Layout({ children }: { children: React.ReactNode }) {
424
+ return (
425
+ <html>
426
+ <body>
427
+ <header>
428
+ <Suspense fallback={<div>Loading navigation...</div>}>
429
+ <Navigation />
430
+ </Suspense>
431
+ </header>
432
+
433
+ <main>
434
+ <Suspense fallback={<div>Loading content...</div>}>
435
+ {children}
436
+ </Suspense>
437
+ </main>
438
+
439
+ <footer>
440
+ <Suspense fallback={<div>Loading footer...</div>}>
441
+ <Footer />
442
+ </Suspense>
443
+ </footer>
444
+ </body>
445
+ </html>
446
+ );
447
+ }
448
+
449
+ // ✅ Good: Progressive enhancement
450
+ function ProductPage({ params }: { params: { id: string } }) {
451
+ return (
452
+ <div>
453
+ {/* Critical content loads first */}
454
+ <Suspense fallback={<ProductSkeleton />}>
455
+ <ProductInfo productId={params.id} />
456
+ </Suspense>
457
+
458
+ {/* Non-critical content streams in later */}
459
+ <Suspense fallback={<div>Loading reviews...</div>}>
460
+ <ProductReviews productId={params.id} />
461
+ </Suspense>
462
+
463
+ <Suspense fallback={<div>Loading recommendations...</div>}>
464
+ <ProductRecommendations productId={params.id} />
465
+ </Suspense>
466
+ </div>
467
+ );
468
+ }
469
+ ```
470
+
471
+ ## SEO and Metadata | SEO 和元数据
472
+
473
+ ### 1. Metadata API | 元数据 API
474
+
475
+ ```tsx
476
+ import type { Metadata } from 'next';
477
+
478
+ // ✅ Good: Static metadata
479
+ export const metadata: Metadata = {
480
+ title: 'My App',
481
+ description: 'The best app ever built',
482
+ keywords: ['nextjs', 'react', 'typescript'],
483
+ authors: [{ name: 'John Doe' }],
484
+ openGraph: {
485
+ title: 'My App',
486
+ description: 'The best app ever built',
487
+ url: 'https://myapp.com',
488
+ siteName: 'My App',
489
+ images: [
490
+ {
491
+ url: 'https://myapp.com/og-image.jpg',
492
+ width: 1200,
493
+ height: 630,
494
+ },
495
+ ],
496
+ locale: 'en_US',
497
+ type: 'website',
498
+ },
499
+ twitter: {
500
+ card: 'summary_large_image',
501
+ title: 'My App',
502
+ description: 'The best app ever built',
503
+ images: ['https://myapp.com/twitter-image.jpg'],
504
+ },
505
+ };
506
+
507
+ // ✅ Good: Dynamic metadata
508
+ export async function generateMetadata(
509
+ { params }: { params: { id: string } }
510
+ ): Promise<Metadata> {
511
+ const product = await getProduct(params.id);
512
+
513
+ return {
514
+ title: product.name,
515
+ description: product.description,
516
+ openGraph: {
517
+ title: product.name,
518
+ description: product.description,
519
+ images: [product.image],
520
+ },
521
+ };
522
+ }
523
+ ```
524
+
525
+ ### 2. Structured Data | 结构化数据
526
+
527
+ ```tsx
528
+ // ✅ Good: JSON-LD structured data
529
+ function ProductPage({ product }: { product: Product }) {
530
+ const jsonLd = {
531
+ '@context': 'https://schema.org',
532
+ '@type': 'Product',
533
+ name: product.name,
534
+ description: product.description,
535
+ image: product.images,
536
+ offers: {
537
+ '@type': 'Offer',
538
+ price: product.price,
539
+ priceCurrency: 'USD',
540
+ availability: 'https://schema.org/InStock',
541
+ },
542
+ aggregateRating: {
543
+ '@type': 'AggregateRating',
544
+ ratingValue: product.rating,
545
+ reviewCount: product.reviewCount,
546
+ },
547
+ };
548
+
549
+ return (
550
+ <>
551
+ <script
552
+ type="application/ld+json"
553
+ dangerouslySetInnerHTML={{ __html: JSON.stringify(jsonLd) }}
554
+ />
555
+ <div>
556
+ {/* Product content */}
557
+ </div>
558
+ </>
559
+ );
560
+ }
561
+ ```
562
+
563
+ ## Anti-Patterns | 反模式
564
+
565
+ ### 1. Avoid Client Components for Static Content | 避免在静态内容中使用客户端组件
566
+
567
+ ```tsx
568
+ // ❌ Bad: Unnecessary client component
569
+ 'use client';
570
+
571
+ function BlogPost({ post }: { post: Post }) {
572
+ return (
573
+ <article>
574
+ <h1>{post.title}</h1>
575
+ <p>{post.content}</p>
576
+ </article>
577
+ );
578
+ }
579
+
580
+ // ✅ Good: Server component for static content
581
+ function BlogPost({ post }: { post: Post }) {
582
+ return (
583
+ <article>
584
+ <h1>{post.title}</h1>
585
+ <p>{post.content}</p>
586
+ </article>
587
+ );
588
+ }
589
+ ```
590
+
591
+ ### 2. Avoid Blocking Data Fetching | 避免阻塞数据获取
592
+
593
+ ```tsx
594
+ // ❌ Bad: Sequential data fetching
595
+ async function BadProductPage({ params }: { params: { id: string } }) {
596
+ const product = await getProduct(params.id);
597
+ const reviews = await getProductReviews(params.id); // Waits for product
598
+ const related = await getRelatedProducts(params.id); // Waits for reviews
599
+
600
+ return (
601
+ <div>
602
+ <ProductDetails product={product} />
603
+ <ProductReviews reviews={reviews} />
604
+ <RelatedProducts products={related} />
605
+ </div>
606
+ );
607
+ }
608
+
609
+ // ✅ Good: Parallel data fetching
610
+ async function GoodProductPage({ params }: { params: { id: string } }) {
611
+ const [product, reviews, related] = await Promise.all([
612
+ getProduct(params.id),
613
+ getProductReviews(params.id),
614
+ getRelatedProducts(params.id)
615
+ ]);
616
+
617
+ return (
618
+ <div>
619
+ <ProductDetails product={product} />
620
+ <ProductReviews reviews={reviews} />
621
+ <RelatedProducts products={related} />
622
+ </div>
623
+ );
624
+ }
625
+ ```
626
+
627
+ ### 3. Avoid Overusing Client Components | 避免过度使用客户端组件
628
+
629
+ ```tsx
630
+ // ❌ Bad: Everything as client component
631
+ 'use client';
632
+
633
+ function Layout({ children }: { children: React.ReactNode }) {
634
+ return (
635
+ <div>
636
+ <Header />
637
+ <main>{children}</main>
638
+ <Footer />
639
+ </div>
640
+ );
641
+ }
642
+
643
+ // ✅ Good: Mix server and client components
644
+ function Layout({ children }: { children: React.ReactNode }) {
645
+ return (
646
+ <div>
647
+ <Header /> {/* Server component */}
648
+ <main>{children}</main>
649
+ <Footer /> {/* Server component */}
650
+ </div>
651
+ );
652
+ }
653
+
654
+ // Only interactive parts are client components
655
+ 'use client';
656
+ function InteractiveHeader() {
657
+ const [isOpen, setIsOpen] = useState(false);
658
+
659
+ return (
660
+ <nav>
661
+ <button onClick={() => setIsOpen(!isOpen)}>
662
+ Menu
663
+ </button>
664
+ {isOpen && <MobileMenu />}
665
+ </nav>
666
+ );
667
+ }
668
+ ```
669
+
670
+ ## Performance Checklist | 性能检查清单
671
+
672
+ - [ ] Server components are used for static content
673
+ - [ ] Client components are minimal and focused
674
+ - [ ] Data fetching is parallelized where possible
675
+ - [ ] Images use Next.js Image component with optimization
676
+ - [ ] Code splitting is implemented for large components
677
+ - [ ] Metadata is properly configured for SEO
678
+ - [ ] Caching strategies are implemented
679
+ - [ ] Bundle size is monitored and optimized
680
+ - [ ] Streaming and Suspense are used for better UX
681
+ - [ ] Core Web Vitals are measured and optimized
682
+
683
+ ## 性能检查清单
684
+
685
+ - [ ] 静态内容使用服务器组件
686
+ - [ ] 客户端组件最小化且专注
687
+ - [ ] 在可能的地方并行化数据获取
688
+ - [ ] 图像使用 Next.js Image 组件进行优化
689
+ - [ ] 为大型组件实现代码分割
690
+ - [ ] 为 SEO 正确配置元数据
691
+ - [ ] 实现缓存策略
692
+ - [ ] 监控和优化包大小
693
+ - [ ] 使用流式传输和 Suspense 改善用户体验
694
+ - [ ] 测量和优化核心 Web 指标