Hexo+Gitee个人博客搭建折腾全纪录

本文最后更新于:2022年7月8日下午5点08分

好长时间没有打理博客,近期因为各种原因重新折腾了一番,做了升级与备份,将这一切记录在这,以后方便查找。

一、本地环境配置

首先需要安装Node.jsGit。需要注意的是,Node.js更新速度较快,hexo可能无法完全兼容最新版本的Node.js,所以推荐使用以前的版本,我这里使用12.14.1。

如果已经安装过新版本,后面遇到问题想退回老版本,下载老版本后直接到C:\Program Files\nodejs目录下覆盖原文件就可以了。

安装完成后确认安装成功

1
2
3
4
5
6
$ git --version
git version 2.34.1.windows.1
$ node -v
v12.14.1
$ npm -v
6.13.4

确认无误后安装Hexo,安装时间可能较长,耐心等待,不要有其他操作

1
$ npm install hexo-cli -g # -g代表全局安装

安装完成后查看版本,确认操作成功

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
$ hexo -v
INFO Validating config
hexo: 5.4.0
hexo-cli: 4.3.0
os: win32 10.0.19044
node: 12.14.1
v8: 7.7.299.13-node.16
uv: 1.33.1
zlib: 1.2.11
brotli: 1.0.7
ares: 1.15.0
modules: 72
nghttp2: 1.40.0
napi: 5
llhttp: 2.0.1
http_parser: 2.8.0
openssl: 1.1.1d
cldr: 35.1
icu: 64.2
tz: 2019c
unicode: 12.1

安装成功后,初始化hexo并部署到本地服务器。新建blog文件夹,在空白处右键Git Bash,执行以下操作

1
2
3
4
hexo init # 初始化hexo,生成相关文件
hexo cl # 清空已有的部署文件 hexo clean
hexo g # 生成网站文件 hexo generate
hexo s # 启动本地服务器 hexo server

在浏览器输入localhost:4000,就可以看到hexo网页了,默认主题是landscape,后续我们在修改它。本地配置基本完成,现在来配置云端。

二、Gitee配置

新建仓库及分支

注册登录Gitee,新建仓库,输入自己的仓库名称,路径会同步生成一样的名字,不用去改动它,其他保持默认即可。仓库建好之后会自动生成master分支,然后我们顺便再新建一个backup分支,待会用于备份源文件,即:

  • master:hexo生成的静态网页
  • backup:博客源文件

添加SSH公钥

首先在Git Bash中生成公钥,生成时会提示输入文件名、用户名和密码,默认文件名是id_rsaid_rsa.pub,建议修改一下,方便管理,后文会提到相关事项。用户名和密码一般不用添加,回车即可

1
ssh-keygen -t rsa -C "xxxxx@gmail.com" # 填写自己的邮箱

生成的公钥在~\.ssh\id_rsa.pub文件中,将其复制。在Gitee首页,进入右上角头像里的设置页面,进入安全设置中的SSH公钥,添加刚才复制的公钥。

开启Gitee Pages

在刚才创建的仓库首页,进入上边菜单栏服务中的Gitee Pages(现在开通要实名认证,我记得几年前我开通的时候还不要的)。开通之后应该能在个人主页头像下方看到一个网址,这就是你的博客了,不过我们还没有将内容推送到这里,现在是看不到任何东西的。

三、本地部署

在blog文件夹下找到_config.yml文件,用记事本打开,找到对应代码作如下修改

1
2
3
4
5
6
7
8
9
# URL
## Set your site url here. For example, if you use GitHub Page, set url as 'https://username.github.io/project'
url: https://xxxxxx.gitee.io # 你的网站的名字
......
deploy:
type: 'git' # 不保留''貌似也没事
repo: git@gitee.com:xxxxx/xxxxx.git # 你的仓库地址(非地址栏地址,而是克隆地址)
branch: master # 确保这里是master分支,因为我们hexo d -g都是默认到这个分支
message: 'web updata: {{now("YYYY-MM-DD HH/mm/ss")}}' # 自定义提交消息,这里是设置更新时间

在Gitee仓库页面下,点击克隆/下载按钮SSH选项下的复制,在blog文件夹右键Git Bash下执行以下命令,这里用SSH能避免用https时需要输入用户名和密码的麻烦

1
git clone git@gitee.com:xxxxx/xxxxx.git # 替换成你的仓库克隆地址

