hexo博客搭建及主题优化(二)

4 篇文章 0 订阅
订阅专栏

个人博客

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

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

HexoBlog

  • 个人博客
    • 主题优化二
      • 21. 网站log设置
      • 22. 网站动态标题行
      • 23. about页面添加个人简历
      • 24. 404页面
      • 25. 网站页脚修改
      • 26. 添加动漫人物
      • 27. 雪花和樱花效果
        • 添加雪花飘落效果
        • 添加樱花飘落效果
      • 28. 鼠标点击爱心效果
        • 鼠标点击样式1
        • 鼠标点击样式2
      • 29. 修改博客文章模板
      • 30. 在线编辑hexo博客
      • 31. 站点统计不显示问题
      • 32. 文章链接部分超长处理
      • 33. 静态/动态彩带, 背景canvas
        • 背景静态彩带
        • 背景动态彩带
        • 背景canvas
      • 34. 提取相册(壁纸)
      • 35. 添加相册列表
        • 添加`清单-相册`菜单
        • 添加相册配置文件
        • 添加相册布局文件
        • 添加相册列表样式
        • 相册列表效果
        • 相册密码设置
      • 36. 修改导航栏不透明
      • 37. 添加快捷导航
      • 38. 添加音乐导航页
      • 39. 生成电影卡片
      • 40. 添加夜间模式切换
      • 41. 添加valine评论功能
      • 42. 添加artitalk说说功能
      • 43. 归档时间轴添加时间列表的切换
    • Front-matter
      • Front-matter 选项详解
      • 最全示例

上篇 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('&#10052;'),
            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="<%- theme.jsDelivr.url %><%- url_for('/js/snow.js' ) %>"></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="<%- theme.jsDelivr.url %><%- url_for('/js/sakura.js') %>"></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="<%- theme.jsDelivr.url %><%- url_for('/js/wenzi.js') %>"></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在线写博客

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="<%- theme.jsDelivr.url %><%- url_for(ribbonSrc) %>" 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="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.ribbon_dynamic) %>" 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="<%- theme.jsDelivr.url %><%- url_for(ribbonSrc) %>" 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 style="color:#97b8b2;border-radius: 10px;>'
                + '<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 的值。

