@tnotesjs/core 0.1.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.

Potentially problematic release.


This version of @tnotesjs/core might be problematic. Click here for more details.

Files changed (117) hide show
  1. package/README.md +105 -0
  2. package/dist/chunk-K3X5OP3N.js +1532 -0
  3. package/dist/cli/index.d.ts +2 -0
  4. package/dist/cli/index.js +4199 -0
  5. package/dist/index.d.ts +138 -0
  6. package/dist/index.js +9 -0
  7. package/package.json +74 -0
  8. package/types/config.ts +61 -0
  9. package/types/index.ts +11 -0
  10. package/types/note.ts +33 -0
  11. package/vitepress/assets/icons/icon__check.svg +3 -0
  12. package/vitepress/assets/icons/icon__clipboard.svg +8 -0
  13. package/vitepress/assets/icons/icon__close.svg +1 -0
  14. package/vitepress/assets/icons/icon__collapse.svg +1 -0
  15. package/vitepress/assets/icons/icon__confirm.svg +1 -0
  16. package/vitepress/assets/icons/icon__copy.svg +4 -0
  17. package/vitepress/assets/icons/icon__focus.svg +1 -0
  18. package/vitepress/assets/icons/icon__fold.svg +3 -0
  19. package/vitepress/assets/icons/icon__folder.svg +1 -0
  20. package/vitepress/assets/icons/icon__fullscreen.svg +1 -0
  21. package/vitepress/assets/icons/icon__fullscreen_exit.svg +1 -0
  22. package/vitepress/assets/icons/icon__github.svg +4 -0
  23. package/vitepress/assets/icons/icon__mindmap.svg +1 -0
  24. package/vitepress/assets/icons/icon__next.svg +1 -0
  25. package/vitepress/assets/icons/icon__number_gray.svg +1 -0
  26. package/vitepress/assets/icons/icon__number_purple.svg +1 -0
  27. package/vitepress/assets/icons/icon__prev.svg +1 -0
  28. package/vitepress/assets/icons/icon__restore.svg +1 -0
  29. package/vitepress/assets/icons/icon__rotate.svg +4 -0
  30. package/vitepress/assets/icons/icon__search.svg +1 -0
  31. package/vitepress/assets/icons/icon__sidebar_collapsed.svg +1 -0
  32. package/vitepress/assets/icons/icon__sidebar_opened.svg +1 -0
  33. package/vitepress/assets/icons/icon__totop.svg +6 -0
  34. package/vitepress/assets/icons/icon__vscode.svg +6 -0
  35. package/vitepress/assets/icons/icon__zoom_fit.svg +1 -0
  36. package/vitepress/assets/icons/icon__zoom_in.svg +1 -0
  37. package/vitepress/assets/icons/icon__zoom_out.svg +1 -0
  38. package/vitepress/assets/icons/icon__zoom_reset.svg +1 -0
  39. package/vitepress/assets/icons/index.ts +38 -0
  40. package/vitepress/components/BilibiliOutsidePlayer/BilibiliOutsidePlayer.vue +20 -0
  41. package/vitepress/components/CodeBlockFullscreen/CodeBlockFullscreen.vue +373 -0
  42. package/vitepress/components/CodeBlockFullscreen/index.ts +115 -0
  43. package/vitepress/components/CodeBlockFullscreen/styles.css +64 -0
  44. package/vitepress/components/Discussions/Discussions.module.scss +32 -0
  45. package/vitepress/components/Discussions/Discussions.vue +211 -0
  46. package/vitepress/components/EnWordList/EnWordList.module.scss +124 -0
  47. package/vitepress/components/EnWordList/EnWordList.vue +543 -0
  48. package/vitepress/components/EnWordList/RightClickMenu.module.scss +22 -0
  49. package/vitepress/components/EnWordList/RightClickMenu.vue +66 -0
  50. package/vitepress/components/Footprints/Footprints.module.scss +93 -0
  51. package/vitepress/components/Footprints/Footprints.vue +377 -0
  52. package/vitepress/components/Layout/AboutModal.module.scss +233 -0
  53. package/vitepress/components/Layout/AboutModal.vue +105 -0
  54. package/vitepress/components/Layout/AboutPanel.vue +266 -0
  55. package/vitepress/components/Layout/ContentCollapse.vue +603 -0
  56. package/vitepress/components/Layout/CustomSidebar.vue +605 -0
  57. package/vitepress/components/Layout/DocBeforeControls.vue +139 -0
  58. package/vitepress/components/Layout/DocFooter.vue +225 -0
  59. package/vitepress/components/Layout/ImagePreview.module.scss +201 -0
  60. package/vitepress/components/Layout/ImagePreview.vue +281 -0
  61. package/vitepress/components/Layout/Layout.module.scss +661 -0
  62. package/vitepress/components/Layout/Layout.vue +542 -0
  63. package/vitepress/components/Layout/NoteStatus.vue +140 -0
  64. package/vitepress/components/Layout/SidebarItems.vue +263 -0
  65. package/vitepress/components/Layout/SidebarNavBefore.vue +92 -0
  66. package/vitepress/components/Layout/Swiper.vue +167 -0
  67. package/vitepress/components/Layout/ToggleFullContent.module.scss +11 -0
  68. package/vitepress/components/Layout/ToggleFullContent.vue +34 -0
  69. package/vitepress/components/Layout/ToggleSidebar.module.scss +11 -0
  70. package/vitepress/components/Layout/ToggleSidebar.vue +35 -0
  71. package/vitepress/components/Layout/composables/useCollapseControl.ts +88 -0
  72. package/vitepress/components/Layout/composables/useNoteConfig.ts +121 -0
  73. package/vitepress/components/Layout/composables/useNoteSave.ts +173 -0
  74. package/vitepress/components/Layout/composables/useNoteValidation.ts +85 -0
  75. package/vitepress/components/Layout/composables/useRedirect.ts +110 -0
  76. package/vitepress/components/Layout/composables/useVSCodeIntegration.ts +85 -0
  77. package/vitepress/components/Layout/homeReadme.data.ts +124 -0
  78. package/vitepress/components/LoadingPage/LoadingPage.vue +192 -0
  79. package/vitepress/components/MarkMap/MarkMap.module.scss +159 -0
  80. package/vitepress/components/MarkMap/MarkMap.vue +404 -0
  81. package/vitepress/components/Mermaid/Mermaid.module.scss +275 -0
  82. package/vitepress/components/Mermaid/Mermaid.vue +364 -0
  83. package/vitepress/components/NotesTable/NotesTable.module.scss +77 -0
  84. package/vitepress/components/NotesTable/NotesTable.vue +98 -0
  85. package/vitepress/components/NotesTable/README.md +67 -0
  86. package/vitepress/components/Settings/Settings.module.scss +433 -0
  87. package/vitepress/components/Settings/Settings.vue +306 -0
  88. package/vitepress/components/SidebarCard/MindMapView.vue +483 -0
  89. package/vitepress/components/SidebarCard/NotesTrendChart.vue +108 -0
  90. package/vitepress/components/SidebarCard/SidebarCard.vue +948 -0
  91. package/vitepress/components/Tooltip/Tooltip.vue +70 -0
  92. package/vitepress/components/constants.ts +91 -0
  93. package/vitepress/components/notesConfig.data.ts +73 -0
  94. package/vitepress/components/sidebar.data.ts +59 -0
  95. package/vitepress/components/tnotes-config.data.ts +21 -0
  96. package/vitepress/components/utils.ts +26 -0
  97. package/vitepress/config/index.ts +126 -0
  98. package/vitepress/configs/constants.ts +26 -0
  99. package/vitepress/configs/head.config.ts +25 -0
  100. package/vitepress/configs/index.ts +9 -0
  101. package/vitepress/configs/markdown-it.d.ts +23 -0
  102. package/vitepress/configs/markdown.config.ts +366 -0
  103. package/vitepress/configs/theme.config.ts +108 -0
  104. package/vitepress/plugins/buildProgressPlugin.ts +390 -0
  105. package/vitepress/plugins/getNoteByConfigIdPlugin.ts +107 -0
  106. package/vitepress/plugins/renameNotePlugin.ts +60 -0
  107. package/vitepress/plugins/updateConfigPlugin.ts +63 -0
  108. package/vitepress/theme/index.ts +95 -0
  109. package/vitepress/theme/styles/base.scss +50 -0
  110. package/vitepress/theme/styles/components/404.scss +31 -0
  111. package/vitepress/theme/styles/components/collapse.scss +175 -0
  112. package/vitepress/theme/styles/components/markmap.scss +101 -0
  113. package/vitepress/theme/styles/components/swiper.scss +255 -0
  114. package/vitepress/theme/styles/index.scss +25 -0
  115. package/vitepress/theme/styles/layout.scss +62 -0
  116. package/vitepress/theme/styles/utilities.scss +39 -0
  117. package/vitepress/theme/styles/vitepress-override.scss +25 -0