执行完成后,会产生一个以仓库名命名的文件夹,将其中的.git文件夹复制到blog目录下,然后删除原仓库文件夹。

我们顺便将默认分支改为backup,因为我们博客的部署(hexo d)是默认到master分支的,而我们执行克隆(git clone)时,都是克隆的默认分支,两者不会产生冲突,如果不改的话,后续较为麻烦,不如索性在这里就改好。

然后我们就可以hexo三连了

1
2
3
hexo cl # 清空已有的部署文件,不一定每次都要执行 hexo clean
hexo g # 生成网站文件 hexo generate
hexo d # 部署到gitee hexo server

可以合并命令

1
hexo cl && hexo d -g # hexo cl && hexo g -d 也行

再次打开你的博客网站应该就可以看到hexo默认主题landscape了,待会换个好看点的主题。先来看一下备份与恢复。

三、备份与恢复

推送到Gitee

假设你最开始搭建博客的电脑为A,那你需要在A电脑上的blog文件夹下执行以下,将本地博客与远程仓库关联

如果已经建立https的关联,可以通过这条命令断开关联(选用)

1
git remote rm origin

然后建立关联

1
git remote add origin git@gitee.com:xxxxx/xxxxx.git

查看.gitignore,通过文件名就知道,这个文件是用来忽略某些文件的推送的,通常是已经配置好的,一般如下

1
2
3
4
5
6
7
.DS_Store
Thumbs.db
db.json
*.log
node_modules/
public/
.deploy*/

然后执行如下git三连

1
2
3
git add . # 将文件添加到本地暂存区
git commit -m 'hexo backup fluid 1.8.13' # 将暂存区的内容提交到本地仓库 ''内自定义
git push origin backup # 将内容推送到远程仓库backup分支

首次git push时,要加上-u参数,让本地分支和远程分支建立关联,形成一个管道,之后 git push可以直接沿着管道到达关联的分支,无需在加-u参数了。

这样A电脑上的博客源文件就上传到Gitee的backup分支中了。

克隆/拉取到本地

如果你换了电脑B或者不小心把A电脑本地博客目录删除了,只要将远程仓库的backup分支克隆下来就好了

1
git clone git@gitee.com:xxxxx/xxxxx.git

因为我们设置了默认分支是backup所以这条命令会直接克隆backup分支,这样在执行正常的博客配置就可以继续使用了。

以后如果你在B电脑上编辑了一些文档,那么你需要执行上面给出的git add/commit/push命令,将远程仓库与B电脑保持一致,然后当你想在A电脑上编辑文档时,就需要先git pull一下,以确保A电脑与远程仓库保持一致,如果不git push,当以后git push时,会产生错误警告。git pull命令如下

1
git pull git@gitee.com:xxxxx/xxxxx.git

切记,更换电脑编辑时,最好每次都要先git push(A电脑)git pull(B电脑)一下。

四、多个SSH公钥配置

这一部分主要参考了这篇文章

当我们同时需要用ssh连接github、gitee和树莓派时,就需要对私钥、公钥进行管理。而且当你用多台电脑进行操作时,每台电脑都需要一个公钥,也要对网站进行相应的配置。

生成SSH key

首先,延续前文提到的概念,你的博客首次搭建在A电脑上,你在A电脑上已经按照前面说的生成了SSH公钥,文件名最好已经修改了,没有的话可以删掉然后重新操作一遍。以下在A电脑上操作

有两种办法生成自定义文件名的公钥,第一种是先输入命令,然后中间提示修改文件名时修改

1
ssh-keygen -t rsa -C "xxxxx@gmail.com"
1
2
Generating public/private rsa key pair.
Enter file in which to save the key (/home/zhao/.ssh/id_rsa): /home/zhao/.ssh/id_rsa_gitee

还有一种是直接用命令自定义文件名

1
ssh-keygen -t rsa -C "YOUR_EMAIL@YOUREMAIL.COM" -f ~/.ssh/id_rsa_gitee

总之,这都会在~/.ssh/目录下生成私钥id_rsa_gitee和公钥id_rsa_gitee.pub。同样的步骤可以生成多个私钥和公钥,比如id_rsa_githubid_rsa_github.pub。不要忘了在对应的网站添加SSH公钥,公钥的标题要确保能让自己记起来是A这台电脑或当前这个用户。

