@coffic/cosy-ui 0.9.3 → 0.9.5

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 (39) hide show
  1. package/dist/app.css +1 -1
  2. package/dist/index-astro.ts +3 -19
  3. package/dist/index-collection.ts +1 -105
  4. package/dist/src-astro/collection/entities/BaseDoc.ts +28 -0
  5. package/dist/src-astro/collection/entities/BlogDoc.ts +221 -0
  6. package/dist/src-astro/collection/entities/CourseDoc.ts +254 -0
  7. package/dist/src-astro/collection/entities/ExperimentDoc.ts +169 -0
  8. package/dist/src-astro/collection/entities/LessonDoc.ts +203 -0
  9. package/dist/src-astro/collection/entities/MetaDoc.ts +115 -0
  10. package/dist/src-astro/{entities → collection/entities}/SidebarItem.ts +17 -17
  11. package/dist/src-astro/collection/entities/Tag.ts +42 -0
  12. package/dist/src-astro/collection/index.ts +11 -0
  13. package/dist/src-astro/collection/repos/BaseRepo.ts +276 -0
  14. package/dist/src-astro/collection/repos/BlogRepo.ts +226 -0
  15. package/dist/src-astro/collection/repos/CourseRepo.ts +137 -0
  16. package/dist/src-astro/collection/repos/ExperimentRepo.ts +121 -0
  17. package/dist/src-astro/collection/repos/LessonRepo.ts +128 -0
  18. package/dist/src-astro/collection/repos/MetaRepo.ts +89 -0
  19. package/dist/src-astro/empty-state/EmptyState.astro +100 -0
  20. package/dist/src-astro/empty-state/index.ts +2 -0
  21. package/dist/src-astro/empty-state/types.ts +70 -0
  22. package/dist/src-astro/layout-dashboard/DashboardLayout.astro +2 -1
  23. package/dist/src-astro/layout-dashboard/DashboardTopNavbar.astro +2 -26
  24. package/package.json +1 -1
  25. package/dist/src-astro/database/BaseDB.ts +0 -264
  26. package/dist/src-astro/database/BlogDB.ts +0 -198
  27. package/dist/src-astro/database/CourseDB.ts +0 -90
  28. package/dist/src-astro/database/ExperimentDB.ts +0 -106
  29. package/dist/src-astro/database/LessonDB.ts +0 -106
  30. package/dist/src-astro/database/MetaDB.ts +0 -74
  31. package/dist/src-astro/database/index.ts +0 -3
  32. package/dist/src-astro/entities/BaseDoc.ts +0 -207
  33. package/dist/src-astro/entities/BlogDoc.ts +0 -107
  34. package/dist/src-astro/entities/CourseDoc.ts +0 -102
  35. package/dist/src-astro/entities/ExperimentDoc.ts +0 -119
  36. package/dist/src-astro/entities/LessonDoc.ts +0 -153
  37. package/dist/src-astro/entities/MetaDoc.ts +0 -93
  38. package/dist/src-astro/entities/Tag.ts +0 -42
  39. /package/dist/src-astro/{entities → collection/entities}/Feature.ts +0 -0
