个人博客

个人博客: https://www.crystalblog.xyz/

备用地址: https://wang-qz.gitee.io/crystal-blog/

上篇hexo博客搭建及主题优化(一)
下篇hexo博客搭建及主题优化(三)

主题优化二

21. 网站log设置

主题目录下的_config.yml 配置文件中:

# 配置网站favicon和网站LOGO
## 本地
#favicon: /favicon.png
#logo: /medias/logo.png
# 此处我用的CDN,也可以使用本地文件
favicon: https://cdn.jsdelivr.net/gh/guixinchn/image/blog/favicon.png
logo: https://cdn.jsdelivr.net/gh/guixinchn/image/blog/logo.png

图片资源在主题目录的\themes\hexo-theme-matery\source\medias下面 , 也可以使用外链图片.

22. 网站动态标题行

实现方法,引入 js 文件,在主题文件下的 /source/js/ 下新建 funnyTitle.js,增加以下代码:

var OriginTitle = document.title;
 var titleTime;
 document.addEventListener('visibilitychange', function () { 
   
     if (document.hidden) { 
   
         $('[rel="icon"]').attr('href', "https://cdn.jsdelivr.net/gh/guixinchn/image/blog/favicon.png");
         document.title = '我相信你还会回来的!';
         clearTimeout(titleTime);
     }
     else { 
   
         $('[rel="icon"]').attr('href', "https://cdn.jsdelivr.net/gh/guixinchn/image/blog/favicon.png");
         document.title = '哈哈,我就知道!' + OriginTitle;
         titleTime = setTimeout(function () { 
   
             document.title = OriginTitle;
         }, 2000);
     }
 });

然后在主题目录下的/layout/layout.ejs 引入

<script src="<%- theme.jsDelivr.url %><%- url_for('/js/funnyTitle.js') %>"></script>

23. about页面添加个人简历

打开主题目录下的 /layout/about.ejs 文件,新增如下代码:

<div class="card-content article-card-content">
   <div class="title center-align" data-aos="zoom-in-up">
      <i class="fa fa-address-book"></i>&nbsp;&nbsp;<%- __('个人简历') %>
   </div>
   <div id="articleContent" data-aos="fade-up">
      <%- page.content %>
   </div>
</div>

在主题目录下的/layout/about.ejs 文件里面关于下面代码中的profile相关信息从主题的 _config.yml 配置文件中获取.

<div class="profile center-align">
    <div class="avatar">
        <img src="<%- theme.jsDelivr.url %><%- url_for(theme.profile.avatar) %>" alt="<%- config.author %>" class="circle responsive-img avatar-img">
    </div>
    <div class="author">
        <div class="post-statis hide-on-large-only" data-aos="zoom-in-right">
            <%- partial('_partial/post-statis') %>
        </div>
        <div class="title"><%- config.author %></div>
        <div class="career"><%- theme.profile.career %></div>
        <div class="social-link hide-on-large-only" data-aos="zoom-in-left">
            <%- partial('_partial/social-link') %>
        </div>
    </div>
</div>

主题目录下的 _config.yml 配置文件中profile信息配置, 可以修改…

# profile in about page, including avatars, career, and personal introductions.
# 在”关于”页面中配置个人信息,包括头像、职业和个人介绍.
profile:
  avatar: /medias/avatar.jpg
  career: Software Engineer
  introduction: If you wish to succeed, you should use persistence as your good friend, experience as your reference, prudence as your brother and hope as your sentry.

24. 404页面

原来的主题没有404页面,首先在主题目录下的/source/目录下新建一个404.md,内容如下:

---
title: 404
date: 2017-07-19 16:41:10
type: "404"
layout: "404"
description: "你访问的页面被外星人叼走了 :("
---

然后在主题目录下新建一个/layout/404.ejs文件,内容如下:

<style type="text/css"> /* don't remove. */ .about-cover { 
      height: 90.2vh; } </style>
<div class="bg-cover pd-header about-cover">
    <div class="container">
        <div class="row">
            <div class="col s10 offset-s1 m8 offset-m2 l8 offset-l2">
                <div class="brand">
                    <div class="title center-align">
                        404
                    </div>
                    <div class="description center-align">
                        <%= page.description %>
                    </div>
                </div>
            </div>
        </div>
    </div>
</div>

<% if (theme.banner.enable) { %>
    <script> // 每天切换 banner 图. Switch banner image every day. var bannerUrl = "<%- theme.jsDelivr.url %><%- url_for('/medias/banner/') %>" + new Date().getDay() + '.jpg'; $('.bg-cover').css('background-image', 'url(' + bannerUrl + ')'); </script>
<% } else { %>
    <script> $('.bg-cover').css('background-image', 'url(<%- theme.jsDelivr.url %><%- url_for('/medias/banner/0.jpg') %>)'); </script>
<% } %>

25. 网站页脚修改

根据自己需要修改主题目录下的/layout/_partial/footer.ejs文件, 可以设置站点访问量, 访问人数, 字数统计, 站点运行时间, 网站备案等信息.

<footer class="page-footer bg-color">
<% if (theme.music.enable && theme.music.fixed) { %>
<%- partial('_widget/music') %>
<% } %>
<div class="container row center-align" style="margin-bottom: <% if (theme.time.enable) { 
 %>15<% } else { 
 %>0<% } %>px !important;">
<div class="col s12 m8 l8 copy-right">
Copyright&nbsp;&copy;
<% if (theme.time.year !== new Date().getFullYear()) { %>
<span id="year"><%- theme.time.year %>-<%- new Date().getFullYear() %></span>
<% } else { %>
<span id="year"><%- theme.time.year %></span>
<% } %>
<i class="fa fa-heart" style="color: #ff71a8"></i>
<a href="<%- url_for('/about') %>" target="_blank"><%- config.author %></a>
|&nbsp;Powered by&nbsp;<a href="https://hexo.io/" target="_blank">Hexo</a>
|&nbsp;&nbsp;<a href="https://github.com/blinkfox/hexo-theme-matery" target="_blank">Matery</a>
<br>
<% if (theme.postInfo.totalCount) { %>
<span style="margin-left: -20px; display: inline">
&nbsp;  <i class="fas fa-chart-area"></i>&nbsp;<%- __('siteTotalWords') %>:&nbsp;<span class="white-color"><%= totalcount(site) %></span>
<span/>
<% } %>
<% let socialClass = '' %>
<% if (theme.busuanziStatistics && theme.busuanziStatistics.enable) { %>
<% socialClass = 'social-statis' %>
<% } %>
<% if (theme.busuanziStatistics && theme.busuanziStatistics.totalTraffic) { %>
<span id="busuanzi_container_site_pv3" style="display: inline">
&nbsp;|&nbsp;<i class="far fa-eye"></i>&nbsp;<%- __('siteTotalVisits') %>:&nbsp;<span id="busuanzi_value_site_pv" class="white-color"><%= totalcount(site) %></span>
</span>
<% } %>
<% if (theme.busuanziStatistics && theme.busuanziStatistics.totalNumberOfvisitors) { %>
<span id="busuanzi_container_site_uv3" style="display: inline">
&nbsp;|&nbsp;<i class="fas fa-users"></i>&nbsp;<%- __('siteTotalVisitors') %>:&nbsp;<span id="busuanzi_value_site_uv" class="white-color"><%= totalcount(site) %></span>
</span>
<% } %>
<br>
<!-- 运行天数提醒. -->
<% if (theme.time.enable) { %>
<span id="sitetime"> Loading ...</span>
<script> var calcSiteTime = function () { 
 var seconds = 1000; var minutes = seconds * 60; var hours = minutes * 60; var days = hours * 24; var years = days * 365; var today = new Date(); var startYear = "<%- theme.time.year %>"; var startMonth = "<%- theme.time.month %>"; var startDate = "<%- theme.time.date %>"; var startHour = "<%- theme.time.hour %>"; var startMinute = "<%- theme.time.minute %>"; var startSecond = "<%- theme.time.second %>"; var todayYear = today.getFullYear(); var todayMonth = today.getMonth() + 1; var todayDate = today.getDate(); var todayHour = today.getHours(); var todayMinute = today.getMinutes(); var todaySecond = today.getSeconds(); var t1 = Date.UTC(startYear, startMonth, startDate, startHour, startMinute, startSecond); var t2 = Date.UTC(todayYear, todayMonth, todayDate, todayHour, todayMinute, todaySecond); var diff = t2 - t1; var diffYears = Math.floor(diff / years); var diffDays = Math.floor((diff / days) - diffYears * 365); var diffHours = Math.floor((diff / hours) - diffYears * 365 * 24 - diffDays * 24); var diffMinutes = Math.floor((diff / minutes) - diffYears * 365 * 24 * 60 - diffDays * 24 * 60 - diffHours * 60); var diffSeconds = Math.floor((diff / seconds) - diffYears * 365 * 24 * 60 * 60 - diffDays * 24 * 60 * 60 - diffHours * 60 * 60 - diffMinutes * 60); // 区分是否有年份. var language = '<%- config.language %>'; if (startYear === String(todayYear)) { 
 document.getElementById("year").innerHTML = todayYear; var daysTip = 'This site has been running for ' + diffDays + ' days'; if (language === 'zh-CN') { 
 daysTip = '本站已运行 ' + diffDays + ' 天'; } else if (language === 'zh-HK') { 
 daysTip = '本站已運行 ' + diffDays + ' 天'; } document.getElementById("sitetime").innerHTML = daysTip; } else { 
 document.getElementById("year").innerHTML = startYear + " - " + todayYear; var yearsAndDaysTip = 'This site has been running for ' + diffYears + ' years and ' + diffDays + ' days'; if (language === 'zh-CN') { 
 yearsAndDaysTip = '本站已运行 ' + diffYears + ' 年 ' + diffDays + ' 天 ' + diffHours + ' 小时 ' + diffMinutes + ' 分钟 ' + diffSeconds + ' 秒'; } else if (language === 'zh-HK') { 
 yearsAndDaysTip = '本站已運行 ' + diffYears + ' 年 ' + diffDays + ' 天'; } document.getElementById("sitetime").innerHTML = yearsAndDaysTip; } } var timer = setInterval(calcSiteTime); // calcSiteTime(); </script>
<% } %>
&nbsp;|&nbsp;
<% if (theme.icp.enable) { %>
<span id="icp">
<img src="<%- theme.jsDelivr.url %><%- url_for('/medias/icp.png') %>" style="vertical-align: text-bottom;"/>
<a href="<%- url_for(theme.icp.url) %>" target="_blank"><%=                                 theme.icp.text %></a>
</span>
<% } %>
</div>
<div class="col s12 m4 l4 social-link <%- socialClass %>">
<%- partial('_partial/social-link') %>
</div>
</div>
</footer>