添加配置文件

~/.ssh/目录下新建config文件,添加以下内容(树莓派暂时不确定配置的对不对)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
# gitee
Host gitee.com # Host就是git仓库的域名或者IP
HostName gitee.com
PreferredAuthentications publickey
IdentityFile ~/.ssh/id_rsa_gitee
User git # User后面的值为访问的git ssh地址的@之前的部分

#github
#Host github.com
#HostName github.com
#PreferredAuthentications publickey
#IdentityFile ~/.ssh/id_rsa_github
#User git

#raspberrypi
#Host 192.168.1.200
#HostName 192.168.1.200
#PreferredAuthentications publickey
#IdentityFile ~/.ssh/id_rsa_pi
#User pi

然后测试一下配置

1
ssh -T git@github.com

看到有You've successfully authenticated这句话就代表成功了,如果报错试一下这条命令

1
chmod 600 ~/.ssh/config

现在,A电脑上就配置完成了。对于B电脑也是同样的操作,不过要注意,B电脑也要在网站上添加SSH公钥。

五、主题配置

fluid基础配置

fluid主题简洁不失美观,个人非常喜欢。

Hexo5.0.0版本以上,在blog文件夹下执行命令

1
npm install --save hexo-theme-fluid # 我安装的版本是1.8.13

然后修改刚才的_config.yml文件

1
2
3
theme: fluid  # 指定主题

language: zh-CN # 指定语言,会影响主题显示的语言,按需修改

其他配置在官方文档查看即可。我这里将我自己做出的修改列出。

自定义修改

特效

对于樱花飘落和雪花飘落二选一,在\blog\themes\fluid\source\js文件夹下新建比如skura.js文件,将对应代码复制到其中并保存。在blog文件夹下的_config.fluid.yml文件中做如下修改

1
custom_js: /js/sakura.js

刷新一下网页,或者Ctrl+C暂停本地服务,然后hexo s重启本地服务就可以看到特效了。

樱花飘落

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
// 樱花飘落bynote.cn
var stop, staticx; var img = new Image(); img.src = ""; function Sakura(x, y, s, r, fn) { this.x = x; this.y = y; this.s = s; this.r = r; this.fn = fn; }
Sakura.prototype.draw = function (cxt) {
cxt.save(); var xc = 40 * this.s / 4; cxt.translate(this.x, this.y); cxt.rotate(this.r); cxt.drawImage(img, 0, 0, 40 * this.s, 40 * this.s)
cxt.restore();
}
Sakura.prototype.update = function () { this.x = this.fn.x(this.x, this.y); this.y = this.fn.y(this.y, this.y); this.r = this.fn.r(this.r); if (this.x > window.innerWidth || this.x < 0 || this.y > window.innerHeight || this.y < 0) { this.r = getRandom('fnr'); if (Math.random() > 0.4) { this.x = getRandom('x'); this.y = 0; this.s = getRandom('s'); this.r = getRandom('r'); } else { this.x = window.innerWidth; this.y = getRandom('y'); this.s = getRandom('s'); this.r = getRandom('r'); } } }
SakuraList = function () { this.list = []; }
SakuraList.prototype.push = function (sakura) { this.list.push(sakura); }
SakuraList.prototype.update = function () { for (var i = 0, len = this.list.length; i < len; i++) { this.list[i].update(); } }
SakuraList.prototype.draw = function (cxt) { for (var i = 0, len = this.list.length; i < len; i++) { this.list[i].draw(cxt); } }
SakuraList.prototype.get = function (i) { return this.list[i]; }
SakuraList.prototype.size = function () { return this.list.length; }
function getRandom(option) {
var ret, random; switch (option) {
case 'x': ret = Math.random() * window.innerWidth; break; case 'y': ret = Math.random() * window.innerHeight; break; case 's': ret = Math.random(); break; case 'r': ret = Math.random() * 6; break; case 'fnx': random = -0.5 + Math.random() * 1; ret = function (x, y) { return x + 0.5 * random - 1.7; }; break; case 'fny': random = 1.5 + Math.random() * 0.7
ret = function (x, y) { return y + random; }; break; case 'fnr': random = Math.random() * 0.03; ret = function (r) { return r + random; }; break;
}
return ret;
}
function startSakura() {
requestAnimationFrame = window.requestAnimationFrame || window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || window.msRequestAnimationFrame || window.oRequestAnimationFrame; var canvas = document.createElement('canvas'), cxt; staticx = true; canvas.height = window.innerHeight; canvas.width = window.innerWidth; canvas.setAttribute('style', 'position: fixed;left: 0;top: 0;pointer-events: none;'); canvas.setAttribute('id', 'canvas_sakura'); document.getElementsByTagName('body')[0].appendChild(canvas); cxt = canvas.getContext('2d'); var sakuraList = new SakuraList(); for (var i = 0; i < 50; i++) { var sakura, randomX, randomY, randomS, randomR, randomFnx, randomFny; randomX = getRandom('x'); randomY = getRandom('y'); randomR = getRandom('r'); randomS = getRandom('s'); randomFnx = getRandom('fnx'); randomFny = getRandom('fny'); randomFnR = getRandom('fnr'); sakura = new Sakura(randomX, randomY, randomS, randomR, { x: randomFnx, y: randomFny, r: randomFnR }); sakura.draw(cxt); sakuraList.push(sakura); }
stop = requestAnimationFrame(function () { cxt.clearRect(0, 0, canvas.width, canvas.height); sakuraList.update(); sakuraList.draw(cxt); stop = requestAnimationFrame(arguments.callee); })
}
window.onresize = function () { var canvasSnow = document.getElementById('canvas_snow'); }
img.onload = function () { startSakura(); }
function stopp() { if (staticx) { var child = document.getElementById("canvas_sakura"); child.parentNode.removeChild(child); window.cancelAnimationFrame(stop); staticx = false; } else { startSakura(); } }

