个人博客: https://www.crystalblog.xyz/
备用地址: https://wang-qz.gitee.io/crystal-blog/
上篇hexo博客搭建及主题优化(一)
下篇hexo博客搭建及主题优化(三)
主题目录下的_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
下面 , 也可以使用外链图片.
实现方法,引入 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>
打开主题目录下的 /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> <%- __('个人简历') %>
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.
原来的主题没有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>
<% } %>
根据自己需要修改主题目录下的/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 ©
<% 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>
| Powered by <a href="https://hexo.io/" target="_blank">Hexoa>
| <a href="https://github.com/blinkfox/hexo-theme-matery" target="_blank">Materya>
<br>
<% if (theme.postInfo.totalCount) { %>
<span style="margin-left: -20px; display: inline">
<i class="fas fa-chart-area">i> <%- __('siteTotalWords') %>: <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">
| <i class="far fa-eye">i> <%- __('siteTotalVisits') %>: <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">
| <i class="fas fa-users">i> <%- __('siteTotalVisitors') %>: <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>
<% } %>
|
<% 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">
| <i class="fas fa-users"></i> <%- __('siteTotalVisitors') %>: <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>
安装插件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
在主题目录下新增/source/js/snow.js
文件, 添加内容:
/*样式一*/
(function ($) {
$.fn.snow = function (options) {
var $flake = $('').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) { %>
<% } %>
在主题目录下_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) { %>
<% } %>
在主题目录下_config.yml
里配置:
# 樱花特效
sakura:
enable: true
在主题目录下/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);
在主题目录下/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 = $("").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) { %>
<% } %>
在主题目录下_config.yml
里配置:
# 鼠标点击特效
wenzi:
enable: true
为了新建文章方便,我们可以修改一下文章模板,可以将/scaffolds/post.md
修改为如下代码:
title: {{ title }}
date: {{ date }}
author:
img:
cover: false
coverImg:
top: false
toc: true
mathjax: false
password:
summary:
keywords:
tags:
categories:
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上还是无法使用.
有时候请求busuanzi数据比较慢,然后浏览量和访问人数就会隐藏,可能是默认的,在matery.css
中增加以下代码可以让它一直显示.
#busuanzi_container_site_pv,
#busuanzi_value_site_pv,
#busuanzi_container_site_uv {
display: inline !important;
}
在/source/css/matery.css
中增加以下代码:
/*文章链接超长部分隐藏*/
.reprint__type {
display: inline-block;
width: 100%;
overflow: hidden;
}
主题目录下的/layout/layout.ejs
文件主题目录下_config.yml
中静态彩带的配置项:
<% if (theme.ribbon.enable) { %>
<% var ribbonSrc = theme.ribbon.clickChange ? theme.libs.js.ribbon : theme.libs.js.ribbonRefresh; %>
<% } %>
主题目录下_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) { %>
<% } %>
主题目录下_config.yml
中动态彩带的配置项:
# 背景动态彩带.
ribbon_dynamic:
enable: true # 改为true即可开启背景动态彩带
主题目录下的/layout/layout.ejs
文件主题目录下_config.yml
中canvas
的配置项:
<% if (theme.ribbon.enable) { %>
<% var ribbonSrc = theme.ribbon.clickChange ? theme.libs.js.ribbon : theme.libs.js.ribbonRefresh; %>
<% } %>
主题目录下_config.yml
中canvas
的配置项:
#背景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
是全局布局文件, 可以自己添加自定义效果, 方式同上面添加雪花飘落.
新建相册(壁纸)文件
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
文件的如下部分:
相册读取的图片配置路径在主题路径下的_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;
}
相册列表
可以参考博客的友情链接
界面, 将友链信息存放在/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> <%- __('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>
相册列表中的每个相册都是一个标签, 点击单个相册会跳转到相册图片展示页面, 在主题目录下新建/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加密.
透明导航栏经常给我造成阅读障碍,可以设置不使用透明导航栏, 指定好看的颜色. 找到主题目录下的/source/css/matery.css
文件,修改如下部分:
header .nav-transparent {
background-color: transparent !important;
/*background-color: #000B3F;*/ /*修改导航栏不透明 #16103f #4cbf30 #7371BC*/
background-image: none;
box-shadow: none;
}
在博客根目录下的\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">xdiv>
<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">51CTOa>
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">社区 · Codediv>
<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">GitHuba>
li>
<li>
<a href="https://coding.net/" class="link-3"
target="_blank">Codinga>
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">CSDNa>
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">在线PSa>
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">WikeHoma>
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">Boosa>
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;
%>
在博客根目录下新增\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> <%- 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 # 隐藏歌词
安装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 相应的电影网址获取
在主题目录下的/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 = '+ '晚上使用深色模式阅读更好哦。(゚▽゚)/'
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*******/
使用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 = `请填写正确的昵称和邮箱!`;
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加密后的博主邮箱 [email protected]
metaPlaceholder: # 输入框的背景文字
nick: 昵称/QQ号(必填)
mail: 邮箱(必填)
link: 网址(https://)
lang: zh-CN
tagMeta: # 评论标签要显示的文字
- 博主
- 小伙伴
- 访客
friends: # md5 加密后的小伙伴邮箱
- 410739a5ad278251b305dab085768c57 # [email protected]
- 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>
参考资料: Hexo博客Matery主题添加ArtiTalk说说模块
具体使用见artitalk官方文档 , 发布说说的数据可以存储在LeanCloud, 上面评论功能已经提供了相关使用参考资料.
首先, 下载artitalk源码
git clone [email protected]: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/
归档的布局文件是主题目录下的/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
选项中的所有内容均为非必填的。但仍然建议至少填写 title
和 date
的值。
配置选项 | 默认值 | 描述 |
---|---|---|
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 中的一个 |
注意:
如果 img 属性不填写的话,文章特色图会根据文章标题的 hashcode 的值取余,然后选取主题中对应的特色图片,从而达到让所有文章都的特色图各有特色。
date 的值尽量保证每篇文章是唯一的,因为本主题中 Gitalk 和 Gitment 识别 id 是通过 date 的值来作为唯一标识的。
如果要对文章设置阅读验证密码的功能,不仅要在 Front-matter 中设置采用了 SHA256 加密的 password 的值,还需要在主题的 _config.yml 中激活了配置。有些在线的 SHA256 加密的地址,可供使用:开源中国在线工具、chahuo、站长工具。
您可以在文章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
---