配置选项默认值描述
titleMarkdown 的文件标题文章标题,强烈建议填写此选项
date文件创建时的日期时间发布时间,强烈建议填写此选项,且最好保证全局唯一
author_config.yml 中的 author文章作者
imgfeatureImages 中的某个值文章特征图
toptrue推荐文章(文章是否置顶),如果 top 值为 true,则会作为首页推荐文章
coverfalse表示该文章是否需要加入到首页轮播封面中
coverImg表示该文章在首页轮播封面需要显示的图片路径,如果没有,则默认使用文章的特色图片
password文章阅读密码,如果要对文章设置阅读验证密码的话,就可以设置 password 的值,该值必须是用 SHA256 加密后的密码,防止被他人识破。前提是在主题的 config.yml 中激活了 verifyPassword 选项
toctrue是否开启 TOC,可以针对某篇文章单独关闭 TOC 的功能。前提是在主题的 config.yml 中激活了 toc 选项
mathjaxfalse是否开启数学公式支持 ,本文章是否开启 mathjax,且需要在主题的 _config.yml 文件中也需要开启才行
summary文章摘要,自定义的文章摘要内容,如果这个属性有值,文章卡片摘要就显示这段文字,否则程序会自动截取文章的部
tags文章标签,一篇文章可以多个标签
categories文章分类,本主题的分类表示宏观上大的分类,只建议一篇文章一个分类
keywords文章标题文章关键字,SEO 时需要
reprintPolicycc_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
---
React图片百叶窗切换特效
07-24
React图片百叶窗切换特效是一款基于React制作的全屏的旅游城市景点图片滑动百叶窗轮播切换效果。
React响应导航栏
02-15
响应式应用程序导航栏 使用和构建 安装 1)克隆存储库 $ git clone https://github.com/randallmlough/react-responsive-navbar.git $ cd react-responsive-navbar 2)安装依赖项 $ yarn 3)预览/开发版本 $ yarn start 4)生产建设 $ yarn build
推荐一款优雅的Hexo主题:AirCloud
最新发布
gitblog_00003的博客
04-17 317
推荐一款优雅的Hexo主题:AirCloud 项目地址:https://gitcode.com/aircloud/hexo-theme-aircloud 如果你是一位热衷于写作,对网站设计有着独特品味的技术爱好者,那么AirCloud这款Hexo主题可能会吸引你的注意力。它是一个简洁而精致的开源主题,旨在为你的个人站点增添一份清新和专业的感觉。 项目简介 AirCloud 是由 AirClou...
Android开发之React Navigation 导航栏样式调整+底部角标消息提示
08-26
主要介绍了React Navigation 导航栏样式调整+底部角标消息提示的相关知识,非常不错,具有一定的参考借鉴价值 ,需要的朋友可以参考下
ReactNative底部导航栏(带消息圆点)
11-10
React Native:使用react-native-scrollable-tab-view库的ScrollableTabView实现了想要的功能(控制Tab页面的显示和隐藏、带消息圆点指示器)
Python Scrapy中yield Request的理解
u014689510的博客
06-07 13万+
最近在看《learn scrapy》中的关于爬虫的部分,对于parse中的yield Request用法不是很理解,现在总结下。 def parse(self, response): # Get the next index URLs and yield Requests next_selector = response.xpath('//*[contains(@class,' '
小程序自定义导航栏公共组件
qq_41232312的博客
10-30 1万+
子组件.js Component({ properties: { myProperty: { type: Object, value: { bg_color: “white”, color: “#000”, flag: 1, name: “” } }, commonHeadHeight: { type: Object, value: {} } }, lifetimes: { // 生命周期函数,可以为函数,或一个在methods段中定义的方法名 attached: function() {}, moved:
vue嵌入app中——首次加载慢的动画
热门推荐
未来以来 futureToday
02-25 45万+
背景:项目首次加载过慢,需要下载的资源比较大;白屏时间过长,导致用户体验不好; 解决办法:index.html 下载完,(首页下载完)就像执行动画 伸手党代码: 加粗样式 <script> //在页面未加载完毕之前显示的loading Html自定义内容 var _LoadingHtml = ` <div id="loadingDiv" style="position:absolute;left:50%;top:50%;margin-left:-35px;margi
Kubeadm 部署 Kubernetes1.20.4集群
Liqiufeng_的博客
02-20 1万+
Kubeadm 部署 Kubernetes1.20.4集群(2021-02-19) 一、环境说明 主机名 IP地址 角色 系统 Kubernetes 192.168.144.156 k8s-master Centos7.6 k8s-node1 192.168.144.157 k8s-node Centos7.6 k8s-node2 192.168.144.158 k8s-node Centos7.6 注意:官方建议每台机器至少双核2G内存,同时需确保MAC和product_uu
160 - 6 aLoNg3x.1
123
02-07 3567
160 - 6 aLoNg3x.1
配置默认路由配置如下:
宛白#永康技术实战型博客
02-13 3162
拓扑图如下: 准备3台AR1220路由器,两台S3700交换机 给pc机配置IP地址和子网掩码 pc1: 192.168.1.2/24 pc2: 192.168.4.2/24 =============================================================== 这里配置简单的 修改主机名 IP地址和子网掩码: AR-1: &...
Hexo搭建文件(next主题
12-07
Hexo需要git和nodejs,这里时next的所有的配置文件,都是自己改过来的。
hexo matery主题解析bug问题及主题优化
01-08
matery主题优化时解析出bug matery是个好的主题这个我就不详细地说了,但是优化也难 越好的主题、越丰富的主题,插件越多,配置越多,就容易出错 最要命的是这种静态网页的部署还要用最原始的方法进行,先在命令行...
hexo nodejs +搭建+butteflr主题自用
07-26
自己的魔改模板备用,hexo nodejs +搭建+butteflr主题自用
基于Hexo搭建系统源码.zip
11-02
基于Hexo搭建系统源码.zip基于Hexo搭建系统源码.zip基于Hexo搭建系统源码.zip基于Hexo搭建系统源码.zip基于Hexo搭建系统源码.zip
Hexo搭建脚本
01-19
写blog虽然经历了N多不同时代的产品,恒久不变的始终是自己无人问津的网站。虽然没几个人看,还是隔断时间就要折腾一下。从最开始的wordpress,到tale,到现在的hexo,网站变得越来越简单,越来越轻量级。
微信小程序实现动态时间滚动
GXQ的博客
05-25 2万+
github地址:https://github.com/axlsdtkl/wechat-mini/tree/master/time_demo 效果为打开微信小程序,实现动态时间滚动: 文件目录: index.js: import { dateFormat } from "../../utils/index.js" const app = getApp() Page({ data: { timer: null, currentDate: "" }, onLoad: funct
http://my.qy6.com/login.php,MaliStore/app.wxss at master · tomzj/MaliStore · GitHub
weixin_34471637的博客
03-13 45万+
src: url(data:font/truetype;charset=utf-8;base64,AAEAAAALAIAAAwAwR1NVQrD+s+0AAAE4AAAAQk9TLzJXBku4AAABfAAAAFZjbWFwX864igAABTQAAA5iZ2x5ZrXmsl0AABVMAADKgGhlYWQR8usHAAAA4AAAADZoaGVhCPIF2QAAALwAAAAkaG10eGX...
signature=112da3f3fab3c7ee4be8e6f16717577b,yarn.lock
weixin_34177886的博客
05-29 4万+
# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.# yarn lockfile v1"7zip-bin@~5.0.3":version "5.0.3"resolved "https://registry.yarnpkg.com/7zip-bin/-/7zip-bin-5.0.3.tgz#bc5b5532ecafd923...
hexo github搭建个人
07-28
要使用 Hexo 和 GitHub 搭建个人,可以按照以下步骤进行操作: 1. 首先,确保你已经安装了 Node.js 和 Git。如果没有安装,可以去官网下载并按照提示进行安装。 2. 打开终端(或命令行界面),使用 npm 命令...

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
写文章

热门文章

  • EasyExcel实现Excel文件导入导出 65472
  • hexo博客搭建及主题优化(二) 38800
  • EasyPoi实现excel文件导入导出 17583
  • idea如何开启Run Dashboard 11698
  • IDEA中实现Mapper接口到映射文件xml的跳转 10525

分类专栏

  • 设计模式 14篇
  • Shell 1篇
  • SpringBoot 15篇
  • 工具篇 11篇
  • Linux 2篇
  • 消息队列MQ 5篇
  • Excel文件导入导出 9篇
  • 面试相关 6篇
  • 微服务 15篇
  • Java 1篇
  • Hexo博客 4篇
  • Redis 4篇
  • tomcat部署 1篇
  • 任务调度 1篇
  • Mysql 1篇

最新评论

  • EasyExcel实现Excel文件导入导出

    m0_68526932: 你配享太庙

  • 新版Intellij IDEA的Plugins搜索不了插件

    锡兰1: 感谢,可以了

  • IDEA启动外部tomcat报异常:javax.management.InstanceNotFoundException: Catalina:type=Server

    hhgang_11: 大神太牛了,致敬表情包

  • 建造者设计模式

    CSDN-Ada助手: 恭喜你这篇博客进入【CSDN每天值得看】榜单,全部的排名请看 https://bbs.csdn.net/topics/617643206。

  • IDEA启动外部tomcat报异常:javax.management.InstanceNotFoundException: Catalina:type=Server

    HYF4038: 感谢大佬, 大佬请收下我的膝盖.

大家在看

  • Java各种基础类的介绍
  • three.js 通用 shaderToy着色器使用,切换
  • Laravel 高级版:鲜为人知但实用的 Composer 命令
  • 您从未使用过的 10 种 Laravel 集合方法
  • Java基础api之String介绍,实现原理,String的方法,StringBuilder类的介绍

最新文章

  • 策略设计模式
  • 模板设计模式
  • 观察者设计模式
2024年3篇
2023年28篇
2022年21篇
2021年37篇

目录

目录

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43元 前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值

深圳SEO优化公司镇江网站改版报价泰安至尊标王哪家好阜阳外贸网站制作报价天水品牌网站设计推荐陇南模板网站建设多少钱秦皇岛设计公司网站哪家好鞍山网站推广方案哪家好云浮网站优化按天计费推荐宝安网站优化排名多少钱眉山营销型网站建设报价德阳如何制作网站多少钱沙井百搜标王报价荷坳百度爱采购价格厦门网站搜索优化多少钱娄底百度网站优化价格安顺SEO按效果付费价格莆田百度竞价价格洛阳优化公司濮阳网站排名优化哪家好恩施阿里店铺运营公司邵阳网站推广工具丹竹头百度标王报价伊犁网站设计模板公司石岩企业网站改版哪家好荆州网站优化排名哪家好周口seo网站优化推荐垦利网站推广方案晋城关键词按天扣费推荐垦利建设网站多少钱丹竹头设计网站推荐歼20紧急升空逼退外机英媒称团队夜以继日筹划王妃复出草木蔓发 春山在望成都发生巨响 当地回应60岁老人炒菠菜未焯水致肾病恶化男子涉嫌走私被判11年却一天牢没坐劳斯莱斯右转逼停直行车网传落水者说“没让你救”系谣言广东通报13岁男孩性侵女童不予立案贵州小伙回应在美国卖三蹦子火了淀粉肠小王子日销售额涨超10倍有个姐真把千机伞做出来了近3万元金手镯仅含足金十克呼北高速交通事故已致14人死亡杨洋拄拐现身医院国产伟哥去年销售近13亿男子给前妻转账 现任妻子起诉要回新基金只募集到26元还是员工自购男孩疑遭霸凌 家长讨说法被踢出群充个话费竟沦为间接洗钱工具新的一天从800个哈欠开始单亲妈妈陷入热恋 14岁儿子报警#春分立蛋大挑战#中国投资客涌入日本东京买房两大学生合买彩票中奖一人不认账新加坡主帅:唯一目标击败中国队月嫂回应掌掴婴儿是在赶虫子19岁小伙救下5人后溺亡 多方发声清明节放假3天调休1天张家界的山上“长”满了韩国人?开封王婆为何火了主播靠辱骂母亲走红被批捕封号代拍被何赛飞拿着魔杖追着打阿根廷将发行1万与2万面值的纸币库克现身上海为江西彩礼“减负”的“试婚人”因自嘲式简历走红的教授更新简介殡仪馆花卉高于市场价3倍还重复用网友称在豆瓣酱里吃出老鼠头315晚会后胖东来又人满为患了网友建议重庆地铁不准乘客携带菜筐特朗普谈“凯特王妃P图照”罗斯否认插足凯特王妃婚姻青海通报栏杆断裂小学生跌落住进ICU恒大被罚41.75亿到底怎么缴湖南一县政协主席疑涉刑案被控制茶百道就改标签日期致歉王树国3次鞠躬告别西交大师生张立群任西安交通大学校长杨倩无缘巴黎奥运

深圳SEO优化公司 XML地图 TXT地图 虚拟主机 SEO 网站制作 网站优化