雪花飘落

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
/*样式二*/
/* 控制下雪 */
function snowFall(snow) {
/* 可配置属性 */
snow = snow || {};
this.maxFlake = snow.maxFlake || 200; /* 最多片数 */
this.flakeSize = snow.flakeSize || 10; /* 雪花形状 */
this.fallSpeed = snow.fallSpeed || 1; /* 坠落速度 */
}
/* 兼容写法 */
requestAnimationFrame = window.requestAnimationFrame ||
window.mozRequestAnimationFrame ||
window.webkitRequestAnimationFrame ||
window.msRequestAnimationFrame ||
window.oRequestAnimationFrame ||
function(callback) { setTimeout(callback, 1000 / 60); };

cancelAnimationFrame = window.cancelAnimationFrame ||
window.mozCancelAnimationFrame ||
window.webkitCancelAnimationFrame ||
window.msCancelAnimationFrame ||
window.oCancelAnimationFrame;
/* 开始下雪 */
snowFall.prototype.start = function(){
/* 创建画布 */
snowCanvas.apply(this);
/* 创建雪花形状 */
createFlakes.apply(this);
/* 画雪 */
drawSnow.apply(this)
}
/* 创建画布 */
function snowCanvas() {
/* 添加Dom结点 */
var snowcanvas = document.createElement("canvas");
snowcanvas.id = "snowfall";
snowcanvas.width = window.innerWidth;
snowcanvas.height = document.body.clientHeight;
snowcanvas.setAttribute("style", "position:absolute; top: 0; left: 0; z-index: 1; pointer-events: none;");
document.getElementsByTagName("body")[0].appendChild(snowcanvas);
this.canvas = snowcanvas;
this.ctx = snowcanvas.getContext("2d");
/* 窗口大小改变的处理 */
window.onresize = function() {
snowcanvas.width = window.innerWidth;
/* snowcanvas.height = window.innerHeight */
}
}
/* 雪运动对象 */
function flakeMove(canvasWidth, canvasHeight, flakeSize, fallSpeed) {
this.x = Math.floor(Math.random() * canvasWidth); /* x坐标 */
this.y = Math.floor(Math.random() * canvasHeight); /* y坐标 */
this.size = Math.random() * flakeSize + 2; /* 形状 */
this.maxSize = flakeSize; /* 最大形状 */
this.speed = Math.random() * 1 + fallSpeed; /* 坠落速度 */
this.fallSpeed = fallSpeed; /* 坠落速度 */
this.velY = this.speed; /* Y方向速度 */
this.velX = 0; /* X方向速度 */
this.stepSize = Math.random() / 30; /* 步长 */
this.step = 0 /* 步数 */
}
flakeMove.prototype.update = function() {
var x = this.x,
y = this.y;
/* 左右摆动(余弦) */
this.velX *= 0.98;
if (this.velY <= this.speed) {
this.velY = this.speed
}
this.velX += Math.cos(this.step += .05) * this.stepSize;

this.y += this.velY;
this.x += this.velX;
/* 飞出边界的处理 */
if (this.x >= canvas.width || this.x <= 0 || this.y >= canvas.height || this.y <= 0) {
this.reset(canvas.width, canvas.height)
}
};
/* 飞出边界-放置最顶端继续坠落 */
flakeMove.prototype.reset = function(width, height) {
this.x = Math.floor(Math.random() * width);
this.y = 0;
this.size = Math.random() * this.maxSize + 2;
this.speed = Math.random() * 1 + this.fallSpeed;
this.velY = this.speed;
this.velX = 0;
};
// 渲染雪花-随机形状(此处可修改雪花颜色!!!)
flakeMove.prototype.render = function(ctx) {
var snowFlake = ctx.createRadialGradient(this.x, this.y, 0, this.x, this.y, this.size);
snowFlake.addColorStop(0, "rgba(255, 255, 255, 0.9)"); /* 此处是雪花颜色,默认是白色 */
snowFlake.addColorStop(.5, "rgba(255, 255, 255, 0.5)"); /* 若要改为其他颜色,请自行查 */
snowFlake.addColorStop(1, "rgba(255, 255, 255, 0)"); /* 找16进制的RGB 颜色代码。 */
ctx.save();
ctx.fillStyle = snowFlake;
ctx.beginPath();
ctx.arc(this.x, this.y, this.size, 0, Math.PI * 2);
ctx.fill();
ctx.restore();
};
/* 创建雪花-定义形状 */
function createFlakes() {
var maxFlake = this.maxFlake,
flakes = this.flakes = [],
canvas = this.canvas;
for (var i = 0; i < maxFlake; i++) {
flakes.push(new flakeMove(canvas.width, canvas.height, this.flakeSize, this.fallSpeed))
}
}
/* 画雪 */
function drawSnow() {
var maxFlake = this.maxFlake,
flakes = this.flakes;
ctx = this.ctx, canvas = this.canvas, that = this;
/* 清空雪花 */
ctx.clearRect(0, 0, canvas.width, canvas.height);
for (var e = 0; e < maxFlake; e++) {
flakes[e].update();
flakes[e].render(ctx);
}
/* 一帧一帧的画 */
this.loop = requestAnimationFrame(function() {
drawSnow.apply(that);
});
}
/* 调用及控制方法 */
var snow = new snowFall({maxFlake:60});
snow.start();

