@qlover/create-app 0.7.14 → 0.7.15
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.
- package/CHANGELOG.md +23 -0
- package/dist/index.cjs +1 -1
- package/dist/index.js +1 -1
- package/dist/templates/next-app/README.en.md +131 -0
- package/dist/templates/next-app/README.md +115 -20
- package/dist/templates/next-app/docs/en/api.md +387 -0
- package/dist/templates/next-app/docs/en/component.md +544 -0
- package/dist/templates/next-app/docs/en/database.md +496 -0
- package/dist/templates/next-app/docs/en/development-guide.md +727 -0
- package/dist/templates/next-app/docs/en/env.md +563 -0
- package/dist/templates/next-app/docs/en/i18n.md +287 -0
- package/dist/templates/next-app/docs/en/index.md +166 -0
- package/dist/templates/next-app/docs/en/page.md +457 -0
- package/dist/templates/next-app/docs/en/project-structure.md +177 -0
- package/dist/templates/next-app/docs/en/router.md +427 -0
- package/dist/templates/next-app/docs/en/theme.md +532 -0
- package/dist/templates/next-app/docs/en/validator.md +478 -0
- package/dist/templates/next-app/docs/zh/api.md +387 -0
- package/dist/templates/next-app/docs/zh/component.md +544 -0
- package/dist/templates/next-app/docs/zh/database.md +496 -0
- package/dist/templates/next-app/docs/zh/development-guide.md +727 -0
- package/dist/templates/next-app/docs/zh/env.md +563 -0
- package/dist/templates/next-app/docs/zh/i18n.md +287 -0
- package/dist/templates/next-app/docs/zh/index.md +166 -0
- package/dist/templates/next-app/docs/zh/page.md +457 -0
- package/dist/templates/next-app/docs/zh/project-structure.md +177 -0
- package/dist/templates/next-app/docs/zh/router.md +427 -0
- package/dist/templates/next-app/docs/zh/theme.md +532 -0
- package/dist/templates/next-app/docs/zh/validator.md +476 -0
- package/package.json +1 -1
- package/dist/templates/next-app/docs/env.md +0 -94
|
@@ -0,0 +1,457 @@
|
|
|
1
|
+
# Next.js 页面开发指南
|
|
2
|
+
|
|
3
|
+
## 目录
|
|
4
|
+
|
|
5
|
+
1. [页面开发概述](#页面开发概述)
|
|
6
|
+
2. [服务端组件详解](#服务端组件详解)
|
|
7
|
+
3. [客户端组件详解](#客户端组件详解)
|
|
8
|
+
4. [数据获取和状态管理](#数据获取和状态管理)
|
|
9
|
+
5. [页面路由和元数据](#页面路由和元数据)
|
|
10
|
+
6. [最佳实践和性能优化](#最佳实践和性能优化)
|
|
11
|
+
|
|
12
|
+
## 页面开发概述
|
|
13
|
+
|
|
14
|
+
### 1. 服务端组件 vs 客户端组件
|
|
15
|
+
|
|
16
|
+
在 Next.js 13+ 中,所有组件默认都是服务端组件(Server Components),除非显式声明为客户端组件。
|
|
17
|
+
|
|
18
|
+
```typescript
|
|
19
|
+
// 服务端组件(默认)
|
|
20
|
+
// page.tsx
|
|
21
|
+
export default function Page() {
|
|
22
|
+
return <div>服务端渲染的内容</div>;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
// 客户端组件
|
|
26
|
+
// 需要在文件顶部添加 'use client' 指令
|
|
27
|
+
'use client';
|
|
28
|
+
|
|
29
|
+
export function ClientComponent() {
|
|
30
|
+
return <div>客户端渲染的内容</div>;
|
|
31
|
+
}
|
|
32
|
+
```
|
|
33
|
+
|
|
34
|
+
### 2. 选择指南
|
|
35
|
+
|
|
36
|
+
**使用服务端组件的场景**:
|
|
37
|
+
|
|
38
|
+
- 需要直接访问后端资源
|
|
39
|
+
- 包含敏感信息(API 密钥、tokens 等)
|
|
40
|
+
- 依赖后端的重量级操作
|
|
41
|
+
- 需要减少客户端 JavaScript 体积
|
|
42
|
+
- 需要 SEO 优化
|
|
43
|
+
- 页面级组件
|
|
44
|
+
- 不需要客户端交互
|
|
45
|
+
- 不需要浏览器 API
|
|
46
|
+
|
|
47
|
+
**使用客户端组件的场景**:
|
|
48
|
+
|
|
49
|
+
- 需要添加交互和事件处理
|
|
50
|
+
- 使用浏览器 API
|
|
51
|
+
- 使用 React hooks
|
|
52
|
+
- 需要维护组件状态
|
|
53
|
+
- 需要使用客户端特有的生命周期
|
|
54
|
+
- 使用依赖浏览器 API 的第三方库
|
|
55
|
+
|
|
56
|
+
## 服务端组件详解
|
|
57
|
+
|
|
58
|
+
### 1. 基本结构
|
|
59
|
+
|
|
60
|
+
```typescript
|
|
61
|
+
// app/[locale]/login/page.tsx
|
|
62
|
+
export default async function LoginPage(props: PageParamsProps) {
|
|
63
|
+
// 1. 参数验证
|
|
64
|
+
if (!props.params) {
|
|
65
|
+
return notFound();
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
// 2. 服务端初始化
|
|
69
|
+
const params = await props.params;
|
|
70
|
+
const pageParams = new PageParams(params);
|
|
71
|
+
const server = new BootstrapServer();
|
|
72
|
+
|
|
73
|
+
// 3. 服务端数据获取和验证
|
|
74
|
+
if (await server.getIOC(ServerAuth).hasAuth()) {
|
|
75
|
+
return redirect({ href: '/', locale: params.locale! });
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// 4. 获取国际化文本
|
|
79
|
+
const tt = await pageParams.getI18nInterface(loginI18n);
|
|
80
|
+
|
|
81
|
+
// 5. 渲染页面
|
|
82
|
+
return (
|
|
83
|
+
<BaseLayout>
|
|
84
|
+
<div>{/* 页面内容 */}</div>
|
|
85
|
+
<ClientComponent /> {/* 嵌入客户端组件 */}
|
|
86
|
+
</BaseLayout>
|
|
87
|
+
);
|
|
88
|
+
}
|
|
89
|
+
```
|
|
90
|
+
|
|
91
|
+
### 2. 服务端数据获取
|
|
92
|
+
|
|
93
|
+
```typescript
|
|
94
|
+
// 直接在服务端组件中获取数据
|
|
95
|
+
export default async function UsersPage() {
|
|
96
|
+
const server = new BootstrapServer();
|
|
97
|
+
|
|
98
|
+
const result = await server
|
|
99
|
+
.use(new AdminAuthPlugin()) // 使用服务端中间件
|
|
100
|
+
.execNoError(async ({ parameters: { IOC } }) => {
|
|
101
|
+
const userService = IOC(UserService);
|
|
102
|
+
return userService.getUsers();
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
// 数据直接传递给客户端组件
|
|
106
|
+
return <UserList initialData={result.data} />;
|
|
107
|
+
}
|
|
108
|
+
```
|
|
109
|
+
|
|
110
|
+
### 3. 静态生成和动态渲染
|
|
111
|
+
|
|
112
|
+
```typescript
|
|
113
|
+
// 静态页面生成
|
|
114
|
+
export const dynamic = 'force-static'; // 强制静态生成
|
|
115
|
+
export const revalidate = 3600; // 每小时重新生成
|
|
116
|
+
|
|
117
|
+
// 动态页面生成
|
|
118
|
+
export const dynamic = 'force-dynamic'; // 强制动态生成
|
|
119
|
+
|
|
120
|
+
// 生成静态路由参数
|
|
121
|
+
export async function generateStaticParams() {
|
|
122
|
+
return [{ locale: 'en' }, { locale: 'zh' }];
|
|
123
|
+
}
|
|
124
|
+
```
|
|
125
|
+
|
|
126
|
+
## 客户端组件详解
|
|
127
|
+
|
|
128
|
+
### 1. 基本结构
|
|
129
|
+
|
|
130
|
+
```typescript
|
|
131
|
+
'use client'; // 声明为客户端组件
|
|
132
|
+
|
|
133
|
+
export function LoginForm(props: { tt: LoginI18nInterface }) {
|
|
134
|
+
// 1. Hooks 使用
|
|
135
|
+
const [loading, setLoading] = useState(false);
|
|
136
|
+
const userService = useIOC(I.UserServiceInterface);
|
|
137
|
+
|
|
138
|
+
// 2. 事件处理
|
|
139
|
+
const handleLogin = async (values: LoginFormData) => {
|
|
140
|
+
try {
|
|
141
|
+
setLoading(true);
|
|
142
|
+
await userService.login(values);
|
|
143
|
+
routerService.gotoHome();
|
|
144
|
+
} catch (error) {
|
|
145
|
+
logger.error(error);
|
|
146
|
+
} finally {
|
|
147
|
+
setLoading(false);
|
|
148
|
+
}
|
|
149
|
+
};
|
|
150
|
+
|
|
151
|
+
// 3. 渲染 UI
|
|
152
|
+
return (
|
|
153
|
+
<Form onFinish={handleLogin}>
|
|
154
|
+
{/* 表单内容 */}
|
|
155
|
+
</Form>
|
|
156
|
+
);
|
|
157
|
+
}
|
|
158
|
+
```
|
|
159
|
+
|
|
160
|
+
### 2. 客户端状态管理
|
|
161
|
+
|
|
162
|
+
```typescript
|
|
163
|
+
'use client';
|
|
164
|
+
|
|
165
|
+
export function UserProfile() {
|
|
166
|
+
// 1. 本地状态
|
|
167
|
+
const [isEditing, setIsEditing] = useState(false);
|
|
168
|
+
|
|
169
|
+
// 2. 服务状态
|
|
170
|
+
const userStore = useIOC(UserStore);
|
|
171
|
+
const user = useStore(userStore, userStore.selector.currentUser);
|
|
172
|
+
|
|
173
|
+
// 3. 表单状态
|
|
174
|
+
const [form] = Form.useForm();
|
|
175
|
+
|
|
176
|
+
// 4. 副作用
|
|
177
|
+
useEffect(() => {
|
|
178
|
+
if (user) {
|
|
179
|
+
form.setFieldsValue(user);
|
|
180
|
+
}
|
|
181
|
+
}, [user, form]);
|
|
182
|
+
|
|
183
|
+
return (
|
|
184
|
+
<div>{/* 组件内容 */}</div>
|
|
185
|
+
);
|
|
186
|
+
}
|
|
187
|
+
```
|
|
188
|
+
|
|
189
|
+
### 3. 与服务端组件交互
|
|
190
|
+
|
|
191
|
+
```typescript
|
|
192
|
+
// 1. 通过属性接收服务端数据
|
|
193
|
+
interface Props {
|
|
194
|
+
initialData: UserData; // 从服务端获取的初始数据
|
|
195
|
+
tt: I18nInterface; // 从服务端获取的翻译文本
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
'use client';
|
|
199
|
+
export function UserList({ initialData, tt }: Props) {
|
|
200
|
+
const [data, setData] = useState(initialData);
|
|
201
|
+
|
|
202
|
+
// 使用服务端数据初始化客户端状态
|
|
203
|
+
useEffect(() => {
|
|
204
|
+
setData(initialData);
|
|
205
|
+
}, [initialData]);
|
|
206
|
+
|
|
207
|
+
return (
|
|
208
|
+
<div>{/* 组件内容 */}</div>
|
|
209
|
+
);
|
|
210
|
+
}
|
|
211
|
+
```
|
|
212
|
+
|
|
213
|
+
## 数据获取和状态管理
|
|
214
|
+
|
|
215
|
+
### 1. 服务端数据获取
|
|
216
|
+
|
|
217
|
+
```typescript
|
|
218
|
+
// 1. 在服务端组件中获取数据
|
|
219
|
+
export default async function DashboardPage() {
|
|
220
|
+
const server = new BootstrapServer();
|
|
221
|
+
|
|
222
|
+
// 并行数据获取
|
|
223
|
+
const [userData, statsData] = await Promise.all([
|
|
224
|
+
server.execNoError(async ({ IOC }) => IOC(UserService).getCurrentUser()),
|
|
225
|
+
server.execNoError(async ({ IOC }) => IOC(StatsService).getStats())
|
|
226
|
+
]);
|
|
227
|
+
|
|
228
|
+
return (
|
|
229
|
+
<Dashboard
|
|
230
|
+
userData={userData}
|
|
231
|
+
statsData={statsData}
|
|
232
|
+
/>
|
|
233
|
+
);
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
// 2. 使用服务端 action
|
|
237
|
+
export async function fetchUserData(userId: string) {
|
|
238
|
+
'use server'; // 声明为服务端 action
|
|
239
|
+
|
|
240
|
+
const server = new BootstrapServer();
|
|
241
|
+
return server.execNoError(async ({ IOC }) => {
|
|
242
|
+
return IOC(UserService).getUserById(userId);
|
|
243
|
+
});
|
|
244
|
+
}
|
|
245
|
+
```
|
|
246
|
+
|
|
247
|
+
### 2. 客户端状态管理
|
|
248
|
+
|
|
249
|
+
```typescript
|
|
250
|
+
'use client';
|
|
251
|
+
|
|
252
|
+
// 1. 使用 Store 管理状态
|
|
253
|
+
@injectable()
|
|
254
|
+
export class UserProfileStore extends StoreInterface<UserProfileState> {
|
|
255
|
+
constructor() {
|
|
256
|
+
super(() => ({
|
|
257
|
+
user: null,
|
|
258
|
+
loading: false,
|
|
259
|
+
error: null
|
|
260
|
+
}));
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
async fetchUser(id: string) {
|
|
264
|
+
this.emit({ ...this.state, loading: true });
|
|
265
|
+
try {
|
|
266
|
+
const user = await this.userService.getUser(id);
|
|
267
|
+
this.emit({ ...this.state, user, loading: false });
|
|
268
|
+
} catch (error) {
|
|
269
|
+
this.emit({ ...this.state, error, loading: false });
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
// 2. 在组件中使用 Store
|
|
275
|
+
export function UserProfile() {
|
|
276
|
+
const store = useIOC(UserProfileStore);
|
|
277
|
+
const user = useStore(store, store.selector.user);
|
|
278
|
+
const loading = useStore(store, store.selector.loading);
|
|
279
|
+
|
|
280
|
+
return (
|
|
281
|
+
<div>
|
|
282
|
+
{loading ? <Loading /> : <UserInfo user={user} />}
|
|
283
|
+
</div>
|
|
284
|
+
);
|
|
285
|
+
}
|
|
286
|
+
```
|
|
287
|
+
|
|
288
|
+
## 页面路由和元数据
|
|
289
|
+
|
|
290
|
+
### 1. 动态路由
|
|
291
|
+
|
|
292
|
+
```typescript
|
|
293
|
+
// app/[locale]/users/[id]/page.tsx
|
|
294
|
+
export default async function UserPage({
|
|
295
|
+
params: { locale, id }
|
|
296
|
+
}: {
|
|
297
|
+
params: { locale: string; id: string };
|
|
298
|
+
}) {
|
|
299
|
+
// 处理路由参数
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
// 生成静态路由
|
|
303
|
+
export async function generateStaticParams() {
|
|
304
|
+
const users = await fetchUsers();
|
|
305
|
+
|
|
306
|
+
return users.map((user) => ({
|
|
307
|
+
locale: ['en', 'zh'],
|
|
308
|
+
id: user.id
|
|
309
|
+
}));
|
|
310
|
+
}
|
|
311
|
+
```
|
|
312
|
+
|
|
313
|
+
### 2. 元数据生成
|
|
314
|
+
|
|
315
|
+
```typescript
|
|
316
|
+
// 1. 静态元数据
|
|
317
|
+
export const metadata: Metadata = {
|
|
318
|
+
title: 'User Profile',
|
|
319
|
+
description: 'User profile page'
|
|
320
|
+
};
|
|
321
|
+
|
|
322
|
+
// 2. 动态元数据
|
|
323
|
+
export async function generateMetadata({
|
|
324
|
+
params
|
|
325
|
+
}: {
|
|
326
|
+
params: PageParamsType;
|
|
327
|
+
}): Promise<Metadata> {
|
|
328
|
+
const pageParams = new PageParams(await params);
|
|
329
|
+
const tt = await pageParams.getI18nInterface(userI18n);
|
|
330
|
+
|
|
331
|
+
return {
|
|
332
|
+
title: tt.pageTitle,
|
|
333
|
+
description: tt.pageDescription
|
|
334
|
+
};
|
|
335
|
+
}
|
|
336
|
+
```
|
|
337
|
+
|
|
338
|
+
## 最佳实践和性能优化
|
|
339
|
+
|
|
340
|
+
### 1. 组件分割原则
|
|
341
|
+
|
|
342
|
+
```typescript
|
|
343
|
+
// ❌ 错误:在一个组件中混合服务端和客户端逻辑
|
|
344
|
+
export default function Page() {
|
|
345
|
+
const [state, setState] = useState(); // 错误:服务端组件不能使用 hooks
|
|
346
|
+
const data = await fetchData(); // 正确:服务端数据获取
|
|
347
|
+
|
|
348
|
+
return <div>{/* ... */}</div>;
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
// ✅ 正确:分离服务端和客户端逻辑
|
|
352
|
+
// page.tsx (服务端组件)
|
|
353
|
+
export default async function Page() {
|
|
354
|
+
const data = await fetchData(); // 服务端数据获取
|
|
355
|
+
return <ClientComponent data={data} />;
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
// ClientComponent.tsx (客户端组件)
|
|
359
|
+
'use client';
|
|
360
|
+
export function ClientComponent({ data }) {
|
|
361
|
+
const [state, setState] = useState(); // 客户端状态管理
|
|
362
|
+
return <div>{/* ... */}</div>;
|
|
363
|
+
}
|
|
364
|
+
```
|
|
365
|
+
|
|
366
|
+
### 2. 性能优化策略
|
|
367
|
+
|
|
368
|
+
```typescript
|
|
369
|
+
// 1. 组件级缓存
|
|
370
|
+
export default async function UserList() {
|
|
371
|
+
const users = await fetchUsers();
|
|
372
|
+
|
|
373
|
+
return (
|
|
374
|
+
<div>
|
|
375
|
+
{users.map(user => (
|
|
376
|
+
<Suspense key={user.id} fallback={<Loading />}>
|
|
377
|
+
<UserCard user={user} />
|
|
378
|
+
</Suspense>
|
|
379
|
+
))}
|
|
380
|
+
</div>
|
|
381
|
+
);
|
|
382
|
+
}
|
|
383
|
+
|
|
384
|
+
// 2. 路由级缓存
|
|
385
|
+
export const revalidate = 3600; // 缓存一小时
|
|
386
|
+
|
|
387
|
+
// 3. 选择性水合
|
|
388
|
+
export default function Page() {
|
|
389
|
+
return (
|
|
390
|
+
<>
|
|
391
|
+
<StaticContent />
|
|
392
|
+
<Suspense fallback={<Loading />}>
|
|
393
|
+
<DynamicContent />
|
|
394
|
+
</Suspense>
|
|
395
|
+
</>
|
|
396
|
+
);
|
|
397
|
+
}
|
|
398
|
+
```
|
|
399
|
+
|
|
400
|
+
### 3. 错误处理
|
|
401
|
+
|
|
402
|
+
```typescript
|
|
403
|
+
// 1. 错误边界
|
|
404
|
+
'use client';
|
|
405
|
+
export function ErrorBoundary({
|
|
406
|
+
error,
|
|
407
|
+
reset
|
|
408
|
+
}: {
|
|
409
|
+
error: Error;
|
|
410
|
+
reset: () => void;
|
|
411
|
+
}) {
|
|
412
|
+
return (
|
|
413
|
+
<div>
|
|
414
|
+
<h2>出错了!</h2>
|
|
415
|
+
<button onClick={reset}>重试</button>
|
|
416
|
+
</div>
|
|
417
|
+
);
|
|
418
|
+
}
|
|
419
|
+
|
|
420
|
+
// 2. 加载状态
|
|
421
|
+
export default async function Page() {
|
|
422
|
+
return (
|
|
423
|
+
<Suspense fallback={<Loading />}>
|
|
424
|
+
<SlowComponent />
|
|
425
|
+
</Suspense>
|
|
426
|
+
);
|
|
427
|
+
}
|
|
428
|
+
```
|
|
429
|
+
|
|
430
|
+
## 总结
|
|
431
|
+
|
|
432
|
+
Next.js 13+ 的页面开发需要注意以下几点:
|
|
433
|
+
|
|
434
|
+
1. **组件类型选择**:
|
|
435
|
+
- 默认使用服务端组件
|
|
436
|
+
- 只在必要时使用客户端组件
|
|
437
|
+
- 合理分割组件职责
|
|
438
|
+
|
|
439
|
+
2. **数据流处理**:
|
|
440
|
+
- 服务端优先获取数据
|
|
441
|
+
- 通过属性传递给客户端
|
|
442
|
+
- 使用 Store 管理客户端状态
|
|
443
|
+
|
|
444
|
+
3. **性能优化**:
|
|
445
|
+
- 合理使用缓存策略
|
|
446
|
+
- 实现选择性水合
|
|
447
|
+
- 优化加载性能
|
|
448
|
+
|
|
449
|
+
4. **开发体验**:
|
|
450
|
+
- 清晰的代码组织
|
|
451
|
+
- 类型安全
|
|
452
|
+
- 完善的错误处理
|
|
453
|
+
|
|
454
|
+
5. **最佳实践**:
|
|
455
|
+
- 遵循单一职责原则
|
|
456
|
+
- 实现优雅降级
|
|
457
|
+
- 保持代码可维护性
|
|
@@ -0,0 +1,177 @@
|
|
|
1
|
+
# 项目结构详细说明
|
|
2
|
+
|
|
3
|
+
## 目录结构概览
|
|
4
|
+
|
|
5
|
+
```
|
|
6
|
+
next-app/
|
|
7
|
+
├── config/ # 全局配置文件
|
|
8
|
+
├── src/ # 源代码目录
|
|
9
|
+
│ ├── app/ # Next.js 应用路由
|
|
10
|
+
│ ├── base/ # 客户端核心代码
|
|
11
|
+
│ ├── core/ # 核心启动和IOC配置
|
|
12
|
+
│ ├── server/ # 服务端代码
|
|
13
|
+
│ ├── styles/ # 样式文件
|
|
14
|
+
│ └── uikit/ # UI组件库
|
|
15
|
+
└── public/ # 静态资源
|
|
16
|
+
```
|
|
17
|
+
|
|
18
|
+
## 详细结构说明
|
|
19
|
+
|
|
20
|
+
### 1. 客户端架构 (src/base)
|
|
21
|
+
|
|
22
|
+
#### 1.1 接口层 (src/base/port)
|
|
23
|
+
|
|
24
|
+
- `AdminLayoutInterface.ts` - 管理布局接口
|
|
25
|
+
- `AdminPageInterface.ts` - 管理页面接口
|
|
26
|
+
- `AppApiInterface.ts` - 应用API接口
|
|
27
|
+
- `AppUserApiInterface.ts` - 用户API接口
|
|
28
|
+
- `AsyncStateInterface.ts` - 异步状态接口
|
|
29
|
+
- `I18nServiceInterface.ts` - 国际化服务接口
|
|
30
|
+
- `IOCInterface.ts` - 依赖注入接口
|
|
31
|
+
- `RouterInterface.ts` - 路由接口
|
|
32
|
+
- `UserServiceInterface.ts` - 用户服务接口
|
|
33
|
+
|
|
34
|
+
#### 1.2 业务逻辑层 (src/base/cases)
|
|
35
|
+
|
|
36
|
+
- 状态管理控制器
|
|
37
|
+
- 对话框处理
|
|
38
|
+
- 路由服务
|
|
39
|
+
- 用户服务
|
|
40
|
+
- 加密服务
|
|
41
|
+
|
|
42
|
+
#### 1.3 服务实现层 (src/base/services)
|
|
43
|
+
|
|
44
|
+
- `AdminUserService.ts` - 管理用户服务
|
|
45
|
+
- `I18nService.ts` - 国际化服务
|
|
46
|
+
- `UserService.ts` - 用户服务
|
|
47
|
+
- `adminApi/` - 管理API实现
|
|
48
|
+
- `appApi/` - 应用API实现
|
|
49
|
+
|
|
50
|
+
### 2. 服务端架构 (src/server)
|
|
51
|
+
|
|
52
|
+
#### 2.1 接口层 (src/server/port)
|
|
53
|
+
|
|
54
|
+
- `CrentialTokenInterface.ts` - 凭证令牌接口
|
|
55
|
+
- `DBBridgeInterface.ts` - 数据库桥接接口
|
|
56
|
+
- `DBTableInterface.ts` - 数据表接口
|
|
57
|
+
- `PaginationInterface.ts` - 分页接口
|
|
58
|
+
- `ParamsHandlerInterface.ts` - 参数处理接口
|
|
59
|
+
- `ServerAuthInterface.ts` - 服务认证接口
|
|
60
|
+
- `ServerInterface.ts` - 服务器接口
|
|
61
|
+
- `UserRepositoryInterface.ts` - 用户仓库接口
|
|
62
|
+
- `ValidatorInterface.ts` - 验证器接口
|
|
63
|
+
|
|
64
|
+
#### 2.2 核心实现
|
|
65
|
+
|
|
66
|
+
- `AppErrorApi.ts` - API错误处理
|
|
67
|
+
- `AppSuccessApi.ts` - API成功响应
|
|
68
|
+
- `ServerAuth.ts` - 服务认证实现
|
|
69
|
+
- `UserCredentialToken.ts` - 用户凭证令牌
|
|
70
|
+
|
|
71
|
+
#### 2.3 服务层 (src/server/services)
|
|
72
|
+
|
|
73
|
+
- `AdminAuthPlugin.ts` - 管理认证插件
|
|
74
|
+
- `AIService.ts` - AI服务
|
|
75
|
+
- `ApiUserService.ts` - API用户服务
|
|
76
|
+
- `UserService.ts` - 用户服务
|
|
77
|
+
|
|
78
|
+
#### 2.4 数据访问层
|
|
79
|
+
|
|
80
|
+
- `repositorys/UserRepository.ts` - 用户数据仓库
|
|
81
|
+
- `sqlBridges/SupabaseBridge.ts` - Supabase数据库桥接
|
|
82
|
+
|
|
83
|
+
#### 2.5 验证器 (src/server/validators)
|
|
84
|
+
|
|
85
|
+
- `LoginValidator.ts` - 登录验证
|
|
86
|
+
- `PaginationValidator.ts` - 分页验证
|
|
87
|
+
|
|
88
|
+
### 3. API路由结构 (src/app/api)
|
|
89
|
+
|
|
90
|
+
```
|
|
91
|
+
api/
|
|
92
|
+
├── admin/ # 管理接口
|
|
93
|
+
│ └── users/ # 用户管理
|
|
94
|
+
├── ai/ # AI相关接口
|
|
95
|
+
│ └── completions/ # AI补全
|
|
96
|
+
└── user/ # 用户接口
|
|
97
|
+
├── login/ # 登录
|
|
98
|
+
├── logout/ # 登出
|
|
99
|
+
└── register/ # 注册
|
|
100
|
+
```
|
|
101
|
+
|
|
102
|
+
### 4. 页面路由结构 (src/app/[locale])
|
|
103
|
+
|
|
104
|
+
```
|
|
105
|
+
[locale]/
|
|
106
|
+
├── admin/ # 管理页面
|
|
107
|
+
│ └── users/ # 用户管理
|
|
108
|
+
├── login/ # 登录页面
|
|
109
|
+
├── register/ # 注册页面
|
|
110
|
+
└── page.tsx # 首页
|
|
111
|
+
```
|
|
112
|
+
|
|
113
|
+
### 5. UI组件库 (src/uikit)
|
|
114
|
+
|
|
115
|
+
#### 5.1 组件 (components)
|
|
116
|
+
|
|
117
|
+
- `AdminLayout.tsx` - 管理布局
|
|
118
|
+
- `BaseHeader.tsx` - 基础头部
|
|
119
|
+
- `BaseLayout.tsx` - 基础布局
|
|
120
|
+
- `ChatRoot.tsx` - 聊天根组件
|
|
121
|
+
- `LanguageSwitcher.tsx` - 语言切换
|
|
122
|
+
- `ThemeSwitcher.tsx` - 主题切换
|
|
123
|
+
- 其他通用组件
|
|
124
|
+
|
|
125
|
+
#### 5.2 Hooks和Context
|
|
126
|
+
|
|
127
|
+
- `useI18nInterface.ts` - 国际化Hook
|
|
128
|
+
- `useIOC.ts` - IOC容器Hook
|
|
129
|
+
- `useStore.ts` - 状态管理Hook
|
|
130
|
+
- `IOCContext.ts` - IOC上下文
|
|
131
|
+
|
|
132
|
+
## 技术特点
|
|
133
|
+
|
|
134
|
+
1. **依赖注入模式**
|
|
135
|
+
- 使用 IOC 容器管理依赖
|
|
136
|
+
- 客户端和服务端分别维护独立的 IOC 容器
|
|
137
|
+
- 通过接口进行解耦
|
|
138
|
+
|
|
139
|
+
2. **分层架构**
|
|
140
|
+
- 清晰的接口定义
|
|
141
|
+
- 职责分明的实现层
|
|
142
|
+
- 松耦合的模块设计
|
|
143
|
+
|
|
144
|
+
3. **状态管理**
|
|
145
|
+
- 基于控制器的状态管理
|
|
146
|
+
- 响应式数据流
|
|
147
|
+
- 可预测的状态变更
|
|
148
|
+
|
|
149
|
+
4. **国际化支持**
|
|
150
|
+
- 基于 next-intl 的多语言支持
|
|
151
|
+
- 动态语言切换
|
|
152
|
+
- 路由级别的语言隔离
|
|
153
|
+
|
|
154
|
+
5. **主题系统**
|
|
155
|
+
- 基于 Tailwind CSS
|
|
156
|
+
- 暗黑模式支持
|
|
157
|
+
- 可配置的主题变量
|
|
158
|
+
|
|
159
|
+
## 开发流程
|
|
160
|
+
|
|
161
|
+
1. **API开发**
|
|
162
|
+
- 定义接口 (port)
|
|
163
|
+
- 实现验证器 (validators)
|
|
164
|
+
- 实现服务 (services)
|
|
165
|
+
- 创建API路由 (api)
|
|
166
|
+
|
|
167
|
+
2. **页面开发**
|
|
168
|
+
- 创建页面组件
|
|
169
|
+
- 实现控制器
|
|
170
|
+
- 注册依赖
|
|
171
|
+
- 添加路由
|
|
172
|
+
|
|
173
|
+
3. **组件开发**
|
|
174
|
+
- 创建UI组件
|
|
175
|
+
- 实现业务逻辑
|
|
176
|
+
- 注入依赖
|
|
177
|
+
- 添加样式
|