还可以添加百度不蒜子统计

找到/layout/_partial/footer.ejs 文件,修改对应样式为

<!--总访问人数-->
<% if (theme.busuanziStatistics && theme.busuanziStatistics.totalNumberOfvisitors) { 
 %>
<span id="busuanzi_container_site_uv" style="display: inline">
&nbsp;|&nbsp;<i class="fas fa-users"></i>&nbsp;<%- __('siteTotalVisitors') %>:&nbsp;<span
id="busuanzi_value_site_uv" class="white-color"></span>
</span>
<% } %>
<!--最后加上-->
<script>
let _hmt = _hmt || [];
(function () { 

var hm = document.createElement("script");
hm.src = "https://hm.baidu.com/hm.js?147475454185ebcf440a27cc35e793ef";
var s = document.getElementsByTagName("script")[0];
s.parentNode.insertBefore(hm, s);
})();
</script>

26. 添加动漫人物

安装插件hexo-helper-live2d

npm install --save hexo-helper-live2d

安装下载动画人物库, 动画人物有很多, 可以网上查询资料, 下面推荐几种.

npm install --save live2d-widget-model-shizuku #课桌女孩
npm install --save live2d-widget-model-hibiki  #御姐
npm install --save live2d-widget-model-wanko   #狗狗
npm install --save live2d-widget-model-haruto  #海军服女孩
npm install --save live2d-widget-model-miku    #萝莉

博客根目录_config.yml文件配置:

## 添加动画live2d模块 npm install --save hexo-helper-live2d
## 下载动画人物库 npm install live2d-widget-model-z16 -D
live2d:
enable: true
scriptFrom: local # 默认
pluginRootPath: live2dw/ # 插件在站点上的根目录(相对路径)
pluginJsPath: lib/ # 脚本文件相对与插件根目录路径
pluginModelPath: assets/ # 模型文件相对与插件根目录路径
tagMode: false # 标签模式, 是否仅替换 live2d tag标签而非插入到所有页面中
debug: false # 调试, 是否在控制台输出日志
model:
use: live2d-widget-model-miku
display:
position: right #动画位置
width: 150
height: 190
# 位置配置,这个在左侧边栏位置很居中
hOffset: 50  # 调节水平位置
vOffset: -5  # 调节垂直位置
mobile:
show: false # 是否在移动设备上显示
scale: 0.5 # 移动设备上的缩放
react:
opacityDefault: 0.7
opacityOnHover: 0.8

27. 雪花和樱花效果

添加雪花飘落效果

在主题目录下新增/source/js/snow.js文件, 添加内容:

/*样式一*/
(function ($) { 

$.fn.snow = function (options) { 

var $flake = $('<div id="snowbox" />').css({ 

'position': 'absolute',
'z-index': '9999',
'top': '-50px'
}).html('❄'),
documentHeight = $(document).height(),
documentWidth = $(document).width(),
defaults = { 

minSize: 10,
maxSize: 20,
newOn: 1000,
flakeColor: "#AFDAEF" /* 此处可以定义雪花颜色,若要白色可以改为#FFFFFF */
},
options = $.extend({ 
}, defaults, options);
var interval = setInterval(function () { 

var startPositionLeft = Math.random() * documentWidth - 100,
startOpacity = 0.5 + Math.random(),
sizeFlake = options.minSize + Math.random() * options.maxSize,
endPositionTop = documentHeight - 200,
endPositionLeft = startPositionLeft - 500 + Math.random() * 500,
durationFall = documentHeight * 10 + Math.random() * 5000;
$flake.clone().appendTo('body').css({ 

left: startPositionLeft,
opacity: startOpacity,
'font-size': sizeFlake,
color: options.flakeColor
}).animate({ 

top: endPositionTop,
left: endPositionLeft,
opacity: 0.2
}, durationFall, 'linear', function () { 

$(this).remove()
});
}, options.newOn);
};
})(jQuery);
$(function () { 

$.fn.snow({ 

minSize: 5, /* 定义雪花最小尺寸 */
maxSize: 50,/* 定义雪花最大尺寸 */
newOn: 300  /* 定义密集程度,数字越小越密集 */
});
});
/*样式二*/
/* 控制下雪 */
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();

在主题目录下/layout/layout.ejs里添加如下代码:

<!-- 雪花特效 -->
<% if (theme.snow.enable) { %>
<script src="<%-%20theme.jsDelivr.url%20%><%-%20url_for('/js/snow.js'%20)%20%>"></script>
<% } %>

在主题目录下_config.yml里配置:

# 雪花特效
snow:
enable: true

添加樱花飘落效果

在主题目录下新增/source/js/sakura.js文件, 添加内容:

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();
}
}

在主题目录下/layout/layout.ejs里添加如下代码:

<!-- 樱花特效 -->
<% if (theme.sakura.enable) { %>
<script src="<%-%20theme.jsDelivr.url%20%><%-%20url_for('/js/sakura.js')%20%>"></script>
<% } %>

在主题目录下_config.yml里配置:

# 樱花特效
sakura:
enable: true

28. 鼠标点击爱心效果

鼠标点击样式1

在主题目录下/source/libs/others/clicklove.js文件中为鼠标点击爱心效果代码.

!function(e,t,a){ 
function r(){ 
for(var e=0;e<n.length;e++)n[e].alpha<=0?(t.body.removeChild(n[e].el),n.splice(e,1)):(n[e].y--,n[e].scale+=.004,n[e].alpha-=.013,n[e].el.style.cssText="left:"+n[e].x+"px;top:"+n[e].y+"px;opacity:"+n[e].alpha+";transform:scale("+n[e].scale+","+n[e].scale+") rotate(45deg);background:"+n[e].color+";z-index:99999");requestAnimationFrame(r)}var n=[];e.requestAnimationFrame=e.requestAnimationFrame||e.webkitRequestAnimationFrame||e.mozRequestAnimationFrame||e.oRequestAnimationFrame||e.msRequestAnimationFrame||function(e){ 
setTimeout(e,1e3/60)},function(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)}(".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;}"),function(){ 
var a="function"==typeof e.onclick&&e.onclick;e.onclick=function(e){ 
a&&a(),function(e){ 
var a=t.createElement("div");a.className="heart",n.push({ 
el:a,x:e.clientX-5,y:e.clientY-5,scale:1,alpha:1,color:"rgb("+~~(255*Math.random())+","+~~(255*Math.random())+","+~~(255*Math.random())+")"}),t.body.appendChild(a)}(e)}}(),r()}(window,document);

鼠标点击样式2

在主题目录下/source/js/wenzi.js文件中为鼠标点击爱心效果代码.

/* 鼠标点击文字特效 */
var a_idx = 0;
jQuery(document).ready(function ($) { 

$("body").click(function (e) { 

// var a = new Array("❤富强❤","❤民主❤","❤文明❤","❤和谐❤","❤自由❤","❤平等❤","❤公正❤","❤法治❤","❤爱国❤","❤敬业❤","❤诚信❤","❤友善❤");
var a = new Array("富强", "民主", "文明", "和谐", "自由", "平等", "公正", "法治", "爱国", "敬业", "诚信", "友善");
var $i = $("<span></span>").text(a[a_idx]);
a_idx = (a_idx + 1) % a.length;
var x = e.pageX,
y = e.pageY;
$i.css({ 

"z-index": 
999999999999999999999999999999999999999999999999999999999999999999999,
"top": y - 20,
"left": x,
"position": "absolute",
"font-weight": "bold",
"color": "rgb(" + ~~(255 * Math.random()) + "," + ~~(255 * Math.random()) 
+ "," + ~~(255 * Math.random()) + ")"
});
$("body").append($i);
$i.animate({ 

"top": y - 180,
"opacity": 0
},
1500,
function () { 

$i.remove();
});
});
});

在主题目录下/layout/layout.ejs里添加如下代码:

<!-- 鼠标点击特效 -->
<% if (theme.wenzi.enable) { %>
<script src="<%-%20theme.jsDelivr.url%20%><%-%20url_for('/js/wenzi.js')%20%>"></script>
<% } %>

在主题目录下_config.yml里配置:

# 鼠标点击特效
wenzi:
enable: true

29. 修改博客文章模板

为了新建文章方便,我们可以修改一下文章模板,可以将/scaffolds/post.md修改为如下代码:

title: {
{ title }} 
date: {
{ date }} 
author: 
img: 
cover: false
coverImg: 
top: false
toc: true 
mathjax: false 
password: 
summary: 
keywords: 
tags: 
categories: 

30. 在线编辑hexo博客

hexo编辑文章时,其原生方式不便利,官网提供了一款插件hexo-admin界面化了markdown编辑器.

首先安装hexo-admin插件

npm install --save hexo-admin

然后启动 hexo s, 访问 http://127.0.0.1:4000/crystalBlog/admin 就可方便快捷的进行博文编辑了.

编辑后还可以快速部署发布. 不过还是更喜欢在typora上面写markdown语法, 如果部署在自己的服务器上面可以使用该功能, 部署在gitee或github上还是无法使用.

hexo-admin在线写博客hexo-admin在线写博客

31. 站点统计不显示问题

有时候请求busuanzi数据比较慢,然后浏览量和访问人数就会隐藏,可能是默认的,在matery.css中增加以下代码可以让它一直显示.

#busuanzi_container_site_pv, #busuanzi_value_site_pv, #busuanzi_container_site_uv { 

display: inline !important;
}

32. 文章链接部分超长处理

/source/css/matery.css中增加以下代码:

/*文章链接超长部分隐藏*/
.reprint__type { 

display: inline-block;
width: 100%;
overflow: hidden;
}

33. 静态/动态彩带, 背景canvas

背景静态彩带

主题目录下的/layout/layout.ejs 文件主题目录下_config.yml中静态彩带的配置项:

<!--背景静止彩带-->
<% if (theme.ribbon.enable) { %>
<% var ribbonSrc = theme.ribbon.clickChange ? theme.libs.js.ribbon : theme.libs.js.ribbonRefresh; %>
<script type="text/javascript" size="<%- theme.ribbon.size %>" alpha='<%- theme.ribbon.alpha %>'
zIndex="<%- theme.ribbon.zIndex %>" src="<%-%20theme.jsDelivr.url%20%><%-%20url_for(ribbonSrc)%20%>" async="async"></script>
<% } %>

主题目录下_config.yml中静态彩带的配置项:

# 背景静止彩带.
ribbon:
enable: false  # 改为true即可开启背景静态彩带
size: 150 # 彩带大小, 默认: 90.
alpha: 0.6 # 彩带透明度 (0 ~ 1), 默认: 0.6.
zIndex: -1 # 背景的z-index属性,css属性用于控制所在层的位置, 默认: -1.
clickChange: false  # 设置是否每次点击都更换彩带.