@@ -0,0 +1,364 @@
1
+ <template>
2
+ <div
3
+ ref="mermaidRef"
4
+ :class="[$style.mermaidWrapper, { [$style.showToolbar]: showToolbar }]"
5
+ @mouseenter="showToolbar = true"
6
+ @mouseleave="showToolbar = false"
7
+ >
8
+ <!-- 工具栏 -->
9
+ <div
10
+ v-if="!loading && !error"
11
+ :class="[$style.toolbar, { [$style.visible]: showToolbar }]"
12
+ >
13
+ <button
14
+ @click="zoomIn"
15
+ :class="[$style.toolbarBtn, $style.iconBtn]"
16
+ title="放大 (Ctrl + +)"
17
+ >
18
+ <img :src="icon__zoom_in" alt="放大" :class="$style.btnIcon" />
19
+ </button>
20
+ <button
21
+ @click="zoomOut"
22
+ :class="[$style.toolbarBtn, $style.iconBtn]"
23
+ title="缩小 (Ctrl + -)"
24
+ >
25
+ <img :src="icon__zoom_out" alt="缩小" :class="$style.btnIcon" />
26
+ </button>
27
+ <button
28
+ @click="resetZoom"
29
+ :class="[$style.toolbarBtn, $style.iconBtn]"
30
+ title="重置缩放 (Ctrl + 0)"
31
+ >
32
+ <img :src="icon__zoom_reset" alt="重置" :class="$style.btnIcon" />
33
+ </button>
34
+ <button
35
+ @click="fitToScreen"
36
+ :class="[$style.toolbarBtn, $style.iconBtn]"
37
+ title="适应屏幕"
38
+ >
39
+ <img :src="icon__zoom_fit" alt="适应屏幕" :class="$style.btnIcon" />
40
+ </button>
41
+ <button
42
+ @click="toggleFullscreen"
43
+ :class="[$style.toolbarBtn, $style.iconBtn]"
44
+ title="全屏 (F11)"
45
+ >
46
+ <img
47
+ v-if="isFullscreen"
48
+ :src="icon__fullscreen_exit"
49
+ alt="退出全屏"
50
+ :class="$style.btnIcon"
51
+ />
52
+ <img
53
+ v-else
54
+ :src="icon__fullscreen"
55
+ alt="全屏"
56
+ :class="$style.btnIcon"
57
+ />
58
+ </button>
59
+ <span :class="$style.zoomLevel">{{ Math.round(scale * 100) }}%</span>
60
+ </div>
61
+
62
+ <!-- 图表容器 -->
63
+ <div
64
+ :class="[$style.mermaidContainer, { [$style.fullscreen]: isFullscreen }]"
65
+ @click="handleContainerClick"
66
+ >
67
+ <div v-if="loading" :class="$style.mermaidLoading">
68
+ <div :class="$style.spinner"></div>
69
+ <p>加载图表中...</p>
70
+ </div>
71
+ <div v-else-if="error" :class="$style.mermaidError">
72
+ <span :class="$style.errorIcon">⚠️</span>
73
+ <p>{{ error }}</p>
74
+ </div>
75
+ <div
76
+ v-show="!loading && !error"
77
+ ref="diagramRef"
78
+ :class="[$style.mermaidDiagram, props.class]"
79
+ :style="{
80
+ transform: `scale(${scale})`,
81
+ transformOrigin: 'center center',
82
+ }"
83
+ @wheel.prevent="handleWheel"
84
+ ></div>
85
+ </div>
86
+ </div>
87
+ </template>
88
+
89
+ <script setup lang="ts">
90
+ import mermaid from 'mermaid'
91
+ import { nextTick, onMounted, onBeforeUnmount, ref, watch } from 'vue'
92
+ import {
93
+ icon__zoom_in,
94
+ icon__zoom_out,
95
+ icon__zoom_reset,
96
+ icon__zoom_fit,
97
+ icon__fullscreen,
98
+ icon__fullscreen_exit,
99
+ } from '../../assets/icons'
100
+
101
+ const props = defineProps({
102
+ graph: {
103
+ type: String,
104
+ required: true,
105
+ },
106
+ id: {
107
+ type: String,
108
+ required: true,
109
+ },
110
+ class: {
111
+ type: String,
112
+ default: 'mermaid',
113
+ },
114
+ })
115
+
116
+ const mermaidRef = ref<HTMLElement | null>(null)
117
+ const diagramRef = ref<HTMLElement | null>(null)
118
+ const loading = ref(true)
119
+ const error = ref<string | null>(null)
120
+ const scale = ref(1)
121
+ const isFullscreen = ref(false)
122
+ const showToolbar = ref(false)
123
+
124
+ // ===================================
125
+ // #region 工具栏显示控制
126
+ // ===================================
127
+ let hideTimer: ReturnType<typeof setTimeout> | null = null
128
+
129
+ function handleContainerClick() {
130
+ // 移动端点击切换工具栏显示
131
+ if (window.innerWidth <= 768) {
132
+ showToolbar.value = !showToolbar.value
133
+
134
+ // 3秒后自动隐藏
135
+ if (showToolbar.value) {
136
+ if (hideTimer) clearTimeout(hideTimer)
137
+ hideTimer = setTimeout(() => {
138
+ showToolbar.value = false
139
+ }, 3000)
140
+ }
141
+ }
142
+ }
143
+ // #endregion
144
+
145
+ // ===================================
146
+ // #region 缩放控制
147
+ // ===================================
148
+ const MIN_SCALE = 0.1
149
+ const MAX_SCALE = 5
150
+ const SCALE_STEP = 0.1
151
+
152
+ function zoomIn() {
153
+ scale.value = Math.min(scale.value + SCALE_STEP, MAX_SCALE)
154
+ }
155
+
156
+ function zoomOut() {
157
+ scale.value = Math.max(scale.value - SCALE_STEP, MIN_SCALE)
158
+ }
159
+
160
+ function resetZoom() {
161
+ scale.value = 1
162
+ }
163
+
164
+ function fitToScreen() {
165
+ if (!diagramRef.value || !mermaidRef.value) return
166
+
167
+ const svg = diagramRef.value.querySelector('svg')
168
+ if (!svg) return
169
+
170
+ const containerRect = mermaidRef.value.getBoundingClientRect()
171
+ const svgRect = svg.getBoundingClientRect()
172
+
173
+ const scaleX = (containerRect.width * 0.9) / svgRect.width
174
+ const scaleY = (containerRect.height * 0.9) / svgRect.height
175
+
176
+ scale.value = Math.min(scaleX, scaleY, 1)
177
+ }
178
+
179
+ function handleWheel(e: WheelEvent) {
180
+ if (e.ctrlKey || e.metaKey) {
181
+ e.preventDefault()
182
+ const delta = e.deltaY > 0 ? -SCALE_STEP : SCALE_STEP
183
+ scale.value = Math.max(MIN_SCALE, Math.min(MAX_SCALE, scale.value + delta))
184
+ }
185
+ }
186
+ // #endregion
187
+
188
+ // ===================================
189
+ // #region 全屏控制
190
+ // ===================================
191
+ function toggleFullscreen() {
192
+ if (!mermaidRef.value) return
193
+
194
+ if (!isFullscreen.value) {
195
+ if (mermaidRef.value.requestFullscreen) {
196
+ mermaidRef.value.requestFullscreen()
197
+ }
198
+ } else {
199
+ if (document.exitFullscreen) {
200
+ document.exitFullscreen()
201
+ }
202
+ }
203
+ }
204
+
205
+ function handleFullscreenChange() {
206
+ isFullscreen.value = !!document.fullscreenElement
207
+ }
208
+ // #endregion
209
+
210
+ // ===================================
211
+ // #region 键盘快捷键
212
+ // ===================================
213
+ function handleKeydown(e: KeyboardEvent) {
214
+ if (e.ctrlKey || e.metaKey) {
215
+ if (e.key === '=' || e.key === '+') {
216
+ e.preventDefault()
217
+ zoomIn()
218
+ } else if (e.key === '-' || e.key === '_') {
219
+ e.preventDefault()
220
+ zoomOut()
221
+ } else if (e.key === '0') {
222
+ e.preventDefault()
223
+ resetZoom()
224
+ }
225
+ } else if (e.key === 'F11') {
226
+ e.preventDefault()
227
+ toggleFullscreen()
228
+ } else if (e.key === 'Escape' && isFullscreen.value) {
229
+ toggleFullscreen()
230
+ }
231
+ }
232
+ // #endregion
233
+
234
+ // ===================================
235
+ // #region Mermaid 渲染
236
+ // ===================================
237
+ // 获取当前主题
238
+ const getCurrentTheme = () => {
239
+ return document.documentElement.classList.contains('dark')
240
+ ? 'dark'
241
+ : 'default'
242
+ }
243
+
244
+ // 初始化 Mermaid
245
+ const initMermaid = async () => {
246
+ try {
247
+ mermaid.initialize({
248
+ startOnLoad: false,
249
+ theme: getCurrentTheme(),
250
+ securityLevel: 'loose',
251
+ fontFamily: 'inherit',
252
+ })
253
+ } catch (err) {
254
+ console.error('Mermaid initialization error:', err)
255
+ }
256
+ }
257
+
258
+ // 渲染图表
259
+ const renderDiagram = async () => {
260
+ // 等待下一个 DOM 更新周期
261
+ await nextTick()
262
+
263
+ if (!diagramRef.value) {
264
+ // 再等待一小段时间确保 DOM 完全渲染
265
+ await new Promise((resolve) => setTimeout(resolve, 0))
266
+ if (!diagramRef.value) {
267
+ // console.warn('diagramRef is still null')
268
+ return
269
+ }
270
+ }
271
+
272
+ try {
273
+ loading.value = true
274
+ error.value = null
275
+
276
+ // 每次渲染前都重新初始化主题
277
+ mermaid.initialize({
278
+ startOnLoad: false,
279
+ theme: getCurrentTheme(),
280
+ securityLevel: 'loose',
281
+ fontFamily: 'inherit',
282
+ })
283
+
284
+ const { svg, bindFunctions } = await mermaid.render(
285
+ props.id,
286
+ decodeURIComponent(props.graph)
287
+ )
288
+
289
+ diagramRef.value.innerHTML = svg
290
+
291
+ // 绑定交互函数(如果有)
292
+ if (bindFunctions) {
293
+ bindFunctions(diagramRef.value)
294
+ }
295
+ } catch (err) {
296
+ error.value = `Failed to render diagram: ${err.message}`
297
+ console.error('Mermaid render error:', err)
298
+ } finally {
299
+ loading.value = false
300
+ }
301
+ }
302
+
303
+ // 监听主题变化
304
+ const observeTheme = () => {
305
+ const observer = new MutationObserver((mutations) => {
306
+ mutations.forEach((mutation) => {
307
+ if (
308
+ mutation.type === 'attributes' &&
309
+ mutation.attributeName === 'class'
310
+ ) {
311
+ // 检查是否是主题变化
312
+ const isDark = document.documentElement.classList.contains('dark')
313
+ const wasDark = mutation.oldValue?.includes('dark')
314
+
315
+ // 只有当主题实际发生变化时才重新渲染
316
+ if ((isDark && !wasDark) || (!isDark && wasDark)) {
317
+ renderDiagram()
318
+ }
319
+ }
320
+ })
321
+ })
322
+
323
+ observer.observe(document.documentElement, {
324
+ attributes: true,
325
+ attributeFilter: ['class'],
326
+ attributeOldValue: true,
327
+ })
328
+
329
+ return observer
330
+ }
331
+
332
+ onMounted(async () => {
333
+ await initMermaid()
334
+ await renderDiagram()
335
+
336
+ // 观察主题变化
337
+ const themeObserver = observeTheme()
338
+
339
+ // 监听键盘事件
340
+ document.addEventListener('keydown', handleKeydown)
341
+
342
+ // 监听全屏变化
343
+ document.addEventListener('fullscreenchange', handleFullscreenChange)
344
+
345
+ // 清理函数
346
+ onBeforeUnmount(() => {
347
+ themeObserver.disconnect()
348
+ document.removeEventListener('keydown', handleKeydown)
349
+ document.removeEventListener('fullscreenchange', handleFullscreenChange)
350
+ })
351
+ })
352
+
353
+ // 当图表内容变化时重新渲染并重置缩放
354
+ watch(
355
+ () => props.graph,
356
+ () => {
357
+ renderDiagram()
358
+ resetZoom()
359
+ }
360
+ )
361
+ // #endregion
362
+ </script>
363
+
364
+ <style module src="./Mermaid.module.scss"></style>
@@ -0,0 +1,77 @@
1
+ .notesTable {
2
+ width: 100%;
3
+ border-collapse: collapse;
4
+ margin: 1.5rem 0;
5
+ font-size: 0.95rem;
6
+
7
+ th,
8
+ td {
9
+ padding: 0.75rem 1rem;
10
+ text-align: left;
11
+ border: 1px solid var(--vp-c-divider);
12
+ }
13
+
14
+ th {
15
+ background-color: var(--vp-c-bg-soft);
16
+ font-weight: 600;
17
+ color: var(--vp-c-text-1);
18
+ }
19
+
20
+ tbody tr {
21
+ transition: background-color 0.2s;
22
+
23
+ &:hover {
24
+ background-color: var(--vp-c-bg-soft);
25
+ }
26
+ }
27
+
28
+ td {
29
+ color: var(--vp-c-text-2);
30
+ }
31
+
32
+ .noteLink {
33
+ color: var(--vp-c-brand-1);
34
+ text-decoration: none;
35
+ font-weight: 500;
36
+ transition: color 0.2s;
37
+
38
+ &:hover {
39
+ color: var(--vp-c-brand-2);
40
+ text-decoration: underline;
41
+ }
42
+ }
43
+
44
+ .noteId {
45
+ font-family: var(--vp-font-family-mono);
46
+ font-size: 0.9em;
47
+ color: var(--vp-c-text-3);
48
+ margin-right: 0.5rem;
49
+ }
50
+
51
+ .description {
52
+ line-height: 1.6;
53
+
54
+ &.empty {
55
+ color: var(--vp-c-text-3);
56
+ font-style: italic;
57
+ }
58
+ }
59
+ }
60
+
61
+ .error {
62
+ padding: 1rem;
63
+ margin: 1rem 0;
64
+ border-left: 4px solid var(--vp-c-danger-1);
65
+ background-color: var(--vp-c-danger-soft);
66
+ color: var(--vp-c-danger-1);
67
+ border-radius: 4px;
68
+ }
69
+
70
+ .warning {
71
+ padding: 1rem;
72
+ margin: 1rem 0;
73
+ border-left: 4px solid var(--vp-c-warning-1);
74
+ background-color: var(--vp-c-warning-soft);
75
+ color: var(--vp-c-warning-1);
76
+ border-radius: 4px;
77
+ }
@@ -0,0 +1,98 @@
1
+ <template>
2
+ <div v-if="errorMessage" :class="$style.error">
3
+ {{ errorMessage }}
4
+ </div>
5
+
6
+ <div v-else-if="notFoundIds.length > 0" :class="$style.warning">
7
+ 以下笔记 ID 未找到配置: {{ notFoundIds.join(', ') }}
8
+ </div>
9
+
10
+ <table v-if="tableData.length > 0" :class="$style.notesTable">
11
+ <thead>
12
+ <tr>
13
+ <th>笔记</th>
14
+ <th>简介</th>
15
+ </tr>
16
+ </thead>
17
+ <tbody>
18
+ <tr v-for="note in tableData" :key="note.id">
19
+ <td>
20
+ <a :href="note.url" :class="$style.noteLink">
21
+ <span :class="$style.noteId">{{ note.id }}.</span>
22
+ <span>{{ note.title }}</span>
23
+ </a>
24
+ </td>
25
+ <td>
26
+ <span
27
+ :class="[$style.description, { [$style.empty]: !note.description }]"
28
+ >
29
+ {{ note.description || '暂无简介' }}
30
+ </span>
31
+ </td>
32
+ </tr>
33
+ </tbody>
34
+ </table>
35
+ </template>
36
+
37
+ <script setup lang="ts">
38
+ import { computed } from 'vue'
39
+ import { useData } from 'vitepress'
40
+ // @ts-expect-error - VitePress data loader exports data at runtime
41
+ import { data as allNotesConfig } from '../notesConfig.data.ts'
42
+
43
+ interface Props {
44
+ ids: string[]
45
+ }
46
+
47
+ const props = defineProps<Props>()
48
+ const vpData = useData()
49
+
50
+ // 错误消息
51
+ const errorMessage = computed(() => {
52
+ if (!props.ids || !Array.isArray(props.ids)) {
53
+ return '错误: ids 属性必须是一个数组'
54
+ }
55
+ if (props.ids.length === 0) {
56
+ return '错误: ids 数组不能为空'
57
+ }
58
+ return null
59
+ })
60
+
61
+ // 未找到的笔记 ID
62
+ const notFoundIds = computed(() => {
63
+ if (errorMessage.value) return []
64
+
65
+ return props.ids.filter((id) => !allNotesConfig[id])
66
+ })
67
+
68
+ // 表格数据
69
+ const tableData = computed(() => {
70
+ if (errorMessage.value) return []
71
+
72
+ return props.ids
73
+ .filter((id) => allNotesConfig[id]) // 只保留存在的笔记
74
+ .map((id) => {
75
+ const config = allNotesConfig[id]
76
+ const base = vpData.site.value.base || '/'
77
+
78
+ // 从 redirect 路径中提取笔记标题
79
+ // redirect 格式: notes/0001. 标题/README
80
+ let title = id
81
+ if (config.redirect) {
82
+ const match = config.redirect.match(/notes\/\d{4}\.\s*([^/]+)\/README/)
83
+ if (match) {
84
+ title = match[1]
85
+ }
86
+ }
87
+
88
+ return {
89
+ id,
90
+ title,
91
+ description: config.description || '',
92
+ url: config.redirect ? `${base}${config.redirect}` : '#',
93
+ }
94
+ })
95
+ })
96
+ </script>
97
+
98
+ <style module src="./NotesTable.module.scss"></style>
@@ -0,0 +1,67 @@
1
+ # NotesTable 组件
2
+
3
+ ## 功能说明
4
+
5
+ `NotesTable` 组件用于在 Markdown 文件中渲染笔记列表表格,支持笔记链接跳转和简介显示。
6
+
7
+ ## 使用方法
8
+
9
+ ### 基础用法
10
+
11
+ ```html
12
+ <NotesTable :ids="['0001', '0002', '0054', '0039']" />
13
+ ```
14
+
15
+ ### 简写形式
16
+
17
+ ```html
18
+ <N :ids="['0001', '0002', '0054', '0039']" />
19
+ ```
20
+
21
+ ## Props
22
+
23
+ | 属性 | 类型 | 必填 | 说明 |
24
+ | ---- | -------- | ---- | -------------------------------------- |
25
+ | ids | string[] | 是 | 笔记 ID 数组,每个 ID 为 4 位数字字符串 |
26
+
27
+ ## 示例效果
28
+
29
+ 渲染后的表格包含两列:
30
+
31
+ 1. 笔记列: 显示笔记 ID 和标题,支持点击跳转
32
+ 2. 简介列: 显示笔记配置中的 `description` 字段内容
33
+
34
+ ## 错误处理
35
+
36
+ ### 1. ids 属性验证
37
+
38
+ - 如果 `ids` 不是数组,显示错误提示
39
+ - 如果 `ids` 为空数组,显示错误提示
40
+
41
+ ### 2. 笔记不存在
42
+
43
+ - 如果某些 ID 在配置中找不到,会显示警告信息
44
+ - 表格只渲染存在的笔记
45
+
46
+ ## 数据来源
47
+
48
+ 组件从 `notesConfig.data.ts` 中读取笔记配置信息,包括:
49
+
50
+ - 笔记标题 (从 redirect 路径解析)
51
+ - 笔记简介 (description 字段)
52
+ - 笔记链接 (redirect 字段)
53
+
54
+ ## 样式特性
55
+
56
+ - 响应式表格设计
57
+ - 悬停高亮效果
58
+ - 支持暗色/亮色主题
59
+ - 笔记链接带品牌色
60
+ - 空简介显示占位文本
61
+
62
+ ## 注意事项
63
+
64
+ 1. 笔记 ID 必须是 4 位数字字符串 (如 "0001", "0054")
65
+ 2. 组件会自动过滤不存在的笔记 ID
66
+ 3. 简介为空时显示 "暂无简介"
67
+ 4. 链接会自动添加网站 base 路径