hexo blog building and theme optimization

crystalBlog

Part I hexo blog building and theme optimization (I)

Theme optimization II

21. Website log settings

Under the subject directory_ In the config.yml configuration file:

# Configure website favicon and website LOGO
## local
#favicon: /favicon.png
#logo: /medias/logo.png
# The CDN I use here can also use local files
favicon: https://cdn.jsdelivr.net/gh/guixinchn/image/blog/favicon.png
logo: https://cdn.jsdelivr.net/gh/guixinchn/image/blog/logo.png

Picture resources are under \ themes \ hexo theme matrix \ source \ medias in the theme directory. You can also use external chain pictures

22. Website dynamic title line

Implementation method: introduce JS file, create funnyTitle.js under / source/js / under the theme file, and add the following code:

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 = 'I believe you will come back!';
         clearTimeout(titleTime);
     }
     else {
         $('[rel="icon"]').attr('href', "https://cdn.jsdelivr.net/gh/guixinchn/image/blog/favicon.png");
         document.title = 'Ha ha, I knew it!' + OriginTitle;
         titleTime = setTimeout(function () {
             document.title = OriginTitle;
         }, 2000);
     }
 });

Then import it in / layout/layout.ejs under the theme directory

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

23. Add a resume to the about page

Open the / layout/about.ejs file in the theme directory and add the following code:

<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;<%- __('curriculum vitae') %>
   </div>
   <div id="articleContent" data-aos="fade-up">
      <%- page.content %>
   </div>
</div>

In the / layout/about.ejs file in the theme directory, the relevant information about the profile in the following code is from the theme_ Get from config.yml configuration file

<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>

Under the subject directory_ The profile information in the config.yml configuration file can be modified

# profile in about page, including avatars, career, and personal introductions.
# Configure personal information in the about page, including avatar, occupation and personal introduction
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. Page 404

The original topic does not have a 404 page. First, create a 404.md in the / source / directory under the topic directory, as follows:

---
title: 404
date: 2017-07-19 16:41:10
type: "404"
layout: "404"
description: "The page you visited was taken away by aliens :("
---

Then create a new / layout/404.ejs file in the theme directory, as follows:

<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>
        // 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. Website footer modification

Modify / layout under the theme directory as needed/_ Partial / footer.ejs file, you can set site visits, number of visitors, word count, site running time, website filing and other information

<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>

            <!-- Running days reminder. -->
            <% 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);
                        // Distinguish whether there is a year
                        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 = 'This station has been running ' + diffDays + ' day';
                            } else if (language === 'zh-HK') {
                                daysTip = 'This station has been running ' + diffDays + ' day';
                            }
                            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 = 'This station has been running ' + diffYears + ' year ' +
                                   diffDays + ' day ' + diffHours + ' hour ' + 
                                   diffMinutes + ' minute ' + diffSeconds + ' second';
                            } else if (language === 'zh-HK') {
                                yearsAndDaysTip = 'This station has been running ' + diffYears + ' year ' + 
                                   diffDays + ' day';
                            }
                            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>

You can also add Baidu garlic statistics

/ layout found/_ Partial / footer.ejs file, modify the corresponding style to

<!--Total number of visitors-->
<% 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>
<% } %>
<!--Finally add-->
<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. Add anime characters

Install plug-ins hexo-helper-live2d

npm install --save hexo-helper-live2d

Install and download the animated character library. There are many animated characters. You can query information online. Several are recommended below

npm install --save live2d-widget-model-shizuku #Desk girl
npm install --save live2d-widget-model-hibiki  #domineering lady
npm install --save live2d-widget-model-wanko   #Dog
npm install --save live2d-widget-model-haruto  #Navy girl
npm install --save live2d-widget-model-miku    #lolita

Blog root_ config.yml file configuration:

## Add animation live2d module NPM install -- save hexo helper live2d
## Download animated character library npm install live2d-widget-model-z16 -D
live2d:
  enable: true
  scriptFrom: local # default
  pluginRootPath: live2dw/ # The root directory (relative path) of the plug-in on the site
  pluginJsPath: lib/ # Script file relative to plug-in root directory path
  pluginModelPath: assets/ # Model file relative to plug-in root directory path
  tagMode: false # Label mode, do you want to replace only live2d tag labels instead of inserting them into all pages
  debug: false # Debug, whether to output logs on the console
  model:
    use: live2d-widget-model-miku
  display:
    position: right #Animation position
    width: 150
    height: 190
    # Position configuration. This is in the middle of the left sidebar
    hOffset: 50  # Adjust horizontal position
    vOffset: -5  # Adjust vertical position
  mobile:
    show: false # Show on mobile device
    scale: 0.5 # Zoom on mobile devices
  react:
    opacityDefault: 0.7
    opacityOnHover: 0.8

27. Snowflake and cherry blossom effect

Add snowflake falling effect

Add a new / source/js/snow.js file in the theme directory. Add the following contents:

/*Style one*/
(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" /* Here you can define the color of snowflakes. If you want white, you can change it to #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, /* Define minimum snowflake size */
        maxSize: 50,/* Define maximum snowflake size */
        newOn: 300  /* Define the density. The smaller the number, the more dense it is */
    });
});
/*Style II*/

/* Control snow */
function snowFall(snow) {
    /* Configurable properties */
    snow = snow || {};
    this.maxFlake = snow.maxFlake || 200;   /* Maximum number of slices */
    this.flakeSize = snow.flakeSize || 10;  /* Snowflake shape */
    this.fallSpeed = snow.fallSpeed || 1;   /* Falling speed */
}

