Quartz 的构建
Tldr
关于官方的构建教程,可以参考:quartz 官方文档,如果看不习惯英文,也可以参考 quartz 文档中文版,但需要注意的是中文文档有些过时了,有很多新的 Features 都没有翻译
在这里主要记录一下我的迁移与配置过程。
首先,我们需要到 官方仓库 中,点击 Use this template 进行仓库的生成

- 如果你没有自己的个人博客网站,那么可以新建的名称为
github-username.github.io,这里github-username是你的 Github 账号名称; - 否则你可以自定义生成一个仓库,例如
quartz,但请牢记仓库的名称
我在这里为了方便,直接使用了第一种方式,因为这个仓库名不需要额外配置 github pages
然后,我们生成了一个仓库后,将其克隆到本地,并安装本地包,尝试在本地构建 运行:
git clone <your-repo-url> blogs
cd blogs
pnpm i
npx quartz build --serve --concurrency 4随后,就能够在 localhost:8080 打开构建完成的网页了,生成的所有 html 文件都在 public 文件夹中
注意
你可以发现这里我的包管理器选择了
pnpm而不是官方默认的npm,你也可以选择 Bun 替换 Nodejs 来作为包管理器(以及运行时)
Quartz 的持续更新
在上一步中,我们将 git 仓库的 remote 设置为了你自己的博客仓库,别名为 origin,这里,如果你想追随 Quartz 的更新的话,可以按照以下方式进行:
-
设置另一个
remote如下:git remote add public git@github.com:jackyzha0/quartz.git -
每天运行
git fetch public获取上游的更新(牢记之前更新的commit,可以通过标签记录,或者 Ob 直接记录) -
将未更新的
commit,使用 VSCode中的版本管理 插件进行cherry pick操作即可,commit信息也可一同保留提交
Important
关于将
content私有化后,通过 git submodule 管理并同步更新,可以参考 通过 submodule 发布博客
Quartz 基础配置
基础配置需要在 quartz.config.ts 中进行配置,需要注意的只有三个部分:
analyticsbaseUrldefaultDateType
Analytics
网站的分析工具,这里推荐使用微软出品的 clarity,配置简单,而且开源免费,分析的感觉也很有意思:

配置只需要填写 {provider: 'clarity', projectId: '<your-clarity-id-code>' } 即可,如下所示:
analytics: {
provider: "clarity",
projectId: "xxxxxx"
},baseUrl
这里尤为重要,主要是填写博客的地址,由于我在 Quartz 的构建 中提到,我使用的是第一种方法,所以在这里直接填写我的博客地址即可(如果是第二种方法,那么需要写成 github-username.github.io/repo-name)
Important
如果你是用自定义域名,在购买域名的地方做好 DNS 解析后,直接填写 pages 中的 custom domain 即可,这里的
baseUrl也使用自定义的域名,否则内部链接路径会出错。注意,允许支持二级域名,例如
blog.virgiling.wiki,只需要在购买域名的地方加一个解析:blog.virgiling.wiki解析到virgiling.github.io即可
defaultDateType
这里我填写的为 defaultDateType: "modified",,其实就是为了在 Meta 数据中显示最近一次更新的时间,我会在 frontmatter 中的 lastmod 进行填写,需要注意的是,原来的创建时间 createAt 现在被更改为 date 了
其他设置
其余的设置暂时都与默认一致,但 Plugin.Latex({ renderEngine: "katex" }), 中,我发现其实数学还支持使用 typst,但由于之前的笔记都是用的 Latex 写的,不想再做兼容了,就不切换了吧
Note
在 2024-12-23 时,typst暂时还无法使用,原因是上游的包 还没有更新,会导致构建失败(问题在于locate这个函数在typst中已经被弃用了)
布局配置
布局配置较为简单,这里主要说明一下评论的增加
我们在这里增加评论组件:
// components shared across all pages
export const sharedPageComponents: SharedLayout = {
head: Component.Head(),
header: [],
afterBody: [],
footer: Component.Comments({
provider: "giscus",
options: {
repo: "xxxxxxx",
repoId: "xxxxxxx",
category: "Announcements",
categoryId: "xxxxxxx",
themeUrl: "https://giscus.app/themes/",
lightTheme: "noborder_light"
}
})
}我们可以在 Giscus 直接填写一些信息,然后网站就会给出配置信息,我们按照名称对应一一填写即可。
Note
其实官网上的配置很多,可以参考 Features List
样式的修改
这里样式的修改主要集中在三个部分:
- 字体的显示
- 主题的颜色
- 一些自定义的设置
字体
Hint
也可以考虑使用
cdn的方式引入,例如 引入霞鹜文楷
对于字体而言,我使用的和 Obsidian 字体 中所述的一致,我们需要在 quartz/style/custom.scss 中进行配置:
@font-face {
font-family: "LXGWWenKaiScreen";
font-style: normal;
font-weight: normal;
font-display: swap;
src: url("/static/fonts/LXGWWenKaiScreen.woff2") format("woff2");
}
@font-face {
font-family: "Monaco";
font-style: normal;
font-weight: normal;
font-display: swap;
src: url("/static/fonts/Monaco.woff2") format("woff2");
}
@font-face {
font-family: "Biro";
font-style: normal;
font-weight: normal;
font-display: swap;
src: url("/static/fonts/Biro_Script.woff2") format("woff2");
}
@font-face {
font-family: "LXGWWenKaiScreen";
font-style: normal;
font-weight: normal;
font-display: swap;
src: url("/static/fonts/Bookerly.woff2") format("woff2");
unicode-range: U+00-7F;
}注意高亮的行,我们通过 unicode-range 将两个字体何为一个,都称为 LXGWWenKaiScreen,这样,我们在 quartz.config.ts 中的写法为:
theme: {
fontOrigin: "local",
cdnCaching: false,
typography: {
header: "LXGWWenKaiScreen",
body: "LXGWWenKaiScreen",
code: "Monaco",
},
随后,我们需要将下载的字体(最好是压缩后的,woff2 的格式就比较小),放在 quartz/static/fonts 中,也就是与 字体配置 中填写的 url 一致,但存放静态文件的路径必须存于 quartz/static 目录下,否则无法被构建到最终的网页文件夹 public 中
主题颜色
暂时只适配了亮色主题,并把暗色主题的切换删除了,强制亮色显示(,
现在也适配了暗色,和原生的暗色配置一样,没有更改
颜色的配置的含义为:
light: 页面背景 lightgray: 边框 gray: 图形链接,较重的边框 darkgray: 正文 dark: 标题文本和图标 secondary: 链接颜色,当前 graph 节点 tertiary: 悬停状态和访问的 graph 节点 highlight: 内部链接背景,高亮显示的文本,高亮显示的代码行
自定义设置
我在 custom.scss 中还添加了一些行内代码的高亮,接着,让 Meta 数据(也就是创建,修改,阅读时间)的字体进行更改:
:root {
//加粗字体、代码块高亮色
--custom-highlight: #8b2e2e;
}
:not(pre)>code {
background-color: var(--lightgray);
border-radius: var(--border-radius);
padding: 0 .3rem;
margin: 0 0.2em;
border: 1px solid var(--gray);
color: var(--custom-highlight);
font-weight: 500;
}
.content-meta {
font-family: 'Biro';
}但这个字体只适用于英文的,于是,我们还需要修改这个组件的语言显示:
if (fileData.dates) {
segments.push(
<span>✏️ <Date date={fileData.dates.created} locale='en-US' /></span>
)
segments.push(
<span>🔧 <Date date={fileData.dates.modified} locale='en-US' /></span>
)
}一些插件
Important
这边的插件几乎都不再使用了(灯箱还在使用),但是你可以在
commit历史中找到,例如在 b3e2c94 之前的提交,下面的内容应当是能找到原版的
这里使用了 8Cats & Me 中开发的一些组件和样式,主要是:
FloatButtonCode-Block
浮动按键
这个按键只会在桌面端显示在右下角,主要是为了更好的阅读体验,包含的内容是:
- 回到顶部
- 回到底部
- 全局图谱
- 查看快捷键
但在这之后,还引入了 kbd (键盘按键)的样式,可以参考以下文件:
- FloatingButtons.tsx
- floatingButtons.inline.ts
- floatingButtons.scss
- custom.scss
- keyboard.csss
自定义代码块
这里使用了 expressive-code ,能够做到与 Obsidian-shiki-plugin 一致的显示,具有更强的表达能力,这在上面的代码块中已经能够发现了,配置也十分简单
首先,我们需要引入包:
pnpm add rehype-expressive-code @expressive-code/plugin-collapsible-sections @expressive-code/plugin-line-numbers然后参照文件 syntax.ts 进行修改即可
Important
需要注意的是,如果使用了
expressive-code,我们需要更改quartz.config.ts中的构建顺序如下:Plugin.Latex({ renderEngine: "katex" }), Plugin.SyntaxHighlighting(),主要是需要保证语法高亮需要在解析数学公式的后面,否则,对于行间公式,就会被解析为
math块,无法被Latex解析器正确处理
灯箱
更新
在 pr#2074 中有另外一个更好的实现(主要是样式不错,但是我感觉代码写的比较一般,不过鉴于这个我也不太需要配置什么,所以就直接复制粘贴过来了
原版的 quartz 对于图片较为匮乏,没办法放大查看,感觉其实不算很友好,但我在 pr#1480 中发现了一个灯箱,然后修改了一下就拿来用了,主要代码可以参考 quartz/plugins/transformers/lightbox.ts,我们只需要在 quartz.config.ts 中的 transform 中最后引入这个插件即可
自定义插件
Quartz 的客制化其实做的很不错,写一个自己的插件也不难,基本上就是遍历生成的 dom 节点树,然后把需要修改的那部分拿出来自定义即可
最简单的插件就可以参考上面的 灯箱 以及 自定义代码块,都是很好的例子。
网站地图 (sitemap.xml)
sitemap.xml 的生成主要在 quartz/plugins/emitters/contentIndex.ts 中,但其实原版的生成有一个问题,这里的 lastmod 字段其实是错误的,代码只使用了 date 字段来生成 lastmod,可以更改如下:
function generateSiteMap(cfg: GlobalConfiguration, idx: ContentIndex): string {
const base = cfg.baseUrl ?? ""
const createURLEntry = (slug: SimpleSlug, content: ContentDetails): string => `<url>
<loc>https://${joinSegments(base, encodeURI(slug))}</loc>
${(content.lastmod && `<lastmod>${content.lastmod.toISOString()}</lastmod>`) ||
(content.date && `<lastmod>${content.date.toISOString()}</lastmod>`)}
</url>`
const urls = Array.from(idx)
.map(([slug, content]) => createURLEntry(simplifySlug(slug), content))
.join("")
return `<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.sitemaps.org/schemas/sitemap/0.9 http://www.sitemaps.org/schemas/sitemap/0.9/sitemap.xsd"> ${urls} </urlset>`
}Question
目前不知道为什么,我新生成的网站地图是正确的,但是提交给
bing解析之后,发现解析和索引的 URL 依然是我老博客的 URL( 😭