config.mts 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  1. import "dotenv/config";
  2. import { fileURLToPath, URL } from "node:url";
  3. import { defineConfig } from "vitepress";
  4. import VueI18nPlugin from "@intlify/unplugin-vue-i18n/vite";
  5. import Icons from "unplugin-icons/vite";
  6. import { FileSystemIconLoader } from "unplugin-icons/loaders";
  7. import { SVG, cleanupSVG, parseColors, runSVGO } from "@iconify/tools";
  8. import {
  9. S3Client,
  10. ListObjectsCommand,
  11. GetObjectCommand,
  12. } from "@aws-sdk/client-s3";
  13. import fs from "node:fs";
  14. import path from "node:path";
  15. import { exec } from "child_process";
  16. import { buildSideBar } from "../utils/sideBar";
  17. const DOC_BASE_PATH = "pages";
  18. if (process.env.NODE_ENV === "production") {
  19. // 执行清理pages命令
  20. await new Promise((resolve, reject) =>
  21. exec(
  22. `git checkout -- ${DOC_BASE_PATH} & git clean -df ${DOC_BASE_PATH}`,
  23. (error, stdout, stderr) => {
  24. if (error) {
  25. console.error(`清理pages时出错: ${error.message}`);
  26. reject(stderr);
  27. }
  28. console.log(`清理pages结果: ${stdout}`);
  29. resolve(stdout);
  30. }
  31. )
  32. );
  33. }
  34. // 从S3构建pages里面的markdown文件
  35. const s3 = new S3Client({
  36. credentials: {
  37. accessKeyId: process.env.VITE_AWS_S3_ACCESS_KEY_ID!,
  38. secretAccessKey: process.env.VITE_AWS_S3_SECRET_ACCESS_KEY!,
  39. },
  40. region: process.env.VITE_AWS_S3_REGION!,
  41. });
  42. const command = new ListObjectsCommand({
  43. Bucket: process.env.VITE_DOCS_LIST_BUCKET,
  44. });
  45. const { Contents: contents } = await s3.send(command);
  46. await Promise.all(
  47. contents!.map((content) => {
  48. return new Promise(async (resolve, reject) => {
  49. try {
  50. const command = new GetObjectCommand({
  51. Bucket: process.env.VITE_DOCS_LIST_BUCKET,
  52. Key: content.Key,
  53. ResponseCacheControl: "no-cache",
  54. });
  55. const file = await s3.send(command);
  56. const writePath = path.join(DOC_BASE_PATH, content.Key!);
  57. const directory = path.dirname(writePath);
  58. fs.mkdirSync(directory, { recursive: true });
  59. fs.writeFileSync(writePath, await file.Body!.transformToString());
  60. resolve(content.Key);
  61. } catch (e) {
  62. reject(e);
  63. }
  64. fs.writeFile;
  65. });
  66. })
  67. );
  68. // 构建sideBar数据
  69. let { rootSideBar, zhHKSideBar } = buildSideBar(contents);
  70. import util from "util";
  71. console.log(
  72. util.inspect(rootSideBar, { showHidden: false, depth: null, colors: true }),
  73. util.inspect(zhHKSideBar, { showHidden: false, depth: null, colors: true })
  74. );
  75. // process.exit();
  76. // https://vitepress.dev/reference/site-config
  77. export default defineConfig({
  78. title: "CocoBlockly帮助文档",
  79. description: "Cococlass help documents",
  80. srcDir: DOC_BASE_PATH,
  81. // themeConfig: {
  82. // search: {
  83. // provider: "local",
  84. // },
  85. // },
  86. appearance: false,
  87. vite: {
  88. publicDir: "../public",
  89. envDir: "../",
  90. // optimizeDeps: {
  91. // include: ["vue-i18n"],
  92. // },
  93. ssr: {
  94. // SSG Vue-i18n workaround
  95. noExternal: [/vue-i18n/],
  96. },
  97. plugins: [
  98. VueI18nPlugin({
  99. /* options */
  100. // locale messages resource pre-compile option
  101. // include: resolve(dirname(fileURLToPath(import.meta.url)), './path/to/src/locales/**'),
  102. }),
  103. Icons({
  104. compiler: "vue3",
  105. customCollections: {
  106. "ccrbi-plain": FileSystemIconLoader(
  107. "assets/icons/plain",
  108. async (svgStr) => {
  109. const svg = new SVG(svgStr);
  110. cleanupSVG(svg);
  111. parseColors(svg, {
  112. defaultColor: "currentColor",
  113. callback: (attr, colorStr, color) => {
  114. // console.log('Color:', colorStr, color);
  115. // 普通图标
  116. return "currentColor";
  117. // Change black to 'currentColor'
  118. // const blackColor = stringToColor("black");
  119. // if (color && compareColors(color, blackColor!)) {
  120. // return "currentColor";
  121. // }
  122. // switch (color?.type) {
  123. // case "none":
  124. // case "current":
  125. // return color;
  126. // }
  127. // throw new Error(
  128. // `Unexpected color "${colorStr}" in attribute ${attr}`
  129. // );
  130. },
  131. });
  132. // Optimise, but do not change shapes because they are animated
  133. runSVGO(svg, {
  134. keepShapes: true,
  135. });
  136. return svg.toMinifiedString({});
  137. }
  138. ),
  139. "ccrbi-colored": FileSystemIconLoader(
  140. "assets/icons/colored",
  141. async (svgStr) => {
  142. const svg = new SVG(svgStr);
  143. cleanupSVG(svg);
  144. // Optimise, but do not change shapes because they are animated
  145. runSVGO(svg, {
  146. keepShapes: true,
  147. });
  148. return svg.toMinifiedString({});
  149. }
  150. ),
  151. },
  152. }),
  153. ],
  154. resolve: {
  155. alias: [
  156. {
  157. find: /^.*\/VPNavBar\.vue$/,
  158. replacement: fileURLToPath(
  159. new URL("../components/CustomNavBar.vue", import.meta.url)
  160. ),
  161. },
  162. {
  163. find: "@/",
  164. replacement: fileURLToPath(new URL("../", import.meta.url)),
  165. },
  166. ],
  167. },
  168. },
  169. locales: {
  170. root: {
  171. label: "简体中文",
  172. lang: "zh-CN",
  173. themeConfig: {
  174. logo: "/logo.png",
  175. siteTitle: false,
  176. // https://vitepress.dev/reference/default-theme-config
  177. // nav: [
  178. // { text: 'Home', link: '/' },
  179. // { text: 'Examples', link: '/markdown-examples' }
  180. // ],
  181. sidebar: [
  182. { text: "关于CocoBlockly X", link: "/docs" },
  183. { text: "常见问题解答", link: "/docs/常见问题解答" },
  184. {
  185. text: "开始使用CocoBlockly X",
  186. link: "/docs/start-using-cocoblockly-x",
  187. collapsed: true,
  188. items: [],
  189. },
  190. {
  191. text: "电子模块基本教学",
  192. collapsed: true,
  193. items: [{ text: "todo", link: "/docs" }],
  194. },
  195. ],
  196. // sidebar: rootSideBar,
  197. // socialLinks: [
  198. // { icon: 'github', link: 'https://github.com/vuejs/vitepress' }
  199. // ]
  200. },
  201. },
  202. "zh-HK": {
  203. label: "繁体中文",
  204. lang: "zh-HK",
  205. themeConfig: {
  206. logo: "/logo.png",
  207. siteTitle: false,
  208. // https://vitepress.dev/reference/default-theme-config
  209. // nav: [
  210. // { text: 'Home', link: '/' },
  211. // { text: 'Examples', link: '/markdown-examples' }
  212. // ],
  213. sidebar: [
  214. { text: "什麽是CocoBlockly X", link: "/zh-HK/docs" },
  215. { text: "常見問題解答", link: "/zh-HK/docs/faq" },
  216. {
  217. text: "開始使用CocoBlockly X",
  218. link: "/zh-HK/docs/start-using-cocoblockly-x",
  219. },
  220. {
  221. text: "電子模組基本教學",
  222. collapsed: true,
  223. items: [{ text: "todo", link: "/zh-HK/docs" }],
  224. },
  225. ],
  226. // socialLinks: [
  227. // { icon: 'github', link: 'https://github.com/vuejs/vitepress' }
  228. // ]
  229. },
  230. },
  231. },
  232. });