/* Compatible writing */
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;
/* It began to snow */
snowFall.prototype.start = function () {
    /* Create canvas */
    snowCanvas.apply(this);
    /* Create a snowflake shape */
    createFlakes.apply(this);
    /* Painting snow */
    drawSnow.apply(this)
}

/* Create canvas */
function snowCanvas() {
    /* Add Dom node */
    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");
    /* Processing of window size change */
    window.onresize = function () {
        snowcanvas.width = window.innerWidth;
        /* snowcanvas.height = window.innerHeight */
    }
}

/* Snow moving object */
function flakeMove(canvasWidth, canvasHeight, flakeSize, fallSpeed) {
    this.x = Math.floor(Math.random() * canvasWidth);   /* x coordinate */
    this.y = Math.floor(Math.random() * canvasHeight);  /* y coordinate */
    this.size = Math.random() * flakeSize + 2;          /* shape */
    this.maxSize = flakeSize;                           /* Maximum shape */
    this.speed = Math.random() * 1 + fallSpeed;         /* Falling speed */
    this.fallSpeed = fallSpeed;                         /* Falling speed */
    this.velY = this.speed;                             /* Y Directional velocity */
    this.velX = 0;                                      /* X Directional velocity */
    this.stepSize = Math.random() / 30;                 /* step */
    this.step = 0                                       /* Number of steps */
}

flakeMove.prototype.update = function () {
    var x = this.x,
        y = this.y;
    /* Swing left and right (cosine) */
    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;
    /* Handling of flying out of boundary */
    if (this.x >= canvas.width || this.x <= 0 || this.y >= canvas.height || this.y <= 0) {
        this.reset(canvas.width, canvas.height)
    }
};
/* Fly out of the boundary - place the top and continue to fall */
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;
};
// Render snowflake - random shape (snowflake color can be modified here!!!)
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)");  /* Here is the snowflake color. The default is white */
    snowFlake.addColorStop(.5, "rgba(255, 255, 255, 0.5)"); /* If you want to change to another color, please check it yourself */
    snowFlake.addColorStop(1, "rgba(255, 255, 255, 0)");    /* Find the hexadecimal RGB color code. */
    ctx.save();
    ctx.fillStyle = snowFlake;
    ctx.beginPath();
    ctx.arc(this.x, this.y, this.size, 0, Math.PI * 2);
    ctx.fill();
    ctx.restore();
};

/* Create snowflakes - define shapes */
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))
    }
}

/* Painting snow */
function drawSnow() {
    var maxFlake = this.maxFlake,
        flakes = this.flakes;
    ctx = this.ctx, canvas = this.canvas, that = this;
    /* Empty snowflakes */
    ctx.clearRect(0, 0, canvas.width, canvas.height);
    for (var e = 0; e < maxFlake; e++) {
        flakes[e].update();
        flakes[e].render(ctx);
    }
    /*  Frame by frame */
    this.loop = requestAnimationFrame(function () {
        drawSnow.apply(that);
    });
}

/* Call and control method */
var snow = new snowFall({maxFlake: 60});
snow.start();

Add the following code in / layout/layout.ejs under the Theme Directory:

<!-- Snow effect -->
<% if (theme.snow.enable) { %>
    <script src="<%- theme.jsDelivr.url %><%- url_for('/js/snow.js' ) %>"></script>
<% } %>

Under the topic directory_ Configuration in config.yml:

# Snow effect
snow:
  enable: true

Add Cherry Blossom falling effect

Add a new / source/js/sakura.js file in the theme directory. Add the following contents:

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

Add the following code in / layout/layout.ejs under the Theme Directory:

<!-- Cherry Blossom effect -->
<% if (theme.sakura.enable) { %>
    <script src="<%- theme.jsDelivr.url %><%- url_for('/js/sakura.js') %>"></script>
<% } %>

Under the topic directory_ Configuration in config.yml:

# Cherry Blossom effect
sakura:
  enable: true

28. Mouse click effect

Mouse click style 1

In the / source/libs/others/clicklove.js file under the theme directory, provide the code for mouse click effect

!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);

Mouse click style 2

In the / source/js/wenzi.js file under the theme directory, provide the code for mouse click effect

/* Mouse click text effect */
var a_idx = 0;
jQuery(document).ready(function ($) {
    $("body").click(function (e) {
        // var a = new Array(" ❤ Prosperity and strength ❤","❤ democratic ❤","❤ civilization ❤","❤ harmonious ❤","❤ free ❤","❤ equality ❤","❤ fair ❤","❤ rule by law ❤","❤ Patriotic ❤","❤ be dedicated ❤","❤ sincerity ❤","❤ friendly ❤");
        var a = new Array("Prosperity and strength", "democratic", "civilization", "harmonious", "free", "equality", "fair", "rule by law", "Patriotic", "be dedicated", "sincerity", "friendly");
        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();
            });
    });
});

Add the following code in / layout/layout.ejs under the Theme Directory:

<!-- Mouse click effect -->
<% if (theme.wenzi.enable) { %>
    <script src="<%- theme.jsDelivr.url %><%- url_for('/js/wenzi.js') %>"></script>
<% } %>

Configure in _config.yml under the topic Directory:

# Mouse click effect
wenzi:
  enable: true

29. Modify blog post template

For the convenience of creating a new article, we can modify the article template. You can modify / Scaffolding / post.md to the following code:

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

30. Edit hexo blog online

When editing articles, hexo's native mode is inconvenient. The official website provides a plug-in hexo admin interface and a markdown editor