鼠标特效

E:\blog\themes\fluid\layout\_partial\plugins新建mouse-click.ejs文件,将下面代码复制保存。然后在E:\blog\themes\fluid\layout\_partial\scripts.ejs文件中对应位置处做如下修改

1
2
3
4
5
<%- partial('_partial/plugins/typed.ejs') %>
<%- partial('_partial/plugins/math.ejs') %>
<%- partial('_partial/plugins/mermaid.ejs') %>
<%- partial('_partial/plugins/analytics.ejs') %>
<%- partial('_partial/plugins/mouse-click.ejs') %> # 添加的代码在这里

还需要修改_config.fluid.yml文件,修改位置其实任意,我实在加载进度条之前修改的

1
2
3
4
5
6
7
mouse_click:  # 鼠标点击动效
enable: true
style: love # 目前设置两个值:values(价值观词汇) | love(爱心)

# 加载进度条
# Progress bar when loading
progressbar:
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
<% if (theme.fun_features.mouse_click && theme.fun_features.mouse_click.enable) { %>
<% var style = theme.fun_features.mouse_click.style %>
<% if (style === 'values') { %>
<script type="text/javascript">
//定义获取词语下标
var a_idx = 0;
jQuery(document).ready(function ($) {
//点击body时触发事件
$("body").click(function (e) {
//需要显示的词语
var a = new Array("富强", "民主", "文明", "和谐", "自由", "平等", "公正", "法治", "爱国", "敬业", "诚信", "友善");
//设置词语给span标签
var $i = $("<span/>").text(a[a_idx]);
//下标等于原来下标+1 余 词语总数
a_idx = (a_idx + 1) % a.length;
//获取鼠标指针的位置,分别相对于文档的左和右边缘。
//获取x和y的指针坐标
var x = e.pageX, y = e.pageY;
//在鼠标的指针的位置给$i定义的span标签添加css样式
$i.css({
"z-index": 999,
"top": y - 20,
"left": x,
"position": "absolute",
"font-weight": "bold",
"color": rand_color()
});
// 随机颜色
function rand_color() {
return "rgb(" + ~~(255 * Math.random()) + "," + ~~(255 * Math.random()) + "," + ~~(255 * Math.random()) + ")"
}
//在body添加这个标签
$("body").append($i);
//animate() 方法执行 CSS 属性集的自定义动画。
//该方法通过CSS样式将元素从一个状态改变为另一个状态。CSS属性值是逐渐改变的,这样就可以创建动画效果。
//详情请看http://www.w3school.com.cn/jquery/effect_animate.asp
$i.animate({
//将原来的位置向上移动180
"top": y - 180,
"opacity": 0
//1500动画的速度
}, 1500, function () {
//时间到了自动删除
$i.remove();
});
});
})
;
</script>
<% } else if (style === 'love') { %>
<script>
!function (e, t, a) {
function r() {
for (var e = 0; e < s.length; e++) s[e].alpha <= 0 ? (t.body.removeChild(s[e].el), s.splice(e, 1)) : (s[e].y--, s[e].scale += .004, s[e].alpha -= .013, s[e].el.style.cssText = "left:" + s[e].x + "px;top:" + s[e].y + "px;opacity:" + s[e].alpha + ";transform:scale(" + s[e].scale + "," + s[e].scale + ") rotate(45deg);background:" + s[e].color + ";z-index:99999");
requestAnimationFrame(r)
}

function n() {
var t = "function" == typeof e.onclick && e.onclick;
e.onclick = function (e) {
t && t(), o(e)
}
}

function o(e) {
var a = t.createElement("div");
a.className = "heart", s.push({
el: a,
x: e.clientX - 5,
y: e.clientY - 5,
scale: 1,
alpha: 1,
color: c()
}), t.body.appendChild(a)
}

function i(e) {
var a = t.createElement("style");
a.type = "text/css";
try {
a.appendChild(t.createTextNode(e))
} catch (t) {
a.styleSheet.cssText = e
}
t.getElementsByTagName("head")[0].appendChild(a)
}

function c() {
return "rgb(" + ~~(255 * Math.random()) + "," + ~~(255 * Math.random()) + "," + ~~(255 * Math.random()) + ")"
}

var s = [];
e.requestAnimationFrame = e.requestAnimationFrame || e.webkitRequestAnimationFrame || e.mozRequestAnimationFrame || e.oRequestAnimationFrame || e.msRequestAnimationFrame || function (e) {
setTimeout(e, 1e3 / 60)
}, i(".heart{width: 10px;height: 10px;position: fixed;background: #f00;transform: rotate(45deg);-webkit-transform: rotate(45deg);-moz-transform: rotate(45deg);}.heart:after,.heart:before{content: '';width: inherit;height: inherit;background: inherit;border-radius: 50%;-webkit-border-radius: 50%;-moz-border-radius: 50%;position: fixed;}.heart:after{top: -5px;}.heart:before{left: -5px;}"), n(), r()
}(window, document);
</script>
<% } %>
<% } %>

