技术提醒
本文代码为撰写时的状态,仅供参考,不代表最终实现。
升级动力 {#升级动力}
Hexo 主题 {#hexo-主题}
2023年5月24日,我使用 Heo 风格的 Acrylic 主题,将个人网站升级为 Hexo 博客。但不久之后,Acrylic 主题停止维护,同时 Heo 风格的博客主题大行其道,令我感到「泯然于众Blog」。
于是在2024年1月9日,我选择了偶然之间瞥见的「千雪的咖啡厅」的上游主题 Stellar。并且惊喜地发现,Stellar 的作者 xaoxuu 也是知名主题 Volantis 的主要贡献者。朋友 Colsrch 负责管理 Volantis 主题交流群,我也曾于2020年加入了交流群。
「站」际关系 {#站际关系}
在更换 Stellar 主题之后,我于2024年1月29日开始积极更新博客,并添加了一些友链,藉以丰富自己的「站」际关系。由于几天后友链之一「微光档案」无法访问,我通过 Internet Archive 找到并加入了其交流群,了解到网站正在备案。自此,命运的齿轮开始转动,我与「微光档案」的站长 KazariEX 在后来结下了更深的联系。
博客美化 {#博客美化}
Stellar 主题的更新迭代速度很快,就如同我博客文章的更新速度一样。在这期间,我通过插入自定义 CSS 来更改样式。随着修改幅度越来越大,我编写了一系列 CSS 文件、Hexo 脚本,甚至还使用 JS 修改 DOM,为博客 Logo 注入动画元素。我的底线是「不修改主题源代码」,以避免拉取新版本时处理合并冲突,就算修改也是向主题提交 Bug 修复 PR。
KazariEX 见此情况,几次建议我使用 Nuxt.js 重构博客并手写主题,但我当时认为博客重心在于文章,因此并未考虑。
重构思考 {#重构思考}
我的想法是,当 Hexo 无法满足我对博客的要求时(可能是几年后),我会考虑用 Astro 重构博客。但目前一切尚好。
2024年6月1日, KazariEX 带领我 10 小时极速上手 Nuxt 重构个人主页。从某种意义上讲,此刻我由原生前端开发转变为一名现代前端开发者。我通过实战了解了 Nuxt 和 Vue 的基础知识,并体验到了现代前端开发的便捷。
8月3日,在 KazariEX 不知第几次建议下,我开始尝试使用 Nuxt Content 构建博客。8月11日,新博客上线。
功能开发 {#功能开发}
深色模式 {#深色模式}
我选择直接使用个人主页项目中的 `` ThemeToggle.vue ``
组件。这个组件的开发过程也很有意思。
我使用原生的 ``
localStorage ``
实现了颜色模式的持久化存储和自动加载,但后来觉得既然有了框架,使用 Pinia 来管理状态会更好。但随之而来遇到的问题是网页会在 `` mounted ``
钩子读取持久化存储的主题,如果主题与系统偏好不匹配时,刷新会导致短暂地使用系统偏好。
简而言之
如果系统偏好浅色模式,而博客选择了深色主题,刷新时会闪白。
不少网站都有这个问题,比如 GitHub Docs 、 VueUse 文档的 `` useColorMode ``
。最令我佩服的是 Astro 主题 Frosti ,我提出 Issue: 深色模式下刷新/切换页面时的屏幕闪烁 后,作者 EveSunMaple 持续跟进,最终解决了问题。
话说回来,我在解决页面刷新时的主题问题时遇到了困难,于是请求 KazariEX 的支援。我们发现 VitePress 没有这个问题,就一起研究它的源码,最终发现 VitePress 使用原生 ``
< script
>
``
预处理主题,避免了在 `` mounted ``
钩子中才切换主题的情况。我们参考这一思路解决了问题。后来 isYangs 引入了现成的 `` @nuxt/color-mode ``
模块,使代码更加优雅。
样式预处理 {#样式预处理}
在开发过程中,我选择基于自己修改的 Stellar 主题编写样式。通过 ``
tokei ./app
``
统计,博客前端排除空行后大约有4000行代码,其中大约一半是样式代码。
我选择了 SCSS 作为样式预处理器,主要使用 SCSS 的嵌套声明块、 ``
@mixin ``
和少量用于媒体查询宽度的变量。 绝大多数语法均兼容 CSS 3。
有群友提出使用原子化 CSS,有助于减少样式代码量。我认为原子化 CSS 能处理简易样式,但不适合设计导向或复杂样式,「风记星辰」的站长「宇」也支持我的观点。
删一行样式都会出问题的哪种?
当我能给样式代码写注释的时候,它已经不是普通的易于改动的代码了
Restent Ou
所以我用 UnoCSS
UnoCSS 能帮助处理这种近乎玄学的问题吗
我昨晚睡前还认真思考了一下要不要上 UnoCSS
最后觉得我的新博客既然注重样式,还是专门使用 `` <style> ``
好
Restent Ou
我提到原子化 CSS 是因为相比手写 CSS,其复杂度会相对降低
毕竟我现在遇到的问题是 CSS 本身特性带来的问题,而不是原不原子化影响对样式的约束/控制能力
宇
是的,其实现在专门设计的网站,几乎都不使用 CSS 框架了,自己写不仅节省代码,而且 HTML 也干净
因此,博客未来只可能会有限地引入原子化 CSS,例如仅对需要设置透明度、字体大小和颜色的元素应用原子化 CSS。
海量组件 {#海量组件}
先前我选择 Stellar 主题的原因之一,便是它「支持丰富的标签和动态数据组件」。没想到八个月前的回旋镖飞回到了自己头上,我望着博客 Markdown 文件里大量通过 Nunjucks 语法 ``
{ %
tag_name
attr1
attr2:val2
%
}
``
调用的组件,思考如何用 Nuxt Content 和 MDC 语法编写用于替代的 Vue 组件。随后,我便开始了漫长的组件编写之旅。
MDC 语法相比传统 Markdown 语法,能够更方便地使用 Vue 组件并传递参数。受限于一些组件(如聊天记录、时间线)的渲染方式,我甚至使用非标准方式获取 VNode 子节点中的默认插槽内容的文本值,写出了 `` as any ``
类型断言。不过总之,各种组件能 以比较优雅的方式调用 了。
使用Chat组件 使用Tab组件 Timeline组件简化源码
blog-using-nuxt.md mdc
自动换行 复制
::
chat
</span>
</span>
<span class="line" line="2">
<span emptylineplaceholder="true">
</span>
</span>
<span class="line" line="3">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
{.}
</span>
</span>
<span class="line" line="4">
<span emptylineplaceholder="true">
</span>
</span>
<span class="line" line="5">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
这是我说的内容,显示在右侧
</span>
</span>
<span class="line" line="6">
<span emptylineplaceholder="true">
</span>
</span>
<span class="line" line="7">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
{用户1}
</span>
</span>
<span class="line" line="8">
<span emptylineplaceholder="true">
</span>
</span>
<span class="line" line="9">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
这是用户1的回复\
</span>
</span>
<span class="line" line="10">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
可以这样换行
</span>
</span>
<span class="line" line="11">
<span emptylineplaceholder="true">
</span>
</span>
<span class="line" line="12">
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
::
</span>
</span>
</code>
<!--]--> </pre> <!---->
``
评论接入 {#评论接入}
2024年6月28日,我为上一版博客接入了 Twikoo 评论系统;在8月11日新博客上线当天,我通过简易代码重新接入了 Twikoo。
后来,KazariEX 提出使用 ``
useScriptTag ()
``
引入 Twikoo,我觉得这就是一种抽象。从积极的插入脚本变为懒加载之后,无论是在外部使用动态 key 调用组件,抑或是在组件内监听 `` path ``
,都会在 SSG 环境下异常,只得如此在 ``
onMounted ()
``
中手动重新调用,倒是显得不太优雅了。
旧代码 新代码
Comment.vue > script setup ts
自动换行 复制
function
initTwikoo
()
{
</span>
</span>
<span class="line" line="2">
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
window
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
twikoo
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
?.
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
init
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
?.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
{
</span>
</span>
<span class="line" line="3">
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
envId
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
:
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
appConfig
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
twikoo
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
envId
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
,
</span>
</span>
<span class="line" line="4">
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
el
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
:
</span>
<span style="--shiki-default:#40A02B;--shiki-dark:#98C379">
'#twikoo'
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
,
</span>
</span>
<span class="line" line="5">
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
}
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
</span>
<span class="line" line="6">
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
}
</span>
</span>
<span class="line" line="7">
<span style="--shiki-default:#9CA0B0;--shiki-default-font-style:italic;--shiki-dark:#7F848E;--shiki-dark-font-style:italic">
// 从其他页面路由至文章页面时,通过 onLoaded 事件初始化,onMounted 不起作用
</span>
</span>
<span class="line" line="8">
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
useScriptTag
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
appConfig
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
twikoo
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
js
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
,
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
()
</span>
<span style="--shiki-default:#179299;--shiki-dark:#C678DD">
=>
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
initTwikoo
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
()
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
,
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
{
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
defer
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
:
</span>
<span style="--shiki-default:#FE640B;--shiki-dark:#D19A66">
true
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
}
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
</span>
<span class="line" line="9">
<span style="--shiki-default:#9CA0B0;--shiki-default-font-style:italic;--shiki-dark:#7F848E;--shiki-dark-font-style:italic">
// 在文章页面之间路由时不会触发 onLoaded 事件,需要手动初始化
</span>
</span>
<span class="line" line="10">
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
onMounted
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
()
</span>
<span style="--shiki-default:#179299;--shiki-dark:#C678DD">
=>
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
initTwikoo
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
())
</span>
</span>
</code>
<!--]--> </pre> <!---->
``
页面布局 {#页面布局}
我参考了 Stellar 主题的左中右三栏布局。其中,右侧栏可以通过元数据或 Front Matter 中的 ``
aside ``
数组来自定义组件。
Stellar 主题使用 Grid 布局,而我选择借鉴 VitePress 默认主题的 Flex 布局实现方式,这样不仅兼容性更好,还能在隐藏右侧栏时让正文占据此空间。
blog-using-nuxt.md yaml
自动换行 复制
---
</span>
</span>
<span class="line" line="2">
<span style="--shiki-default:#9CA0B0;--shiki-default-font-style:italic;--shiki-dark:#7F848E;--shiki-dark-font-style:italic">
# hideAside: true # 隐藏右侧栏
</span>
</span>
<span class="line" line="3">
<span style="--shiki-default:#1E66F5;--shiki-dark:#E06C75">
aside
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
:
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
[
</span>
<span style="--shiki-default:#40A02B;--shiki-dark:#98C379">
toc
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
,
</span>
<span style="--shiki-default:#40A02B;--shiki-dark:#98C379">
github_card
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
]
</span>
</span>
<span class="line" line="4">
<span style="--shiki-default:#EA76CB;--shiki-dark:#ABB2BF">
---
</span>
</span>
</code>
<!--]--> </pre> <!---->
在处理侧边栏显隐的逻辑时,我起初复用了
``
aside ``
,其值为 ``
false ``
时会隐藏侧边栏,但后来我发现这种写法意义不够明确,不利于其他人识别,尤其是代码中 ``
route .
meta
.
aside
!==
false
``
的判断经常会被简化为 ``
route .
meta
.
aside
``
。在 isYangs 的建议下,我将隐藏侧边栏的逻辑移到了 ``
hideAside ``
中,并修改了类型定义。
types/index.ts diff
自动换行 复制
declare module 'vue-router' {
</span>
</span>
<span class="line" line="2">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
interface RouteMeta {
</span>
</span>
<span class="line" line="3">
<span style="--shiki-default:#7C7F93;--shiki-dark:#E06C75">
-
</span>
<span style="--shiki-default:#D20F39;--shiki-dark:#E06C75">
aside: string[] | false
</span>
</span>
<span class="line" line="4">
<span style="--shiki-default:#7C7F93;--shiki-dark:#98C379">
+
</span>
<span style="--shiki-default:#40A02B;--shiki-dark:#98C379">
hideAside?: boolean
</span>
</span>
<span class="line" line="5">
<span style="--shiki-default:#7C7F93;--shiki-dark:#98C379">
+
</span>
<span style="--shiki-default:#40A02B;--shiki-dark:#98C379">
aside: string[]
</span>
</span>
<span class="line" line="6">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
}
</span>
</span>
<span class="line" line="7">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
}
</span>
</span>
</code>
<!--]--> </pre> <!---->
带动效应 {#带动效应}
汲取灵感 {#汲取灵感}
我使用 Nuxt 和 Nuxt Content 开发博客时遇到了不少问题,但好在发现了不少使用 Nuxt Content 的小伙伴,他们的文章提供了不少灵感。
- 从Hexo到Nuxt,正式升级为动态博客!- Z次元 :这篇文章来自我的友链,他的博客间接鼓励了我使用 Nuxt Content。
- 博客从 Ghost 迁移至 Nuxt Content - HADB.ME :我开发博客时,他在 Wakatime 排行榜上每日代码时长、语言排名与我极其相似,再访问他的个人主页,原来他也在使用 Nuxt Content 构建博客,非常巧合。我参考了他的
`` atom.xml ``
生成逻辑。 - 使用 Nuxt 重构我的博客 - Restent's Notebook :刚开始参考了他的目录逻辑,但后来通过递归使用复用组件完成了逻辑,并做出了伴随滚动切换高亮项的目录。
开源反哺 {#开源反哺}
我在使用 Nuxt Content 开发博客时,为其他人提供了一些样式思路。博客交流群里也有不少小伙伴受到鼓舞,使用现代的前端框架重构博客。
兼容优化 {#兼容优化}
样式特性 {#样式特性}
当然,也少不了我们的老伙计------安卓上的夸克浏览器。
如果一个现代的前端网站能在夸克APP上正常运行,那么它的兼容性 十分 良好;如果它将兼容夸克APP,那么我认为这实在 万分 厉害。
它除了不支持五年前的 ``
prefers-color-scheme ``
特性外,我还在测试环节留意到诸如 `` dvh ``
的动态视口单位、 ``
: has
()
``
函数式伪类等新特性也不受支持。对于动态视口单位,我通过多个声明语句渐进覆盖,「风记星辰」的站长「宇」推荐我阅读 2023年应该重新定义的CSS习惯(一) ,使用更优雅的媒体查询 ``
@supports ``
写法。对于 ``
: has
()
``
函数式伪类,我通过 Vue 的动态 class 绑定解决。
代码越少,越使用新特性。越使用新特性,兼容性越差。兼容性越差,越需要更多的兼容性代码。所以代码越少,代码就越多。
SSG != SSR {#ssg-ssr}
Nuxt 之 SSR 开发体验,善也。然限于 Vercel 之速,以 SSG 部署尔。
上文样式之扰仅为滥觞,代码之兼容性亦困我于囹圄。幸有 KazariEX 助我。
其一为 ``
Object .
groupBy
()
``
------本地体验大善,上云部署错误,观之为 Node.js 20 环境,果断升至 22。此后正常部署,本地体验正常。 喜笑颜开之时,掏出夸克APP。不喜。 ``
import {
group
}
from
'radash'
``
解决之。
其二为 ``
Array .
prototype
.
sort
()
``
。此为修改方法而非复制方法,会破坏响应性。忽闻 ``
Array .
prototype
.
toSorted
()
``
,大喜,如获至宝,遂部署。 复掏出夸克APP。不喜。 ``
import {
alphabetical
}
from
'radash'
``
解决之。
其三为 ``
Array .
prototype
.
at
()
``
。开启 Chrome 88,发觉友链页面先正常加载后报错。此为 SSG 与 CSR 水合不匹配 ,经查,友链页面提取域名之函数使用 ``
Array .
prototype
.
at
()
``
,而 Chrome 92 之前无此方法。遂改之。
其四为 SSG 下 HTML 属性 (attrs) 响应性之缺失。初,使用 URL 之查询参数 `` ?page= ``
以定页面,点击翻页功能如常,然刷新时渲染不符预期。测试 attrs 时,发现其难以具备响应性,遂弃查询参数,以 Vue Router 之动态路由匹配 `` /page/:page ``
替之。
其五为 SSG 下侧栏组件篡位。侧栏目录 (TOC) 组件在换文之时,会悄然以新文数据覆于旧文缓存中,偶以极其复杂之触发条件令其原形毕现,遂通过侧栏动态组件之 key 断其生命周期,乃造新 TOC 于新文而非旧 TOC 监听 URL 路径以篡其数据也。
SSG 之 Bug 种种,实扰人耳。
Bug 排查 {#bug-排查}
发现了 Bug,当然要快速定位问题。还好我使用的是 Vercel,它提供了历史部署版本的快照,通过访问旧部署,即可快速定位到 Commit 和出现问题的代码。
然后只需要修复就行。至少听起来十分轻松。
嗯,我还会留下注释,防止自己后续优化代码时又改回可能导致 Bug 的写法。
在这个项目里,没有一行注释是无辜的。
列表负载 {#列表负载}
我善于观察异常的苗头,因此发现了许多容易被忽视的问题。例如,我的博客上线第一天,相比于旧博客的带宽消耗翻了番。经过排查,首页、归档页面加载时,会附带一个约 2MB 的 `` payload.json ``
,细看发现竟含有我全站的文章元数据和正文内容。
KazariEX 猜我没学过数据库,因为我犯了类似 ``
select *
``
的错误。而我也确实没学过,毕竟这学期才会有《数据库原理与应用》课程。
不过凭直觉,应该在查询结果中排除 `` body ``
字段,后来我改成只选择指定字段,需要什么就加什么。虽然 Nuxt 相比于 Vue 不用手动书写 Vue 内置函数、组件、工具函数、组合式函数的导入语句,但在 Nuxt Content 的编程模式下,我仍需审慎组合、使用它提供的 API。
SEO {#seo}
2024年8月24日是我博客历史上灰暗的一天,因为我的必应和谷歌搜索流量几乎置零。
我的所有页面被错误的打上了 `` noindex ``
标记,这将阻止搜索引擎索引我的博客;我的 `` sitemap.xml ``
为空,这相当于博客告诉搜索引擎自己没有内容。
在我起初引入 `` @nuxtjs/seo@2.0.0-rc.16 ``
时一切检查正常,但在一次依赖更新后,改变悄然发生。有用户在 `` @nuxtjs/seo@2.0.0-rc.18 ``
报告了空 sitemap 问题 ,开发者讲虽然有一个"轻微的 (slightly) 破坏性 (breaking) 更改",但造成空 sitemap 另有其因,应当升级依赖。
来自未来的文章 {#来自未来的文章}
YAML 会以 UTC 时区解析 Front Matter 中的时间,刚更新的文章常常显示来自"8小时后",但我希望以本地时区解析。刚开始修改了 ``
getPostDate ()
``
函数,但语义化的 ~~``
< time
datetime
=
"{{ date }}"
>
``
~~ ``
< time
:datetime
=
"date"
>
``
标签依旧不符合规范,因此编写了这个 Nitro 插件用于修正时区。
server/plugins/fixPostTimezone.ts ts
自动换行 复制
import
{
getTimezoneOffset
}
from
'date-fns-tz'
</span>
</span>
<span class="line" line="2">
<span style="--shiki-default:#8839EF;--shiki-dark:#C678DD">
import
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
blogConfig
</span>
<span style="--shiki-default:#8839EF;--shiki-dark:#C678DD">
from
</span>
<span style="--shiki-default:#40A02B;--shiki-dark:#98C379">
'~~/blog.config'
</span>
</span>
<span class="line" line="3">
<span emptylineplaceholder="true">
</span>
</span>
<span class="line" line="4">
<span style="--shiki-default:#8839EF;--shiki-dark:#C678DD">
const
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
timezoneOffset
</span>
<span style="--shiki-default:#179299;--shiki-dark:#56B6C2">
=
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
getTimezoneOffset
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
blogConfig
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
timezone
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
</span>
<span class="line" line="5">
<span emptylineplaceholder="true">
</span>
</span>
<span class="line" line="6">
<span style="--shiki-default:#8839EF;--shiki-dark:#C678DD">
export
</span>
<span style="--shiki-default:#8839EF;--shiki-dark:#C678DD">
default
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
defineNitroPlugin
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#E64553;--shiki-default-font-style:italic;--shiki-dark:#E06C75;--shiki-dark-font-style:italic">
nitroApp
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
)
</span>
<span style="--shiki-default:#179299;--shiki-dark:#C678DD">
=>
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
{
</span>
</span>
<span class="line" line="7">
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
nitroApp
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
hooks
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
hook
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#40A02B;--shiki-dark:#98C379">
'content:file:afterParse'
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
,
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#E64553;--shiki-default-font-style:italic;--shiki-dark:#E06C75;--shiki-dark-font-style:italic">
file
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
)
</span>
<span style="--shiki-default:#179299;--shiki-dark:#C678DD">
=>
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
{
</span>
</span>
<span class="line" line="8">
<span style="--shiki-default:#8839EF;--shiki-dark:#C678DD">
if
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
file
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
date
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
{
</span>
</span>
<span class="line" line="9">
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
file
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
date
</span>
<span style="--shiki-default:#179299;--shiki-dark:#56B6C2">
=
</span>
<span style="--shiki-default:#8839EF;--shiki-default-font-weight:bold;--shiki-dark:#C678DD;--shiki-dark-font-weight:inherit">
new
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
Date
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#8839EF;--shiki-default-font-weight:bold;--shiki-dark:#C678DD;--shiki-dark-font-weight:inherit">
new
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
Date
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
file
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
date
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
getTime
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
()
</span>
<span style="--shiki-default:#179299;--shiki-dark:#56B6C2">
-
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
timezoneOffset
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
</span>
<span class="line" line="10">
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
}
</span>
</span>
<span class="line" line="11">
<span style="--shiki-default:#8839EF;--shiki-dark:#C678DD">
if
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
file
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
updated
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
{
</span>
</span>
<span class="line" line="12">
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
file
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
updated
</span>
<span style="--shiki-default:#179299;--shiki-dark:#56B6C2">
=
</span>
<span style="--shiki-default:#8839EF;--shiki-default-font-weight:bold;--shiki-dark:#C678DD;--shiki-dark-font-weight:inherit">
new
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
Date
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#8839EF;--shiki-default-font-weight:bold;--shiki-dark:#C678DD;--shiki-dark-font-weight:inherit">
new
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
Date
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
file
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
updated
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
getTime
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
()
</span>
<span style="--shiki-default:#179299;--shiki-dark:#56B6C2">
-
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
timezoneOffset
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
</span>
<span class="line" line="13">
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
}
</span>
</span>
<span class="line" line="14">
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
}
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
</span>
<span class="line" line="15">
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
}
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
</span>
</code>
<!--]--> </pre> <!---->
Nuxt 虽然是热更新,但我甚至重启开发环境都不能确保其生效,必须手动清除缓存。
颜色追求 {#颜色追求}
行内代码 `` inline-code ``
的需求
其 ``
border-color ``
和 ``
background-color ``
应跟随文本颜色变化,而不是单调的灰色。
起初我想写为 `` rgba(currentcolor, 0.2) ``
,但一方面它不符合 SCSS 的函数参数个数要求,另一方面 CSS 仅支持自定义属性 `` rgba(var(--color), 0.2) ``
而不支持在其中使用 `` currentColor ``
。故只能使用 `` #{"rgba(var(--color), 0.2)"} ``
的形式绕过 SCSS 检查,还要通过 JS 计算出当前颜色并写入 `` --color ``
中,并且在切换深色模式后还需要重新计算值,可谓十分不优雅。
旧代码 新代码
ProseCodeInline.vue vue
自动换行 复制
<
script
setup
lang
=
"ts"
>
</span>
</span>
<span class="line" line="2">
<span style="--shiki-default:#8839EF;--shiki-dark:#C678DD">
const
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
code
</span>
<span style="--shiki-default:#179299;--shiki-dark:#56B6C2">
=
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
ref
</span>
<span style="--shiki-default:#04A5E5;--shiki-dark:#ABB2BF">
<
</span>
<span style="--shiki-default:#DF8E1D;--shiki-default-font-style:italic;--shiki-dark:#E5C07B;--shiki-dark-font-style:inherit">
HTMLElement
</span>
<span style="--shiki-default:#04A5E5;--shiki-dark:#ABB2BF">
>
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
()
</span>
</span>
<span class="line" line="3">
<span style="--shiki-default:#8839EF;--shiki-dark:#C678DD">
const
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
color
</span>
<span style="--shiki-default:#179299;--shiki-dark:#56B6C2">
=
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
ref
</span>
<span style="--shiki-default:#04A5E5;--shiki-dark:#ABB2BF">
<
</span>
<span style="--shiki-default:#8839EF;--shiki-dark:#E5C07B">
string
</span>
<span style="--shiki-default:#04A5E5;--shiki-dark:#ABB2BF">
>
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
()
</span>
</span>
<span class="line" line="4">
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
onMounted
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
()
</span>
<span style="--shiki-default:#179299;--shiki-dark:#C678DD">
=>
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
{
</span>
</span>
<span class="line" line="5">
<span style="--shiki-default:#8839EF;--shiki-dark:#C678DD">
const
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
rgbColorStr
</span>
<span style="--shiki-default:#179299;--shiki-dark:#56B6C2">
=
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
getComputedStyle
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
code
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
value
</span>
<span style="--shiki-default:#179299;--shiki-dark:#56B6C2">
!
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
color
</span>
</span>
<span class="line" line="6">
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
color
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
value
</span>
<span style="--shiki-default:#179299;--shiki-dark:#56B6C2">
=
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E5C07B">
rgbColorStr
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
match
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#EA76CB;--shiki-dark:#E06C75">
/\d
</span>
<span style="--shiki-default:#179299;--shiki-dark:#D19A66">
+
</span>
<span style="--shiki-default:#EA76CB;--shiki-dark:#E06C75">
/
</span>
<span style="--shiki-default:#8839EF;--shiki-dark:#C678DD">
g
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
<span style="--shiki-default:#179299;--shiki-dark:#56B6C2">
!
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
map
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
Number
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
.
</span>
<span style="--shiki-default:#1E66F5;--shiki-default-font-style:italic;--shiki-dark:#61AFEF;--shiki-dark-font-style:inherit">
join
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
(
</span>
<span style="--shiki-default:#40A02B;--shiki-dark:#98C379">
','
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
</span>
<span class="line" line="7">
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
}
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
)
</span>
</span>
<span class="line" line="8">
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
</
</span>
<span style="--shiki-default:#1E66F5;--shiki-dark:#E06C75">
script
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
>
</span>
</span>
<span class="line" line="9">
<span emptylineplaceholder="true">
</span>
</span>
<span class="line" line="10">
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
<
</span>
<span style="--shiki-default:#1E66F5;--shiki-dark:#E06C75">
template
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
>
</span>
</span>
<span class="line" line="11">
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
<
</span>
<span style="--shiki-default:#1E66F5;--shiki-dark:#E06C75">
code
</span>
<span style="--shiki-default:#DF8E1D;--shiki-dark:#D19A66">
ref
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
=
</span>
<span style="--shiki-default:#40A02B;--shiki-dark:#98C379">
"code"
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
:
</span>
<span style="--shiki-default:#DF8E1D;--shiki-dark:#D19A66">
style
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
=
</span>
<span style="--shiki-default:#40A02B;--shiki-dark:#ABB2BF">
"
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
{
</span>
<span style="--shiki-default:#40A02B;--shiki-dark:#98C379">
'--color'
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
:
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#E06C75">
color
</span>
<span style="--shiki-default:#7C7F93;--shiki-dark:#ABB2BF">
}
</span>
<span style="--shiki-default:#40A02B;--shiki-dark:#ABB2BF">
"
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
><
</span>
<span style="--shiki-default:#1E66F5;--shiki-dark:#E06C75">
slot
</span>
<span style="--shiki-default:#4C4F69;--shiki-dark:#FFFFFF">
/
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
></
</span>
<span style="--shiki-default:#1E66F5;--shiki-dark:#E06C75">
code
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
>
</span>
</span>
<span class="line" line="12">
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
</
</span>
<span style="--shiki-default:#1E66F5;--shiki-dark:#E06C75">
template
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
>
</span>
</span>
<span class="line" line="13">
<span emptylineplaceholder="true">
</span>
</span>
<span class="line" line="14">
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
<
</span>
<span style="--shiki-default:#1E66F5;--shiki-dark:#E06C75">
style
</span>
<span style="--shiki-default:#DF8E1D;--shiki-dark:#D19A66">
scoped
</span>
<span style="--shiki-default:#DF8E1D;--shiki-dark:#D19A66">
lang
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
=
</span>
<span style="--shiki-default:#40A02B;--shiki-dark:#98C379">
"scss"
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
>
</span>
</span>
<span class="line" line="15">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
code {
</span>
</span>
<span class="line" line="16">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
padding: 0.1em 0.3em;
</span>
</span>
<span class="line" line="17">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
border: 1px solid #{"rgba(var(--color), 0.2)"};
</span>
</span>
<span class="line" line="18">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
border-radius: 4px;
</span>
</span>
<span class="line" line="19">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
background-color: #{"rgba(var(--color), 0.05)"};
</span>
</span>
<span class="line" line="20">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
font-size: 0.875em;
</span>
</span>
<span class="line" line="21">
<span style="--shiki-default:#4C4F69;--shiki-dark:#ABB2BF">
}
</span>
</span>
<span class="line" line="22">
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
</
</span>
<span style="--shiki-default:#1E66F5;--shiki-dark:#E06C75">
style
</span>
<span style="--shiki-default:#179299;--shiki-dark:#ABB2BF">
>
</span>
</span>
</code>
<!--]--> </pre> <button type="button" class="toggle-btn" aria-label="展开代码块" data-v-09e64c36=""> <span class="iconify i-ph:caret-double-up-bold toggle-icon is-collapsed" aria-hidden="true" style="" data-v-09e64c36=""> </span> <span class="toggle-tip" data-v-09e64c36=""> 23 行 </span> </button>
``
后来,KazariEX 提到了 CSS
`` color-mix() ``
函数,我认为他又一次给出了真正优雅的答案。至于兼容性?不担心的,通过上文「多个声明语句渐进覆盖」的方法即可解决。
阶段总结 {#阶段总结}
经过约一个月/200小时的开发,我的博客由 Hexo 转为基于 Nuxt Content 开发的项目。项目选型、主要功能完成后,我选择不断优化已有代码,修复 Bug、优化性能、语义化和兼容性。我认为,开发就是不断的做加法和减法,在项目起步阶段,减法比加法更重要,这样才有利于后续功能的扩展,而不是面对「独木桥上架起的二层小洋房」无从下手。
另外,我认为,博客的精髓在于内容,技术与框架不过是其内在的骨架,样式也仅为锦上添花的一笔, 真正优秀的博客并不会因为架构、样式的陈旧或简单而蒙尘 。我仍将持续投入内容创作与功能建构,以期形成自己的独特表达。
`` ``