First install the hexo admin plug-in

npm install --save hexo-admin

Then start hexo s to access http://127.0.0.1:4000/crystalBlog/admin You can edit blog posts conveniently and quickly

After editing, you can also deploy and publish quickly. However, you prefer to write markdown syntax on typera. If you deploy it on your own server, you can use this function, but you still can't use it on gitee or github

31. Site statistics do not show problems

Sometimes it is slow to request busuanzi data, and then the number of views and visitors will be hidden. It may be the default. Add the following code in matrix.css to keep it displayed all the time

#busuanzi_container_site_pv,
#busuanzi_value_site_pv,
#busuanzi_container_site_uv {
    display: inline !important;
}

32. The article link is too long

Add the following code in / source / CSS / material.css:

/*Article link super long part hidden*/
.reprint__type {
    display: inline-block;
    width: 100%;
    overflow: hidden;
}

33. Static / dynamic ribbon, background canvas

Background static ribbon

The configuration item of static ribbon in _config.yml in the theme directory of / layout/layout.ejs file in the Theme Directory:

<!--Background still ribbon-->
    <% 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>
    <% } %>

Configuration items of static ribbon in _config.yml under the Theme Directory:

# Background still ribbon
ribbon:
  enable: false  # Change to true to open the background static ribbon
  size: 150 # RIBBON size, default: 90
  alpha: 0.6 # Ribbon transparency (0 ~ 1), default: 0.6
  zIndex: -1 # The background z-index attribute and css attribute are used to control the position of the layer. The default is: - 1
  clickChange: false  # Set whether to change the ribbon every time you click

Background dynamic ribbon

/ layout/layout.ejs file in the theme directory_ Configuration items of dynamic ribbon in config.yml:

<!--Background dynamic ribbon-->
<% if (theme.ribbon_dynamic.enable) { %>
<script type="text/javascript" src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.ribbon_dynamic) %>" async="async"></script>
<% } %>

Subject directory_ Configuration items of dynamic ribbon in config.yml:

# Background dynamic ribbon
ribbon_dynamic:
  enable: true # Change to true to open the background dynamic ribbon

Background canvas

/ layout/layout.ejs file in the theme directory_ Configuration item of canvas in config.yml:

<!--Background still ribbon-->
<% 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>
<% } %>

Subject directory_ Configuration item of canvas in config.yml:

#Background canvas nest
canvas_nest:
  enable: true
  color: 0,0,255 # Line color, default: '0,0,0'; The three numbers are (R,G,B). Pay attention to the use of and segmentation
  pointColor: 0,0,255 # Intersection color, default: '0,0,0'; The three numbers are (R,G,B). Pay attention to the use of and segmentation
  opacity: 0.7 # Line transparency (0 ~ 1), default: 0.5
  zIndex: -1 # The background z-index attribute and css attribute are used to control the position of the layer. The default is: - 1
  count: 99 # Total number of lines, default: 99

/ layout/layout.ejs in the theme directory is a global layout file. You can add custom effects yourself in the same way as above Snowflakes fall.

34. Extract photo album (wallpaper)

New album (wallpaper) file

hexo new page wallpaper

Modify the under the topic directory_ config.yml file, mine is extracted to the list - Album navigation

Lists:  ##detailed list
  url: /
  icon: fas fa-list
  children:
    - name: music
      url: /musics
      icon: fas fa-music
    - name: film
      url: /movies
      icon: fas fa-film
    - name: read
      url: /books
      icon: fas fa-book
    - name: wallpaper
      url: /wallpaper
      icon: fas fa-image

Modify the site / galleries/index.md file

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

Create a new / layout/wallpaper.ejs file under the theme directory, and add the following contents:

<style type="text/css">
    /* don't remove. */
    .about-cover {
        height: 75vh;
    }
</style>
 
<%- partial('_partial/bg-cover') %>
<!--The following is the code to extract the photo album function of the blog-->
<main class="content">
<% if (theme.myGallery && theme.myGallery.enable) { %>
<%- partial('_widget/my-gallery') %>
<% } %>
</main>
 
<% if (page.total > 1) { %>
<%- partial('_partial/paging') %>
<% } %>

At the same time, note the following parts of the / layout/about.ejs file in the topic Directory:

<!--gallery Function migration to[detailed list-album]Navigation Division-->
<!--<% if (theme.myGallery && theme.myGallery.enable) { %>
<%- partial('_widget/my-gallery') %>
<% } %>-->

The picture configuration path read from the album is under the theme path_ In config.yml file:

# Configure my album pictures on the about page. If you don't need this information, you can set it inactive or delete it
myGallery:
  enable: true
  data:
    - /medias/featureimages/0.jpg
    - /medias/featureimages/1.jpg
    - /medias/featureimages/2.jpg

Modify the album layout, find the / source / CSS / material.css file, and modify the following parts:

.my-gallery {
    margin: 4.5rem auto 1rem;
    padding: 0 1.2rem; /*This is the display width, the front is the page width, and the back is the picture width*/
    max-width: 1100px;
    /*position: relative;*/
}

.my-gallery .photo {
    margin: .5rem 0; /*This is the space between the top and bottom lines*/
    /*position: relative;
   overflow: hidden;*/
}

.my-gallery .photo img {
    width: 100%;
    height: 200px; /*Limit the height and keep the peers at the same height, otherwise it will be very chaotic*/
    border-radius: 10px;
    cursor: pointer;
}

35. Add album list