背景动态彩带

主题目录下的/layout/layout.ejs 文件主题目录下_config.yml中动态彩带的配置项:

<!--背景动态彩带-->
<% if (theme.ribbon_dynamic.enable) { %>
<script type="text/javascript" src="<%-%20theme.jsDelivr.url%20%><%-%20url_for(theme.libs.js.ribbon_dynamic)%20%>" async="async"></script>
<% } %>

主题目录下_config.yml中动态彩带的配置项:

# 背景动态彩带.
ribbon_dynamic:
enable: true # 改为true即可开启背景动态彩带

背景canvas

主题目录下的/layout/layout.ejs 文件主题目录下_config.ymlcanvas的配置项:

<!--背景静止彩带-->
<% if (theme.ribbon.enable) { %>
<% var ribbonSrc = theme.ribbon.clickChange ? theme.libs.js.ribbon : theme.libs.js.ribbonRefresh; %>
<script type="text/javascript" size="<%- theme.ribbon.size %>" alpha='<%- theme.ribbon.alpha %>'
zIndex="<%- theme.ribbon.zIndex %>" src="<%-%20theme.jsDelivr.url%20%><%-%20url_for(ribbonSrc)%20%>" async="async"></script>
<% } %>

主题目录下_config.ymlcanvas的配置项:

#背景canvas-nest
canvas_nest:
enable: true
color: 0,0,255 # 线条颜色, 默认: '0,0,0' ;三个数字分别为(R,G,B),注意用,分割
pointColor: 0,0,255 # 交点颜色, 默认: '0,0,0' ;三个数字分别为(R,G,B),注意用,分割
opacity: 0.7 # 线条透明度(0~1), 默认: 0.5
zIndex: -1 # 背景的 z-index 属性,css 属性用于控制所在层的位置, 默认: -1.
count: 99 # 线条的总数量, 默认: 99

主题目录下/layout/layout.ejs 是全局布局文件, 可以自己添加自定义效果, 方式同上面添加雪花飘落.

34. 提取相册(壁纸)

新建相册(壁纸)文件

hexo new page wallpaper

修改主题目录下的_config.yml文件, 我的是提取到清单-相册导航处.

Lists:  ##清单
url: /
icon: fas fa-list
children:
- name: 音乐
url: /musics
icon: fas fa-music
- name: 电影
url: /movies
icon: fas fa-film
- name: 阅读
url: /books
icon: fas fa-book
- name: 壁纸
url: /wallpaper
icon: fas fa-image

修改站点 /galleries/index.md文件

---
title: 壁纸
date: 2019-02-04 21:35:22
layout: wallpaper
---

主题目录下新建/layout/wallpaper.ejs文件,添加内容如下:

<style type="text/css"> /* don't remove. */ .about-cover { 
 height: 75vh; } </style>
<%- partial('_partial/bg-cover') %>
<!--下面就是提取博客自带相册功能的代码-->
<main class="content">
<% if (theme.myGallery && theme.myGallery.enable) { %>
<%- partial('_widget/my-gallery') %>
<% } %>
</main>
<% if (page.total > 1) { %>
<%- partial('_partial/paging') %>
<% } %>

同时注释主题目录下的/layout/about.ejs文件的如下部分:

<!--gallery功能迁移到[清单-相册]导航处-->
<!--<% if (theme.myGallery && theme.myGallery.enable) { %> <%- partial('_widget/my-gallery') %> <% } %>-->

相册读取的图片配置路径在主题路径下的_config.yml文件中:

# 在“关于”页面配置"我的相册"图片,如果你不需要这些信息则可以将其设置为不激活或者将其删除.
myGallery:
enable: true
data:
- /medias/featureimages/0.jpg
- /medias/featureimages/1.jpg
- /medias/featureimages/2.jpg

修改相册布局, 找到/source/css/matery.css 文件,修改如下部分:

.my-gallery { 

margin: 4.5rem auto 1rem;
padding: 0 1.2rem; /*这是显示宽度,前边是页面宽度,后边是图片宽度*/
max-width: 1100px;
/*position: relative;*/
}
.my-gallery .photo { 

margin: .5rem 0; /*这是上下两行的行距*/
/*position: relative; overflow: hidden;*/
}
.my-gallery .photo img { 

width: 100%;
height: 200px; /*限制高度,使同行保持等高,不然会很乱*/
border-radius: 10px;
cursor: pointer;
}

35. 添加相册列表

相册列表可以参考博客的友情链接界面, 将友链信息存放在/source/_data/friends.json文件中, 然后hexo会按照friends.ejs模板文件里的结构渲染出来友链列表. 效果如下:

参考友链的卡片列表参考友链的卡片列表

原理其实就是三个a标签, 里面包含头像, 地址等信息, 点击后跳转到对于的地址. 那么我们也可以自定义一个相册列表的配置文件(galleries.json)和模板文件(galleries.ejs), 然后hexo读取配置文件, 自动生成相册列表界面, 如果可以这样优化, 后面新增相册就只需要在配置文件galleries.json中维护相册信息即可. (增删改)

那我们说干就干, 下面开始咯.

添加清单-相册菜单

这里要修改几个文件:

主题目录下的配置文件_config.yml ,不要跟站点根目录下的同名文件搞混了,在menu下添加以下代码:

menu: 
Lists:  ##清单
url: /crystal-blog
icon: fas fa-list
children:
# 此处省略其他菜单
- name: 壁纸
url: /wallpaper
icon: fas fa-image
- name: 相册
url: /galleries
icon: fas fa-camera

在站点根目录source下新建galleries目录,然后在该目录下新建index.md,就会生成index.html文件了

hexo new page "galleries"

修改/galleries/index.md文件, 指定布局界面:

---
title: 我的相册
date: 2021-08-25 19:56:35
type: galleries
layout: galleries
---

添加相册配置文件

在主题目录下新建/source/_data/galleries.json文件, 按照自定义约定添加如下的相册配置内容(我维护了三个相册):

[
{ 

"name": "博客背景图",
"url": "/gallery2",
"cover": "https://www.bing.com/th?id=OHR.Mpumalanga_ZH-CN9666962271_tmb.jpg&rf=",
"description": "博客背景图",
"photos": [
"https://www.bing.com/th?id=OHR.Mpumalanga_ZH-CN9666962271_tmb.jpg&rf=",
]
},
{ 

"name": "彭于晏",
"url": "/gallery0",
"cover": "https://uploadfile.bizhizu.cn/2016/0106/20160106033828391.jpg",
"description": "彭于晏写生",
"photos": [
"http://i.52desktop.cn:81/upimg/allimg/20191204/2019124151645578778013.jpg",
"http://i.52desktop.cn:81/upimg/allimg/20191204/2019124151645687778016.jpg"
]
},
{ 

"name": "刘德华",
"url": "/gallery1",
"cover": "https://tu1.whhost.net/uploads/20181029/18/1540809870-NgSCnhWkcJ.jpg",
"description": "刘德华写生",
"photos": [
"https://uploadfile.bizhizu.cn/2015/0306/20150306103233272.jpg",
"https://www.beihaiting.com/uploads/allimg/150401/10723-150401195426203.jpg"
]
}
]

添加相册布局文件

在主题目录下新建/layout/galleries.ejs模板文件, 参照friends.ejs文件修改后内容如下:

<link rel="stylesheet" href="<%- theme.jsDelivr.url %><%- url_for(theme.libs.css.gallery) %>">
<%- partial('_partial/bg-cover') %>
<main class="content">
<div class="container">
<div class="title center-align" data-aos="zoom-in-up">
<i class="fas fa-camera"></i>&nbsp;&nbsp;<%- __('galleries') %>
</div>
<% if (site.data && site.data.galleries) { %>
<% var galleries = site.data.galleries; %>
<div class="gallery-wrapper row">
<% for (var i = 0, len = galleries.length; i < len; i++) { %>
<% var gallery = galleries[i]; %>
<div class="col s6 m4 l4 xl3 gallery-box">
<a href="./<%- gallery.url %>" class="gallery-item" data-aos="zoom-in-up">
<div class="gallery-cover-box" style="background-image: url(<%- gallery.cover %>);">
</div>
<p class="gallery-name" style="font-size: 22px; font-family: '华文行楷'">
<%- gallery.name %>
</p>
</a>
</div>
<% } %>
</div>
<% } %>
</div>
</main>

相册列表中的每个相册都是一个<a></a>标签, 点击单个相册会跳转到相册图片展示页面, 在主题目录下新建/layout/gallery.ejs, 这里我们可以参考壁纸my-gallery.ejs文件的布局和渲染方式, 优化后的代码如下:

<%- partial('_partial/bg-cover') %>
<%
var galleries = site.data.galleries;
var pageTitle = page.title;
var currentGallery = getCurrentGallery(galleries, pageTitle)
var photos = currentGallery.photos;
function getCurrentGallery(galleries, pageTitle) {
for (let i = 0; i < galleries.length; i++) {
if (galleries[i]['name'] == pageTitle) {
return galleries[i];
}
}
}
/***/
%>
<div id="myGallery" class="my-gallery">
<div class="title center-align" data-aos="zoom-in-up">
<p class="gallery-name">
<b style="font-size: 35px;"><%- currentGallery.name %></b>
</p>
</div>
<div class="row">
<% if (photos) { %>
<% Object.keys(photos).forEach(function(photo) { %>
<div class="photo col s12 m6 l4" data-aos="fade-up">
<div class="img-item" data-src="<%- photos[photo] %>">
<img src="<%- theme.jsDelivr.url %><%- url_for(photos[photo]) %>" class="responsive-img">
</div>
</div>
<% }); %>
<% } %>
</div>
</div>
<script> $(function () { 
 let animateClass = 'animated pulse'; $('#myGallery .photo').hover(function () { 
 $(this).addClass(animateClass); }, function () { 
 $(this).removeClass(animateClass); }); }); </script>

添加相册列表样式

/layout/galleries.ejs模板文件模板引用了gallery.css样式文件, 所以需要在主题目录下新增/source/css/gallery.css文件, 添加内容如下:

.gallery-wrapper { 

padding-top: 30px;
}
.gallery-wrapper .gallery-box { 

padding: 5px !important;
}
.gallery-wrapper .gallery-item { 

display: block;
overflow: hidden;
background-color: #fff;
padding: 5px;
padding-bottom: 0;
position: relative;
-moz-box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.22);
-webkit-box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.22);
box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.22);
}
.gallery-cover-box { 

width: 100%;
padding-top: 60%;
text-align: center;
overflow: hidden;
position: relative;
background: center center no-repeat;
-webkit-background-size: cover;
background-size: cover;
}
.gallery-cover-box .gallery-cover-img { 

display: inline-block;
width: 100%;
position: absolute;
left: 50%;
top: 50%;
transform: translate(-50%, -50%);
}
.gallery-item .gallery-name { 

font-size: 14px;
line-height: 24px;
text-align: center;
color: #666;
margin: 0;
}
.waterfall { 

column-count: 3;
column-gap: 1em;
}
.photo-wrapper { 

padding-top: 20px;
}
.photo-item { 

display: block;
padding: 10px;
padding-bottom: 0;
margin-bottom: 14px;
font-size: 0;
-moz-page-break-inside: avoid;
-webkit-column-break-inside: avoid;
break-inside: avoid;
background: white;
-moz-box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.22);
-webkit-box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.22);
box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.22);
}
.photo-item img { 

width: 100%;
}
.photo-item .photo-name { 

font-size: 14px;
line-height: 30px;
text-align: center;
margin-top: 10px;
margin-bottom: 10px;
border-top: 1px solid #dddddd;
}
/*适配移动端布局*/
@media only screen and (max-width: 601px) { 

.waterfall { 

column-count: 2;
column-gap: 1em;
}
}

相册列表效果

相册列表效果相册列表效果

相册展示效果

相册展示效果相册展示效果

相册密码设置

给相册设置密码, 可以参考博客文章密码访问. 在主题目录下的/layout/gallery.ejs文件中添加如下代码:

<% if (theme.verifyPassword.enable) { 
 %>
<script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.crypto) %>"></script>
<script>
(function() { 

/*pwd是博客中配置的sha256加密后的密码*/
let pwd = '<%- page.password %>';
if (pwd && pwd.length > 0) { 

if (pwd !== CryptoJS.SHA256(prompt('请输入访问本相册的密
码')).toString(CryptoJS.enc.Hex)) { 

alert('密码错误!');
location.href = '<%- url_for("/galleries") %>';
}
}
})();
</script>
<% } %>

然后在需要设置密码的gallery3相册目录下的/source/galleries/gallery3/index.md文件中设置password即可. 密码需要使用SHA256加密.

36. 修改导航栏不透明

透明导航栏经常给我造成阅读障碍,可以设置不使用透明导航栏, 指定好看的颜色. 找到主题目录下的/source/css/matery.css 文件,修改如下部分:

header .nav-transparent { 

background-color: transparent !important;
/*background-color: #000B3F;*/  /*修改导航栏不透明 #16103f #4cbf30 #7371BC*/
background-image: none;
box-shadow: none;
}

37. 添加快捷导航

在博客根目录下的\source\_posts\navigate\index.md目录下新建快捷导航:

hexo new page navigate

修改主题目录下的_config.yml文件, 添加快捷导航菜单:

## 快捷导航
menu:
// .... 此处省略其他菜单
Navigate:
url: /navigate
icon: fas fa-location-arrow

修改站点 /navigate/index.md文件

---
title: 快捷导航
date: 2021-08-29 16:25:05
layout: navigate
---

主题目录下新建/layout/navigate.ejs文件,添加内容如下:

<div class="navi-height bg-cover pd-header">
<div class="link-box container">
<div class="baidu baidu-2 large-screen">
<form name="f" action="https://www.baidu.com/" target="_blank">
<div id="Select-2">
<div class="Select-box-2" id="baidu">
<ul style="height:46px">
<li class="this_s">百 度</li>
<li class="bing_s">必 应</li>
<li class="google_s">谷 歌</li>
<li class="baidu_s">百 度</li>
</ul>
</div>
<input name="wd" id="kw-2" maxlength="100" autocomplete="off" type="text">
</div>
<div class="qingkong" id="qingkong" title="清 · 空" style="display:block">x</div>
<input value="搜 索" id="su-2" type="submit"/>
<ul class="keylist"></ul>
</form>
</div>
<div class="row tags-posts">
<div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up">
<div class="card">
<div class="jj-list-tit">编程 · 学习</div>
<ul class="jj-list-con">
<li>
<a href="https://www.oschina.net/" class="link-3" target="_blank">开源中国</a>
</li>
<li>
<a href="https://htmldog.com/" class="link-3" target="_blank">HTML狗</a>
</li>
<li>
<a href="https://www.icourse163.org/" class="link-3" target="_blank">中国大学慕课</a>
</li>
<li>
<a href="https://www.imooc.com/" class="link-3" target="_blank">慕课网</a>
</li>
<li>
<a href="http://www.wxapp-union.com/" class="link-3" target="_blank">程序</a>
</li>
<li>
<a href="https://www.runoob.com/" class="link-3" target="_blank">菜鸟教程</a>
</li>
<li>
<a href="https://blog.51cto.com/" class="link-3" target="_blank">51CTO</a>
</li>
<li>
<a href="https://www.shiyanlou.com/library/" class="link-3" target="_blank">实验楼</a>
</li>
<li>
<a href="/posts/2d1a17c5.html" class="link-3" target="_blank">个人收藏页</a>
</li>
</ul>
</div>
</div>
<div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up">
<div class="card">
<div class="jj-list-tit">社区 · Code</div>
<ul class="jj-list-con">
<li>
<a href="https://www.zhangxiaocai.com/contact/" class="link-3" target="_blank">留言
</a>
</li>
<li>
<a href="https://github.com/" class="link-3" target="_blank">GitHub</a>
</li>
<li>
<a href="https://coding.net/" class="link-3" target="_blank">Coding</a>
</li>
<li>
<a href="https://juejin.im/" class="link-3" target="_blank">掘金</a>
</li>
<li>
<a href="https://gitee.com/" class="link-3" target="_blank">码云</a>
</li>
<li>
<a href="https://www.csdn.net/" class="link-3" target="_blank">CSDN</a>
</li>
<li>
<a href="https://www.jianshu.com/" class="link-3" target="_blank">简书</a>
</li>
<li>
<a href="https://segmentfault.com/" class="link-3" target="_blank">思否</a>
</li>
<li>
<a href="https://cloud.tencent.com/developer/" class="link-3" target="_blank">云+社区
</a>
</li>
</ul>
</div>
</div>
<div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up">
<div class="card">
<div class="jj-list-tit">实用 · 工具</div>
<ul class="jj-list-con">
<li>
<a href="https://mdnice.com/" class="link-3" target="_blank">Nice编辑器</a>
</li>
<li>
<a href="https://translate.google.cn/" class="link-3" target="_blank">谷歌翻译</a>
</li>
<li>
<a href="https://www.uupoop.com/" class="link-3" target="_blank">在线PS</a>
</li>
<li>
<a href="https://www.processon.com/" class="link-3" target="_blank">思维导图</a>
</li>
<li>
<a href="https://wallhaven.cc/" class="link-3" target="_blank">超清壁纸</a>
</li>
<li>
<a href="https://cli.im/" class="link-3" target="_blank">二维码</a>
</li>
<li>
<a href="http://www.yinfans.me/" class="link-3" target="_blank">音范思</a>
</li>
<li>
<a href="https://www.extfans.com" class="link-3" target="_blank">谷歌插件</a>
</li>
<li>
<a href="https://my.openwrite.cn/" class="link-3" target="_blank">OW分发</a>
</li>
</ul>
</div>
</div>
<div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up">
<div class="card">
<div class="jj-list-tit">娱乐 · 影视</div>
<ul class="jj-list-con">
<li>
<a href="https://www.jd.com/" class="link-3" target="_blank">京东</a>
</li>
<li>
<a href="https://www.taobao.com/" class="link-3" target="_blank">淘宝</a>
</li>
<li>
<a href="https://www.tmall.com/" class="link-3" target="_blank">天猫</a>
</li>
<li>
<a href="https://v.qq.com/" class="link-3" target="_blank">腾讯视频</a>
</li>
<li>
<a href="http://www.iqiyi.com/" class="link-3" target="_blank">爱奇艺</a>
</li>
<li>
<a href="https://www.bilibili.com/" class="link-3" target="_blank">哔哩哔哩</a>
</li>
<li>
<a href="https://music.163.com/" class="link-3" target="_blank">网易云音乐</a>
</li>
<li>
<a href="https://y.qq.com/" class="link-3" target="_blank">QQ音乐</a>
</li>
<li>
<a href="http://www.kugou.com/" class="link-3" target="_blank">酷狗音乐</a>
</li>
</ul>
</div>
</div>
<div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up">
<div class="card">
<div class="jj-list-tit">资讯 · 趋势</div>
<ul class="jj-list-con">
<li>
<a href="https://www.huxiu.com/" class="link-3" target="_blank">虎嗅</a>
</li>
<li>
<a href="https://insights.stackoverflow.com/" class="link-3" target="_blank">技术调查
</a>
</li>
<li>
<a href="http://www.asciiworld.com/" class="link-3" target="_blank">摸鱼</a>
</li>
<li>
<a href="https://sspai.com/" class="link-3" target="_blank">少数派</a>
</li>
<li>
<a href="https://zh.wikihow.com/" class="link-3" target="_blank">WikeHom</a>
</li>
<li>
<a href="https://www.awesomes.cn/rank?sort=hot" class="link-3" target="_blank">
前端趋势
</a>
</li>
<li>
<a href="https://github-trending.com/" class="link-3" target="_blank">GitHub趋势</a>
</li>
<li>
<a href="https://www.tiobe.com/" class="link-3" target="_blank">编程趋势</a>
</li>
<li>
<a href="https://trends.google.com/" class="link-3" target="_blank">Google趋势</a>
</li>
</ul>
</div>
</div>
<div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up">
<div class="card">
<div class="jj-list-tit">搜索 · 其他</div>
<ul class="jj-list-con">
<li>
<a href="https://ac.scmor.com/" class="link-3" target="_blank">谷歌镜像</a>
</li>
<li>
<a href="http://www.pansoso.com/" class="link-3" target="_blank">网盘搜索</a>
</li>
<li>
<a href="http://music.ifkdy.com/" class="link-3" target="_blank">音乐搜索</a>
</li>
<li>
<a href="https://www.dytt8.net/" class="link-3" target="_blank">电影天堂</a>
</li>
<li>
<a href="https://carbon.now.sh/" class="link-3" target="_blank">代码图片</a>
</li>
<li>
<a href="https://www.zhipin.com/" class="link-3" target="_blank">Boos</a>
</li>
<li>
<a href="https://fontawesome.dashgame.com/" class="link-3" target="_blank">图标库</a>
</li>
<li>
<a href="https://www.qvdv.com/tools/qvdv-guid.html" class="link-3" target="_blank">
在线工具
</a>
</li>
<li>
<a href="http://zhongguose.com/" class="link-3" target="_blank">中国色</a>
</li>
</ul>
</div>
</div>
</div>
<script> $(".Select-box ul").hover(function () { 
 $(this).css("height", "auto") }, function () { 
 $(this).css("height", "40px") }), $(".Select-box-2 ul").hover(function () { 
 $(this).css("height", "auto") }, function () { 
 $(this).css("height", "46px") }), $(".Select-box li").click(function () { 
 var t = $(this).attr("class"), s = $(this).html(); "baidu_s" == t && (t = "https://www.baidu.com/s", _name = "wd"), "google_s" == t && (t = "https://www.google.com/search", _name = "q"), "bing_s" == t && (t = "https://www.bing.com/search", _name = "q"), $(".baidu form").attr("action", t), $(".this_s").html(s), $("#kw").attr("name", _name), $(".Select-box ul").css("height", "40px") }), $(".Select-box-2 li").click(function () { 
 var t = $(this).attr("class"), s = $(this).html(); "baidu_s" == t && (t = "https://www.baidu.com/s", _name = "wd"), "google_s" == t && (t = "https://www.google.com/search", _name = "q"), "bing_s" == t && (t = "https://www.bing.com/search", _name = "q"), $(".baidu form").attr("action", t), $(".this_s").html(s), $("#kw-2").attr("name", _name), $(".Select-box-2 ul").css("height", "48px") }); $("#qingkong").click(function () { 
 $("input[ name='wd' ] ").val(""); }); let timer; function backgroundImgRandom() { 
 clearInterval(timer); timer = setInterval(function () { 
 // [0-9) $("body").css("background-image", "url(<%- theme.background.url[parseInt(Math.random() * 9)] %>"); }, 5000); } window.onload = backgroundImgRandom; </script>
</div>
</div>
<style> * { 
 margin: 0; padding: 0; font-family: consolas, hl, "微软雅黑" } dd, dl, dt, form, h1, h2, h3, h4, h5, h6, li, p, ul { 
 margin: 0; padding: 0; font-size: 14px; font-weight: 400 } img { 
 border-style: none } li { 
 list-style: none; float: left } a { 
 text-decoration: none } .card { 
 background-color: rgba(25, 240, 229, 0); width: 96%; margin-left: 2% } .baidu { 
 float: left; margin-left: 100px } .baidu form { 
 position: relative } .Select-box ul { 
 height: 40px; position: absolute; left: -1px; top: 0; z-index: 9999; background: #FFF; border: 1px solid #ccc; border-top: none; overflow: hidden } .Select-box li { 
 width: 60px; line-height: 40px; font-size: 14px; color: #484848; border: 0; cursor: pointer } .Select-box li:hover { 
 background: #3385ff; color: #FFF } .Select-box .this_s { 
 color: #317ef3 } .Select-box .this_s:hover { 
 background: #FFF; color: #317ef3 } .qingkong { 
 position: absolute; right: 120px; top: 12px; width: 18px; height: 18px; background: rgba(0, 0, 0, .1); border-radius: 18px; line-height: 16px; color: #666; cursor: pointer; text-align: center; font-size: 14px; display: none; color: #881509; } .qingkong:hover { 
 background: rgba(0, 0, 0, .2) } .qingkong:active { 
 background: rgba(0, 0, 0, .3) } .baidu-2 { 
 width: 100%; height: 110px; margin: 0 auto; background: 0 0; padding-top: 50px } .baidu-2 form { 
 width: 520px; margin: 0 auto } .baidu-2 input { 
 padding: 13px 8px; opacity: .9; font-size: 15px } #Select-2 { 
 float: left } .Select-box-2 { 
 text-align: center; float: left; position: relative } .Select-box-2 ul { 
 height: 46px; position: absolute; left: 0; top: 1px; z-index: 9999; background: rgba(255, 255, 255, .9); border: 1px solid #ccc; border-top: none; overflow: hidden } .Select-box-2 li { 
 width: 60px; line-height: 46px; font-size: 15px; color: #484848; border: 0; cursor: pointer } .Select-box-2 li:hover { 
 background: #3385ff; color: #FFF } .Select-box-2 .this_s { 
 color: #317ef3 } .Select-box-2 .this_s:hover { 
 background: 0 0; color: #317ef3 } #kw-2 { 
 width: 335px; outline: 0; border: 1px solid #ccc; background: rgba(255, 255, 255, .2); color: #000; padding-left: 70px; font-weight: 700 } #su-2 { 
 width: 90px; background: #4e6ef2; border: none; border-top: #3385ff 1px solid; border-bottom: 1px solid #2d78f4; color: #FFF; cursor: pointer; outline: 0 } #su-2:hover { 
 background: #00f; border-bottom: 1px solid #00f } #su-2:active { 
 background: #00f; box-shadow: inset 1px 1px 3px #00f; -webkit-box-shadow: inset 1px 1px 3px #00f } #su-3 { 
 width: 90px; background: #00f; border: none; border-top: #3385ff 1px solid; border-bottom: 1px solid #2d78f4; color: #FFF; cursor: pointer; outline: 0 } .jj-list-tit { 
 font-size: 16px; line-height: 25px; color: #fff; width: 100%; padding-left: 38.5% } .jj-list-con { 
 overflow: hidden; margin: 0 auto } .jj-list-con li { 
 width: 31.333%; margin: 1% } .link-3 { 
 display: block; background: rgba(0, 0, 0, .35); color: #FFF; font-size: 13px; text-align: center; line-height: 35px; padding: 4px 0; border-radius: 2px; transition: all .2s } .link-3:hover { 
 background: rgba(0, 0, 0, .45); font-size: 15px; font-weight: 700 } @media only screen and (max-width: 584px) { 
 .navi-height { 
 height: 1300px } .link-box { 
 margin-top: 5% } .large-screen { 
 display: none } } @media only screen and (min-width: 584px) and (max-width: 993px) { 
 .navi-height { 
 height: 800px } .link-box { 
 margin-top: 5% } .large-screen { 
 display: none } } @media only screen and (min-width: 993px) { 
 .navi-height { 
 position: absolute; width: 100%; height: 100% } } .page-footer { 
 display: none } </style>

主题目录下的/layout/_partial/navigation.ejs文件添加快捷导航菜单中文名称. 以及移动客户端文件/layout/_partial/mobile-nav.ejs一样添加.

<%
var menuMap = new Map();
menuMap.set("Index", "首页");
menuMap.set("Tags", "标签");
menuMap.set("Categories", "分类");
menuMap.set("Archives", "归档");
menuMap.set("About", "关于");
menuMap.set("Contact", "留言板");
menuMap.set("Friends", "友情链接");
menuMap.set("Lists", "清单");
menuMap.set("Navigate", "快捷导航");
var configRoot = config.root
configRoot = (configRoot === null || configRoot === undefined 
|| configRoot === '/') ? '' : configRoot;
%>

38. 添加音乐导航页

在博客根目录下新增\source\_posts\musics\index.md文件

hexo new page musics

主题目录下的_config.yml文件中添加[清单-音乐]菜单:

menu:
// 此处省略其他菜单项
Lists:  ##清单
url: /
icon: fas fa-list
children:
- name: 音乐
url: /musics
icon: fas fa-music

修改音乐文件/musics/index.md指定布局配置:

---
title: musics
date: 2021-08-25 19:55:53
layout: musics
---

主题目录下新增音乐布局文件 /layout/musics.ejs, 添加如下内容:

<style type="text/css"> /* don't remove. */ .about-cover { 
 height: 75vh; } </style>
<%- partial('_partial/bg-cover') %>
<main class="content">
<% if (theme.mymusic.enable) { %>
<%- partial('_widget/mymusic') %>
<% } %>
</main>

主题目录下新增自定义音乐播放器文件 /layout/_widget/mymusic.ejs, 添加如下内容:

<link rel="stylesheet" href="<%- theme.jsDelivr.url %><%- url_for(theme.libs.css.aplayer) %>">
<style> .aplayer .aplayer-lrc p { 
 <%if(theme.mymusic.hideLrc){ 
%> display: none; <%}%> font-size: 12px; font-weight: 700; line-height: 16px !important; } .aplayer .aplayer-lrc p.aplayer-lrc-current { 
 <%if(theme.mymusic.hideLrc){ 
%> display: none; <%}%> font-size: 15px; color: <%- theme.mymusic.theme %>; } <%if(theme.mymusic.autoHide){ 
%> .aplayer.aplayer-fixed.aplayer-narrow .aplayer-body { 
 left: -66px !important; } .aplayer.aplayer-fixed.aplayer-narrow .aplayer-body:hover { 
 left: 0px !important; } <%}%> </style>
<div class="<% if(!theme.mymusic.fixed) { %>music-player<% } %>">
<% if (!theme.mymusic.fixed && theme.mymusic.title.enable) { %>
<div class="title center-align">
<i class="fas fa-music"></i>&nbsp;&nbsp;<%- theme.mymusic.title.show %>
</div>
<% } %>
<div class="row">
<meting-js class="col l8 offset-l2 m10 offset-m1 s12" server="<%- theme.mymusic.server %>" type="<%- theme.mymusic.type %>" id="<%- theme.mymusic.id %>" fixed='<%- theme.mymusic.fixed ? 'true' : 'false' %>'
autoplay='<%- theme.mymusic.autoplay === true %>'
theme='<%- theme.mymusic.theme %>'
loop='<%- theme.mymusic.loop %>'
order='<%- theme.mymusic.order %>'
preload='<%- theme.mymusic.preload %>'
volume='<%- Number(theme.mymusic.volume) %>'
list-folded='<%- theme.mymusic.listFolded ? 'true' : 'false' %>'
>
</meting-js>
</div>
</div>
<script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.aplayer) %>"></script>
<script src="https://cdn.jsdelivr.net/npm/meting@2/dist/Meting.min.js"></script>

主题目录下配置文件_config.yml添加自定义音乐播放器配置:

# 自定义音乐组件, 展示在自己需要的页面
mymusic:
enable: true
title: #非吸底模式有效
enable: true
show: 轻松时刻
autoHide: true    # hide automaticaly
server: netease   #require music platform: netease, tencent, kugou, xiami, baidu
type: playlist    #require song, playlist, album, search, artist
id: 4965675848     #require song id / playlist id (useful playlist 4965675848 2888085740)/ album id / search keyword
fixed: false       # 开启吸底模式 true 播放器会在站点侧边,点击会出现
autoplay: false   # 是否自动播放
theme: '#42b983'
loop: 'all'       # 音频循环播放, 可选值: 'all', 'one', 'none'
order: 'random'   # 音频循环顺序, 可选值: 'list', 'random'
preload: 'auto'   # 预加载,可选值: 'none', 'metadata', 'auto'
volume: 0.7       # 默认音量,请注意播放器会记忆用户设置,用户手动设置音量后默认音量即失效
listFolded: false  # 列表默认折叠
hideLrc: false     # 隐藏歌词

39. 生成电影卡片

安装hexo-tag-mtime插件

npm install hexo-tag-mtime --save

新增电影导航页面, 博客根目录下生成/source/movies/index.md文件:

hexo new page movies

/source/movies/index.md文件添加如下内容:

---
title: 电影推荐
date: 2021-08-25 19:56:04
type: movies
---
## 精彩电影推荐
###  怒火·重案
{% mtime 263501 %}
###  再见,少年
{% mtime 259370 %}
### 流浪地球
{% mtime 218707 %}
id可以在时光网 https://www.mtime.com 相应的电影网址获取

40. 添加夜间模式切换

在主题目录下的/layout/layout.ejs文件添加夜间模式切换按钮:

<!-- 切换夜间/白天模式按钮 -->
<a onclick="switchNightMode()" id="sma"> <i class="fa fa-moon-o" id="nightMode" aria-hidden="true"></i> </a>

在主题目录下的/source/js/matery.js文件添加js代码:

// 深色模式按钮设置
if (localStorage.getItem('dark') === '1') { 

document.body.classList.add('dark');
} else if (new Date().getHours() >= 22 || new Date().getHours() < 7) { 

/*定时开启暗色模式<默认晚22点至早6点默认开启>*/
// document.body.classList.add('dark');
// $("#nightMode").removeClass("fa-moon-o").addClass("fa-lightbulb");
} else if (matchMedia('(prefers-color-scheme: dark)').matches) { 

document.body.classList.add('dark');
}
// 深色模式设置
function switchNightMode() { 

var body = document.body;
if (body.classList.contains('dark')) { 

document.body.classList.remove('dark');
localStorage.setItem('dark', '0');
$('#nightMode').removeClass("fa-lightbulb").addClass("fa-moon-o");
return;
} else { 

document.body.classList.add('dark');
localStorage.setItem('dark', '1');
$('#nightMode').removeClass("fa-moon-o").addClass("fa-lightbulb");
return;
}
}
/*提醒开启夜间模式功能*/
setTimeout(
function () { 

if ((new Date().getHours() >= 19 || new Date().getHours() < 7) && !$('body').hasClass('DarkMode')) { 

let toastHTML = '<span token operator">+ '<i class="fa fa-bell" aria-hidden="true"></i>晚上使用深色模式阅读更好哦。(゚▽゚)/</span>'
M.toast({ 
html: toastHTML})
}
}, 2200);

在主题目录下的/source/css/matery.css文件添加夜间模式切换的样式:

/******夜间模式切换样式 start*******/
/* 深色模式按钮设置 */
#sma { 

background: #000;
width: 38px;
height: 38px;
display: block;
position: fixed;
border-radius: 50%;
right: 15px;
bottom: 170px;
padding-top: 15px;
margin-bottom: 0;
z-index: 998;
cursor: pointer;
}
#sma .fa-moon-o { 

position: absolute;
right: 8px;
bottom: 8px;
font-size: 1.48rem !important;
}
#sma .fa-lightbulb { 

