butterfly 主题适合娱乐宣传,纯粹记笔记还是用其它框架,这里使用 VitePress 搭建

安装

  1. 项目初始化(一路回车)
1
npx vitepress init
  1. 安装 vitepress 及依赖
1
npm add -D vitepress
  1. 启动
1
npm run docs:dev

项目初始结构(当前目录./中搭建站点生成的文件结构)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
.
├── node_modules
│ ├── .bin
│ ├── ...
│ ├── ...
│ ├── ...
│ └── vue
├── .vitepress
│ ├── cache
│ └── config.mts # 配置文件
├── api-examples.md # 示例API文档
├── index.md # 文档首页
├── markdown-examples.md # 示例文档
├── public # 静态资源,默认网页路径为"/"
├── package-lock.json
└── package.json

基本设置

文档首页

修改(./index.md

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
---
# https://vitepress.dev/reference/default-theme-home-page
layout: home

hero:
name: "Xdogの文档站"
text: "学习笔记整理"
tagline: 结束即是开始
actions:
- theme: brand
text: 查看文档
link: /butterfly/
- theme: alt
text: 关于
link: /
image:
src: /logo.svg
alt: VitePress

features:
- title: 📚 阅读
details: 书是人类进步的阶梯,书籍是知识的摇篮。
- title: 📝 写作
details: 写作是最好的学习方式。
- title: 🎉 编码
details: 程序员的能力与他写过的代码是正比的。
---

我的配置文件

.vitepress\config.mts

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
import { defineConfig, type DefaultTheme } from 'vitepress'

// https://vitepress.dev/reference/site-config
export default defineConfig({
head: [['link', { rel: 'icon', href: '/favicon.svg' }]], // 站点图标设置
title: "Xdogの文档站",
description: "结束即是开始",
outDir: 'public', // 设置输出目录为 public 文件夹
markdown: {
image: {
// 默认禁用图片懒加载
lazyLoading: true
}
},
// 忽略死链检测,防止编译失败
ignoreDeadLinks: true,
themeConfig: {
// https://vitepress.dev/reference/default-theme-config
nav: [
{ text: '主页', link: '/' },
{ text: 'Butterfly美化', link: '/butterfly/' },
{ text: '作者博客', link: 'https://xdog.top/' }
],
// 搜索
search: {
provider: 'local',
options: {
translations: {
button: {
buttonText: '搜索文档',
buttonAriaLabel: '搜索文档'
},
modal: {
noResultsText: '无法找到相关结果',
resetButtonTitle: '清除查询条件',
footer: {
selectText: '选择',
navigateText: '切换',
closeText: '关闭',
}
}
}
}
},
// 文档脚
docFooter: {
prev: '上一页',
next: '下一页'
},
outline: {
label: '目录'
},
lastUpdated: {
text: '最后更新于',
formatOptions: {
dateStyle: 'short',
timeStyle: 'medium'
}
},
returnToTopLabel: '回到顶部',
sidebarMenuLabel: '菜单',
darkModeSwitchLabel: '主题',
lightModeSwitchTitle: '切换到浅色模式',
darkModeSwitchTitle: '切换到深色模式',
// 页脚
footer: {
message: '<img style="display:inline-block;position:relative;height:16px;top:3px;padding-right:3px;" src="https://vitepress.dev/vitepress-logo-mini.svg" alt="ICP" data-ll-status="loading" class="entered loading"><a style="text-decoration-line: none;" href="https://vitepress.dev/zh/">VitePress</a> | <img style="display:inline-block;position:relative;top:3px;padding-right:3px;" src="https://www.upyun.com/static/favicon-16x16.png" alt="ICP" data-ll-status="loading" class="entered loading"><a style="text-decoration-line: none;" href="https://www.upyun.com/?utm_source=lianmeng&utm_medium=referral">又拍云</a>',
copyright: 'Copyright © 2023 Xdog'
},

sidebar: {
'/butterfly/': { base: '/butterfly/', items: sidebarButterfly() },
},

// sidebar: {'/blog': auto_set_sidebar('blog')},

socialLinks: [
{ icon: 'github', link: 'https://github.com/vuejs/vitepress' }
]
}
})

function sidebarButterfly(): DefaultTheme.SidebarItem[] {
return [
{
text: '简介',
collapsed: false,
items: [
{ text: '介绍', link: 'index' },
{ text: '准备', link: 'prepare' }
]
},
{
text: '盒子调整',
collapsed: false,
items: [
{ text: '整体布局', link: '盒子调整-整体布局' },
{ text: '导航栏', link: '盒子调整-导航栏' },
{ text: '首页', link: '盒子调整-首页' },
{ text: '分类和标签页', link: '盒子调整-分类和标签页' },
{ text: '关于页', link: '盒子调整-关于页' },
{ text: '底部快捷导航', link: '盒子调整-底部快捷导航' }
]
},
{
text: '功能拓展',
collapsed: false,
items: [
{ text: '侧边栏小工具', link: '功能拓展-侧边栏小工具' },
{ text: '首页动态分类条', link: '功能拓展-首页动态分类条' },
{ text: '最新文章标识', link: '功能拓展-最新文章标识' },
{ text: 'Volantis标签移植', link: '功能拓展-Volantis标签移植' },
{ text: '51la统计&灵雀统计', link: '功能拓展-51la统计&灵雀统计' },
{ text: '图标', link: '功能拓展-图标' },
{ text: '杂项', link: '功能拓展-杂项' }
]
},
{
text: '美化特效',
collapsed: false,
items: [
{ text: '背景', link: '美化特效-背景' },
{ text: '加载动画', link: '美化特效-加载动画' },
{ text: '文章页', link: '美化特效-文章页' },
{ text: '彩色标签', link: '美化特效-彩色标签' },
{ text: '页脚', link: '美化特效-页脚' },
{ text: '杂项', link: '美化特效-杂项' }
]
},
// { text: '配置和 API 参考', base: '/reference/', link: 'site-config' }
]
}

自定义主题

通过创建一个.vitepress/theme/index.js.vitepress/theme/index.ts文件来启用自定义主题,启用后默认主题失效

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
.
├── .vitepress
│ ├── cache
│ ├─ theme
│ │ ├── index.ts # 主题入口
│ │ └── style.css # 主题样式
│ └── config.mts # 配置文件
├── api-examples.md # 示例API文档
├── index.md # 文档首页
├── markdown-examples.md # 示例文档
├── node_modules
│ ├── .bin
│ ├── ...
│ ├── ...
│ ├── ...
│ └── vue
├── package-lock.json
└── package.json

自定义CSS

.vitepress/theme/index.ts

1
2
3
4
5
// .vitepress/theme/index.ts
import DefaultTheme from 'vitepress/theme'
import './custom.css'

export default DefaultTheme

.vitepress/theme/custom.css

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
/**
* Customize default theme styling by overriding CSS variables:
* https://github.com/vuejs/vitepress/blob/main/src/client/theme-default/styles/vars.css
*/

/**
* Colors
*
* Each colors have exact same color scale system with 3 levels of solid
* colors with different brightness, and 1 soft color.
*
* - `XXX-1`: The most solid color used mainly for colored text. It must
* satisfy the contrast ratio against when used on top of `XXX-soft`.
*
* - `XXX-2`: The color used mainly for hover state of the button.
*
* - `XXX-3`: The color for solid background, such as bg color of the button.
* It must satisfy the contrast ratio with pure white (#ffffff) text on
* top of it.
*
* - `XXX-soft`: The color used for subtle background such as custom container
* or badges. It must satisfy the contrast ratio when putting `XXX-1` colors
* on top of it.
*
* The soft color must be semi transparent alpha channel. This is crucial
* because it allows adding multiple "soft" colors on top of each other
* to create a accent, such as when having inline code block inside
* custom containers.
*
* - `default`: The color used purely for subtle indication without any
* special meanings attched to it such as bg color for menu hover state.
*
* - `brand`: Used for primary brand colors, such as link text, button with
* brand theme, etc.
*
* - `tip`: Used to indicate useful information. The default theme uses the
* brand color for this by default.
*
* - `warning`: Used to indicate warning to the users. Used in custom
* container, badges, etc.
*
* - `danger`: Used to show error, or dangerous message to the users. Used
* in custom container, badges, etc.
* -------------------------------------------------------------------------- */

:root {
--vp-c-default-1: var(--vp-c-gray-1);
--vp-c-default-2: var(--vp-c-gray-2);
--vp-c-default-3: var(--vp-c-gray-3);
--vp-c-default-soft: var(--vp-c-gray-soft);

--vp-c-brand-1: var(--vp-c-indigo-1);
--vp-c-brand-2: var(--vp-c-indigo-2);
--vp-c-brand-3: var(--vp-c-indigo-3);
--vp-c-brand-soft: var(--vp-c-indigo-soft);

--vp-c-tip-1: var(--vp-c-brand-1);
--vp-c-tip-2: var(--vp-c-brand-2);
--vp-c-tip-3: var(--vp-c-brand-3);
--vp-c-tip-soft: var(--vp-c-brand-soft);

--vp-c-warning-1: var(--vp-c-yellow-1);
--vp-c-warning-2: var(--vp-c-yellow-2);
--vp-c-warning-3: var(--vp-c-yellow-3);
--vp-c-warning-soft: var(--vp-c-yellow-soft);

--vp-c-danger-1: var(--vp-c-red-1);
--vp-c-danger-2: var(--vp-c-red-2);
--vp-c-danger-3: var(--vp-c-red-3);
--vp-c-danger-soft: var(--vp-c-red-soft);
}

/**
* Component: Button
* -------------------------------------------------------------------------- */

:root {
--vp-button-brand-border: transparent;
--vp-button-brand-text: var(--vp-c-white);
--vp-button-brand-bg: var(--vp-c-brand-3);
--vp-button-brand-hover-border: transparent;
--vp-button-brand-hover-text: var(--vp-c-white);
--vp-button-brand-hover-bg: var(--vp-c-brand-2);
--vp-button-brand-active-border: transparent;
--vp-button-brand-active-text: var(--vp-c-white);
--vp-button-brand-active-bg: var(--vp-c-brand-1);
}

/**
* Component: Home
* -------------------------------------------------------------------------- */

:root {
--vp-home-hero-name-color: transparent;
--vp-home-hero-name-background: -webkit-linear-gradient(
120deg,
#bd34fe 30%,
#41d1ff
);

--vp-home-hero-image-background-image: linear-gradient(
-45deg,
#bd34fe 50%,
#47caff 50%
);
--vp-home-hero-image-filter: blur(44px);
}

@media (min-width: 640px) {
:root {
--vp-home-hero-image-filter: blur(56px);
}
}

@media (min-width: 960px) {
:root {
--vp-home-hero-image-filter: blur(68px);
}
}

/**
* Component: Custom Block
* -------------------------------------------------------------------------- */

:root {
--vp-custom-block-tip-border: transparent;
--vp-custom-block-tip-text: var(--vp-c-text-1);
--vp-custom-block-tip-bg: var(--vp-c-brand-soft);
--vp-custom-block-tip-code-bg: var(--vp-c-brand-soft);
}

/**
* Component: Algolia
* -------------------------------------------------------------------------- */

.DocSearch {
--docsearch-primary-color: var(--vp-c-brand-1) !important;
}

自动生成侧边栏

自动设置侧边栏原理

  1. 项目根目录新建文件scripts/auto_set_sidebar.mjs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
import path from "node:path";
import fs from "node:fs";

// 文件根目录
const DIR_PATH = path.resolve();
// 白名单,过滤不是文章的文件和文件夹
const WHITE_LIST = [
"index.md",
".vitepress",
"node_modules",
".idea",
"assets",
];

// 判断是否是文件夹
const isDirectory = (path) => fs.lstatSync(path).isDirectory();

// 取差值
const intersections = (arr1, arr2) =>
Array.from(new Set(arr1.filter((item) => !new Set(arr2).has(item))));

// 把方法导出直接使用
function getList(params, path1, pathname) {
// 存放结果
const res = [];
// 开始遍历params
for (let file in params) {
// 拼接目录
const dir = path.join(path1, params[file]);
// 判断是否是文件夹
const isDir = isDirectory(dir);
if (isDir) {
// 如果是文件夹,读取之后作为下一次递归参数
const files = fs.readdirSync(dir);
res.push({
text: params[file],
collapsible: true,
items: getList(files, dir, `${pathname}/${params[file]}`),
});
} else {
// 获取名字
const name = path.basename(params[file]);
// 排除非 md 文件
const suffix = path.extname(params[file]);
if (suffix !== ".md") {
continue;
}
res.push({
text: name,
link: `${pathname}/${name}`,
});
}
}
// 对name做一下处理,把后缀删除
res.map((item) => {
item.text = item.text.replace(/\.md$/, "");
});
return res;
}

export const set_sidebar = (pathname) => {
// 获取pathname的路径
const dirPath = path.join(DIR_PATH, pathname);
// 读取pathname下的所有文件或者文件夹
const files = fs.readdirSync(dirPath);
// 过滤掉
const items = intersections(files, WHITE_LIST);
// getList 函数后面会讲到
return getList(items, dirPath, pathname);
};
  1. 导入配置文件``.vitepress\config.mts`
1
import {auto_set_sidebar} from '../scripts/auto_set_sidebar.mjs'
  1. 若想让目录blog下的文档自动生成侧边栏,首先目录下要有首页index.md,否则404,然后修改主题配置的的 sidebar 配置项
1
sidebar: {'/blog': auto_set_sidebar('blog')}, 

这个auto_set_sidebar传递的参数是相对于根路径的文件夹路径,返回的是每个文件夹中文件的名称和链接

部署

免费部署服务里Gitlab+Vercel大概是目前最优解,为什么不用Github,懂得都懂

.gitignore

1
2
3
4
5
6
7
8
node_modules/
.DS_Store
.vitepress/dist
.vitepress/dist-ssr
cache
.cache
.temp
*.local

首次推送到 Gitlab 仓库

1
2
3
4
git init
git add .
git commit -m "$(date)"
git push -u -f https://gitlab.com/pbloods/docs.git main

GitLab默认不允许强制推送,后台关闭即可,详见

后续使用shell脚本一键推送,项目根目录新建文件push.ps1

1
2
3
git add ./
git commit -m "$(date)"
git push

需要推送时执行以下命令即可

1
./push.ps1

Gitlab CI

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
# The Docker image that will be used to build your app
image: node:lts
# Functions that should be executed before the build script is run
before_script:
- npm install
pages:
script:
- npm run docs:build
artifacts:
paths:
# The folder that contains the files to be exposed at the Page URL
- .vitepress/dist # 注意Windows目录分隔符和Linux是相反的
publish: .vitepress/dist
rules:
# This ensures that only pushes to the default branch will trigger
# a pages deploy
- if: $CI_COMMIT_REF_NAME == $CI_DEFAULT_BRANCH

导入Vercel的设置
构建命令:npm run docs:build
输出目录(初始化时选择/时):.vitepress/dist