For the album list, you can refer to the friendship link interface of the blog and store the friendship link information in / source/_ In the data / friends.json file, and then hexo will render the friend chain list according to the structure in the friends.ejs template file. The effect is as follows:

The principle is actually three a tags, which contain avatar, address and other information. Click and jump to the corresponding address. Then we can also customize the configuration file (galleries.json) and template file (galleries.ejs) of an album list, and then hexo reads the configuration file to automatically generate the album list interface. If this can be optimized, To add an album later, you only need to maintain the album information in the configuration file galleries.json. (add, delete and modify)

Let's just do it. Let's start now

Add list - Album menu

Here are several files to modify:

Configuration files under the topic directory_ config.yml. Don't confuse it with the file with the same name in the root directory of the site. Add the following code under menu:

menu: 
 	Lists:  ##detailed list
       url: /crystal-blog
       icon: fas fa-list
       children:
         # Other menus are omitted here
         - name: wallpaper
           url: /wallpaper
           icon: fas fa-image
         - name: album
           url: /galleries
           icon: fas fa-camera

Create a new galleries directory under the site root directory source, and then create index.md under this directory to generate the index.html file

hexo new page "galleries"

Modify the / galleries/index.md file to specify the layout interface:

---
title: My album 
date: 2021-08-25 19:56:35
type: galleries
layout: galleries
---

Add album profile

Create a new / source under the topic directory/_ Data / galleries.json file, add the following album configuration content according to the custom Convention (I maintain three albums):

[
  {
    "name": "Blog background",
    "url": "/gallery2",
    "cover": "https://www.bing.com/th?id=OHR.Mpumalanga_ZH-CN9666962271_tmb.jpg&rf=",
    "description": "Blog background",
    "photos": [
      "https://www.bing.com/th?id=OHR.Mpumalanga_ZH-CN9666962271_tmb.jpg&rf=",
    ]
  },
  {
    "name": "Peng Yuyan",
    "url": "/gallery0",
    "cover": "https://uploadfile.bizhizu.cn/2016/0106/20160106033828391.jpg",
    "description": "Peng Yuyan sketching",
    "photos": [
      "http://i.52desktop.cn:81/upimg/allimg/20191204/2019124151645578778013.jpg",
      "http://i.52desktop.cn:81/upimg/allimg/20191204/2019124151645687778016.jpg"
    ]
  },
  {
    "name": "Lau Andy",
    "url": "/gallery1",
    "cover": "https://tu1.whhost.net/uploads/20181029/18/1540809870-NgSCnhWkcJ.jpg",
    "description": "Andy Lau sketching",
    "photos": [
      "https://uploadfile.bizhizu.cn/2015/0306/20150306103233272.jpg",
      "https://www.beihaiting.com/uploads/allimg/150401/10723-150401195426203.jpg"
    ]
  }
]

Add album layout file

Create a new / layout/galleries.ejs template file in the theme directory. Refer to the friends.ejs file. The modified content is as follows:

<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: 'Chinese block letters'">
                                <%- gallery.name %>
                            </p>
                        </a>
                    </div>

                <% } %>
            </div>
        <% } %>
    </div>
</main>

Each album in the album list is a < a > < / a > label. Clicking a single album will jump to the album picture display page and create / layout/gallery.ejs under the theme directory. Here we can refer to the layout and rendering method of the wallpaper my-gallery.ejs file. The optimized code is as follows:

<%- 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>

Add album list style

/The layout/galleries.ejs template file refers to the gallery.css style file, so you need to add a new / source/css/gallery.css file in the theme directory. The contents are as follows:

.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;
}

/*Adapt mobile terminal layout*/
@media only screen and (max-width: 601px) {
    .waterfall {
        column-count: 2;
        column-gap: 1em;
    }
}

Album list effect

Album display effect

Album password settings

To set the password for the album, you can refer to the blog post password. Add the following code in the / layout/gallery.ejs file under the Theme Directory:

<% if (theme.verifyPassword.enable) { %>
    <script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.crypto) %>"></script>
    <script>
        (function() {
            /*pwd Is the encrypted password of sha256 configured in the blog*/
            let pwd = '<%- page.password %>';
            if (pwd && pwd.length > 0) {
                if (pwd !== CryptoJS.SHA256(prompt('Please enter a password to access this album
                                                   code')).toString(CryptoJS.enc.Hex)) {
                    alert('Wrong password!');
                    location.href = '<%- url_for("/galleries")  %>';
                }
            }
        })();
    </script>
<% } %>

Then set the password in the / source/galleries/gallery3/index.md file in the gallery3 album directory where the password needs to be set. The password needs to be encrypted with SHA256

36. Modify navigation bar opacity

The transparent navigation bar often causes me reading difficulties. You can set not to use the transparent navigation bar and specify good-looking colors. Find the / source / CSS / matrix.css file in the theme directory and modify the following part:

header .nav-transparent {
    background-color: transparent !important;
    /*background-color: #000B3F;*/  /*Modify navigation bar opacity #16103f #4cbf30 #7371BC*/
    background-image: none;
    box-shadow: none;
}

37. Add shortcut navigation

\ source in the blog root directory\_ Create a new shortcut navigation in posts \ navigation \ index.md Directory:

hexo new page navigate

Modify the under the topic directory_ config.yml file, add shortcut navigation menu:

## Quick navigation
menu:
	// ... other menus are omitted here
   Navigate:
     url: /navigate
     icon: fas fa-location-arrow

Modify the site / navigator / index.md file

---
title: Quick navigation
date: 2021-08-29 16:25:05
layout: navigate
---

Create a new / layout / navigation.ejs file under the theme directory, and add the following contents:

<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">Baidu</li>
                            <li class="bing_s">Bing</li>
                            <li class="google_s">Google</li>
                            <li class="baidu_s">Baidu</li>
                        </ul>
                    </div>
                    <input name="wd" id="kw-2" maxlength="100" 
                           autocomplete="off" type="text">
                </div>
                <div class="qingkong" id="qingkong" title="clear · empty" 
                     style="display:block">x</div>
                <input value="Search" 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">programming · study</div>
                    <ul class="jj-list-con">
                        <li>
                            <a href="https://www.oschina.net/" class="link-3" 
                               target="_blank">Open source China</a>
                        </li>
                        <li>
                            <a href="https://htmldog.com/" class="link-3" 
                               target="_blank">HTML dog</a>
                        </li>
                        <li>
                            <a href="https://www.icourse163.org/" class="link-3" 
                               target="_blank">Chinese University Course</a>
                        </li>
                        <li>
                            <a href="https://www.imooc.com/" class="link-3" 
                               target="_blank">Muke network</a>
                        </li>
                        <li>
                            <a href="http://www.wxapp-union.com/" class="link-3" 
                               target="_blank">Applet</a>
                        </li>
                        <li>
                            <a href="https://www.runoob.com/" class="link-3" 
                               target="_blank">Rookie tutorial</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">Laboratory Building</a>
                        </li>
                        <li>
                            <a href="/posts/2d1a17c5.html" class="link-3" 
                               target="_blank">Personal collection page</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">community · Code</div>
                    <ul class="jj-list-con">
                        <li>
                            <a href="https://www.zhangxiaocai.com/contact/" 
                               class="link-3" target="_blank">Leaving a message.
                            </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">Nuggets</a>
                        </li>
                        <li>
                            <a href="https://gitee.com/" class="link-3" 
                               target="_blank">Code cloud</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">Jian Shu</a>
                        </li>
                        <li>
                            <a href="https://segmentfault.com/" class="link-3" 
                               target="_blank">Si no</a>
                        </li>
                        <li>
                            <a href="https://cloud.tencent.com/developer/" 
                               class="link-3" target="_blank">cloud+community
                            </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">practical · tool</div>
                    <ul class="jj-list-con">
                        <li>
                            <a href="https://mdnice.com/" class="link-3"
                               target="_blank">Nice editor </a>
                        </li>
                        <li>
                            <a href="https://translate.google.cn/" class="link-3"
                               target="_blank">Google translation</a>
                        </li>
                        <li>
                            <a href="https://www.uupoop.com/" class="link-3"
                               target="_blank">on-line PS</a>
                        </li>
                        <li>
                            <a href="https://www.processon.com/" class="link-3" 
                               target="_blank">Mind map</a>
                        </li>
                        <li>
                            <a href="https://wallhaven.cc/" class="link-3" 
                               target="_blank">Ultra clear wallpaper</a>
                        </li>
                        <li>
                            <a href="https://cli.im/" class="link-3" 
                               target="_blank">QR code</a>
                        </li>
                        <li>
                            <a href="http://www.yinfans.me/" class="link-3" 
                               target="_blank">Yin Fansi</a>
                        </li>
                        <li>
                            <a href="https://www.extfans.com" class="link-3" 
                               target="_blank">Google plugin</a>
                        </li>
                        <li>
                            <a href="https://my.openwrite.cn/" class="link-3" 
                               target="_blank">OW distribute</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">entertainment · Movies</div>
                    <ul class="jj-list-con">
                        <li>
                            <a href="https://www.jd.com/" class="link-3" 
                               target="_blank">JD.COM</a>
                        </li>
                        <li>
                            <a href="https://www.taobao.com/" class="link-3" 
                               target="_blank">TaoBao</a>
                        </li>
                        <li>
                            <a href="https://www.tmall.com/" class="link-3" 
                               target="_blank">Tmall</a>
                        </li>
                        <li>
                            <a href="https://v.qq.com/" class="link-3" 
                               target="_blank">Tencent video</a>
                        </li>
                        <li>
                            <a href="http://www.iqiyi.com/" class="link-3" 
                               target="_blank">Iqiyi</a>
                        </li>
                        <li>
                            <a href="https://www.bilibili.com/" class="link-3"
                               target="_blank">Bilibili</a>
                        </li>
                        <li>
                            <a href="https://music.163.com/" class="link-3" 
                               target="_blank">NetEase cloud music</a>
                        </li>
                        <li>
                            <a href="https://y.qq.com/" class="link-3" 
                               target="_blank">QQ music</a>
                        </li>
                        <li>
                            <a href="http://www.kugou.com/" class="link-3" 
                               target="_blank">Cool dog music</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">real-time info · trend</div>
                    <ul class="jj-list-con">
                        <li>
                            <a href="https://www.huxiu.com/" class="link-3" 
                               target="_blank">Tiger smell</a>
                        </li>
                        <li>
                            <a href="https://insights.stackoverflow.com/" 
                               class="link-3" target="_blank">Technical investigation
                            </a>
                        </li>
                        <li>
                            <a href="http://www.asciiworld.com/" class="link-3" 
                               target="_blank">loaf on a job</a>
                        </li>
                        <li>
                            <a href="https://sspai.com/" class="link-3"
                               target="_blank">Minority </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">
                                Front end trend
                            </a>
                        </li>
                        <li>
                            <a href="https://github-trending.com/" class="link-3"
                               target="_blank">GitHub trend</a>
                        </li>
                        <li>
                            <a href="https://www.tiobe.com/" class="link-3" 
                               target="_blank">Programming trends</a>
                        </li>
                        <li>
                            <a href="https://trends.google.com/" class="link-3" 
                               target="_blank">Google trend</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">search · other</div>
                    <ul class="jj-list-con">
                        <li>
                            <a href="https://ac.scmor.com/" class="link-3" 
                               target="_blank">Google Image</a>
                        </li>
                        <li>
                            <a href="http://www.pansoso.com/" class="link-3" 
                               target="_blank">Network disk search</a>
                        </li>
                        <li>
                            <a href="http://music.ifkdy.com/" class="link-3" 
                               target="_blank">Music search</a>
                        </li>
                        <li>
                            <a href="https://www.dytt8.net/" class="link-3" 
                               target="_blank">Movie paradise</a>
                        </li>
                        <li>
                            <a href="https://carbon.now.sh/" class="link-3"
                               target="_blank">Code picture</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">Icon Library</a>
                        </li>
                        <li>
                            <a href="https://www.qvdv.com/tools/qvdv-guid.html"
                               class="link-3" target="_blank">
                                Online tools
                            </a>
                        </li>
                        <li>
                            <a href="http://zhongguose.com/"
                               class="link-3" target="_blank">Chinese color</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, "Microsoft YaHei "
    }

    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 under Theme Directory/_ Add Chinese name of shortcut navigation menu and mobile client file / layout to partial / navigation.ejs file/_ Add as partial / mobile-nav.ejs