文章版权信息

E:\blog\themes\fluid\layout\post.ejs文件中对应位置处添加如下代码

1
2
3
4
5
6
7
8
9
10
11
<% if(theme.post.copyright.enable && theme.post.copyright.content && page.copyright !== false) { %>
<p class="note note-warning">
<% if (typeof page.copyright === 'string' && page.copyright !== '') { %>
<%- page.copyright %>
<% } else { %>
<strong>本文作者: </strong><a href="<%- url_for() %>"><%- theme.about.name || config.author || config.title %></a><br>
<strong>本文链接: </strong><a href="<%- full_url_for(page.path) %>"><%- full_url_for(page.path) %></a><br>
<strong>版权声明: </strong><%- theme.post.copyright.content %>
<% } %>
</p>
<% } %>

这样在每篇文章的评论区上方就会生成版权信息,hexo自己生成的文章链接是根据permalink: :year/:month/:day/:title/这个格式来的,这样对于中文标题来说,解码之后的链接特别长,我们可以用hexo-abbrlink这个插件解决这个问题

安装插件,安装位置在E:\blog\node_modules\hexo-abbrlink,有需要可以自己调整代码

1
npm install hexo-abbrlink --save

然后修改hexo配置文件E:\blog\_config.yml

1
permalink: posts/:abbrlink/

