博客遇到的问题 | 风尘孤狼
0%

博客遇到的问题

制作博客的过程中遇到的一部分问题及其解决办法

常用指令

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

hexo server #启动本地服务器,用于预览主题。Hexo 会监视文件变动并自动更新,除修改站点配置文件外,无须重启服务器,直接刷新网页即可生效。

hexo server -s #以静态模式启动

hexo server -p 5000 #更改访问端口 (默认端口为4000,'ctrl + c’关闭server)

hexo server -i IP地址 #自定义 IP

hexo clean #清除缓存 ,网页正常情况下可以忽略此条命令,执行该指令后,会删掉站点根目录下的public文件夹

hexo g #生成静态网页 (执行 $ hexo g后会在站点根目录下生成public文件夹, hexo会将"/blog/source/" 下面的.md后缀的文件编译为.html后缀的文件,存放在"/blog/public/ " 路径下)

hexo d #自动生成网站静态文件,并将本地数据部署到设定的仓库(如github)

hexo init 文件夹名称 #初始化XX文件夹名称

npm update hexo -g#升级

npm install hexo -g #安装

node-v #查看node.js版本号

npm -v #查看npm版本号

git --version #查看git版本号

hexo -v #查看hexo版本号

简写指令

hexo n "我的第一篇文章"等价于hexo new "我的第一篇文章"还等价于hexo new post “我的第一篇文章”
hexo p等价于hexo publish
hexo g等价于hexo generate
hexo s等价于 hexo server
hexo d等价于hexo deploy
hexo g -d等价于hexo generate --deploy
注: hexo clean 没有 简写, git --version 没有简写

hexo+next设置网站运行时间

找到hexo\themes\next\layout\_partials\footer.swig添加如下代码:

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

然后部署上传即可!

添加樱花飘落效果

themes/matery/source/js目录下新建sakura.js文件,将内容复制粘贴到sakura.js即可:

var stop, staticx;
var img = new Image();
img.src =
    "";

function Sakura(x, y, s, r, fn) {
    this.x = x;
    this.y = y;
    this.s = s;
    this.r = r;
    this.fn = fn;
}
Sakura.prototype.draw = function (cxt) {
    cxt.save();
    var xc = 40 * this.s / 4;
    cxt.translate(this.x, this.y);
    cxt.rotate(this.r);
    cxt.drawImage(img, 0, 0, 40 * this.s, 40 * this.s)
    cxt.restore();
}
Sakura.prototype.update = function () {
    this.x = this.fn.x(this.x, this.y);
    this.y = this.fn.y(this.y, this.y);
    this.r = this.fn.r(this.r);
    if (this.x > window.innerWidth || this.x < 0 || this.y > window.innerHeight || this.y < 0) {
        this.r = getRandom('fnr');
        if (Math.random() > 0.4) {
            this.x = getRandom('x');
            this.y = 0;
            this.s = getRandom('s');
            this.r = getRandom('r');
        } else {
            this.x = window.innerWidth;
            this.y = getRandom('y');
            this.s = getRandom('s');
            this.r = getRandom('r');
        }
    }
}
SakuraList = function () {
    this.list = [];
}
SakuraList.prototype.push = function (sakura) {
    this.list.push(sakura);
}
SakuraList.prototype.update = function () {
    for (var i = 0, len = this.list.length; i < len; i++) {
        this.list[i].update();
    }
}
SakuraList.prototype.draw = function (cxt) {
    for (var i = 0, len = this.list.length; i < len; i++) {
        this.list[i].draw(cxt);
    }
}
SakuraList.prototype.get = function (i) {
    return this.list[i];
}
SakuraList.prototype.size = function () {
    return this.list.length;
}

function getRandom(option) {
    var ret, random;
    switch (option) {
        case 'x':
            ret = Math.random() * window.innerWidth;
            break;
        case 'y':
            ret = Math.random() * window.innerHeight;
            break;
        case 's':
            ret = Math.random();
            break;
        case 'r':
            ret = Math.random() * 6;
            break;
        case 'fnx':
            random = -0.5 + Math.random() * 1;
            ret = function (x, y) {
                return x + 0.5 * random - 1.7;
            };
            break;
        case 'fny':
            random = 1.5 + Math.random() * 0.7
            ret = function (x, y) {
                return y + random;
            };
            break;
        case 'fnr':
            random = Math.random() * 0.03;
            ret = function (r) {
                return r + random;
            };
            break;
    }
    return ret;
}

function startSakura() {
    requestAnimationFrame = window.requestAnimationFrame || window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame ||
        window.msRequestAnimationFrame || window.oRequestAnimationFrame;
    var canvas = document.createElement('canvas'),
        cxt;
    staticx = true;
    canvas.height = window.innerHeight;
    canvas.width = window.innerWidth;
    canvas.setAttribute('style', 'position: fixed;left: 0;top: 0;pointer-events: none;');
    canvas.setAttribute('id', 'canvas_sakura');
    document.getElementsByTagName('body')[0].appendChild(canvas);
    cxt = canvas.getContext('2d');
    var sakuraList = new SakuraList();
    for (var i = 0; i < 50; i++) {
        var sakura, randomX, randomY, randomS, randomR, randomFnx, randomFny;
        randomX = getRandom('x');
        randomY = getRandom('y');
        randomR = getRandom('r');
        randomS = getRandom('s');
        randomFnx = getRandom('fnx');
        randomFny = getRandom('fny');
        randomFnR = getRandom('fnr');
        sakura = new Sakura(randomX, randomY, randomS, randomR, {
            x: randomFnx,
            y: randomFny,
            r: randomFnR
        });
        sakura.draw(cxt);
        sakuraList.push(sakura);
    }
    stop = requestAnimationFrame(function () {
        cxt.clearRect(0, 0, canvas.width, canvas.height);
        sakuraList.update();
        sakuraList.draw(cxt);
        stop = requestAnimationFrame(arguments.callee);
    })
}
window.onresize = function () {
    var canvasSnow = document.getElementById('canvas_snow');
}
img.onload = function () {
    startSakura();
}