<%
    var menuMap = new Map();
    menuMap.set("Index", "home page");
    menuMap.set("Tags", "label");
    menuMap.set("Categories", "classification");
    menuMap.set("Archives", "file");
    menuMap.set("About", "about");
    menuMap.set("Contact", "Message Board");
    menuMap.set("Friends", "Links");
    menuMap.set("Lists", "detailed list");
    menuMap.set("Navigate", "Quick navigation");

    var configRoot = config.root
    configRoot = (configRoot === null || configRoot === undefined 
                  || configRoot === '/') ? '' : configRoot;
%>

38. Add music navigation page

Add \ source under the blog root directory\_ Posts \ music \ index.md file

hexo new page musics

Under the subject directory_ Add the [list - music] menu to the config.yml file:

menu:
// Other menu items are omitted here
Lists:  ##detailed list
    url: /
    icon: fas fa-list
    children:
      - name: music
        url: /musics
        icon: fas fa-music

Modify the music file / music / index.md to specify the layout configuration:

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

Add the music layout file / layout / music.ejs under the theme directory, and add the following contents:

<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>

Add a custom music player file / layout in the theme directory/_ Widget / mymusic.ejs, add the following:

<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>

Configuration file under subject directory_ config.yml add custom music player configuration:

# Customize music components and display them on the page you need
mymusic:
  enable: true
  title: #Non bottom suction mode is effective
    enable: true
    show: Relaxed moment
  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       # Turn on the bottom suction mode. The true player will be on the side of the site, and click it to appear
  autoplay: false   # Auto play
  theme: '#42b983'
  loop: 'all'       # Audio cycle playback, optional values: 'all', 'one', 'none'
  order: 'random'   # Audio cycle sequence, optional values: 'list', 'random'
  preload: 'auto'   # Preload, optional values: 'none', 'metadata', 'auto'
  volume: 0.7       # Default volume. Please note that the player will remember the user's settings. After the user manually sets the volume, the default volume will become invalid
  listFolded: false  # List default collapse
  hideLrc: false     # Hide lyrics

39. Generate movie cards

Install the hexo tag mtime plug-in

npm install hexo-tag-mtime --save

Add a movie navigation page, and generate the / source/movies/index.md file in the blog root directory:

hexo new page movies

/Add the following content to the source/movies/index.md file:

---
title: Film recommendation
date: 2021-08-25 19:56:04
type: movies
---
## Wonderful film recommendation
###  Anger · serious case
{% mtime 263501 %}
###  Bye, boy
{% mtime 259370 %}
### Wandering the earth
{% mtime 218707 %}

id Can be on time net https://www.mtime.com the corresponding movie website

40. Add night mode switching

Add the night mode switch button in the / layout/layout.ejs file under the Theme Directory:

<!-- Switch night/Day mode button -->
<a onclick="switchNightMode()" id="sma"> <i class="fa fa-moon-o" id="nightMode" aria-hidden="true"></i> </a>

Add JS code to the / source / JS / material.js file in the topic Directory:

// Dark mode button settings
if (localStorage.getItem('dark') === '1') {
    document.body.classList.add('dark');
} else if (new Date().getHours() >= 22 || new Date().getHours() < 7) {
    /*Regularly turn on dark mode < defau lt on from 22:00 p.m. to 6:00 a.m. >*/
    // 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');
}

// Dark mode settings
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;
    }
}

/*Remind to turn on the night mode function*/
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>It's better to read in dark mode at night.(゚▽゚)/</span>'
            M.toast({html: toastHTML})
        }
    }, 2200);

Add the style of night mode switching in the / source / CSS / material.css file under the Theme Directory:

/******Night mode switching style start*******/
/* Dark mode button settings */
#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";
}

/* Dark mode settings */ /* The font color turns grayish white */
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);
}

/* The background color turns gray */
body.dark .card,
body.dark .block-with-text:after {
    background-color: #282c34;
}

/* The background color turns black */
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;
   /**Some pages cannot be switched to dark background because of my background image. You need to cancel the background image**/
    background-image: url(#); 
}