position: absolute;
right: 13px;
bottom: 8px;
font-size: 1.5rem !important;
}
.fa-moon-o:before { 

content: "\f186";
}
.fa-comments:before { 

content: "\f086";
}
/* 深色模式设置 */ /* 字体颜色变灰白色 */
body.dark .fas, body.dark .title, body.dark .row .text, body.dark article .article-content .summary, body.dark .card .card-image .card-title, body.dark .fa-moon-o:before, body.dark .fa-lightbulb:before, body.dark article .article-tags .chip, body.dark .chip-container .tag-title, body.dark div.jqcloud a, body.dark .friends-container .tag-title, body.dark .frind-ship .title h1, body.dark .card .card-content p, body.dark .v[data-class=v] .vcount, body.dark .v[data-class=v] .vcount .vnum, body.dark pre code, body.dark h1, body.dark h2, body.dark h3, body.dark h4, body.dark h5, body.dark h6, body.dark li, body.dark p, body.dark header .side-nav .mobile-head .logo-name, body.dark header .side-nav .mobile-head .logo-desc, body.dark header .side-nav .menu-list a, body.dark .bg-cover .post-title, body.dark a { 

color: rgba(255, 255, 255, 0.6);
}
/* 背景颜色变灰色 */
body.dark .card, body.dark .block-with-text:after { 

background-color: #282c34;
}
/* 背景颜色变黑色 */
body.dark, body.dark .v[data-class=v] .vcount, body.dark #rewardModal .modal-content, body.dark .modal, body.dark header .side-nav, body.dark header .side-nav .menu-list .m-nav-show { 

background-color: #12121c;
/**因为我的背景图导致部分页面无法全部切换成深色背景, 需要取消背景图片**/
background-image: url(#); 
}
/* 改变透明度 */
body.dark .aplayer { 

background: #2f3742 !important;
}
body.dark img, body.dark strong { 

filter: brightness(.7);
}
/******夜间模式切换样式 end*******/

41. 添加valine评论功能

使用valine评论功能, 可以使用leanCloud国际版存储评论数据, 具体申请ID和KEY的教程如下:

文字教程: https://cndrew.cn/2020/04/10/hexo-shuoshuo/

B站视频: https://www.bilibili.com/video/BV16A411b7UF

在主题目录下创建/layout/_partial/valine.ejs文件, 添加如下内容:

<style> .valine-card { 
 margin: 1.5rem auto; } .valine-card .card-content { 
 padding: 20px 20px 5px 20px; } #vcomments textarea { 
 box-sizing: border-box; background: url("<%- url_for(theme.valine.background) %>") 100% 100% no-repeat; } #vcomments p { 
 margin: 2px 2px 10px; font-size: 1.05rem; line-height: 1.78rem; } #vcomments blockquote p { 
 text-indent: 0.2rem; } #vcomments a { 
 padding: 0 2px; color: #4cbf30; font-weight: 500; text-decoration: none; } #vcomments img { 
 max-width: 100%; height: auto; cursor: pointer; } #vcomments ol li { 
 list-style-type: decimal; } #vcomments ol, ul { 
 display: block; padding-left: 2em; word-spacing: 0.05rem; } #vcomments ul li, ol li { 
 display: list-item; line-height: 1.8rem; font-size: 1rem; } #vcomments ul li { 
 list-style-type: disc; } #vcomments ul ul li { 
 list-style-type: circle; } #vcomments table, th, td { 
 padding: 12px 13px; border: 1px solid #dfe2e5; } #vcomments table, th, td { 
 border: 0; } table tr:nth-child(2n), thead { 
 background-color: #fafafa; } #vcomments table th { 
 background-color: #f2f2f2; min-width: 80px; } #vcomments table td { 
 min-width: 80px; } #vcomments h1 { 
 font-size: 1.85rem; font-weight: bold; line-height: 2.2rem; } #vcomments h2 { 
 font-size: 1.65rem; font-weight: bold; line-height: 1.9rem; } #vcomments h3 { 
 font-size: 1.45rem; font-weight: bold; line-height: 1.7rem; } #vcomments h4 { 
 font-size: 1.25rem; font-weight: bold; line-height: 1.5rem; } #vcomments h5 { 
 font-size: 1.1rem; font-weight: bold; line-height: 1.4rem; } #vcomments h6 { 
 font-size: 1rem; line-height: 1.3rem; } #vcomments p { 
 font-size: 1rem; line-height: 1.5rem; } #vcomments hr { 
 margin: 12px 0; border: 0; border-top: 1px solid #ccc; } #vcomments blockquote { 
 margin: 15px 0; border-left: 5px solid #42b983; padding: 1rem 0.8rem 0.3rem 0.8rem; color: #666; background-color: rgba(66, 185, 131, .1); } #vcomments pre { 
 font-family: monospace, monospace; padding: 1.2em; margin: .5em 0; background: #272822; overflow: auto; border-radius: 0.3em; tab-size: 4; } #vcomments code { 
 font-family: monospace, monospace; padding: 1px 3px; font-size: 0.92rem; color: #e96900; background-color: #f8f8f8; border-radius: 2px; } #vcomments pre code { 
 font-family: monospace, monospace; padding: 0; color: #e8eaf6; background-color: #272822; } #vcomments pre[class*="language-"] { 
 padding: 1.2em; margin: .5em 0; } #vcomments code[class*="language-"], pre[class*="language-"] { 
 color: #e8eaf6; } #vcomments [type="checkbox"]:not(:checked), [type="checkbox"]:checked { 
 position: inherit; margin-left: -1.3rem; margin-right: 0.4rem; margin-top: -1px; vertical-align: middle; left: unset; visibility: visible; } #vcomments b, strong { 
 font-weight: bold; } #vcomments dfn { 
 font-style: italic; } #vcomments small { 
 font-size: 85%; } #vcomments cite { 
 font-style: normal; } #vcomments mark { 
 background-color: #fcf8e3; padding: .2em; } #vcomments table, th, td { 
 padding: 12px 13px; border: 1px solid #dfe2e5; } table tr:nth-child(2n), thead { 
 background-color: #fafafa; } #vcomments table th { 
 background-color: #f2f2f2; min-width: 80px; } #vcomments table td { 
 min-width: 80px; } #vcomments [type="checkbox"]:not(:checked), [type="checkbox"]:checked { 
 position: inherit; margin-left: -1.3rem; margin-right: 0.4rem; margin-top: -1px; vertical-align: middle; left: unset; visibility: visible; } .v[data-class="v"] .vwrap .vheader .vinput { 
 width: 32%; border-bottom: 1px dashed #dedede; } </style>
<div class="card valine-card" data-aos="fade-up">
<div class="comment_headling" style="font-size: 20px; font-weight: 700; position: relative; padding-left: 20px; top: 15px; padding-bottom: 5px;">
<i class="fas fa-comments fa-fw" aria-hidden="true"></i>
<span>评论</span>
</div>
<div id="vcomments" class="card-content" style="display: grid">
</div>
</div>
<script src="<%- theme.jsDelivr.url %><%- url_for('/libs/valine/av-min.js') %>"></script>
<script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.valine) %>"></script>
<script> let metaPlaceholder = <%- JSON.stringify(theme.valine.metaPlaceholder) %> ; //这里要换行 new Valine({ 
 el: '#vcomments', appId: '<%- theme.valine.appId %>', appKey: '<%- theme.valine.appKey %>', notify: '<%- theme.valine.notify %>' === 'true', verify: '<%- theme.valine.verify %>' === 'true', visitor: '<%- theme.valine.visitor %>' === 'true', avatar: '<%- theme.valine.avatar %>', pageSize: '<%- theme.valine.pageSize %>', lang: '<%- theme.valine.lang %>', placeholder: '<%= theme.valine.placeholder %>', meta: <%- '["' + theme.valine.guest_info.join('", "') + '"]' %>, recordIP: '<%- theme.valine.recordIP %>' === 'true', enableQQ: '<%- theme.valine.avatar %>', requiredFields: <%- '["' + theme.valine.guest_info.join('", "') + '"]' %>, master: <%- '["' + theme.valine.master.join('", "') + '"]' %>, friends: <%- '["' + theme.valine.friends.join('", "') + '"]' %>, tagMeta: <%- '["' + theme.valine.tagMeta.join('", "') + '"]' %>, metaPlaceholder: metaPlaceholder, }); document.body.addEventListener('click', function (e) { 
 if (e.target.classList.contains('vsubmit')) { 
 const email = document.querySelector('input[type=email]'); const nick = document.querySelector('input[name=nick]'); const reg = /^[A-Za-z0-9_-\u4e00-\u9fa5]+@[a-zA-Z0-9_-]+(\.[a-zA-Z0-9_-]+)+$/; if (!email.value || !nick.value || !reg.test(email.value)) { 
 const str = `<div class="valert txt-center"><div class="vtext">请填写正确的昵称和邮箱!</div></div>`; const vmark = document.querySelector('.vmark'); vmark.innerHTML = str; vmark.style.display = 'block'; e.stopPropagation(); setTimeout(function () { 
 vmark.style.display = 'none'; vmark.innerHTML = ''; }, 2500); } } }, true); </script>