保存后需要hexo cl一下缓存,然后重新部署会看到短标题了

页脚

网站版权、人数统计及运行时间

E:\blog\themes\fluid\layout\_partial中新建status.ejs文件,将以下代码复制粘贴保存。

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
<% if(theme.footer.statistics.enable) { %>
<div>
<br>© 2020-2022 <a href="https://babblingme.gitee.io" target="_blank" rel="nofollow noopener">Shixin</a>. All Rights Reserved.<br/>
<% if (theme.footer.statistics.pv_format) { %>
<!-- 不蒜子统计PV -->
<% var pv_texts = theme.footer.statistics.pv_format.split('{}') %>
<span id="busuanzi_container_site_pv" style="display: none">
<%- pv_texts[0] %><span id="busuanzi_value_site_pv"></span><%- pv_texts[1] %>
</span>
<% } %>
<% if (theme.footer.statistics.uv_format) { %>
<!-- 不蒜子统计UV -->
<% var uv_texts = theme.footer.statistics.uv_format.split('{}') %>
<span id="busuanzi_container_site_uv" style="display: none">
<%- uv_texts[0] %><span id="busuanzi_value_site_uv"></span><%- uv_texts[1] %>
</span>
<% } %>

<!-- 网站运行时间 -->
<span id="timeDate">载入天数...</span><span id="times">载入时分秒...</span>
<script>
var now = new Date();
function createtime() {
var grt= new Date("03/25/2020 00:00:00");//在此处修改你的建站时间
now.setTime(now.getTime()+250);
days = (now - grt ) / 1000 / 60 / 60 / 24; dnum = Math.floor(days);
hours = (now - grt ) / 1000 / 60 / 60 - (24 * dnum); hnum = Math.floor(hours);
if(String(hnum).length ==1 ){hnum = "0" + hnum;} minutes = (now - grt ) / 1000 /60 - (24 * 60 * dnum) - (60 * hnum);
mnum = Math.floor(minutes); if(String(mnum).length ==1 ){mnum = "0" + mnum;}
seconds = (now - grt ) / 1000 - (24 * 60 * 60 * dnum) - (60 * 60 * hnum) - (60 * mnum);
snum = Math.round(seconds); if(String(snum).length ==1 ){snum = "0" + snum;}
document.getElementById("timeDate").innerHTML = " 本站已安全运行 "+dnum+" 天 ";
document.getElementById("times").innerHTML = hnum + " 小时 " + mnum + " 分 " + snum + " 秒";
}
setInterval("createtime()",250);
</script>
</div>
<% } %>

在同级目录下footer.ejs文件对应位置处修改代码。