@@ -1,264 +0,0 @@
1
- import {
2
- getCollection,
3
- getEntry,
4
- type CollectionEntry,
5
- type DataEntryMap,
6
- } from 'astro:content';
7
- import { cosyLogger, ERROR_PREFIX } from '../cosy';
8
- import type { BaseDoc } from '../../index-astro';
9
-
10
- /**
11
- * BaseDB 是所有数据库类的基类,提供了通用的文档操作功能。
12
- *
13
- * 使用方法:
14
- * ```typescript
15
- * class MyDB extends BaseDB<'collection', MyEntry, MyDoc> {
16
- * protected collectionName = 'collection' as const;
17
- * protected createDoc(entry: MyEntry): MyDoc {
18
- * return new MyDoc(entry);
19
- * }
20
- * }
21
- *
22
- * // 使用单例模式获取实例
23
- * const db = MyDB.getInstance();
24
- * const docs = await db.allTopLevelDocs();
25
- * ```
26
- *
27
- * 类型参数说明:
28
- * @template Collection - Astro content collection 的名称,必须是 DataEntryMap 的键
29
- * @template Entry - Collection 对应的条目类型
30
- * @template Doc - 文档类型,通常是自定义的文档类
31
- */
32
- export abstract class BaseDB<
33
- Collection extends keyof DataEntryMap,
34
- Entry extends CollectionEntry<Collection>,
35
- Doc extends BaseDoc<Collection, Entry>,
36
- > {
37
- /** 集合名称,必须在子类中指定 */
38
- protected abstract collectionName: Collection;
39
-
40
- /**
41
- * 创建文档实例的方法,必须在子类中实现
42
- * @param entry - 集合条目
43
- * @returns 文档实例
44
- */
45
- protected abstract createDoc(entry: Entry): Doc;
46
-
47
- /**
48
- * 获取所有文档的ID
49
- * @returns 返回所有文档的ID数组
50
- */
51
- protected async getAllIds(): Promise<string[]> {
52
- const entries = await getCollection(this.collectionName);
53
- return entries.map((entry) => entry.id);
54
- }
55
-
56
- /**
57
- * 获取集合中的所有条目
58
- * @returns 返回所有条目的数组
59
- */
60
- protected async getEntries(): Promise<Entry[]> {
61
- return await getCollection(this.collectionName);
62
- }
63
-
64
- /**
65
- * 获取指定深度的文档
66
- * 深度是指文档 ID 中斜杠的数量加1
67
- * 例如:
68
- * - "blog.md" 深度为1
69
- * - "zh-cn/blog.md" 深度为2
70
- * - "zh-cn/tech/blog.md" 深度为3
71
- *
72
- * @param depth - 文档深度
73
- * @returns 返回指定深度的文档数组
74
- */
75
- protected async getDocsByDepth(depth: number): Promise<Doc[]> {
76
- const entries = await getCollection(
77
- this.collectionName,
78
- ({ id }: { id: string }) => id.split('/').length === depth
79
- );
80
-
81
- if (entries.length === 0) {
82
- cosyLogger.warn(
83
- `[BaseDB] 没有找到深度为${depth}的文档(collection=${this.collectionName as string})`
84
- );
85
- const allEntries = await getCollection(this.collectionName);
86
- cosyLogger.array(
87
- '[BaseDB] 所有文档',
88
- allEntries.map((entry) => entry.id)
89
- );
90
- return [];
91
- }
92
-
93
- return entries.map((entry) => this.createDoc(entry as Entry));
94
- }
95
-
96
- /**
97
- * 根据ID查找单个文档
98
- * @param id - 文档ID
99
- * @param debug - 是否启用调试模式, 默认为false
100
- * @throws 如果ID不是字符串类型,则抛出错误
101
- * @throws 如果文档不存在,则返回null
102
- * @throws 如果发生其他错误,则抛出错误
103
- * @returns 返回找到的文档,如果不存在则返回null
104
- */
105
- async find(id: string, debug: boolean = false): Promise<Doc | null> {
106
- if (debug) {
107
- cosyLogger.info(`查找文档,ID: ${id}`);
108
- }
109
-
110
- if (id == undefined) {
111
- cosyLogger.error('ID is undefined');
112
- throw new Error(ERROR_PREFIX + 'ID is undefined');
113
- }
114
-
115
- if (typeof id !== 'string') {
116
- cosyLogger.error('ID must be a string, but got ' + typeof id);
117
- cosyLogger.debug(id);
118
- throw new Error(
119
- ERROR_PREFIX + 'ID must be a string, but got ' + typeof id
120
- );
121
- }
122
-
123
- // 获取所有文档的ID并排好顺序
124
- if (debug) {
125
- const allIds = (await this.getAllIds()).sort();
126
- cosyLogger.array('所有文档的ID', allIds);
127
- }
128
-
129
- // 根据ID查找文档
130
- const entry = await getEntry(this.collectionName, id);
131
- return entry ? this.createDoc(entry as Entry) : null;
132
- }
133
-
134
- /**
135
- * 获取指定文档的直接子文档(不包括更深层级的文档)
136
- * 例如对于文档 "zh-cn/blog":
137
- * - "zh-cn/blog/post1.md" 会被包含
138
- * - "zh-cn/blog/2024/post2.md" 不会被包含
139
- *
140
- * @param parentId - 父文档ID
141
- * @returns 返回子文档数组
142
- */
143
- async getChildren(parentId: string): Promise<Doc[]> {
144
- const parentLevel = parentId.split('/').length;
145
- const childrenLevel = parentLevel + 1;
146
-
147
- const entries = await getCollection(
148
- this.collectionName,
149
- ({ id }: { id: string }) =>
150
- id.startsWith(parentId) && id.split('/').length === childrenLevel
151
- );
152
-
153
- return entries.map((entry) => this.createDoc(entry as Entry));
154
- }
155
-
156
- /**
157
- * 获取指定文档的所有后代文档(包括所有层级)
158
- * 例如对于文档 "zh-cn/blog",以下都会被包含:
159
- * - "zh-cn/blog/post1.md"
160
- * - "zh-cn/blog/2024/post2.md"
161
- * - "zh-cn/blog/2024/tech/post3.md"
162
- *
163
- * @param parentId - 父文档ID
164
- * @returns 返回所有后代文档数组
165
- */
166
- async getDescendantDocs(parentId: string): Promise<Doc[]> {
167
- const entries = await getCollection(
168
- this.collectionName,
169
- ({ id }: { id: string }) => id.startsWith(parentId)
170
- );
171
- return entries.map((entry) => this.createDoc(entry as Entry));
172
- }
173
-
174
- /**
175
- * 获取指定语言和级别的文档
176
- * 通过检查文档ID是否以指定语言代码开头来筛选
177
- *
178
- * @param lang - 语言代码(如 'zh-cn', 'en')
179
- * @param level - 文档级别
180
- * @returns 返回指定语言和级别的文档数组
181
- */
182
- protected async allDocsByLangAndLevel(
183
- lang: string,
184
- level: number = 1,
185
- debug: boolean = false
186
- ): Promise<Doc[]> {
187
- const collectionName = this.collectionName as string;
188
- const docs = await this.getDocsByDepth(level);
189
-
190
- if (debug) {
191
- cosyLogger.array(
192
- `[BaseDB] 所有${level}级文档(lang=any,collection=${collectionName})`,
193
- docs
194
- );
195
- }
196
-
197
- if (docs.length === 0) {
198
- cosyLogger.warn(
199
- `[BaseDB] 没有找到${level}级文档(lang=any,collection=${collectionName})`
200
- );
201
- return [];
202
- }
203
-
204
- const filteredDocs = docs.filter((doc) => {
205
- const id = (doc as any).getId();
206
- return id && typeof id === 'string' && id.startsWith(lang);
207
- });
208
-
209
- if (debug) {
210
- cosyLogger.array(
211
- `[BaseDB] 所有${level}级文档(lang=${lang},collection=${collectionName})`,
212
- filteredDocs
213
- );
214
- }
215
-
216
- if (filteredDocs.length === 0) {
217
- cosyLogger.warn(
218
- `[BaseDB] 没有找到${level}级文档(lang=${lang},collection=${collectionName})`
219
- );
220
- cosyLogger.array(
221
- `[BaseDB] 所有${level}级文档`,
222
- docs.map((doc) => doc.getId())
223
- );
224
- return [];
225
- }
226
-
227
- return filteredDocs;
228
- }
229
-
230
- /**
231
- * 获取用于 Astro 静态路由生成的路径参数
232
- * 为每个文档生成包含语言和slug的路径参数
233
- *
234
- * @returns 返回路径参数数组,每个元素包含 lang 和 slug
235
- * @example
236
- * ```typescript
237
- * const paths = await db.getStaticPaths();
238
- * // 返回格式:
239
- * // [
240
- * // { params: { lang: 'zh-cn', slug: 'post1' } },
241
- * // { params: { lang: 'en', slug: 'post1' } }
242
- * // ]
243
- * ```
244
- */
245
- async getStaticPaths() {
246
- const debug = false;
247
- const docs = await this.getDescendantDocs('');
248
- const paths = docs.map((doc) => {
249
- const docWithMethods = doc as any;
250
- return {
251
- params: {
252
- lang: docWithMethods.getLang?.() || '',
253
- slug: docWithMethods.getSlug?.() || '',
254
- },
255
- };
256
- });
257
-
258
- if (debug) {
259
- cosyLogger.array('所有文档的路径', paths);
260
- }
261
-
262
- return paths;
263
- }
264
- }
@@ -1,198 +0,0 @@
1
- import BlogDoc from '../entities/BlogDoc';
2
- import type Tag from '../entities/Tag';
3
- import { cosyLogger } from '../cosy';
4
- import { type CollectionEntry } from 'astro:content';
5
- import { BaseDB } from './BaseDB';
6
-
7
- export const COLLECTION_BLOG = 'blogs' as const;
8
- export type BlogEntry = CollectionEntry<typeof COLLECTION_BLOG>;
9
-
10
- /**
11
- * 博客数据库类,用于管理博客内容集合。
12
- *
13
- * 目录结构:
14
- * ```
15
- * blogs/
16
- * ├── zh-cn/
17
- * │ ├── typescript-intro.md # 文章内容
18
- * │ ├── images/ # 文章相关图片
19
- * │ └── web-performance.md
20
- * └── en/
21
- * └── typescript-intro.md
22
- * └── images/
23
- * └── web-performance.md
24
- * ```
25
- */
26
- class BlogDB extends BaseDB<typeof COLLECTION_BLOG, BlogEntry, BlogDoc> {
27
- protected collectionName = COLLECTION_BLOG;
28
-
29
- protected createDoc(entry: BlogEntry): BlogDoc {
30
- return BlogDoc.fromEntry(entry);
31
- }
32
-
33
- /**
34
- * 获取所有博客
35
- *
36
- * @returns {Promise<BlogDoc[]>} 返回所有博客
37
- */
38
- async allBlogs(): Promise<BlogDoc[]> {
39
- const debug = false;
40
- const entries = await this.getDocsByDepth(2);
41
-
42
- if (debug) {
43
- cosyLogger.array('所有博客文档', entries);
44
- }
45
-
46
- return entries;
47
- }
48
-
49
- /**
50
- * 获取指定语言的所有博客
51
- *
52
- * @param lang - 语言代码
53
- * @returns {Promise<BlogDoc[]>} 返回指定语言的所有博客
54
- */
55
- async allBlogsByLang(lang: string): Promise<BlogDoc[]> {
56
- const docs = await this.allBlogs();
57
- const filteredEntries = docs.filter((doc) => doc.getId().startsWith(lang));
58
-
59
- return filteredEntries;
60
- }
61
-
62
- /**
63
- * 获取用于 Astro 静态路由生成的路径参数,专门配合 [lang]/blogs/[slug].astro 使用
64
- *
65
- * @param debug - 是否开启调试模式
66
- * @returns 返回路径参数数组
67
- */
68
- async getStaticPaths(debug: boolean = false) {
69
- const docs = await this.allBlogs();
70
-
71
- const paths = docs.map((doc) => {
72
- return {
73
- params: {
74
- lang: doc.getLang(),
75
- slug: doc.getSlug(),
76
- },
77
- };
78
- });
79
-
80
- if (debug) {
81
- cosyLogger.array('所有博客文档的路径', paths);
82
- }
83
-
84
- return paths;
85
- }
86
-
87
- /**
88
- * 获取所有博客标签
89
- * 标签会根据语言和名称去重,使用复合键 "lang:name" 确保唯一性
90
- *
91
- * @returns 返回所有标签数组
92
- * 返回格式:
93
- * [
94
- * { name: 'typescript', lang: 'zh-cn', count: 5 },
95
- * { name: 'javascript', lang: 'en', count: 3 }
96
- * ]
97
- */
98
- async getTags(): Promise<Tag[]> {
99
- const tagsMap = new Map<string, Tag>();
100
- const posts = await this.allBlogs();
101
-
102
- posts.forEach((post) => {
103
- post.getTags().forEach((tag) => {
104
- const key = `${tag.lang}:${tag.name}`;
105
- if (!tagsMap.has(key)) {
106
- tagsMap.set(key, tag);
107
- }
108
- });
109
- });
110
-
111
- return Array.from(tagsMap.values());
112
- }
113
-
114
- /**
115
- * 获取指定语言的博客标签
116
- *
117
- * @param lang - 语言代码(如 'zh-cn', 'en')
118
- * @returns 返回指定语言的标签数组
119
- */
120
- async getTagsByLang(lang: string): Promise<Tag[]> {
121
- const debug = false;
122
- const tagsMap = new Map<string, Tag>();
123
- const posts = await this.allBlogsByLang(lang);
124
-
125
- if (debug) {
126
- cosyLogger.array('posts', posts);
127
- }
128
-
129
- if (posts.length === 0) {
130
- return [];
131
- }
132
-
133
- posts.forEach((post) => {
134
- post.getTags().forEach((tag) => {
135
- const key = `${tag.lang}:${tag.name}`;
136
- if (!tagsMap.has(key)) {
137
- tagsMap.set(key, tag);
138
- }
139
- });
140
- });
141
-
142
- if (debug) {
143
- cosyLogger.array('tags', Array.from(tagsMap.values()));
144
- }
145
-
146
- return Array.from(tagsMap.values());
147
- }
148
-
149
- /**
150
- * 获取指定标签和语言的博客文章
151
- *
152
- * @param tag - 标签名称
153
- * @param lang - 语言代码
154
- * @returns 返回匹配的博客文档数组
155
- * @example
156
- * ```typescript
157
- * const posts = await blogDB.getBlogsByTag('typescript', 'zh-cn');
158
- * // 返回所有包含 'typescript' 标签的中文博客
159
- * ```
160
- */
161
- async getBlogsByTag(tag: string, lang: string): Promise<BlogDoc[]> {
162
- const posts = await this.allBlogsByLang(lang);
163
- return posts.filter((post) => post.getTags().some((t) => t.name === tag));
164
- }
165
-
166
- /**
167
- * 获取标签的静态路径参数,用于生成标签页面的路由,专门配合 [lang]/blogs/tag/[name].astro 使用
168
- *
169
- * @returns 返回所有标签的路径参数数组
170
- * 返回格式:
171
- * [
172
- * { params: { lang: 'zh-cn', name: 'typescript' } },
173
- * { params: { lang: 'en', name: 'javascript' } }
174
- * ]
175
- */
176
- async getTagsStaticPaths() {
177
- const debug = false;
178
- const tags = await this.getTags();
179
-
180
- const paths = tags.map((tag) => {
181
- return {
182
- params: {
183
- lang: tag.lang,
184
- name: tag.name,
185
- },
186
- };
187
- });
188
-
189
- if (debug) {
190
- cosyLogger.array('所有的标签路径', paths);
191
- }
192
-
193
- return paths;
194
- }
195
- }
196
-
197
- // 创建并导出单例实例
198
- export const blogDB = new BlogDB();
@@ -1,90 +0,0 @@
1
- import CourseDoc from '../entities/CourseDoc';
2
- import { getCollection, type CollectionEntry } from 'astro:content';
3
- import { BaseDB } from './BaseDB';
4
-
5
- export const COLLECTION_COURSE = 'courses' as const;
6
- export type CourseEntry = CollectionEntry<typeof COLLECTION_COURSE>;
7
-
8
- /**
9
- * 课程数据库类,用于管理课程内容集合。
10
- *
11
- * 目录结构:
12
- * ```
13
- * courses/
14
- * ├── zh-cn/ # 中文版本
15
- * │ ├── web-development/ # 课程1
16
- * │ │ ├── index.md # 课程文档
17
- * │ │ ├── chapter1
18
- * │ │ │ ├── index.md
19
- * │ │ │ ├── content.md
20
- * │ │ │ └── ...
21
- * │ │ └── chapter2
22
- * │ │ ├── index.md
23
- * │ │ ├── content.md
24
- * │ │ └── ...
25
- * │ └── mobile-dev/ # 课程2
26
- * │ ├── index.md
27
- * │ └── ...
28
- * └── en/ # 英文版本
29
- * └── ...
30
- * ```
31
- */
32
- class CourseDB extends BaseDB<
33
- typeof COLLECTION_COURSE,
34
- CourseEntry,
35
- CourseDoc
36
- > {
37
- protected collectionName = COLLECTION_COURSE;
38
-
39
- protected createDoc(entry: CourseEntry): CourseDoc {
40
- return new CourseDoc(entry);
41
- }
42
-
43
- /**
44
- * 获取指定语言的所有顶级课程
45
- *
46
- * @param lang - 语言代码
47
- * @returns 返回指定语言的顶级课程数组
48
- */
49
- async allCoursesByLang(lang: string): Promise<CourseDoc[]> {
50
- const entries = await getCollection(COLLECTION_COURSE, ({ id }) => {
51
- return id.startsWith(lang) && id.split('/').length === 2;
52
- });
53
- return entries.map((entry) => new CourseDoc(entry));
54
- }
55
-
56
- /**
57
- * 获取用于 Astro 静态路由生成的路径参数,专门配合 [lang]/courses/[...slug].astro 使用
58
- *
59
- * @returns 返回路径参数数组
60
- */
61
- async getStaticPaths(): Promise<
62
- { params: { lang: string; slug: string } }[]
63
- > {
64
- const entries = await getCollection(COLLECTION_COURSE);
65
- return entries.map((entry) => {
66
- const doc = new CourseDoc(entry);
67
- return {
68
- params: {
69
- lang: doc.getLang(),
70
- slug: doc.getSlug(),
71
- },
72
- };
73
- });
74
- }
75
-
76
- /**
77
- * 获取精选课程
78
- * 返回指定语言的前4个顶级课程文档
79
- *
80
- * @param lang - 语言代码(如 'zh-cn', 'en')
81
- * @returns 返回精选课程文档数组(最多4个)
82
- */
83
- async getFamousCourses(lang: string): Promise<CourseDoc[]> {
84
- const courses = await this.allCoursesByLang(lang);
85
- return courses.slice(0, 4);
86
- }
87
- }
88
-
89
- // 创建并导出单例实例
90
- export const courseDB = new CourseDB();
@@ -1,106 +0,0 @@
1
- import { type CollectionEntry } from 'astro:content';
2
- import { BaseDB } from './BaseDB';
3
- import ExperimentDoc from '../entities/ExperimentDoc';
4
- import { cosyLogger } from '../cosy';
5
-
6
- export const COLLECTION_EXPERIMENT = 'experiments' as const;
7
- export type ExperimentEntry = CollectionEntry<typeof COLLECTION_EXPERIMENT>;
8
-
9
- /**
10
- * 实验数据库类,用于管理实验内容集合
11
- *
12
- * 目录结构:
13
- * ```
14
- * experiments/
15
- * ├── safari_itp/ # 实验目录
16
- * │ ├── images/ # 实验图片资源
17
- * │ ├── components/ # 课程组件
18
- * │ ├── en/ # 英文版本
19
- * │ │ ├── index.mdx # 课程首页
20
- * │ │ ├── 1.mdx # 第一章
21
- * │ │ └── 2.mdx # 第二章
22
- * │ └── zh-cn/ # 中文版本
23
- * │ ├── index.mdx # 课程首页
24
- * │ ├── 1.mdx # 第一章
25
- * │ └── 2.mdx # 第二章
26
- * └── learn_astro/ # 另一个课程
27
- * ├── en/
28
- * │ ├── index.mdx
29
- * │ ├── 1.mdx
30
- * │ └── 2.mdx
31
- * └── zh-cn/
32
- * ├── index.mdx
33
- * ├── 1.mdx
34
- * └── 2.mdx
35
- * ```
36
- *
37
- * 说明:
38
- * - 每个课程(如 build_your_own_web_toolbox)是一个独立的目录
39
- * - 课程目录可以包含多语言版本(en, zh-cn 等)
40
- * - 每个语言版本包含完整的课程内容
41
- * - 课程目录可以作为 git 子模块独立管理
42
- */
43
- class ExperimentDB extends BaseDB<
44
- typeof COLLECTION_EXPERIMENT,
45
- ExperimentEntry,
46
- ExperimentDoc
47
- > {
48
- protected collectionName = COLLECTION_EXPERIMENT;
49
-
50
- protected createDoc(entry: ExperimentEntry): ExperimentDoc {
51
- return new ExperimentDoc(entry);
52
- }
53
-
54
- /**
55
- * 获取指定语言的所有课程
56
- *
57
- * @param {string} lang - 语言代码
58
- * @returns {Promise<ExperimentDoc[]>} 返回指定语言的所有课程
59
- */
60
- async allExperiments(lang: string): Promise<ExperimentDoc[]> {
61
- const docs = await this.getDocsByDepth(2);
62
- return docs.filter((doc) => doc.getId().endsWith(lang));
63
- }
64
-
65
- /**
66
- * 获取用于 Astro 静态路由生成的路径参数
67
- *
68
- * @param debug - 是否开启调试模式
69
- * @returns 返回路径参数数组
70
- */
71
- async getStaticPaths(debug: boolean = false) {
72
- const docs = await this.getEntries();
73
-
74
- if (debug) {
75
- cosyLogger.array('所有文档', docs);
76
- }
77
-
78
- const paths = docs.map((doc) => {
79
- const id = doc.id;
80
- const lang = id.split('/')[1];
81
-
82
- let slug = '';
83
- if (id.endsWith(lang)) {
84
- slug = id.replace(`${lang}`, '');
85
- } else {
86
- slug = id.replace(`${lang}/`, '');
87
- }
88
-
89
- return {
90
- params: {
91
- lang: lang,
92
- slug: slug,
93
- },
94
- };
95
- });
96
-
97
- if (debug) {
98
- cosyLogger.array('所有文档的路径', paths);
99
- }
100
-
101
- return paths;
102
- }
103
- }
104
-
105
- // 创建并导出单例实例
106
- export const experimentDB = new ExperimentDB();