然后在需要评论功能的页面引用valine.ejs, 我的在留言版博客文章中添加了评论功能, 分别是主题目录下的/layout/contact.ejs/layout/_partial/post-detail.ejs文件添加如下内容:

<% if (theme.valine && theme.valine.enable) { %>
<%- partial('_partial/valine') %>
<% } %>

然后在主题目录下的_config.yml文件中启用valine评论功能, 并添加一部分valine.ejs中用到的自定义属性.

# Valine 评论模块的配置,默认为不激活,如要使用,就请激活该配置项,并设置 appId 和 appKey.
valine:
enable: true
appId: NiVMV7aR8ipVDy9EywBYtLwI-MdYXbMMI
appKey: P7BGA1hMeBAFbHYz3dKY8doJ
notify: false # 是否开启邮件提醒(https://valine.js.org/notify.html)
verify: true # 是否启用防垃圾验证
visitor: true
avatar: monsterid # 默认头像展示方式,小怪物 # 'mm' # Gravatar style : mm/identicon/monsterid/wavatar/retro/hide
pageSize: 10
placeholder: '填写QQ号就能评论,快来一发吧~'  # Comment Box placeholder
/medias/comment_bg.png #背景图
background: https://cdn.jsdelivr.net/gh/drew233/cdn/20200409110727.webp 
coolpushkey:
# 新增属性 wang-qz
comment_count: true
enableQQ: true  # 强制QQ
recordIP: true
requiredFields: # 必须输入的信息,默认 [昵称, 邮箱, link]
- nick
- mail
guest_info: # 评论框展示的输入项, 默认[nick, mail, link], 可以设置不填某些选项
- nick
- mail
- link
master:
- e4de1f6da602d22e423d6dfb24611b6b  # md5加密后的博主邮箱 wang-qz@foxmail.com
metaPlaceholder:  # 输入框的背景文字
nick: 昵称/QQ号(必填)
mail: 邮箱(必填)
link: 网址(https://)
lang: zh-CN
tagMeta: # 评论标签要显示的文字
- 博主
- 小伙伴
- 访客
friends: # md5 加密后的小伙伴邮箱
- 410739a5ad278251b305dab085768c57 # 1848276756@qq.com
- 410739a5ad278251b305dab085768c57
- 410739a5ad278251b305dab085768c57

另外, valine.min.js 需要进行升级, 我使用的版本是 valine-1.4版本:

<script src="https://cdn.jsdelivr.net/gh/small-rose/small-rose.github.io/libs/valine/Valine.min.js"></script>

因为上面“valine.ejs文件中是引用的/lib/js/valine/valine.min.js`, 所以也可以下载下来替换该js文件.

下面这个版本也可以, 去gitHub克隆下来, 然后使用他的valine.min.js替换本地的即可.

https://github.com/LuckyZmj/LuckyBlog/blob/master/themes/matery/source/libs/valine/Valine.min.js

valine.ejs文件中是引用valine.min.js的方式需要在主题目录下的_config.yml文件中搜索关键词 libs, 找到 js , 在js引用配置的最后一行然后添加:

valine: /libs/valine/Valine.min.js 

这样下面的引用方式就能找到/lib/js/valine/valine.min.js文件

<script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.valine) %>"></script>

42. 添加artitalk说说功能

参考资料: Hexo博客Matery主题添加ArtiTalk说说模块

具体使用见artitalk官方文档 , 发布说说的数据可以存储在LeanCloud, 上面评论功能已经提供了相关使用参考资料.

首先, 下载artitalk源码

git clone git@github.com:ArtitalkJS/Artitalk.git

在主题目录下新建/source/libs/artitalk文件夹, 找到刚才下载artitalk源码,进入dist 目录,里面有2个文件夹:css 和 js, 然后进行如下操作:

/Artitalk/dist/css/ 下的 artitalk.min.css 复制到主题目录 /source/libs/artitalk下;

/Artitalk/dist/js/下的 artitalk.min.js 复制到主题目录 /source/libs/artitalk下;
因为要和matery主题引入风格保持一致, 修改主题配置_config.yml文件, 搜索关键词libs,

找到css在最后一行添加:

artitalk: /libs/artitalk/artitalk.min.css

找到js, 在最后一行添加:

artitalk: /libs/artitalk/artitalk.min.js

我列一下最终效果,因为原来有很多,我就不全部列出了,只要知道最后一行加就可以了,注意对齐,如下:

libs:
css:
fontAwesome: /libs/awesome/css/all.css # V5.11.1
materialize: /libs/materialize/materialize.min.css # 1.0.0
artitalk: /libs/artitalk/artitalk.min.css # 最后一行添加
js:
jquery: /libs/jquery/jquery.min.js
materialize: /libs/materialize/materialize.min.js # 1.0.0
artitalk: /libs/artitalk/artitalk.min.js  # 最后一行添加

找到主题目录下 /layout/_partial/head.ejs,在头部引入css:

<link rel="stylesheet" type="text/css" href="<%- theme.jsDelivr.url %><%- url_for(theme.libs.css.artitalk) %>">

模块准备, 在主题目录下新建一个/layout/artitalk.ejs文件, 添加如下内容:

<style type="text/css"> /* don't remove. */ .about-cover { 
 height: 75vh; } </style>
<%- partial('_partial/bg-cover') %>
<main class="content">
<% if (theme.artitalk && theme.artitalk.enable) { %>
<%- partial('_widget/artitalk') %>
<% } %>
</main>

在主题目录下新建一个/layout/_widget/artitalk.ejs文件, 在上面代码中已经引入了该文件, 文件的内容如下:

<style type="text/css"> #artitalk_main .cbp_tmtimeline > li .cbp_tmlabel::after { 
 right: 100%; border: solid transparent; z-index: -1; content: " "; height: 0; width: 0; position: absolute; pointer-events: none; border-right-color: #0bb7fbd6 ; border-width: 10px; top: 4px; } #pubShuo { 
 margin-right: 5px; } #operare_artitalk .shuoshuo_input_log { 
 outline-style: none; margin-top: 5px; border: 1px solid #ccc; border-radius: 6px; padding: 8px 16px; font-size: 12px; background-color: transparent; color: #0bb7fbd6; width: 70%; height: 28px; margin-left: 10px; } #artitalk_main { 
 margin-top: 5px; margin-left: 5%; margin-right: 5%; } #lazy { 
 margin-top: 40px; } </style>
<script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.artitalk) %>"></script>
<article id="articles11" class="container chip-container">
<div class="row ">
<div class=" card">
<div class="card-content">
<div class="tag-title center-align">
<i class="fas fa-pen-alt"></i> 说说
</div>
<div id="artitalk_main"></div>
</div>
</div>
</div>
</article>
<script> new Artitalk({ 
 appId: "<%= theme.artitalk.appId %>", appKey: "<%= theme.artitalk.appKey %>", <% if (theme.artitalk.serverURL) { 
 %> serverURL: "<%= theme.artitalk.serverURL %>", <% } %> <% if (theme.artitalk.lang) { 
 %> lang: "<%= theme.artitalk.lang %>", <% } %> <% if (theme.artitalk.pageSize) { 
 %> pageSize: "<%= theme.artitalk.pageSize %>", <% } %> <% if (theme.artitalk.shuoPla) { 
 %> shuoPla: "<%= theme.artitalk.shuoPla %>", <% } %> <% if (theme.artitalk.avatarPla) { 
 %> avatarPla: "<%= theme.artitalk.avatarPla %>", <% } %> <% if (theme.artitalk.motion == 0) { 
 %> motion: 0, <% } else { 
 %> motion: 1, <% } %> <% if (theme.artitalk.bgImg) { 
 %> bgImg: "<%= theme.artitalk.bgImg %>", <% } %> <% if (theme.artitalk.color1) { 
 %> color1: "<%= theme.artitalk.color1 %>", <% } %> <% if (theme.artitalk.color2) { 
 %> color2: "<%= theme.artitalk.color2 %>", <% } %> <% if (theme.artitalk.color3) { 
 %> color3: "<%= theme.artitalk.color3 %>", <% } %> <% if (theme.artitalk.cssUrl) { 
 %> cssUrl: "<%= theme.artitalk.cssUrl %>", <% } %> atEmoji: { 
 baiyan: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/baiyan.png", bishi: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/bishi.png", bizui: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/bizui.png", chan: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/chan.png", daku: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/daku.png", dalao: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/dalao.png", dalian: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/dalian.png", dianzan: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/dianzan.png", doge: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/doge.png", facai: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/facai.png", fadai: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/fadai.png", fanu: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/fanu.png", }, }) </script>

在主题目录下的_config.yml文件中添加如下配置:

# 说说 https://artitalk.js.org/
artitalk:
enable: true
appId: NiVMV7aR8ipVDy9EywBYtLwI-MdYXbMMI # leancloud的应用appId
appKey: P7BGA1hMeBAFbHYz3dKY8doJ # leancloud的应用appKey
serverURL: # #leancloud绑定的安全域名,使用国际版的话不需要填写
lang: zh # 语言设置,zh为汉语,en为英语,es为西班牙语。默认为汉语
pageSize: 6 # 每页显示说说的数量
shuoPla: '只有王子才能发布说说哦' # 在编辑说说的输入框中的占位符
motion: 1 #加载动画的开关,1为开,0为关,默认为开
#说说输入框背景图片url
bgImg: https://cdn.jsdelivr.net/gh/drew233/cdn/20200409110727.webp 
avatarPla: https://cdn.jsdelivr.net/gh/small-rose/small-rose.github.io/medias/avatar.jpg #自定义头像url的输入框的占位符
color1: linear-gradient(45deg, rgb(109, 208, 242) 15%, rgb(245, 154, 190) 85%) #说说背景颜色1&按钮颜色1
color2: linear-gradient(45deg, rgb(109, 208, 242) 15%, rgb(245, 154, 190) 85%) #说说背景颜色2&按钮颜色2
color3: black #说说字体颜色

更多配置项参考官网:配置项说明

最后创建页面, 手工创建或者执行hexo命令创建都可以.

hexo new page "artitalk"

在 hexo 的 source 目录会生成一个 artitalk 文件夹,修改里面的 index.md :

---
title: artitalk
date: 2021-09-03 20:31:58
type: artitalk
layout: artitalk
---

主题目录下的_config.yml文件配置菜单:

menu:
##关于
About:
url: /crystal-blog
icon: fas fa-rocket
children:
- name: 关于我
url: /about
icon: fas fa-user-circle
- name: artitalk
url: /artitalk
icon: fas fa-pen-alt

配色参考: https://colordrop.io/

43. 归档时间轴添加时间列表的切换

归档的布局文件是主题目录下的/layout/archive.ejs, 原来里面的逻辑是以卡片的形式循环展示所有的博客文章. 具体代码如下:

 <!--时间轴区域块-->
<div id="cd-timeline" class="container" style="display: none;">
<% page.posts.each(function(post) { %>
<div class="cd-timeline-block">
<%# year. %>
<% if (date(post.date, 'YYYY') != year) { %>
<% year = date(post.date, 'YYYY'); %>
<div class="cd-timeline-img year" data-aos="zoom-in-up">
<a href="<%- url_for('/archives/' + year) %>"><%- year %></a>
</div>
<% } %>
<%# month. %>
<% if (date(post.date, 'YYYY-MM') != month) { %>
<%
month = date(post.date, 'YYYY-MM');
var m = date(post.date, 'MM')
%>
<div class="cd-timeline-img month" data-aos="zoom-in-up">
<a href="<%- url_for('/archives/' + year + '/' + m) %>">
<%- m %></a>
</div>
<% } %>
<%# every day posts. %>
<div class="cd-timeline-img day" data-aos="zoom-in-up">
<span><%- date(post.date, 'YYYY-MM-DD').substring(8, 10) %></span>
</div>
<article class="cd-timeline-content" data-aos="fade-up">
<div class="article col s12 m6">
<div class="card">
<a href="<%- url_for(post.path) %>">
<div class="card-image">
<% if (post.img) { %>
<img src="<%- url_for(post.img) %>" class="responsive-img" alt="<%= post.title %>">
<% } else { %>
<%
featureimg =                                           featureImages[Math.abs(hashCode(post.title) 
% featureImages.length)];
%>
<img src="<%- theme.jsDelivr.url %> <%- url_for(featureimg) %>" class="responsive-img" alt="<%= post.title %>">
<% } %>
<span class="card-title"><%= post.title %>
</span>
</div>
</a>
<div class="card-content article-content">
<div class="summary block-with-text">
<% if (post.summary && post.summary.length > 0)
{ %>
<%- post.summary %>
<% } else { %>
<%- strip_html(post.content).substring(0, 
120) %>
<% } %>
</div>
<div class="publish-info">
<span class="publish-date">
<i class="far fa-clock fa-fw icon-date"></i>
<%= date(post.date, config.date_format) %>
</span>
<span class="publish-author">
<% if (post.categories && 
post.categories.length > 0) { %>
<i class="fas fa-bookmark fa-fw icon-category"></i>
<% post.categories.forEach(category => { 
%>
<a href="<%- url_for(category.path) %>" class="post-category">
<%- category.name %>
</a>
<% }); %>
<% } else if (post.author && 
post.author.length > 0) { %>
<i class="fas fa-user fa-fw"></i>
<%- post.author %>
<% } else { %>
<i class="fas fa-user fa-fw"></i>
<%- config.author %>
<% } %>
</span>
</div>
</div>
<% if (post.tags && post.tags.length) { %>
<div class="card-action article-tags">
<% post.tags.forEach(tag => { %>
<a href="<%- url_for(tag.path) %>">
<span class="chip bg-color">
<%= tag.name %></span></a>
<% }); %>
</div>
<% } %>
</div>
</div>
</article>
</div>
<% }); %>
</div>

我看到网上有些博客的归档是以时间列表方式展现的, 这样对所有博客文章有更佳的查看体验. 于是通过debug模式将网上大佬的博客效果代码copy下来, 再经过自己的修改后, 达到了想要的效果. 下面操作.

先添加时间列表时间轴的切换按钮, 在主题目录下的/layout/archive.ejs文件中添加如下代码:

<div class="container">
<div class="card">
<div class="card-content">
<div class="tag-chips">
<span onclick="showTable()" id="sp-table" class="chip center-align waves-effect waves-light default" data-tagname="时间列表" style="background: linear-gradient(to right, rgb(76, 191, 48) 0%, rgb(15, 157, 88) 100%); color: rgb(255, 255, 255);">时间列表
</span>
<span onclick="showTime()" id="sp-timeline" class="chip center-align waves-effect waves-light default" data-tagname="时间轴" style="background: rgb(249, 235, 234); color: rgba(0, 0, 0, 0.6);">时间轴
</span>
</div>
</div>
</div>
</div>

实现切换效果的js代码:

<script>
function showTime() { 

$("#cd-timeline").show();
$("#cd-table").hide();
$("#sp-timeline").css('background', 'linear-gradient(to right, #4cbf30 0%, 
#0f9d58 100%)');
$("#sp-timeline").css('color', '#fff');
$("#sp-table").css('background', '#F9EBEA')
$("#sp-table").css('color', 'rgba(0,0,0,0.6)');
}
function showTable() { 

$("#cd-timeline").hide();
$("#cd-table").show();
$("#sp-table").css('background', 'linear-gradient(to right, #4cbf30 0%, 
#0f9d58 100%)');
$("#sp-table").css('color', '#fff');
$("#sp-timeline").css('background', '#F9EBEA')
$("#sp-timeline").css('color', 'rgba(0,0,0,0.6)');
}
</script>

切换到时间列表的代码:

<div id="cd-table" class="container archive-container">
<% page.posts.each(function(post) { %>
<div class="card">
<div class="card-content">
<div class="archive">
<%# year. %>
<% if (date(post.date, 'YYYY') != year) { %>
<% year = date(post.date, 'YYYY'); %>
<h4 class="archive-year" id="<%- year %>"><%- year %>年
</h4>
<% } %>
<div class="articles">
<div class="article content>">
<div class="article-sort-post">
<div class="article-sort-item_title">
<a href="<%- url_for(post.path) %>">
<h5>
<i class="fa fa-clock" style="font-size: 1rem;cursor: pointer;">														 </i>
<time class="is-text-small" datetime="2021-08-20T20:20:00.000Z" itemprop="datePublished" style="color: #ff542d;">
<%- date(post.date, 'YYYY-MM-DD') %>
</time>
</h5>
</a>
<h6 class="is-6">
<a href="<%- url_for(post.path) %>"></a>
<a href="<%- url_for(post.path) %>">
<%= post.title %></a>
</h6>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
<% }); %>
</div>

查看我的效果:

归档时间列表效果归档时间列表效果

Front-matter

Front-matter 选项详解

Front-matter 选项中的所有内容均为非必填的。但仍然建议至少填写 titledate 的值。

配置选项 默认值 描述
title Markdown 的文件标题 文章标题,强烈建议填写此选项
date 文件创建时的日期时间 发布时间,强烈建议填写此选项,且最好保证全局唯一
author _config.yml 中的 author 文章作者
img featureImages 中的某个值 文章特征图
top true 推荐文章(文章是否置顶),如果 top 值为 true,则会作为首页推荐文章
cover false 表示该文章是否需要加入到首页轮播封面中
coverImg 表示该文章在首页轮播封面需要显示的图片路径,如果没有,则默认使用文章的特色图片
password 文章阅读密码,如果要对文章设置阅读验证密码的话,就可以设置 password 的值,该值必须是用 SHA256 加密后的密码,防止被他人识破。前提是在主题的 config.yml 中激活了 verifyPassword 选项
toc true 是否开启 TOC,可以针对某篇文章单独关闭 TOC 的功能。前提是在主题的 config.yml 中激活了 toc 选项
mathjax false 是否开启数学公式支持 ,本文章是否开启 mathjax,且需要在主题的 _config.yml 文件中也需要开启才行
summary 文章摘要,自定义的文章摘要内容,如果这个属性有值,文章卡片摘要就显示这段文字,否则程序会自动截取文章的部
tags 文章标签,一篇文章可以多个标签
categories 文章分类,本主题的分类表示宏观上大的分类,只建议一篇文章一个分类
keywords 文章标题 文章关键字,SEO 时需要
reprintPolicy cc_by 文章转载规则, 可以是 cc_by, cc_by_nd, cc_by_sa, cc_by_nc, cc_by_nc_nd, cc_by_nc_sa, cc0, noreprint 或 pay 中的一个

注意:

  1. 如果 img 属性不填写的话,文章特色图会根据文章标题的 hashcode 的值取余,然后选取主题中对应的特色图片,从而达到让所有文章都的特色图各有特色。

  2. date 的值尽量保证每篇文章是唯一的,因为本主题中 Gitalk 和 Gitment 识别 id 是通过 date 的值来作为唯一标识的。

  3. 如果要对文章设置阅读验证密码的功能,不仅要在 Front-matter 中设置采用了 SHA256 加密的 password 的值,还需要在主题的 _config.yml 中激活了配置。有些在线的 SHA256 加密的地址,可供使用:开源中国在线工具、chahuo、站长工具。

  4. 您可以在文章md文件的front-mater中指定reprintPolicy来给单个文章配置转载规则.

最全示例

---
title: 基于Hexo的hexo-theme-matery主题搭建博客并优化
date: 2019-10-03 14:25:00
author: 悟尘
img: /source/images/xxx.jpg
top: true
cover: true
coverImg: /images/1.jpg
password: 8d969eef6ecad3c29a3a629280e686cf0c3f5d5a86aff3ca12020c923adc6c92
toc: false
mathjax: false
summary: 这是你自定义的文章摘要内容,如果这个属性有值,文章卡片摘要就显示这段文字,否则程序会自动截取文章的部分内容作为摘要
categories: 工具
tags:
- blog
- hexo
---