1
2
3
<%- partial('_partial/status.ejs') %> # 将statistics改为status
<%- partial('_partial/beian.ejs') %>
<% if(theme.web_analytics.cnzz) { %>

老版的fluid只有busuanzi.ejs一个文件,对应卜算子这一个人数统计插件,新版的是statistics.ejs文件,里面多了一个LeanCloud,但是我直接在这个文件中添加运行时间代码,显示比较奇怪,所以我干脆用之前的busuanzi.ejs文件,并添加代码后改名为status.ejs文件。我这里主要的修改是将三行信息变为两行,看起来更美观、简约一些(至少我这么觉得)。

关于页、友链页和自定义页

联系方式

卑微的小博主不好意思留其他联系方式,只留个邮箱,但是需要折腾一番,在E:\blog\themes\fluid\source\css下新建email.css文件,将以下代码添加进去,然后修改_config.fluid.yml文件(两处)

1
2
3
4
5
6
custom_css: /css/email.css
# ---------- #
icons:
#- { class: "iconfont icon-github-fill", link: "https://github.com", tip: "GitHub" }
- { class: "iconfont icon-youxiang", link: 'mailto:1837085828@qq.com' }
#- { class: "iconfont icon-wechat-fill", qrcode: "/img/favicon.png" }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
@font-face {font-family: "youxiang";
src: url('iconfont.eot?t=1586331751983'); /* IE9 */
src: url('iconfont.eot?t=1586331751983#iefix') format('embedded-opentype'), /* IE6-IE8 */
url('data:application/x-font-woff2;charset=utf-8;base64,d09GMgABAAAAAAK4AAsAAAAABqAAAAJrAAEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAHEIGVgCCcAqBKIEgATYCJAMICwYABCAFhG0HMhvIBRHVky9kPxKyk8z7w90Smwyh08Sb4PnXv/73SXJn8mh+GN3vCFRaPdixpMsJgM3N0DUqQrEk53k/fUc6VrgQfvjEvdM/CiiQ+c67HNfeNKkLMA6kgPbGKE6khDJuGLvAJdyHAN7Ek4u0tvcOxWKwlgkgc2fNmITNhTAYlrUIrpq9BtmKg1Xr1ENgS/h9+UyxWFA4Gqtu4PS2qTS+G2/n6aWepWN9gjAI6LADaCAXMCBDa1P9aISxNN4pQzLsq1rg3XiPx7cgezUi7K+zKn0BiIWiexK7klsVVAAYGA5FQOWo94MODAwacft5vD7yoiLg1rM4dfBp6fyAMxdELzuf7n/yLGrpudG7D19Im7QrYsqZcwdDJ55cn/6w9Nlrvzt+r59NDxdMa5ieuDfkTsjeRDAmb/X/1BfuAScWL8aL3C9S//efiEyPpBH/TB9VDxBA4NfEb80TAmr/Wh8DwMsbx2Ygg6GfB+w53AL/Styzy+hSy6yMKq7NKYrQ99DgjTfgwb5g3o7Vhd2YQ7A5MYPCTRw0llTSMHPh4EMpXFga4E0OHYd9CGExGjFeQDarAIRArkDhz31oAnlEGuYbOITzFS4CRcOb4RJypg/p4iXVwqgFHZg/JEvdO1ZRUfUdw1U1Z6Uh+418pigs41zM3rAjz7Hh/IRVxINnarCDx7BWgoMpo5UxihzbNPm6F42W2qBoYdSCDswfkqXu/f6iSp+/Y7iq5paOmvAb+UyjwzLOPcibqvfquJdnzk9YRTx4pgY7mIW1Ehz1szJaGeOIxLFNdi/fVzWur2lfdwDwxtIrsadr7BK6fkn3MAwA') format('woff2'),
url('iconfont.woff?t=1586331751983') format('woff'),
url('iconfont.ttf?t=1586331751983') format('truetype'), /* chrome, firefox, opera, Safari, Android, iOS 4.2+ */
url('iconfont.svg?t=1586331751983#iconfont') format('svg'); /* iOS 4.1- */
}

.icon-youxiang {
font-family: "youxiang" !important;
font-size: 16px;
font-style: normal;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
}

.icon-youxiang:before {
content: "\e65e";
}

自定义页面

然后在关于页自定义了一个网页,在E:\blog\source文件夹下新建文件夹rhapsody(名字自定义),在其中新建文件index.md文件,里面可以随意编辑。

然后在关于页的源文件E:\blog\source\about\index.md中添加如下html代码,就可以从关于页进入这个自己创建的页面了(搞博客还是要懂一些前端的知识的)

1
<a class="btn" href="/rhapsody/" title="不成文的思绪">遐想</a>

友链页

对于没人气的小博主,友链直接显示在导航栏感觉有点多余,因此对友链也做了类似自定义页面的处理

1
<a class="btn" href="/links/" title="优质博主up主出没">友链</a>

友链不用创建新文件及文件夹,只需在_config.fluid.yml中修改links: enable: true即可。要想在导航栏显示友链,再去导航栏相关配置区,将menu下的links那一行注释去掉即可。

结束语

费了近一天时间,终于写好了。博客需要折腾,生命亦如此。

参考文档

  1. https://www.cnblogs.com/bndong/p/9947729.html ↩︎
  2. https://zhuanlan.zhihu.com/p/299161193 ↩︎
  3. https://zhuanlan.zhihu.com/p/136552969 ↩︎
  4. https://www.yangbing.club/2019/06/29/save-hexo-source-post-with-git-branch/ ↩︎
  5. https://blog.csdn.net/weixin_43471926/article/details/109798811 ↩︎

本文作者: Shixin
本文链接: https://physxz.github.io/posts/10010/
版权声明: 本博客所有文章除特别声明外,均采用 CC BY-SA 4.0 协议 ,转载请注明出处!