function stopp() {
    if (staticx) {
        var child = document.getElementById("canvas_sakura");
        child.parentNode.removeChild(child);
        window.cancelAnimationFrame(stop);
        staticx = false;
    } else {
        startSakura();
    }
}

然后再themes/matery/layout/layout.ejs文件内添加下面的内容:

<script type="text/javascript">
//只在桌面版网页启用特效
var windowWidth = $(window).width();
if (windowWidth > 768) {
    document.write('<script type="text/javascript" src="/js/sakura.js"><\/script>');
}
</script>

添加鼠标彩虹星星掉落跟随效果

themes/matery/source/js目录下新建cursor.js文件,将内容复制粘贴到cursor.js即可:

/*!
 * Fairy Dust Cursor.js
 * - 90's cursors collection
 * -- https://github.com/tholman/90s-cursor-effects
 * -- http://codepen.io/tholman/full/jWmZxZ/
 */

//鼠标点击雪花特效
(function fairyDustCursor() {
  
  var possibleColors = ["#D61C59", "#E7D84B", "#1B8798"]
  var width = window.innerWidth;
  var height = window.innerHeight;
  var cursor = {x: width/2, y: width/2};
  var particles = [];
  
  function init() {
    bindEvents();
    loop();
  }
  
  // Bind events that are needed
  function bindEvents() {
    document.addEventListener('mousemove', onMouseMove);
    document.addEventListener('touchmove', onTouchMove);
    document.addEventListener('touchstart', onTouchMove);
    
    window.addEventListener('resize', onWindowResize);
  }
  
  function onWindowResize(e) {
    width = window.innerWidth;
    height = window.innerHeight;
  }
  
  function onTouchMove(e) {
    if( e.touches.length > 0 ) {
      for( var i = 0; i < e.touches.length; i++ ) {
        addParticle( e.touches[i].clientX, e.touches[i].clientY, possibleColors[Math.floor(Math.random()*possibleColors.length)]);
      }
    }
  }
  
  function onMouseMove(e) {    
    cursor.x = e.clientX;
    cursor.y = e.clientY;
    
    addParticle( cursor.x, cursor.y, possibleColors[Math.floor(Math.random()*possibleColors.length)]);
  }
  
  function addParticle(x, y, color) {
    var particle = new Particle();
    particle.init(x, y, color);
    particles.push(particle);
  }
  
  function updateParticles() {
    
    // Updated
    for( var i = 0; i < particles.length; i++ ) {
      particles[i].update();
    }
    
    // Remove dead particles
    for( var i = particles.length -1; i >= 0; i-- ) {
      if( particles[i].lifeSpan < 0 ) {
        particles[i].die();
        particles.splice(i, 1);
      }
    }
    
  }
  
  function loop() {
    requestAnimationFrame(loop);
    updateParticles();
  }
  
  /**
   * Particles
   */
  
  function Particle() {

    this.character = "*";
    this.lifeSpan = 120; //ms
    this.initialStyles ={
      "position": "fixed",
      "top": "0", //必须加
      "display": "block",
      "pointerEvents": "none",
      "z-index": "10000000",
      "fontSize": "20px",
      "will-change": "transform"
    };

    // Init, and set properties
    this.init = function(x, y, color) {

      this.velocity = {
        x:  (Math.random() < 0.5 ? -1 : 1) * (Math.random() / 2),
        y: 1
      };
      
      this.position = {x: x - 10, y: y - 20};
      this.initialStyles.color = color;
      console.log(color);

      this.element = document.createElement('span');
      this.element.innerHTML = this.character;
      applyProperties(this.element, this.initialStyles);
      this.update();
      
      document.body.appendChild(this.element);
    };
    
    this.update = function() {
      this.position.x += this.velocity.x;
      this.position.y += this.velocity.y;
      this.lifeSpan--;
      
      this.element.style.transform = "translate3d(" + this.position.x + "px," + this.position.y + "px,0) scale(" + (this.lifeSpan / 120) + ")";
    }
    
    this.die = function() {
      this.element.parentNode.removeChild(this.element);
    }
    
  }
  
  /**
   * Utils
   */
  
  // Applies css `properties` to an element.
  function applyProperties( target, properties ) {
    for( var key in properties ) {
      target.style[ key ] = properties[ key ];
    }
  }
  
  init();
})();

然后再themes/matery/layout/layout.ejs文件内添加下面的内容:

<script src="/js/cursor.js"></script>

点击爆炸效果

首先在themes/next/source/js/src里面建一个叫fireworks.js的文件,代码如下:

"use strict";

function updateCoords(e) {
    pointerX = (e.clientX || e.touches[0].clientX) - canvasEl.getBoundingClientRect().left, pointerY = e.clientY || e.touches[
        0].clientY - canvasEl.getBoundingClientRect().top
}

function setParticuleDirection(e) {
    var t = anime.random(0, 360) * Math.PI / 180,
        a = anime.random(50, 180),
        n = [-1, 1][anime.random(0, 1)] * a;
    return {
        x: e.x + n * Math.cos(t),
        y: e.y + n * Math.sin(t)
    }
}

function createParticule(e, t) {
    var a = {};
    return a.x = e, a.y = t, a.color = colors[anime.random(0, colors.length - 1)], a.radius = anime.random(16, 32), a.endPos =
        setParticuleDirection(a), a.draw = function () {
            ctx.beginPath(), ctx.arc(a.x, a.y, a.radius, 0, 2 * Math.PI, !0), ctx.fillStyle = a.color, ctx.fill()
        }, a
}

function createCircle(e, t) {
    var a = {};
    return a.x = e, a.y = t, a.color = "#F00", a.radius = 0.1, a.alpha = 0.5, a.lineWidth = 6, a.draw = function () {
        ctx.globalAlpha = a.alpha, ctx.beginPath(), ctx.arc(a.x, a.y, a.radius, 0, 2 * Math.PI, !0), ctx.lineWidth =
            a.lineWidth, ctx.strokeStyle = a.color, ctx.stroke(), ctx.globalAlpha = 1
    }, a
}

function renderParticule(e) {
    for (var t = 0; t < e.animatables.length; t++) {
        e.animatables[t].target.draw()
    }
}

function animateParticules(e, t) {
    for (var a = createCircle(e, t), n = [], i = 0; i < numberOfParticules; i++) {
        n.push(createParticule(e, t))
    }
    anime.timeline().add({
        targets: n,
        x: function (e) {
            return e.endPos.x
        },
        y: function (e) {
            return e.endPos.y
        },
        radius: 0.1,
        duration: anime.random(1200, 1800),
        easing: "easeOutExpo",
        update: renderParticule
    }).add({
        targets: a,
        radius: anime.random(80, 160),
        lineWidth: 0,
        alpha: {
            value: 0,
            easing: "linear",
            duration: anime.random(600, 800)
        },
        duration: anime.random(1200, 1800),
        easing: "easeOutExpo",
        update: renderParticule,
        offset: 0
    })
}

function debounce(e, t) {
    var a;
    return function () {
        var n = this,
            i = arguments;
        clearTimeout(a), a = setTimeout(function () {
            e.apply(n, i)
        }, t)
    }
}
var canvasEl = document.querySelector(".fireworks");
if (canvasEl) {
    var ctx = canvasEl.getContext("2d"),
        numberOfParticules = 30,
        pointerX = 0,
        pointerY = 0,
        tap = "mousedown",
        colors = ["#FF1461", "#18FF92", "#5A87FF", "#FBF38C"],
        setCanvasSize = debounce(function () {
            canvasEl.width = 2 * window.innerWidth, canvasEl.height = 2 * window.innerHeight, canvasEl.style.width =
                window.innerWidth + "px", canvasEl.style.height = window.innerHeight + "px", canvasEl.getContext(
                    "2d").scale(2, 2)
        }, 500),
        render = anime({
            duration: 1 / 0,
            update: function () {
                ctx.clearRect(0, 0, canvasEl.width, canvasEl.height)
            }
        });
    document.addEventListener(tap, function (e) {
        "sidebar" !== e.target.id && "toggle-sidebar" !== e.target.id && "A" !== e.target.nodeName && "IMG" !==
            e.target.nodeName && (render.play(), updateCoords(e), animateParticules(pointerX, pointerY))
    }, !1), setCanvasSize(), window.addEventListener("resize", setCanvasSize, !1)
}
"use strict";

function updateCoords(e) {
    pointerX = (e.clientX || e.touches[0].clientX) - canvasEl.getBoundingClientRect().left, pointerY = e.clientY || e.touches[
        0].clientY - canvasEl.getBoundingClientRect().top
}

function setParticuleDirection(e) {
    var t = anime.random(0, 360) * Math.PI / 180,
        a = anime.random(50, 180),
        n = [-1, 1][anime.random(0, 1)] * a;
    return {
        x: e.x + n * Math.cos(t),
        y: e.y + n * Math.sin(t)
    }
}

function createParticule(e, t) {
    var a = {};
    return a.x = e, a.y = t, a.color = colors[anime.random(0, colors.length - 1)], a.radius = anime.random(16, 32), a.endPos =
        setParticuleDirection(a), a.draw = function () {
            ctx.beginPath(), ctx.arc(a.x, a.y, a.radius, 0, 2 * Math.PI, !0), ctx.fillStyle = a.color, ctx.fill()
        }, a
}

function createCircle(e, t) {
    var a = {};
    return a.x = e, a.y = t, a.color = "#F00", a.radius = 0.1, a.alpha = 0.5, a.lineWidth = 6, a.draw = function () {
        ctx.globalAlpha = a.alpha, ctx.beginPath(), ctx.arc(a.x, a.y, a.radius, 0, 2 * Math.PI, !0), ctx.lineWidth =
            a.lineWidth, ctx.strokeStyle = a.color, ctx.stroke(), ctx.globalAlpha = 1
    }, a
}