/* Change transparency */
body.dark .aplayer {
    background: #2f3742 !important;
}

body.dark img, body.dark strong {
    filter: brightness(.7);
}
/******Night mode switching style end*******/

41. Add valine comment function

Using the valine comment function, you can use leanCloud International Store comment data. The specific application ID and KEY tutorials are as follows:

Text tutorial: https://cndrew.cn/2020/04/10/hexo-shuoshuo/

Station B video: https://www.bilibili.com/video/BV16A411b7UF

Create / layout under the theme directory/_ Partial / valine.ejs file, add the following:

<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>comment</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) %> ;
    //There's a new line here
    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">Please fill in the correct nickname and email!</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>

Then, valine.ejs is referenced on the page that needs comment function. My has added comment function to the message board and blog posts, namely / layout/contact.ejs and / layout under the theme directory/_ Add the following to the partial / post-detail.ejs file:

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

Then under the topic directory_ Enable valine comment function in config.yml file, and add some custom attributes used in valine.ejs

# The configuration of Valine comment module is not activated by default. If you want to use it, please activate the configuration item and set appId and appKey
valine:
  enable: true
  appId: NiVMV7aR8ipVDy9EywBYtLwI-MdYXbMMI
  appKey: P7BGA1hMeBAFbHYz3dKY8doJ
  notify: false # Enable email reminder( https://valine.js.org/notify.html)
  verify: true # Enable anti spam verification
  visitor: true
  avatar: monsterid # Default avatar display mode, little monster  # 'mm' # Gravatar style : mm/identicon/monsterid/wavatar/retro/hide
  pageSize: 10
  placeholder: 'fill in QQ You can comment on No. 1. Come on~'  # Comment Box placeholder
  /medias/comment_bg.png #Background map
  background: https://cdn.jsdelivr.net/gh/drew233/cdn/20200409110727.webp 
  coolpushkey:
  # New attribute wang-qz
  comment_count: true
  enableQQ: true  # Forced QQ
  recordIP: true
  requiredFields: # Required information, default [nickname, email, link]
    - nick
    - mail
  guest_info: # The input items displayed in the comment box are [nick, mail, link] by default. Some options can be set not to be filled in
    - nick
    - mail
    - link
  master:
    - e4de1f6da602d22e423d6dfb24611b6b  # md5 encrypted blogger mailbox wang-qz@foxmail.com
  metaPlaceholder:  # Background text of the input box
    nick: nickname/QQ number(Required)
    mail: mailbox(Required)
    link: website(https://)
  lang: zh-CN
  tagMeta: # The text to display in the comment label
    - Blogger
    - buddy
    - visitor
  friends: # md5 encrypted small partner mailbox
    - 410739a5ad278251b305dab085768c57 # 1848276756@qq.com
    - 410739a5ad278251b305dab085768c57
    - 410739a5ad278251b305dab085768c57

In addition, valine.min.js needs to be upgraded. The version I use is valine-1.4:

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

Because the above ` ` valine.ejs file refers to / lib/js/valine/valine.min.js', you can also download it to replace the JS file

You can also clone the following version from gitHub, and then replace the local version with its valine.min.js

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

valine.ejs file refers to valine.min.js, which needs to be in the topic directory_ Search the keyword libs in the config.yml file, find JS, reference the last line of the configuration in JS, and then add:

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

In this way, the / lib/js/valine/valine.min.js file can be found by the following reference method

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

42. Add arttalk function

reference material: Add ArtiTalk talk module to the theme of matrix in Hexo blog

See for specific use Arttalk official documentation , published data can be stored in LeanCloud , the above comment function has provided relevant reference materials

First, download Arttalk source code

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

Create a new / source / LIBS / arttalk folder under the theme directory and find the folder you just downloaded Arttalk source code , enter the dist directory, where there are two folders: css and js, and then perform the following operations:

Copy arttalk.min.css under / arttalk / dist / CSS / to the topic directory / source / LIBS / arttalk;

Copy arttalk.min.js under / artalk / dist / JS / to the topic directory / source / LIBS / arttalk;
To keep consistent with the introduction style of the theme, modify the theme configuration_ config.yml file, search for the keyword libs,

Find css and add on the last line:

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

Find js and add in the last line:

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

I'll list the final effects. I won't list all of them because there are many. Just know the addition of the last line. Pay attention to alignment, as follows:

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 # Add last line

  js:
    jquery: /libs/jquery/jquery.min.js
    materialize: /libs/materialize/materialize.min.js # 1.0.0
    artitalk: /libs/artitalk/artitalk.min.js  # Add last line

Find / layout under the theme directory/_ Partial / head.ejs, introduce css in the header:

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

Module preparation, create a new / layout / arttalk.ejs file in the theme directory, and add the following contents:

<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>

Create a new / layout under the theme directory/_ Widget / arttalk.ejs file, which has been introduced in the above code. The contents of the file are as follows:

<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> say something
                </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>

Under the topic directory_ Add the following configuration to the config.yml file:

# Tell me https://artitalk.js.org/
artitalk:
  enable: true
  appId: NiVMV7aR8ipVDy9EywBYtLwI-MdYXbMMI # App appId of leancloud
  appKey: P7BGA1hMeBAFbHYz3dKY8doJ # Application appKey of leancloud
  serverURL: #  #The security domain name bound to leancloud does not need to be filled in if the international version is used
  lang: zh # Language setting: zh is Chinese, en is English and es is Spanish. The default is Chinese
  pageSize: 6 # Quantity per page
  shuoPla: 'Only the prince can release it' # Placeholder in the input box of the edit table
  motion: 1 #Switch for loading animation, 1 is on, 0 is off, and the default is on
  #Tell me about the url of the background picture of the input box
  bgImg: https://cdn.jsdelivr.net/gh/drew233/cdn/20200409110727.webp 
  avatarPla: https://Cdn.jsdelivr.net/gh/small-rose/small-rose.github.io/media/avatar.jpg # placeholder for input box of custom avatar url
  color1: linear-gradient(45deg, rgb(109, 208, 242) 15%, rgb(245, 154, 190) 85%) #Talk about background color 1 & button color 1
  color2: linear-gradient(45deg, rgb(109, 208, 242) 15%, rgb(245, 154, 190) 85%) #Talk about background color 2 & button color 2
  color3: black #Talk about font color

For more configuration items, please refer to the official website: Configuration item description

Finally, you can create the page manually or execute the hexo command

hexo new page "artitalk"

An arttalk folder will be generated in the source directory of hexo. Modify the index.md in it:

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

Under the subject directory_ config.yml file configuration menu:

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

Color reference: https://colordrop.io/

43. Switching of adding time list to archive timeline

The archived layout file is / layout/archive.ejs under the theme directory. The original logic is to display all blog posts in the form of cards. The specific code is as follows:

 <!--Timeline area block-->
    <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>

I see that the archiving of some blogs on the Internet is displayed in the form of time list, so I can have a better viewing experience for all blog articles. So I copy the blog effect code of the online boss through the debug mode, and then modify it by myself to achieve the desired effect. The following operation

First add the switching buttons of the time list and timeline, and add the following code in the / layout/archive.ejs file under the topic Directory:

<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="Time list"
                  style="background: linear-gradient(to right, rgb(76, 191, 								48) 0%, rgb(15, 157, 88) 100%); color: rgb(255, 255, 255);">Time list
            </span>
            <span onclick="showTime()" id="sp-timeline"
                  class="chip center-align waves-effect waves-light default"
                  data-tagname="time axis"
                  style="background: rgb(249, 235, 234); color: rgba(0, 0, 0, 																0.6);">time axis
            </span>
         </div>
      </div>
   </div>
</div>

js code to achieve switching effect:

<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>

Code to switch to time list:

<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 %>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>

View my effects:

Front-matter

Detailed explanation of front matter options

All contents in the front matter option are not required. However, it is still recommended to fill in at least the values of title and date.

configuration optionDefault valuedescribe
titleMarkdown's file titleThis option is highly recommended for article title
dateDate and time when the file was createdIt is strongly recommended to fill in this option for publishing time, and it is best to ensure global uniqueness
authorRoot_ author in config.ymlArticle author
imgA value in featureImagesArticle feature map
toptrueRecommended articles (whether the article is at the top or not). If the top value is true, it will be used as the recommended article on the home page
coverfalseIndicates whether the article needs to be added to the rotation cover of the home page
coverImgnothingIndicates the image path to be displayed on the rotation cover of the article on the home page. If not, the characteristic image of the article will be used by default
passwordnothingThe article reading password. If you want to set the reading verification password for the article, you can set the value of password, which must be the password encrypted with SHA256 to prevent others from seeing through. If the verifyPassword option is activated in config.yml of the topic
toctrueWhether TOC is enabled or not, you can turn off the function of TOC separately for an article. The premise is that the TOC option is activated in the config.yml of the topic
mathjaxfalseWhether mathematical formula support is enabled, whether mathjax is enabled in this article, and it needs to be in the_ The config.yml file also needs to be opened
summarynothingArticle summary, a customized article summary content. If this attribute has a value, the article card summary will display this text, otherwise the program will automatically intercept the content of the article
tagsnothingArticle tags. An article can have multiple tags
categoriesnothingArticle classification. The classification of this topic represents a macro classification. Only one classification is recommended for an article
keywordsArticle titleArticle keywords, required for SEO
reprintPolicycc_byThe article reprint rule can be CC_ by, cc_ by_ nd, cc_ by_ sa, cc_ by_ nc, cc_ by_ nc_ nd, cc_ by_ nc_ One of SA, cc0, noreprint or pay

be careful:

  1. If the img attribute is not filled in, the article feature map will take the remainder according to the hashcode value of the article title, and then select the corresponding feature picture in the topic, so as to make all articles have their own features.

  2. The value of date should ensure that each article is unique as far as possible, because Gitalk and Gitment identification IDS in this topic are uniquely identified by the value of date.

  3. If you want to set the function of reading verification password for the article, you should not only set the value of password encrypted with SHA256 in front matter, but also in the_ Configuration is activated in config.yml. Some online SHA256 encrypted addresses are available: Open source China online tools,chahuo,Webmaster Tools .

  4. You can specify reprint policy in the front mate of the article md file to configure reprint rules for a single article

Most complete example

---
title: be based on Hexo of hexo-theme-matery Theme build blog and optimize
date: 2019-10-03 14:25:00
author: Alont 
img: /source/images/xxx.jpg
top: true
cover: true
coverImg: /images/1.jpg
password: 8d969eef6ecad3c29a3a629280e686cf0c3f5d5a86aff3ca12020c923adc6c92
toc: false
mathjax: false
summary: This is your customized article summary content. If this attribute has a value, the article card summary will display this text, otherwise the program will automatically intercept part of the article as a summary
categories: tool
tags:
  - blog
  - hexo
---

Keywords: Javascript html5 hexo

Added by lazersam on Sat, 20 Nov 2021 22:44:34 +0200