@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,121 @@
1
+ import { ref, computed, watch } from 'vue'
2
+ import type { Ref } from 'vue'
3
+ import { useData } from 'vitepress'
4
+ import type { NoteConfig } from '../../../../types'
5
+
6
+ /**
7
+ * 管理笔记配置状态
8
+ * 包括可编辑字段、原始值、变更检测等
9
+ */
10
+ export function useNoteConfig(
11
+ currentNoteId: Ref<string>,
12
+ currentNoteConfig: Ref<NoteConfig | null>,
13
+ currentNoteTitle: Ref<string>,
14
+ timeModalOpen: Ref<boolean>
15
+ ) {
16
+ const vpData = useData()
17
+
18
+ // 可编辑的配置项
19
+ const editableNoteStatus = ref(false)
20
+ const editableDiscussionsEnabled = ref(false)
21
+ const editableNoteTitle = ref('')
22
+ const editableDescription = ref('')
23
+
24
+ // 原始配置(用于检测是否有变更)
25
+ const originalNoteStatus = ref(false)
26
+ const originalDiscussionsEnabled = ref(false)
27
+ const originalNoteTitle = ref('')
28
+ const originalDescription = ref('')
29
+
30
+ // 标题验证错误信息
31
+ const titleError = ref('')
32
+
33
+ // 检测是否有配置变更
34
+ const hasConfigChanges = computed(() => {
35
+ return (
36
+ editableNoteStatus.value !== originalNoteStatus.value ||
37
+ editableDiscussionsEnabled.value !== originalDiscussionsEnabled.value ||
38
+ editableDescription.value.trim() !== originalDescription.value ||
39
+ (editableNoteTitle.value.trim() !== originalNoteTitle.value &&
40
+ !titleError.value)
41
+ )
42
+ })
43
+
44
+ // 初始化可编辑字段
45
+ function initEditableFields() {
46
+ if (!currentNoteConfig.value) return
47
+
48
+ editableNoteStatus.value = currentNoteConfig.value.done || false
49
+ editableDiscussionsEnabled.value =
50
+ currentNoteConfig.value.enableDiscussions || false
51
+ editableNoteTitle.value = currentNoteTitle.value || ''
52
+ editableDescription.value = currentNoteConfig.value.description || ''
53
+
54
+ // 保存原始值
55
+ originalNoteStatus.value = editableNoteStatus.value
56
+ originalDiscussionsEnabled.value = editableDiscussionsEnabled.value
57
+ originalNoteTitle.value = editableNoteTitle.value
58
+ originalDescription.value = editableDescription.value
59
+
60
+ // 清除错误信息
61
+ titleError.value = ''
62
+ }
63
+
64
+ // 重置笔记配置
65
+ function resetNoteConfig() {
66
+ editableNoteStatus.value = originalNoteStatus.value
67
+ editableDiscussionsEnabled.value = originalDiscussionsEnabled.value
68
+ editableNoteTitle.value = originalNoteTitle.value
69
+ editableDescription.value = originalDescription.value
70
+ titleError.value = ''
71
+ }
72
+
73
+ // 更新原始值(保存成功后调用)
74
+ function updateOriginalValues() {
75
+ originalNoteStatus.value = editableNoteStatus.value
76
+ originalDiscussionsEnabled.value = editableDiscussionsEnabled.value
77
+ originalNoteTitle.value = editableNoteTitle.value.trim()
78
+ originalDescription.value = editableDescription.value.trim()
79
+ }
80
+
81
+ // 监听笔记配置变化,重新初始化字段
82
+ watch(
83
+ () => currentNoteConfig.value,
84
+ () => {
85
+ initEditableFields()
86
+ },
87
+ { immediate: true }
88
+ )
89
+
90
+ // modal 打开时重新初始化(确保数据最新)
91
+ watch(timeModalOpen, (isOpen) => {
92
+ if (isOpen) {
93
+ initEditableFields()
94
+ }
95
+ })
96
+
97
+ return {
98
+ // 可编辑字段
99
+ editableNoteStatus,
100
+ editableDiscussionsEnabled,
101
+ editableNoteTitle,
102
+ editableDescription,
103
+
104
+ // 原始值
105
+ originalNoteStatus,
106
+ originalDiscussionsEnabled,
107
+ originalNoteTitle,
108
+ originalDescription,
109
+
110
+ // 验证
111
+ titleError,
112
+
113
+ // 计算属性
114
+ hasConfigChanges,
115
+
116
+ // 方法
117
+ initEditableFields,
118
+ resetNoteConfig,
119
+ updateOriginalValues,
120
+ }
121
+ }
@@ -0,0 +1,173 @@
1
+ import { ref, computed } from 'vue'
2
+ import type { Ref, ComputedRef } from 'vue'
3
+ import { useData } from 'vitepress'
4
+ import type { NoteConfig } from '../../../../types'
5
+
6
+ /**
7
+ * 处理笔记配置的保存和重置逻辑
8
+ */
9
+ export function useNoteSave(
10
+ currentNoteId: Ref<string>,
11
+ isDev: Ref<boolean> | ComputedRef<boolean>,
12
+ hasConfigChanges: ComputedRef<boolean>,
13
+ titleError: Ref<string>,
14
+ editableNoteTitle: Ref<string>,
15
+ originalNoteTitle: Ref<string>,
16
+ editableNoteStatus: Ref<boolean>,
17
+ originalNoteStatus: Ref<boolean>,
18
+ editableDiscussionsEnabled: Ref<boolean>,
19
+ originalDiscussionsEnabled: Ref<boolean>,
20
+ editableDescription: Ref<string>,
21
+ originalDescription: Ref<string>,
22
+ allNotesConfig: Record<string, NoteConfig & { redirect?: string }>,
23
+ updateOriginalValues: () => void
24
+ ) {
25
+ const vpData = useData()
26
+
27
+ // 保存状态
28
+ const isSaving = ref(false)
29
+ const showSuccessToast = ref(false)
30
+ const savingMessage = ref('') // 保存进度提示
31
+
32
+ // 保存按钮文本
33
+ const saveButtonText = computed(() => {
34
+ if (isSaving.value) return '保存中...'
35
+ if (!hasConfigChanges.value) return '无更改'
36
+ return '保存配置'
37
+ })
38
+
39
+ // 保存笔记配置
40
+ async function saveNoteConfig() {
41
+ if (!currentNoteId.value || !isDev.value || !hasConfigChanges.value) return
42
+
43
+ // 验证标题
44
+ if (titleError.value) {
45
+ alert('❌ 请修正标题错误后再保存')
46
+ return
47
+ }
48
+
49
+ const titleChanged =
50
+ editableNoteTitle.value.trim() !== originalNoteTitle.value &&
51
+ editableNoteTitle.value.trim()
52
+
53
+ isSaving.value = true
54
+ savingMessage.value = '正在保存配置...'
55
+
56
+ try {
57
+ // 如果标题有变化,先重命名文件夹
58
+ if (titleChanged) {
59
+ savingMessage.value = '正在重命名文件夹...'
60
+
61
+ const renameResponse = await fetch('/__tnotes_rename_note', {
62
+ method: 'POST',
63
+ headers: {
64
+ 'Content-Type': 'application/json',
65
+ },
66
+ body: JSON.stringify({
67
+ noteIndex: currentNoteId.value,
68
+ newTitle: editableNoteTitle.value.trim(),
69
+ }),
70
+ })
71
+
72
+ if (!renameResponse.ok) {
73
+ const error = await renameResponse.text()
74
+ throw new Error(`重命名失败: ${error}`)
75
+ }
76
+
77
+ // 后端已经完成所有更新,包括文件系统同步
78
+ const result = await renameResponse.json()
79
+ console.log('重命名完成:', result)
80
+
81
+ savingMessage.value = '文件已同步,准备跳转...'
82
+ }
83
+
84
+ // 检查是否需要更新配置(无论标题是否改变)
85
+ const needConfigUpdate =
86
+ editableNoteStatus.value !== originalNoteStatus.value ||
87
+ editableDiscussionsEnabled.value !== originalDiscussionsEnabled.value ||
88
+ editableDescription.value.trim() !== originalDescription.value
89
+
90
+ if (needConfigUpdate) {
91
+ savingMessage.value = '正在更新笔记配置...'
92
+
93
+ const response = await fetch('/__tnotes_update_config', {
94
+ method: 'POST',
95
+ headers: {
96
+ 'Content-Type': 'application/json',
97
+ },
98
+ body: JSON.stringify({
99
+ noteIndex: currentNoteId.value,
100
+ config: {
101
+ done: editableNoteStatus.value,
102
+ enableDiscussions: editableDiscussionsEnabled.value,
103
+ description: editableDescription.value.trim(),
104
+ },
105
+ }),
106
+ })
107
+
108
+ if (!response.ok) {
109
+ const error = await response.text()
110
+ throw new Error(error || '保存失败')
111
+ }
112
+ }
113
+
114
+ // 更新原始值
115
+ updateOriginalValues()
116
+
117
+ // 更新本地配置(立即反映在页面上)
118
+ if (allNotesConfig[currentNoteId.value]) {
119
+ allNotesConfig[currentNoteId.value].done = editableNoteStatus.value
120
+ allNotesConfig[currentNoteId.value].enableDiscussions =
121
+ editableDiscussionsEnabled.value
122
+ allNotesConfig[currentNoteId.value].description =
123
+ editableDescription.value.trim()
124
+ }
125
+
126
+ savingMessage.value = '保存成功!'
127
+
128
+ // 显示成功提示
129
+ showSuccessToast.value = true
130
+ setTimeout(() => {
131
+ showSuccessToast.value = false
132
+ }, 3000)
133
+
134
+ // 如果标题改变了,先跳转到loading页,再由loading页根据configId查询目标URL
135
+ if (titleChanged) {
136
+ // 获取当前笔记的 configId (UUID)
137
+ const configId = allNotesConfig[currentNoteId.value]?.id
138
+
139
+ if (!configId) {
140
+ throw new Error('无法获取笔记的 configId')
141
+ }
142
+
143
+ // 跳转到loading页,传递 configId 参数
144
+ const base = vpData.site.value.base || '/'
145
+ const loadingUrl = `${base}loading?configId=${encodeURIComponent(
146
+ configId
147
+ )}`
148
+ window.location.href = loadingUrl
149
+ }
150
+ } catch (error) {
151
+ console.error('保存配置失败:', error)
152
+ savingMessage.value = ''
153
+
154
+ alert(
155
+ '❌ 保存失败: ' +
156
+ (error instanceof Error ? error.message : String(error))
157
+ )
158
+ } finally {
159
+ if (!titleChanged) {
160
+ isSaving.value = false
161
+ savingMessage.value = ''
162
+ }
163
+ }
164
+ }
165
+
166
+ return {
167
+ isSaving,
168
+ showSuccessToast,
169
+ savingMessage,
170
+ saveButtonText,
171
+ saveNoteConfig,
172
+ }
173
+ }
@@ -0,0 +1,85 @@
1
+ import type { Ref } from 'vue'
2
+
3
+ /**
4
+ * 笔记标题验证
5
+ */
6
+ export function useNoteValidation() {
7
+ // 标题验证函数
8
+ function validateTitle(title: string): string | null {
9
+ // 非法字符正则
10
+ const invalidChars = /[<>:"/\\|?*\x00-\x1F]/
11
+ const windowsReservedNames = new Set([
12
+ 'CON',
13
+ 'PRN',
14
+ 'AUX',
15
+ 'NUL',
16
+ 'COM1',
17
+ 'COM2',
18
+ 'COM3',
19
+ 'COM4',
20
+ 'COM5',
21
+ 'COM6',
22
+ 'COM7',
23
+ 'COM8',
24
+ 'COM9',
25
+ 'LPT1',
26
+ 'LPT2',
27
+ 'LPT3',
28
+ 'LPT4',
29
+ 'LPT5',
30
+ 'LPT6',
31
+ 'LPT7',
32
+ 'LPT8',
33
+ 'LPT9',
34
+ ])
35
+
36
+ if (!title || title.trim().length === 0) {
37
+ return '标题不能为空'
38
+ }
39
+
40
+ const trimmedTitle = title.trim()
41
+
42
+ if (trimmedTitle.length > 200) {
43
+ return '标题过长(最多200个字符)'
44
+ }
45
+
46
+ if (invalidChars.test(trimmedTitle)) {
47
+ return '标题包含非法字符(不允许: < > : " / \\ | ? *)'
48
+ }
49
+
50
+ if (/^[.\s]|[.\s]$/.test(trimmedTitle)) {
51
+ return '标题不能以点或空格开头/结尾'
52
+ }
53
+
54
+ const upperTitle = trimmedTitle.toUpperCase()
55
+ if (windowsReservedNames.has(upperTitle)) {
56
+ return `"${trimmedTitle}" 是 Windows 系统保留名称`
57
+ }
58
+
59
+ const baseName = trimmedTitle.split('.')[0].toUpperCase()
60
+ if (windowsReservedNames.has(baseName)) {
61
+ return `"${trimmedTitle}" 包含 Windows 系统保留名称`
62
+ }
63
+
64
+ return null
65
+ }
66
+
67
+ // 标题输入事件处理器
68
+ function onTitleInput(editableNoteTitle: Ref<string>, titleError: Ref<string>) {
69
+ const error = validateTitle(editableNoteTitle.value)
70
+ titleError.value = error || ''
71
+ }
72
+
73
+ // 标题失焦事件处理器
74
+ function onTitleBlur(editableNoteTitle: Ref<string>, titleError: Ref<string>) {
75
+ // 去除首尾空格
76
+ editableNoteTitle.value = editableNoteTitle.value.trim()
77
+ onTitleInput(editableNoteTitle, titleError)
78
+ }
79
+
80
+ return {
81
+ validateTitle,
82
+ onTitleInput,
83
+ onTitleBlur,
84
+ }
85
+ }
@@ -0,0 +1,110 @@
1
+ import { ref, watch } from 'vue'
2
+ import { useRoute, useData } from 'vitepress'
3
+
4
+ import type { NoteConfig } from '../../../../types'
5
+
6
+ /**
7
+ * 处理 404 重定向逻辑
8
+ * 当用户访问旧的笔记 URL 时,自动重定向到新的 URL
9
+ */
10
+ export function useRedirect(allNotesConfig: Record<string, NoteConfig & { redirect?: string }>) {
11
+ const route = useRoute()
12
+ const vpData = useData()
13
+
14
+ // 控制是否显示 404 内容
15
+ const showNotFound = ref(false)
16
+ const currentPath = ref('')
17
+ const matchedId = ref('')
18
+ const redirectPath = ref('')
19
+
20
+ // 解码后的当前路径(用于调试)
21
+ const decodedCurrentPath = ref('')
22
+
23
+ // 重定向检查函数
24
+ function checkRedirect() {
25
+ if (typeof window === 'undefined') return false
26
+
27
+ currentPath.value = window.location.pathname
28
+
29
+ // 匹配路径格式:/TNotes.*/notes/四个数字{任意内容}/README
30
+ const match = currentPath.value.match(
31
+ /\/TNotes[^/]+\/notes\/(\d{4})[^/]*(?:\/.*)?$/
32
+ )
33
+
34
+ if (match) {
35
+ matchedId.value = match[1]
36
+ const targetNote = allNotesConfig[matchedId.value]
37
+ redirectPath.value = targetNote ? targetNote.redirect : ''
38
+
39
+ if (targetNote && targetNote.redirect) {
40
+ const base = vpData.site.value.base
41
+ // 构建目标路径(包含基础路径)
42
+ const targetPath = `${base}${targetNote.redirect}`
43
+
44
+ // 避免重定向死循环
45
+ if (currentPath.value !== targetPath) {
46
+ console.log(`Redirecting from ${currentPath.value} to ${targetPath}`)
47
+
48
+ // 使用完整的页面跳转(强制刷新)
49
+ window.location.href = targetPath
50
+ return true
51
+ }
52
+ }
53
+ }
54
+ return false
55
+ }
56
+
57
+ // 更新解码后的路径
58
+ function updateDecodedPath() {
59
+ try {
60
+ decodedCurrentPath.value = decodeURIComponent(currentPath.value)
61
+ } catch (e) {
62
+ console.error('Failed to decode URI:', e)
63
+ decodedCurrentPath.value = currentPath.value
64
+ }
65
+ }
66
+
67
+ // 初始化检查(在组件挂载后调用)
68
+ function initRedirectCheck() {
69
+ // 延迟执行以确保路由状态稳定
70
+ setTimeout(() => {
71
+ // 如果是 404 页面,尝试重定向
72
+ if (vpData.page.value.isNotFound) {
73
+ const redirected = checkRedirect()
74
+
75
+ // 如果重定向失败,显示原始 404 内容
76
+ if (!redirected) {
77
+ showNotFound.value = true
78
+ }
79
+ }
80
+ }, 1000)
81
+ }
82
+
83
+ // 监听路由变化
84
+ watch(
85
+ () => route.path,
86
+ () => {
87
+ // 延迟检查以确保路由更新完成
88
+ setTimeout(() => {
89
+ if (vpData.page.value.isNotFound) {
90
+ const redirected = checkRedirect()
91
+ if (!redirected) {
92
+ showNotFound.value = true
93
+ }
94
+ }
95
+ }, 1000)
96
+ }
97
+ )
98
+
99
+ // 监听 currentPath 变化,更新解码后的路径
100
+ watch(currentPath, updateDecodedPath, { immediate: true })
101
+
102
+ return {
103
+ showNotFound,
104
+ currentPath,
105
+ matchedId,
106
+ redirectPath,
107
+ decodedCurrentPath,
108
+ initRedirectCheck,
109
+ }
110
+ }
@@ -0,0 +1,85 @@
1
+ import { ref } from 'vue'
2
+ import type { Ref, ComputedRef } from 'vue'
3
+ import { useData } from 'vitepress'
4
+ import type { Router } from 'vitepress'
5
+ import { NOTES_DIR_KEY } from '../../constants'
6
+
7
+ /**
8
+ * VSCode 集成和 GitHub 链接拦截
9
+ */
10
+ export function useVSCodeIntegration() {
11
+ const vpData = useData()
12
+ const vscodeNotesDir = ref('')
13
+
14
+ // 更新 VSCode 笔记目录
15
+ const updateVscodeNoteDir = () => {
16
+ if (typeof window !== 'undefined') {
17
+ const notesDir = localStorage.getItem(NOTES_DIR_KEY)
18
+ vscodeNotesDir.value = notesDir
19
+ ? `vscode://file/${notesDir}/${vpData.page.value.relativePath}`
20
+ : ''
21
+ }
22
+ }
23
+
24
+ // 拦截 home README 中的笔记链接,将 GitHub 链接转换为站点内跳转
25
+ const interceptHomeReadmeLinks = (
26
+ isHomeReadme: Ref<boolean> | ComputedRef<boolean>,
27
+ router: Router,
28
+ ) => {
29
+ if (typeof window === 'undefined') return
30
+
31
+ // 只在 home README 页面执行
32
+ if (!isHomeReadme.value) return
33
+
34
+ // 延迟执行,确保 DOM 已渲染
35
+ setTimeout(() => {
36
+ const content = document.querySelector('.vp-doc')
37
+ if (!content) return
38
+
39
+ // 查找所有指向 GitHub 的笔记链接
40
+ const links = content.querySelectorAll(
41
+ 'a[href*="github.com"][href*="/notes/"]',
42
+ )
43
+
44
+ links.forEach((link) => {
45
+ const href = link.getAttribute('href')
46
+ if (!href) return
47
+
48
+ // 匹配 GitHub 链接格式:https://github.com/{owner}/{repo}/tree/main/notes/{noteDir}/README.md
49
+ const match = href.match(
50
+ /github\.com\/[^/]+\/[^/]+\/tree\/main\/notes\/([^/]+)\/README\.md/,
51
+ )
52
+
53
+ if (match) {
54
+ const encodedNoteDir = match[1]
55
+ const noteDir = decodeURIComponent(encodedNoteDir)
56
+
57
+ // 构建站点内的相对路径
58
+ const base = vpData.site.value.base || '/'
59
+ const internalPath = `${base}notes/${noteDir}/README`
60
+
61
+ // 移除原有的点击事件监听器(如果有)
62
+ const newLink = link.cloneNode(true) as HTMLElement
63
+
64
+ // 添加点击事件拦截
65
+ newLink.addEventListener('click', (e) => {
66
+ e.preventDefault()
67
+ router.go(internalPath)
68
+ })
69
+
70
+ // 更新 href 属性(用于悬停显示和右键菜单)
71
+ newLink.setAttribute('href', internalPath)
72
+
73
+ // 替换原链接
74
+ link.parentNode?.replaceChild(newLink, link)
75
+ }
76
+ })
77
+ }, 100)
78
+ }
79
+
80
+ return {
81
+ vscodeNotesDir,
82
+ updateVscodeNoteDir,
83
+ interceptHomeReadmeLinks,
84
+ }
85
+ }
@@ -0,0 +1,124 @@
1
+ // .vitepress/components/Layout/homeReadme.data.ts
2
+ import fs from 'node:fs'
3
+ import { execSync } from 'node:child_process'
4
+ import path from 'node:path'
5
+
6
+ const rootPath = process.cwd()
7
+
8
+ interface ReadmeData {
9
+ fileContent: string
10
+ doneNotesID: string[]
11
+ doneNotesLen: number
12
+ totalNotesLen: number
13
+ created_at?: number
14
+ updated_at?: number
15
+ }
16
+
17
+ export default {
18
+ watch: [path.resolve(rootPath, 'README.md'), path.resolve(rootPath, 'notes')],
19
+ load(watchedFiles: string[]): ReadmeData {
20
+ let readmeData: ReadmeData = {
21
+ fileContent: '',
22
+ doneNotesID: [],
23
+ doneNotesLen: 0,
24
+ totalNotesLen: 0,
25
+ }
26
+
27
+ watchedFiles.forEach((file) => {
28
+ if (file.endsWith('README.md')) {
29
+ // console.log('file:', file) // => file: README.md
30
+
31
+ const fileContent = fs.readFileSync(file, 'utf-8')
32
+ const doneNotesID = getDoneNotesID(fileContent)
33
+ const doneNotesLen = doneNotesID.length
34
+
35
+ // 获取 git 仓库的时间戳
36
+ const timestamps = getGitTimestamps()
37
+
38
+ // 计算总笔记数
39
+ const notesDir = path.join(path.dirname(file), 'notes')
40
+ const totalNotesLen = getTotalNotesCount(notesDir)
41
+
42
+ readmeData = {
43
+ fileContent,
44
+ doneNotesID,
45
+ doneNotesLen,
46
+ totalNotesLen,
47
+ ...timestamps,
48
+ }
49
+ }
50
+ })
51
+ return readmeData
52
+ },
53
+ }
54
+
55
+ /**
56
+ * 获取 notes 目录下的笔记总数
57
+ * @param notesDir notes 目录路径
58
+ * @returns 笔记总数
59
+ */
60
+ function getTotalNotesCount(notesDir: string): number {
61
+ try {
62
+ if (!fs.existsSync(notesDir)) return 0
63
+
64
+ const dirs = fs.readdirSync(notesDir, { withFileTypes: true })
65
+ // 统计符合格式 "0001. xxx" 的目录
66
+ return dirs.filter(
67
+ (dirent) => dirent.isDirectory() && /^\d{4}\./.test(dirent.name),
68
+ ).length
69
+ } catch (error) {
70
+ console.error(`获取笔记总数失败:`, error)
71
+ return 0
72
+ }
73
+ }
74
+
75
+ /**
76
+ * 获取 git 仓库的时间戳
77
+ * @returns 包含 created_at 和 updated_at 的对象
78
+ */
79
+ function getGitTimestamps(): {
80
+ created_at?: number
81
+ updated_at?: number
82
+ } {
83
+ const now = Date.now()
84
+ let created_at = now
85
+ let updated_at = now
86
+
87
+ try {
88
+ // 仓库的首次提交时间(最早的提交)
89
+ // 先获取所有提交,然后取第一行
90
+ const createdStdout = execSync(`git log --reverse --format=%ct`, {
91
+ encoding: 'utf-8',
92
+ })
93
+ const createdTs = createdStdout.toString().trim().split('\n')[0]
94
+ if (createdTs) {
95
+ created_at = parseInt(createdTs, 10) * 1000
96
+ }
97
+
98
+ // 仓库的最近一次提交时间
99
+ const updatedStdout = execSync(`git log -1 --format=%ct`, {
100
+ encoding: 'utf-8',
101
+ })
102
+ const updatedTs = updatedStdout.toString().trim()
103
+ if (updatedTs) {
104
+ updated_at = parseInt(updatedTs, 10) * 1000
105
+ }
106
+
107
+ return { created_at, updated_at }
108
+ } catch (error) {
109
+ console.error(`获取 git 仓库时间戳失败:`, error)
110
+ return { created_at, updated_at }
111
+ }
112
+ }
113
+
114
+ /**
115
+ * 返回已完成的笔记的 ID 列表
116
+ * @param fileContent 文件内容
117
+ * @returns 已完成的笔记的 ID 列表
118
+ */
119
+ function getDoneNotesID(fileContent: string): string[] {
120
+ const matches = fileContent.match(/- \[x\]\s\[(\d{4})\./g)
121
+ return matches
122
+ ? [...new Set(matches.map((match) => match.slice(-5, -1)))]
123
+ : []
124
+ }