function renderParticule(e) {
    for (var t = 0; t < e.animatables.length; t++) {
        e.animatables[t].target.draw()
    }
}

function animateParticules(e, t) {
    for (var a = createCircle(e, t), n = [], i = 0; i < numberOfParticules; i++) {
        n.push(createParticule(e, t))
    }
    anime.timeline().add({
        targets: n,
        x: function (e) {
            return e.endPos.x
        },
        y: function (e) {
            return e.endPos.y
        },
        radius: 0.1,
        duration: anime.random(1200, 1800),
        easing: "easeOutExpo",
        update: renderParticule
    }).add({
        targets: a,
        radius: anime.random(80, 160),
        lineWidth: 0,
        alpha: {
            value: 0,
            easing: "linear",
            duration: anime.random(600, 800)
        },
        duration: anime.random(1200, 1800),
        easing: "easeOutExpo",
        update: renderParticule,
        offset: 0
    })
}

function debounce(e, t) {
    var a;
    return function () {
        var n = this,
            i = arguments;
        clearTimeout(a), a = setTimeout(function () {
            e.apply(n, i)
        }, t)
    }
}
var canvasEl = document.querySelector(".fireworks");
if (canvasEl) {
    var ctx = canvasEl.getContext("2d"),
        numberOfParticules = 30,
        pointerX = 0,
        pointerY = 0,
        tap = "mousedown",
        colors = ["#FF1461", "#18FF92", "#5A87FF", "#FBF38C"],
        setCanvasSize = debounce(function () {
            canvasEl.width = 2 * window.innerWidth, canvasEl.height = 2 * window.innerHeight, canvasEl.style.width =
                window.innerWidth + "px", canvasEl.style.height = window.innerHeight + "px", canvasEl.getContext(
                    "2d").scale(2, 2)
        }, 500),
        render = anime({
            duration: 1 / 0,
            update: function () {
                ctx.clearRect(0, 0, canvasEl.width, canvasEl.height)
            }
        });
    document.addEventListener(tap, function (e) {
        "sidebar" !== e.target.id && "toggle-sidebar" !== e.target.id && "A" !== e.target.nodeName && "IMG" !==
            e.target.nodeName && (render.play(), updateCoords(e), animateParticules(pointerX, pointerY))
    }, !1), setCanvasSize(), window.addEventListener("resize", setCanvasSize, !1)
};

打开themes/next/layout/_layout.swig,在</body>上面写下如下代码:

<canvas class="fireworks" style="position: fixed;left: 0;top: 0;z-index: 1; pointer-events: none;" ></canvas> 
<script type="text/javascript" src="//cdn.bootcss.com/animejs/2.2.0/anime.min.js"></script> 
<script type="text/javascript" src="/js/fireworks.js"></script>

打开主题配置文件,最后填上:

# Fireworks
fireworks: true

添加雪花飘落效果

、在themes/matery/source/js目录下新建snow.js文件,将内容复制粘贴到snow.js即可:

(function($){
   $.fn.snow = function(options){
   var $flake = $('<div id="snowbox" />').css({'position': 'absolute','z-index':'9999', 'top': '-50px'}).html('&#10052;'),
   documentHeight 	= $(document).height(),
   documentWidth	= $(document).width(),
   defaults = {
      minSize		: 10,
      maxSize		: 20,
      newOn		: 1000,
      flakeColor	: "#AFDAEF" /* 此处可以定义雪花颜色,若要白色可以改为#FFFFFF */
   },
   options	= $.extend({}, defaults, options);
   var interval= setInterval( function(){
   var startPositionLeft = Math.random() * documentWidth - 100,
   startOpacity = 0.5 + Math.random(),
   sizeFlake = options.minSize + Math.random() * options.maxSize,
   endPositionTop = documentHeight - 200,
   endPositionLeft = startPositionLeft - 500 + Math.random() * 500,
   durationFall = documentHeight * 10 + Math.random() * 5000;
   $flake.clone().appendTo('body').css({
      left: startPositionLeft,
      opacity: startOpacity,
      'font-size': sizeFlake,
      color: options.flakeColor
   }).animate({
      top: endPositionTop,
      left: endPositionLeft,
      opacity: 0.2
   },durationFall,'linear',function(){
      $(this).remove()
   });
   }, options.newOn);
    };
})(jQuery);
$(function(){
    $.fn.snow({ 
       minSize: 5, /* 定义雪花最小尺寸 */
       maxSize: 50,/* 定义雪花最大尺寸 */
       newOn: 300  /* 定义密集程度,数字越小越密集 */
    });
});

然后再themes/matery/layout/layout.ejs文件内添加下面的内容:

<script src="/js/snow.js"></script>

设置博文内链接为蓝色

打开themes/next/source/css/_common/components/post/post.styl文件,将下面的代码复制到文件最后:

.post-body p a{
     color: #0593d3;
     border-bottom: none;
     &:hover {
       color: #0477ab;
       text-decoration: underline;
     }
   }

圆角

source/_data/variables.styl中输入以下代码

// 圆角设置
$border-radius-inner     = 20px 20px 20px 20px;
$border-radius           = 20px;
123

然后在 NexT 的配置文件 _config.next.yml 中取消 variables.styl 的注释:

custom_file_path:
  variable: source/_data/variables.styl

个性化回顶

将 back-to-top 按钮添加图片背景,并添加 CSS3 动效即可。

首先,找到自己喜欢的图片素材放到 source\images\ 目录下。

下载图片

然后在自定义样式文件中添加如下代码:

themes\next\source\css\_custom\custom.styl//自定义回到顶部样式
.back-to-top {
  right: 60px;
  width: 70px;  //图片素材宽度
  height: 900px;  //图片素材高度
  top: -900px;
  bottom: unset;
  transition: all .5s ease-in-out;
  background: url("/images/scroll.png");

  //隐藏箭头图标
  > i {
    display: none;
  }

  &.back-to-top-on {
    bottom: unset;
    top: 100vh < (900px + 200px) ? calc( 100vh - 900px - 200px ) : 0px;
  }
}

打字特效

点击下方按钮下载相应的脚本,并置于 themes\next\source\js\ 目录下:

打字特效

在主题自定义布局文件中添加以下代码:

themes\next\layout\_custom\custom.swig{# 打字特效 #}
{% if theme.typing_effect %}
  <script src="/js/activate-power-mode.min.js"></script>
  <script>
    POWERMODE.colorful = {{ theme.typing_effect.colorful }};
    POWERMODE.shake = {{ theme.typing_effect.shake }};
    document.body.addEventListener('input', POWERMODE);
  </script>
{% endif %}

如果 custom.swig 文件不存在,需要手动新建并在布局页面中 body 末尾引入:

themes\next\layout\_layout.swig      ...
      {% include '_third-party/exturl.swig' %}
      {% include '_third-party/bookmark.swig' %}
      {% include '_third-party/copy-code.swig' %}

+     {% include '_custom/custom.swig' %}
    </body>
  </html>

在主题配置文件中添加以下代码:

themes\next\_config.yml# typing effect
typing_effect:
  colorful: true  # 礼花特效
  shake: false  # 震动特效

置顶文章

安装插件

npm install hexo-generator-topindex --save

给需要置顶的文章加top参数!

添加置顶标志

打开:/bolg/themes/next/layout/_macro/post.swig 文件
<div class="post-meta"> 标签下,插入以下代码:

 {% if post.top %}
      <span class="post-meta-item-icon">
      <i class="fa fa-thumb-tack"></i>
      </span>
      <font color=red>置顶</font>
      <span class="post-meta-divider">|</span>
{% endif %}

encrypt

npm install --save hexo-blog-encrypt
  • 将 “password” 字段添加到文章信息
---
title: Hello World
date: 2016-03-30 21:18:02
password: 1
---
  • 再使用 hexo clean && hexo g && hexo s 在本地预览加密的文章.

  • title: VNCTF2022—WP
    date: 2022-02-12 13:27:31
    tags:

       - wp
              - CTF
    

    categories: CTF-wp
    password: 1111111111111111
    wrong_pass_message: 抱歉, 这个密码看着不太对, 请再试试.
    abstract: 有东西被加密了, 请输入密码查看.

加密主题

  • [default]
  • [blink]
  • [shrink]
  • [flip]
  • [up]
  • [surge]
  • [wave]
  • [xray]

添加图片查看器

~~

对于next主题的fancybox插件我找了很多资料发现都不能正常安装插件,总是显示克隆的错误的字典,想必应该是fancybox插件的地址换了吧,然后我就找到了fancybox的GitHub地址,直接用这个地址下载,顺利成功使用fancybox插件!

cd themes/next
git clone https://github.com/theme-next/theme-next-fancybox3.git

当然也可能会失效,我把官网放出来

fancybox官网:https://github.com/theme-next/theme-next-fancybox3

~~

上面这个方法退休了,国内访问如果用上面这个方法会影响站点加载速度,有更好的方法麻烦评论区告诉我,谢谢哦

友链

在博客页面新建页面

hexo new page links

这点需要注意,输入这个指令后会在根目录/source下生成一个links文件夹,打开其中的index.md文件,注意一定要在头部写上如下,尤其注意type!

---
title: 友链
date: 2022-01-18 10:00:39
type: "links"
---

如果不想让这个页面有评论,就再加一个comment。

配置 menu

主题配置文件下在menu下添加

links: /links/ || link

/themes/next/languages/zh-CN.yml 文件中 menu 下增加中文描述

links: 友链

新增 links.swig 页

/themes/next/layout/ 新建 links.swig,内容如下:

 {% block content %}
  {######################}
  {### LINKS BLOCK ###}
  {######################}

    <div id="links">
        <style>

            #links{
               margin-top: 5rem;
            }

            .links-content{
                margin-top:1rem;
            }

            .link-navigation::after {
                content: " ";
                display: block;
                clear: both;
            }

            .card {
                width: 300px;
                font-size: 1rem;
                padding: 10px 20px;
                border-radius: 4px;
                transition-duration: 0.15s;
                margin-bottom: 1rem;
                display:flex;
            }
            .card:nth-child(odd) {
                float: left;
            }
            .card:nth-child(even) {
                float: right;
            }
            .card:hover {
                transform: scale(1.1);
                box-shadow: 0 2px 6px 0 rgba(0, 0, 0, 0.12), 0 0 6px 0 rgba(0, 0, 0, 0.04);
            }
            .card a {
                border:none;
            }
            .card .ava {
                width: 3rem!important;
                height: 3rem!important;
                margin:0!important;
                margin-right: 1em!important;
                border-radius:4px;

            }
            .card .card-header {
                font-style: italic;
                overflow: hidden;
                width: 236px;
            }
            .card .card-header a {
                font-style: normal;
                color: #2bbc8a;
                font-weight: bold;
                text-decoration: none;
            }
            .card .card-header a:hover {
                color: #d480aa;
                text-decoration: none;
            }
            .card .card-header .info {
                font-style:normal;
                color:#a3a3a3;
                font-size:14px;
                min-width: 0;
                text-overflow: ellipsis;
                overflow: hidden;
                white-space: nowrap;
            }
        </style>
        <div class="links-content">
            <div class="link-navigation">

                {% for link in theme.mylinks %}

                    <div class="card">
                        <img class="ava" src="{{ link.avatar }}"/>
                        <div class="card-header">
                           <div>
                              <a href="{{ link.site }}" target="_blank"> {{ link.nickname }}</a>
                              <a href="{{ link.site }}" target="_blank"><span class="focus-links">关注</span></a>
                           </div>
                           <div class="info">{{ link.info }}</div>
                        </div>
                    </div>

                {% endfor %}

            </div>
            {{ page.content }}
            </div>
        </div>

  {##########################}
  {### END LINKS BLOCK ###}
  {##########################}
{% endblock %}

修改 page.swig

修改 /themes/next/layout/page.swig 文件,在开头附近添加如下代码:

{% elif page.type === 'links' and not page.title %}
{{ __('title.links') + page_title_suffix }}

大概位置如图:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fNBLrXPG-1648815291377)(博客制作遇到的问题/image-20220118100857169.png)]

引入 links.swig

接着在 /themes/next/layout/page.swigPAGE BODY 内部,引入刚才新建的 page.swig :

{% elif page.type === 'links' %}
    {% include 'links.swig' %}

添加位置不唯一,我添加的是这:

image-20220118100921527

配置友链

到这里关于友链的配置就完成了,接下来,在 /themes/next/_config.yml 文件中配置友链,末尾处新增 mylinks ,如下

mylinks:
# 名称:故里
# 地址:https://hzy2003628.top/
# 描述:
# 头像:https://hzy2003628.top/images/avatar.png


- nickname: 影影的家  #友链名称
  site: https://wsy2003328.top/  #友链地址
  info: 流年,谁给过的倾城。  #友链说明
  avatar: https://wsy2003328.top/images/avatar.png  #友链头像

添加标签云

安装插件

npm install hexo-tag-cloud@^2.0.* --save

配置插件(因为我的博客是swig,所以只说明swig的相关配置)

在主题文件夹找到文件 theme/next/layout/_macro/sidebar.swig, 然后添加如下代码:

{% if site.tags.length > 1 %}
<script type="text/javascript" charset="utf-8" src="/js/tagcloud.js"></script>
<script type="text/javascript" charset="utf-8" src="/js/tagcanvas.js"></script>
<div class="widget-wrap">
    <h3 class="widget-title">标签云</h3>
    <div id="myCanvasContainer" class="widget tagcloud">
        <canvas width="250" height="250" id="resCanvas" style="width=100%">
            {{ list_tags() }}
        </canvas>
    </div>
</div>
{% endif %}

主题配置

博文置在next主题根目录,找到 _config.yml配置文件然后在最后添加如下的配置项,可以自定义标签云的字体和颜色,还有突出高亮:

# hexo-tag-cloud
tag_cloud:
    textFont: Trebuchet MS, Helvetica
    textColor: '#333' //字体颜色
    textHeight: 25 //字体高度
    outlineColor: '#E2E1D1'
    maxSpeed: 0.1 //文字滚动速度

博文置顶

安装插件

在根目录Blog打开Git Bash,执行下面的命令:

npm uninstall hexo-generator-index --save
npm install hexo-generator-index-pin-top --save

设置置顶标志

打开blog/themes/next/layout/_macro目录下的post.swig文件,定位到

{% if post.top %}
  <i class="fa fa-thumb-tack"></i>
  <font color=7D26CD>置顶</font>
  <span class="post-meta-divider">|</span>
{% endif %}

在文章中添加top

然后在需要置顶的文章的Front-matter中加上top: true即可,如下:

---
title: Hello World

top: true

---

博客不显示图片

在写博客的过程中难免需要插图,而有时候在插图时在博客中无法正常显示图片,这个时候就需要以下几个步骤:

修改博客根目录中_config.yml文件的配置项post_asset_foldertrue

post_asset_folder: true

完成此设置后,当你通过hexo new 文件名新建博客后,会产生一个和文件同名的文件夹。

在博客根目录中下使用npm安装插件

(回到博客根目录下在空白部分点右键,进入git bush here

npm install https://github.com/CodeFalling/hexo-asset-image --save

完成以上步骤后就可以正常插图了!在这里再强调一下插图的语法是

![ ]( )

然后选择对路径之后就会正常显示图片!

Hexo博客添加搜索功能

安装插件,在博客根目录执行命令

npm install hexo-generator-searchdb --save

修改根目录下的_config.yml

search:
  path: search.xml
  field: post
  format: html
  limit: 10000
  content: true

修改主题配置文件./themes/next下的_config.yml文件

local_search:
   enable: true

hexo文章目录点击不跳转,html没有生成href

hexo搭建博客,发现文章目录点击无反应,f12看一下会发现toc-link后面没有href属性image-20220620095747989

点击测试出现报错

image-20220620100220507

进入根目录的 node_modules\hexo-toc\lib\filter.js 中,把 28 行~31 行修改为:

$title.attr('id', id);
// $title.children('a').remove();
// $title.html( '<span id="' + id + '">' + $title.html() + '</span>' );
// $title.removeAttr('id');

然后部署一下就可以了

有的人可能没有hexo-toc,就像我,这个时候下载一下就可以了

npm install hexo-toc --save

部署时发生error:spawn failed错误

首先用ssh -T git@github.com命令测试是否连接,如果不能连接的话,在存放key的目录下新建config文件

MO22Ie.png

填入以下内容

Host github.com
User 你GitHub的邮箱
Hostname ssh.github.com
PreferredAuthentications publickey
IdentityFile ~/.ssh/id_rsa
Port 443

如果可以正常连接的话,回到博客的根目录,打开_config.yml配置文件

修改以下内容

deploy:

type: git

repo: https://github.com/yourname/yourname.github.io.git


branch: master

其中的repo修改为

git@github.com:yourname/yourname.github.io.git

页面加载性能

博客上线后,需要关注页面的加载速度。Google PageSpeed Insights 可用于帮助分析网页加载速度瓶颈,该网站将会针对指定域名进行在线测试,并提供一份详细的页面加载分析报告,报告中还会根据页面资源加载情况给出合理的优化建议及预期优化效果,因此用户可以有的放矢的进行性能优化专项整改。

Google PageSpeed Insights

虽然可以从报告中了解到站点的加载性能瓶颈,但是,报告中仅是提供优化建议,具体的落实还须自己实施。本章节主要介绍 CDN 加速和代码合并压缩两种比较常见的优化方案。

CDN 加速

在控制台的 Network Tab 页中可以查看到页面加载瀑布流。在此需要勾选 Disable cache 以避免缓存干扰分析。另外,在所有静态资源中,对加载速度影响较大且存在大幅优化空间的主要还是 JS 脚本,所以我们先拿它开刀。

页面加载瀑布流

首先理解一下底部信息栏中的这行文字:

21 / 50 requests | 518 KB / 818KB transferred | Finish: 8.44 s | DOMContentLoaded: 4.36 s | Load: 6.80 s

主要包括以下几条信息:

  • 加载页面总共发起了 50 次请求,下载了 818 KB 资源,其中 JS 脚本资源共计 518 KB,占据了 21 次请求。
  • 在 4.36 s 时 DOM 树渲染完毕(蓝线),此时已经可以看到正常的页面,由于 JS 脚本会阻塞 DOM 解析,所以这段时间是包含了 JS 脚本的下载与执行过程的。
  • 在 6.80 s 时所有资源加载完毕(红线),包括异步的脚本和其他非阻塞的页面资源,如图片、CSS 文件等。
  • 在 8.44 s 时所有 HTTP 请求响应完毕,包括 XHR 请求。

关于 Finish / DOMContentLoaded / Load 的讨论请参见 How to analyzing Page Speed in Chrome Dev toolschrome devtools 中 network 一栏中的 Finish 时间代表着什么,两者说法不一致,我倾向于认同后者的观点。

分析以上页面加载瀑布流可知,DOM 树的解析渲染主要是被从 VPS 本地下载 jquery、velocity、fancybox 等脚本所滞后,所以对症下药,我们可以改为从开源公共 CDN 加载脚本。

对于在站点中用到的三方插件,在主题配置文件中为其添加 CDN 加载源:

themes\next\_config.ymlvendors:
  jquery: //cdn.jsdelivr.net/npm/jquery@2.1.3/dist/jquery.min.js

  fancybox: https://cdn.jsdelivr.net/npm/@fancyapps/fancybox@3.5.2/dist/jquery.fancybox.min.js
  fancybox_css: https://cdn.jsdelivr.net/npm/@fancyapps/fancybox@3.5.2/dist/jquery.fancybox.min.css

  velocity: https://cdn.jsdelivr.net/npm/velocity-animate@1.2.1/velocity.min.js

  velocity_ui: //cdn.jsdelivr.net/npm/velocity-animate@1.2.1/velocity.ui.min.js

  fontawesome: //cdn.jsdelivr.net/npm/font-awesome@4.7.0/css/font-awesome.min.css

  algolia_instant_js: https://cdn.jsdelivr.net/npm/instantsearch.js@2.4.1/dist/instantsearch.js
  algolia_instant_css: https://cdn.jsdelivr.net/npm/instantsearch.js@2.4.1/dist/instantsearch.min.css

  pace: //cdn.jsdelivr.net/npm/pace-js@1.0.2/pace.min.js
  pace_css: //cdn.jsdelivr.net/npm/pace-js@1.0.2/themes/blue/pace-theme-minimal.css

  pangu: //cdn.jsdelivr.net/npm/pangu@3.3.0/dist/browser/pangu.min.js

  valine: https://cdn.jsdelivr.net/npm/valine@1.3.3/dist/Valine.min.js

如果开启了 Live2D 看板娘,一样可以通过将脚本来源改为 CDN 来完成性能优化:[](// 修改选中字符的颜色
/* webkit, opera, IE9 /
::selection {
background: #06d633;
color: #f7f7f7;
}
/
firefox */
::-moz-selection {
background: #06d633;
color: #f7f7f7;
}

// 代码块选中颜色
.highlight *::selection {
background: #06d633;
})

_config.yml  # Live2D
  ## https://github.com/EYHN/hexo-helper-live2d
  live2d:
    enable: true
    pluginRootPath: live2dw/ # Root path of plugin to be on the site (Relative)
    pluginJsPath: lib/ # JavaScript path related to plugin's root (Relative)
    pluginModelPath: assets/ # Relative model path related to plugin's root (Relative)
-   scriptFrom: local # Default
+   scriptFrom: jsdelivr # jsdelivr CDN
    tagMode: true # Whether only to replace live2d tag instead of inject to all pages
    log: false # Whether to show logs in console
    model:
      use: live2d-widget-model-shizuku  # 萌系少女
    display:
      position: left
      width: 100
      height: 180
    mobile:
      show: false
    react:
      opacityDefault: 0.7 # 默认透明度

这样可以加快进入博客的速度!

接入DaoVoice

接入这个玩意的时候有个大坑,网上也有其他网友反应这个问题,但是直接搜这个问题网上目前是没有任何解决方法的,我在这里记录一下吧

刚开始就是注册个号

http://dashboard.daovoice.io/app/6f081c2c/inboxes/all/conversations/

注册之后找到这里,安装到网站,找到app_id的值,后面会用到

image-20230102142313468

在这个路径之下的这个文件里加入如下代码\themes\next\layout_partials\head\head.swig

{% if theme.daovoice %}
 <script>(function(i,s,o,g,r,a,m){i["DaoVoiceObject"]=r;i[r]=i[r]||function(){(i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o),m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;a.charset="utf-8";m.parentNode.insertBefore(a,m)})(window,document,"script",('https:' == document.location.protocol ? 'https:' : 'http:') + "//widget.daovoice.io/widget/b6dbddb6.js","daovoice")
 daovoice('init', {
  app_id: "app_id"  //这里填写刚才说的app_id
});
daovoice('update');
 </script>
{% endif %}

然后进入到主题配置文件,最后加上

daovoice: true
daovoice_app_id: #这地方填上app_id

最重要的坑来了,这个时候如果直接部署上去就会发现虽然这个功能出来了但是博客白页了,不显示文章了,原因其实是velocity这个next插件与这个在线聊天功能不兼容

在主题配置文件中找到之后关上再部署上去就可以了,而且也不会有啥影响,这个插件就是动画效果,其实开着也是摆设,基本用不上

image-20230102142753884

这个时候再部署上去就没问题了

又一问题的解决


  • 2024.01.19 博客开启HTTP3.0时代,但是再写文章部署的时候就传不上去github,报错如下
ssh: connect to host github.com port 22: Connection timed out
fatal: Could not read from remote repository.

Please make sure you have the correct access rights
and the repository exists.
FATAL {
  err: Error: Spawn failed
      at ChildProcess.<anonymous> (D:\博客\node_modules\hexo-util\lib\spawn.js:51:21)
      at ChildProcess.emit (node:events:513:28)
      at cp.emit (D:\博客\node_modules\cross-spawn\lib\enoent.js:34:29)
      at ChildProcess._handle.onexit (node:internal/child_process:291:12) {
    code: 128
  }
} Something's wrong. Maybe you can find the solution here: %s https://hexo.io/docs/troubleshooting.html

解决办法就是换端口,这情况就是22端口被防火墙拦截了,换成443端口

先在 C:\Users\25963/.ssh/这个文件夹下建个名为config的文件,文件内容如下

Host github.com
User git
Hostname ssh.github.com
PreferredAuthentications publickey
IdentityFile ~/.ssh/id_rsa
Port 443
 
Host gitlab.com
Hostname altssh.gitlab.com
User git
Port 443
PreferredAuthentications publickey
IdentityFile ~/.ssh/id_rsa

然后在博客根目录输

ssh -T git@github.com

回个yes,就会发现连上了

D:\博客>ssh -T git@github.com
The authenticity of host '[ssh.github.com]:443 ([20.205.243.160]:443)' can't be established.
ED25519 key fingerprint is SHA256:+DiY3wvvV6TuJJhbpZisF/zLDA0zPMSvHdkr4UvCOqU.
This host key is known by the following other names/addresses:
    C:\Users\25963/.ssh/known_hosts:1: github.com
Are you sure you want to continue connecting (yes/no/[fingerprint])? yes
Warning: Permanently added '[ssh.github.com]:443' (ED25519) to the list of known hosts.
Hi hzy030628! You've successfully authenticated, but GitHub does not provide shell access.

这样再重新d就可以了。

推荐:

NexT主题的优化定制修改指南

Hexo-Next 主题博客个性化配置超详细,超全面

[hexo 主题 next7.8 版本配置美化]((11条消息) hexo 主题 next7.8 版本配置美化_uKnow_AL的博客-CSDN博客_next7.8)

hexo theme next7.8 主题美化

next主题配置

hexo插件

制作不易,如若感觉写的不错,欢迎打赏