1 Star 2 Fork 0

crysw / hexo-blog-crystal

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
贡献代码
同步代码
取消
提示: 由于 Git 不支持空文件夾,创建文件夹后会生成空的 .keep 文件
Loading...
README
Apache-2.0

crystalBlog

介绍

个人博客

github地址 https://wang-qz.github.io/

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

域名地址 https://www.crystalblog.xyz/

软件架构

基于hexo框架, hexo-theme-matery主题搭建的个人博客网站, 其他主题选择请进入hexo-theme查看.

安装教程

1. 安装js, hexo, git

网上很多环境安装教程, 此处省略...., 参考Easy Hexo

2. 设置npm全局下载目录

npm config set prefix "D:\softwareInstalled\nodeJs\node_global"
npm config set cache "D:\softwareInstalled\nodeJs\node_cache"

3. cnpm安装

cnpm读取的是国内的镜像站, 比国外镜像站访问快很多.

npm install -g cnpm --registry=https://registry.npm.taobao.org
cnpm -v

4 .安装hexo(两种方式)

npm install -g hexo-cli
cnpm install -g hexo-cli
##查看hexo版本
hexo -v

5. hexo初始化

创建MyBlog博客目录, 进入目录进行hexo初始化:

cd MyBlog
hexo init

6. 清除博客缓存

会清除博客public目录下生成的html等静态文件

hexo clean

7. md文件生成html页面

hexo g
hexo generate

8. 本地部署hexo

部署后访问http://localhost:4000, 就可以进入搭建的博客主页.

hexo s
hexo serve

9. 新建分类 categories

categories 页是用来展示所有的分类,如果 source 目录下还没有 /categories/index.md 文件,那么就需要新建一个,命令如下:

hexo new page "categories"

10. 新增标签tags

会生成 /source/tags/index.md , 其他page页也是一样的新增.

hexo new page "tags"

11. 新增博客

创建一个博客文章, 就可以写文章了, 文章的一些属性设置详见Front-matter

hexo new 'myblog'

12. 发布到远程仓库

需要先安装插件hexo-deployer-git

cnpm install --save hexo-deployer-git

hexo根目录下面_config.yml关联仓库文件配置

deploy:
  type: git
  repository: git@gitee.com:wang-qz/crystal-blog.git
  branch: master  

开启域名url, 在博客根目录 _config.yml 文件中:

url: https://wang-qz.gitee.io/crystal-blog #后期修改为域名

发布到远程仓库

hexo d
hexo deploy

13. 切换主题

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

## landscape matery next butterfly sakura yilia
theme: hexo-theme-matery

14. 博客站点相关信息

在博客根目录下的 _config.yml 配置文件中:

# Site
title: 王子的博客 #网站标题
subtitle: be yourself, be better #网站副标题
#网站描述description 主要用于5E0,告诉搜索引擎一个关于您站点的简单描述
description: 本网站是个人兴趣爱好,总结分享经验,记录生活点滴的平台
 #网站的关键词。使用半角逗号“,”分隔多个关键词
keywords: [王子的博客,HTML, CSS, JavaScript, JQuery, java, linux等]
author: 王子 #您的名字
language: zh-CN #网站使用的语言。建议修改为zh-CN
timezone: Asia/Shanghai #网站时区。Hexo默认使用您电脑的时区。

主题优化

1. 代码高亮配置

安装hexo代码高亮插件

cnpm i -S hexo-prism-plugin

修改 Hexo 根目录下 _config.yml 文件中 highlight.enable 的值为 false,并新增 prism 插件相关的配置,主要配置如下:

highlight:
  enable: false

prism_plugin:
  mode: 'preprocess'    # realtime/preprocess
  theme: 'tomorrow'
  line_number: false    # default false
  custom_css:

2. 添加搜索功能

安装hexo-generator-search 搜索插件

cnpm install hexo-generator-search --save

在 Hexo 根目录下的 _config.yml 文件中,新增以下的配置项:

search:
  path: search.xml
  field: post

3. 社交链接修改

在主题的 _config.yml 文件中,默认支持 QQGitHub 和邮箱的配置,

# 首页 banner 中的第二行个人信息配置,留空即不启用
socialLink:
  github:  #https://github.com/wang-qz
  wechat:    #实际替换为cdn路径
  qq: 1848276756
  email: wang-qz@foxmail.com
  gitee: https://gitee.com/wang-qz
  csdn: https://blog.csdn.net/u013044713
  facebook: # https://www.facebook.com/xxx
  twitter: # https://twitter.com/xxx
  weibo: # https://weibo.com/xxx
  zhihu: # https://www.zhihu.com/xxx
  rss: true # true、false

可以在主题文件的 /layout/_partial/social-link.ejs 文件中,新增、修改需要的社交链接地址,增加链接可参考如下代码:

<% if (theme.socialLink.gitee) { %>
    <a href="<%= theme.socialLink.gitee %>" class="tooltipped" target="_blank" 
       data-tooltip="码云" data-position="top" data-delay="50">
        <i class="fab fa-github-alt"></i>
    </a>
<% } %>

<% if (theme.socialLink.csdn) { %>
    <a href="<%= theme.socialLink.csdn %>" class="tooltipped" target="_blank" 
       data-tooltip="关注我的CSDN" data-position="top" data-delay="50">
        <i class="fab fa-csdn">C</i>
    </a>
<% } %>

其中,社交图标(如:fa-github)可以在 Font Awesome 中搜索找到。以下是常用社交图标的标识,供参考:

Facebook: fab fa-facebook
Twitter: fab fa-twitter
Google-plus: fab fa-google-plus
Linkedin: fab fa-linkedin
Tumblr: fab fa-tumblr
Linkedin: fab fa-linkedin
Slack: fab fa-slack
新浪微博: fab fa-weibo
微信: fab fa-wechat 或 fab fa-weixin
QQ: fab fa-qq
知乎: fab fa-zhihu

5. 打赏二维码修改

在主题文件的 source/medias/reward 文件中,可以替换成你的的微信和支付宝的打赏二维码图片。

主题目录下的_config.yml文件中打赏设置

# 是否激活文章末尾的打赏功能,默认激活
#(你替换为的你自己的微信、支付宝二维码图片、或者使用网络图片也可以).
reward:
  enable: true
  title: 你的赏识是我前进的动力
  wechat: /medias/reward/wechatPay.jpg
  alipay: /medias/reward/alipay.jpg

6. 文章链接转静态短地址

如果文章名称是中文的,那么 Hexo 默认生成的永久链接也会有中文,这样不利于 SEO,且 gitment 评论对中文链接也不支持。我们可以用hexo-permalink-pinyin插件生成文章时生成中文拼音的永久链接,或者用hexo-abbrlink 生成静态文章链接。以下结合hexo-abbrlink生成类似 /yyyy/mmdd+随机数.html 的文章链接地址。

npm install hexo-permalink-pinyin --save
npm install hexo-abbrlink --save

Hexo 根目录下的 _config.yml 文件中,新增以下的配置项:

permalink_pinyin:
  enable: true
  separator: '-' # default: '-'

在 Hexo 根目录下的 _config.yml 文件中,修改 permalink: ,并在文件末尾新增 abbrlink:配置项:

permalink: :year/:month:day:abbrlink.html
abbrlink: 
  alg: crc16 #算法选项:crc16丨crc32
  rep: dec #输出进制:dec为十进制,hex为十六进制

7. 文章字数统计插件

如果你想要在文章中显示文章字数、阅读时长信息,可以安装 hexo-wordcount插件。

npm i --save hexo-wordcount

然后只需在本主题下的 _config.yml 文件中,激活以下配置项即可:

postInfo:
  date: true # 发布日期
  update: true # 更新日期
  wordCount: true # 文章字数统计
  totalCount: true # 站点总文章字数
  min2read: true # 文章阅读时长
  readCount: true # 文章阅读次数

8. 添加 RSS 订阅支持

本主题中还使用到了 hexo-generator-feed 的 Hexo 插件来做 RSS,安装命令如下:

npm install hexo-generator-feed --save

在 Hexo 根目录下的 _config.yml 文件中,新增以下的配置项:

feed:
  type: atom
  path: atom.xml
  limit: 20
  hub:
  content:
  content_limit: 140
  content_limit_delim: ' '
  order_by: -date

执行 hexo clean && hexo g 重新生成博客文件,然后在 public 文件夹中即可看到 atom.xml 文件,说明已经安装成功了。

9. 导航栏/标签页/尾页颜色修改

主题目录 /source/css/matery.css文件中修改:

/* 整体背景颜色,包括导航、移动端的导航、页尾、标签页等的背景颜色. */
.bg-color {
    /**background-image: linear-gradient(to right, #4cbf30 0%, #0f9d58 100%);**/
    background-image: linear-gradient(to right, #7371BC 0%, #284D95 100%);
}

10. 主题背景颜色修改

在主题文件的 /source/css/matery.css 文件中:

@-webkit-keyframes rainbow {
   /* 动态切换背景颜色.即滤镜颜色,不想要可以全部注释,或者换成你喜欢的颜色 */
}
@keyframes rainbow {
    /* 动态切换背景颜色.,不想要可以全部注释,或者换成你喜欢的颜色 */
}

11. banner 图和文章特色图修改

可以直接在/source/medias/banner文件夹中更换喜欢的 banner 图片,主题代码中是每天动态切换一张,只需 7 张即可。如果会 JavaScript 代码,可以修改成自己喜欢切换逻辑,如:随机切换等,banner 切换的代码位置在 /layout/_partial/bg-cover-content.ejs 文件的 <script></script> 代码中:

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

12. 整体布局的背景图设置

在主题目录下的/layout/layout.ejs文件中, 找到下面的代码就是设置背景图的地方:

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

上面代码会加载主题目录下的背景图设置文件/layout/_partial/background.ejs:

<style>
    body {
        /*background-image: url(<%- theme.background.url %> );*/
       /*我将background.url配置为数组,配置了多个背景图,在navigate.ejs中做了随机切换背景图*/
        background-image: url(<%- theme.background.url[0]%>);
        background-repeat: no-repeat;
        background-size: cover;
        background-attachment: fixed;
    }
</style>

13. 取消背景色蒙版特效

默认情况下banner图上面感觉覆盖了一层雾色 ,其实是给banner部分设置了背景色变化, 给人朦胧的特效感觉 , 在主题目录下的 /source/css/matery.css 文件中可以取消朦胧特效设置,搜索 .bg-cover:after 注释它:

/**注释后, 取消背景色蒙版特效**/
.bg-cover:after {
   /* -webkit-animation: rainbow 60s infinite;
    animation: rainbow 60s infinite;*/
}

14. 添加emoji表情支持

Matery 主题新增了对emoji表情的支持,使用到了 hexo-filter-github-emojis 的 Hexo 插件来支持 emoji表情的生成,把对应的markdown emoji语法(::,例如::smile:)转变成会跳跃的emoji表情,安装命令如下:

npm install hexo-filter-github-emojis --save

15. 站点侧边配置音乐播放器

音乐播放器的代码在主题目录下的/layout/_widget/music.ejs文件中.

要支持音乐播放,在主题目录下的 _config.yml 配置文件中激活music配置即可:

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

server可选netease(网易云音乐),tencent(QQ音乐),kugou(酷狗音乐),xiami(虾米音乐), baidu(百度音乐)。 type可选song(歌曲),playlist(歌单),album(专辑),search(搜索关键字),artist(歌手) id获取示例: 打开手机版网易云音乐,选择喜欢的歌单,然后点击分享.

我这里随便选了一个歌单,分享后的文字长这样:

分享真咸鱼饼干的歌单《青年节:致逐梦人,有志者事竟成》http://music.163.com/playlist/4965675848/1548006936/?userid=120124365 (@网易云音乐)

4965675848 这就是歌单的id,文件里默认设置的歌单其实也还不错,歌挺多的,所以如果没什么特殊要求,使用默认歌单也不错。

然后博客是在主题目录下的/layout/index.ejs文件中引用了/layout/_widget/music.ejs音乐文件:

<div id="indexCard" class="index-card">
        <div class="container ">
            <div class="card">
                <div class="card-content">
                    <% if (theme.dream.enable) { %>
                        <%- partial('_widget/dream') %>
                    <% } %>

                    <% if (theme.music.enable && !theme.music.fixed) { %>
                        <%- partial('_widget/music') %>
                    <% } %>

                    <% if (theme.recommend.enable) { %>
                    <div id="recommend-sections" class="recommend">
                        <%- partial('_widget/recommend') %>
                    </div>
                    <% } %>
                </div>
            </div>
        </div>
    </div>

上面代码中的recommend是主页的文章推荐, dream我的梦想, 在主题目录下的_config.yml文件中配置是否激活启用, 在编写MD博客时设置top属性为true即可在主页看到文章推荐.

16. 主页网站标题及副标题

在博客根目录下的 _config.yml 配置文件中:

# Site
title: 王子的博客 #网站标题
subtitle: be yourself, be better #网站副标题
description: 本网站是个人兴趣爱好,总结分享经验,记录生活点滴的平台,希望在以后的学习旅途中,走出自己的风景 #网站描述description 主要用于5E0,告诉搜索引擎一个关于您站点的简单描述
keywords: [王子的博客,HTML, CSS, JavaScript, JQuery, java, linux等] #网站的关键词。使用半角逗号“,”分隔多个关键词
author: 王子 #您的名字
language: zh-CN #网站使用的语言。建议修改为zh-CN
timezone: Asia/Shanghai #网站时区。Hexo默认使用您电脑的时区。

17. 主页网站副标题及打字效果

在博客根目录 _config.yml 文件维护的副标题下方的打字效果, 在主题的 _config.yml 配置文件中配置如下:

# 打字效果副标题.
# 如果有符号 ‘ ,请在 ’ 前面加上 \
subtitle:
  enable: true
  loop: true # 是否循环
  showCursor: true # 是否显示光标
  startDelay: 300 # 开始延迟
  typeSpeed: 100 # 打字速度
  backSpeed: 50 # 删除速度
  sub:
    - 从来没有真正的绝境, 只有心灵的迷途.
    - Never really desperate, only the lost of the soul.
    - 如果放弃太早,你永远都不知道自己会错过什么.
    - 没有伞的孩子必须努力奔跑!
    - 花开不是为了花落,而是为了开的更加灿烂.
    - 没有礁石,就没有美丽的浪花;没有挫折,就没有壮丽的人生.

获取sub打字内容的js代码在主题目录下/layout/_partial/bg-cover-content.ejs

<div class="description center-align">
    <% if (theme.subtitle.enable) { %>
        <span id="subtitle"></span>
        <script src="https://cdn.jsdelivr.net/npm/typed.js@2.0.11"></script>
        <script>
            var typed = new Typed("#subtitle", {
                strings: [
                    <% theme.subtitle.sub.forEach(function(item){ %>
                    "<%= item %>",
                    <% }) %>
                ],
                startDelay: <%= theme.subtitle.startDelay %>,
                typeSpeed: <%= theme.subtitle.typeSpeed %>,
                loop: <%= theme.subtitle.loop %>,
                backSpeed: <%= theme.subtitle.backSpeed %>,
                showCursor: <%= theme.subtitle.showCursor %>
            });
        </script>
    <% } else { %>
        <%= config.description %>
    <% } %>
</div>

打字效果, 打印位置在根目录设置的副标题下面轮播打印主题目录设置的sub值

副标题打字效果

18. 副标题打字效果改为展示今日诗词

修改主题配置文件_config.yml, 将副标题的动态打字enable改为false

# 打字效果副标题.
# 如果有符号 ‘ ,请在 ’ 前面加上 \
subtitle:
  enable: false
  loop: true # 是否循环
  showCursor: true # 是否显示光标
  startDelay: 300 # 开始延迟
  typeSpeed: 100 # 打字速度
  backSpeed: 50 # 删除速度
  sub:
    - 从来没有真正的绝境, 只有心灵的迷途.
    - Never really desperate, only the lost of the soul.
    - 如果放弃太早,你永远都不知道自己会错过什么.
    - 没有伞的孩子必须努力奔跑!
    - 花开不是为了花落,而是为了开的更加灿烂.
    - 没有礁石,就没有美丽的浪花;没有挫折,就没有壮丽的人生.

/themes/hexo-theme-matery/layout/_partial/bg-cover-content.ejs文件中将<%= config.description %>改为:

<span id="jinrishici-sentence">正在加载今日诗词....</span>
<script src="https://sdk.jinrishici.com/v2/browser/jinrishici.js" charset="utf-8"></script>

参考今日诗词接口官方资料

19. 推荐文章设置

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

# 是否显示推荐文章的标题
recommend:
  enable: true
  showTitle: true
  useConfig: false # 是否使用配置文件, 在 _data/recommends.json 下配置推荐文章, false则会走主题配置的 top 属性

top 属性配置详见Front-matter

20. 设置文章阅读密码

主题目录_config.yml配置如下:

# 阅读文章的密码验证功能,如要使用此功能请激活该配置项,并在对应文章的Front-matter中写上'password'的键和加密后的密文即可.
# 请注意:为了保证密码原文不会被泄露到网页中,文章的密码必须是通过'SHA256'加密的,这样就不会被破解.
verifyPassword:
  enable: true
  promptMessage: 请输入访问本文章的密码
  errorMessage: 密码错误,将返回主页!

在博客文章中指定加密后的密文密码password , 博客文件密码属性设置详情见Front-matter

SHA256 加密的地址: 开源中国在线工具chahuo站长工具TTmd5

21. 网站log设置

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

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

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

22. 网站动态标题行

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

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

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

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

23. about页面添加个人简历

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

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

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

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

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

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

24. 404页面

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

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

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

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

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

25. 网站页脚修改

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

<footer class="page-footer bg-color">
    <% if (theme.music.enable && theme.music.fixed) { %>
        <%- partial('_widget/music') %>
    <% } %>

    <div class="container row center-align" style="margin-bottom: <% if (theme.time.enable) { %>15<% } else { %>0<% } %>px !important;">
        <div class="col s12 m8 l8 copy-right">
            Copyright&nbsp;&copy;
            <% if (theme.time.year !== new Date().getFullYear()) { %>
                <span id="year"><%- theme.time.year %>-<%- new Date().getFullYear() %></span>
            <% } else { %>
                <span id="year"><%- theme.time.year %></span>
            <% } %>
            <i class="fa fa-heart" style="color: #ff71a8"></i>
            <a href="<%- url_for('/about') %>" target="_blank"><%- config.author %></a>
            |&nbsp;Powered by&nbsp;<a href="https://hexo.io/" target="_blank">Hexo</a>
            |&nbsp;&nbsp;<a href="https://github.com/blinkfox/hexo-theme-matery" target="_blank">Matery</a>
            <br>

            <% if (theme.postInfo.totalCount) { %>
                <span style="margin-left: -20px; display: inline">
                &nbsp;  <i class="fas fa-chart-area"></i>&nbsp;<%- __('siteTotalWords') %>:&nbsp;<span  class="white-color"><%= totalcount(site) %></span>
                <span/>
            <% } %>

            <% let socialClass = '' %>
            <% if (theme.busuanziStatistics && theme.busuanziStatistics.enable) { %>
                <% socialClass = 'social-statis' %>
            <% } %>
            <% if (theme.busuanziStatistics && theme.busuanziStatistics.totalTraffic) { %>
                <span id="busuanzi_container_site_pv3" style="display: inline">
                &nbsp;|&nbsp;<i class="far fa-eye"></i>&nbsp;<%- __('siteTotalVisits') %>:&nbsp;<span id="busuanzi_value_site_pv" class="white-color"><%= totalcount(site) %></span>
                </span>
            <% } %>
            <% if (theme.busuanziStatistics && theme.busuanziStatistics.totalNumberOfvisitors) { %>
                <span id="busuanzi_container_site_uv3" style="display: inline">
                &nbsp;|&nbsp;<i class="fas fa-users"></i>&nbsp;<%- __('siteTotalVisitors') %>:&nbsp;<span id="busuanzi_value_site_uv" class="white-color"><%= totalcount(site) %></span>
                </span>
            <% } %>
            <br>

            <!-- 运行天数提醒. -->
            <% if (theme.time.enable) { %>
                <span id="sitetime"> Loading ...</span>
                <script>
                    var calcSiteTime = function () {
                        var seconds = 1000;
                        var minutes = seconds * 60;
                        var hours = minutes * 60;
                        var days = hours * 24;
                        var years = days * 365;
                        var today = new Date();
                        var startYear = "<%- theme.time.year %>";
                        var startMonth = "<%- theme.time.month %>";
                        var startDate = "<%- theme.time.date %>";
                        var startHour = "<%- theme.time.hour %>";
                        var startMinute = "<%- theme.time.minute %>";
                        var startSecond = "<%- theme.time.second %>";
                        var todayYear = today.getFullYear();
                        var todayMonth = today.getMonth() + 1;
                        var todayDate = today.getDate();
                        var todayHour = today.getHours();
                        var todayMinute = today.getMinutes();
                        var todaySecond = today.getSeconds();
                        var t1 = Date.UTC(startYear, startMonth, startDate,
                                          startHour, startMinute, startSecond);
                        var t2 = Date.UTC(todayYear, todayMonth, todayDate,
                                          todayHour, todayMinute, todaySecond);
                        var diff = t2 - t1;
                        var diffYears = Math.floor(diff / years);
                        var diffDays = Math.floor((diff / days) - diffYears * 365);
                        var diffHours = Math.floor((diff / hours) - diffYears * 365
                                                   * 24 - diffDays * 24);
                        var diffMinutes = Math.floor((diff / minutes) - diffYears *                                365 * 24 * 60 - diffDays * 24 * 60 - diffHours * 60);
                        var diffSeconds = Math.floor((diff / seconds) - diffYears * 									365 * 24 * 60 * 60 - diffDays * 24 * 60 * 60  - diffHours
                                                     * 60 * 60 - diffMinutes * 60);
                        // 区分是否有年份.
                        var language = '<%- config.language %>';
                        if (startYear === String(todayYear)) {
                            document.getElementById("year").innerHTML = todayYear;
                            var daysTip = 'This site has been running for ' + 
                                diffDays + ' days';
                            if (language === 'zh-CN') {
                                daysTip = '本站已运行 ' + diffDays + '';
                            } else if (language === 'zh-HK') {
                                daysTip = '本站已運行 ' + diffDays + '';
                            }
                            document.getElementById("sitetime").innerHTML = daysTip;
                        } else {
                            document.getElementById("year").innerHTML = startYear +
                               " - " + todayYear;
                            var yearsAndDaysTip = 'This site has been running for ' 
                            + diffYears + ' years and '
                                + diffDays + ' days';
                            if (language === 'zh-CN') {
                                yearsAndDaysTip = '本站已运行 ' + diffYears + '' +
                                   diffDays + '' + diffHours + ' 小时 ' + 
                                   diffMinutes + ' 分钟 ' + diffSeconds + '';
                            } else if (language === 'zh-HK') {
                                yearsAndDaysTip = '本站已運行 ' + diffYears + '' + 
                                   diffDays + '';
                            }
                            document.getElementById("sitetime").innerHTML = 
                               yearsAndDaysTip;
                        }
                    }
                    var timer = setInterval(calcSiteTime);
                    // calcSiteTime();
                </script>
            <% } %>
                    &nbsp;|&nbsp;
            <% if (theme.icp.enable) { %>
                <span id="icp">
                    <img src="<%- theme.jsDelivr.url %><%- 		url_for('/medias/icp.png') %>" style="vertical-align: text-bottom;"/>
                    <a href="<%- url_for(theme.icp.url) %>" target="_blank"><%=                                 theme.icp.text %></a>
                </span>
            <% } %>
        </div>

        <div class="col s12 m4 l4 social-link <%- socialClass %>">
            <%- partial('_partial/social-link') %>
        </div>
    </div>
</footer>

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

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

<!--总访问人数-->
<% if (theme.busuanziStatistics && theme.busuanziStatistics.totalNumberOfvisitors) { %>
<span id="busuanzi_container_site_uv" style="display: inline">
   &nbsp;|&nbsp;<i class="fas fa-users"></i>&nbsp;<%- __('siteTotalVisitors') %>:&nbsp;<span
                                                                                             id="busuanzi_value_site_uv" class="white-color"></span>
</span>
<% } %>
<!--最后加上-->
<script>
    let _hmt = _hmt || [];
    (function () {
        var hm = document.createElement("script");
        hm.src = "https://hm.baidu.com/hm.js?147475454185ebcf440a27cc35e793ef";
        var s = document.getElementsByTagName("script")[0];
        s.parentNode.insertBefore(hm, s);
    })();
</script>

26. 添加动漫人物

安装插件hexo-helper-live2d

npm install --save hexo-helper-live2d

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

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

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

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

27. 雪花和樱花效果

添加雪花飘落效果

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

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

/* 控制下雪 */
function snowFall(snow) {
    /* 可配置属性 */
    snow = snow || {};
    this.maxFlake = snow.maxFlake || 200;   /* 最多片数 */
    this.flakeSize = snow.flakeSize || 10;  /* 雪花形状 */
    this.fallSpeed = snow.fallSpeed || 1;   /* 坠落速度 */
}

/* 兼容写法 */
requestAnimationFrame = window.requestAnimationFrame ||
    window.mozRequestAnimationFrame ||
    window.webkitRequestAnimationFrame ||
    window.msRequestAnimationFrame ||
    window.oRequestAnimationFrame ||
    function (callback) {
        setTimeout(callback, 1000 / 60);
    };

cancelAnimationFrame = window.cancelAnimationFrame ||
    window.mozCancelAnimationFrame ||
    window.webkitCancelAnimationFrame ||
    window.msCancelAnimationFrame ||
    window.oCancelAnimationFrame;
/* 开始下雪 */
snowFall.prototype.start = function () {
    /* 创建画布 */
    snowCanvas.apply(this);
    /* 创建雪花形状 */
    createFlakes.apply(this);
    /* 画雪 */
    drawSnow.apply(this)
}

/* 创建画布 */
function snowCanvas() {
    /* 添加Dom结点 */
    var snowcanvas = document.createElement("canvas");
    snowcanvas.id = "snowfall";
    snowcanvas.width = window.innerWidth;
    snowcanvas.height = document.body.clientHeight;
    snowcanvas.setAttribute("style", "position:absolute; top: 0; left: 0; 
                            z-index: 1; pointer-events: none;");
    document.getElementsByTagName("body")[0].appendChild(snowcanvas);
    this.canvas = snowcanvas;
    this.ctx = snowcanvas.getContext("2d");
    /* 窗口大小改变的处理 */
    window.onresize = function () {
        snowcanvas.width = window.innerWidth;
        /* snowcanvas.height = window.innerHeight */
    }
}

/* 雪运动对象 */
function flakeMove(canvasWidth, canvasHeight, flakeSize, fallSpeed) {
    this.x = Math.floor(Math.random() * canvasWidth);   /* x坐标 */
    this.y = Math.floor(Math.random() * canvasHeight);  /* y坐标 */
    this.size = Math.random() * flakeSize + 2;          /* 形状 */
    this.maxSize = flakeSize;                           /* 最大形状 */
    this.speed = Math.random() * 1 + fallSpeed;         /* 坠落速度 */
    this.fallSpeed = fallSpeed;                         /* 坠落速度 */
    this.velY = this.speed;                             /* Y方向速度 */
    this.velX = 0;                                      /* X方向速度 */
    this.stepSize = Math.random() / 30;                 /* 步长 */
    this.step = 0                                       /* 步数 */
}

flakeMove.prototype.update = function () {
    var x = this.x,
        y = this.y;
    /* 左右摆动(余弦) */
    this.velX *= 0.98;
    if (this.velY <= this.speed) {
        this.velY = this.speed
    }
    this.velX += Math.cos(this.step += .05) * this.stepSize;

    this.y += this.velY;
    this.x += this.velX;
    /* 飞出边界的处理 */
    if (this.x >= canvas.width || this.x <= 0 || this.y >= canvas.height || this.y <= 0) {
        this.reset(canvas.width, canvas.height)
    }
};
/* 飞出边界-放置最顶端继续坠落 */
flakeMove.prototype.reset = function (width, height) {
    this.x = Math.floor(Math.random() * width);
    this.y = 0;
    this.size = Math.random() * this.maxSize + 2;
    this.speed = Math.random() * 1 + this.fallSpeed;
    this.velY = this.speed;
    this.velX = 0;
};
// 渲染雪花-随机形状(此处可修改雪花颜色!!!)
flakeMove.prototype.render = function (ctx) {
    var snowFlake = ctx.createRadialGradient(this.x, this.y, 0, this.x, this.y, this.size);
    snowFlake.addColorStop(0, "rgba(255, 255, 255, 0.9)");  /* 此处是雪花颜色,默认是白色 */
    snowFlake.addColorStop(.5, "rgba(255, 255, 255, 0.5)"); /* 若要改为其他颜色,请自行查 */
    snowFlake.addColorStop(1, "rgba(255, 255, 255, 0)");    /* 找16进制的RGB 颜色代码。 */
    ctx.save();
    ctx.fillStyle = snowFlake;
    ctx.beginPath();
    ctx.arc(this.x, this.y, this.size, 0, Math.PI * 2);
    ctx.fill();
    ctx.restore();
};

/* 创建雪花-定义形状 */
function createFlakes() {
    var maxFlake = this.maxFlake,
        flakes = this.flakes = [],
        canvas = this.canvas;
    for (var i = 0; i < maxFlake; i++) {
        flakes.push(new flakeMove(canvas.width, canvas.height, this.flakeSize, this.fallSpeed))
    }
}

/* 画雪 */
function drawSnow() {
    var maxFlake = this.maxFlake,
        flakes = this.flakes;
    ctx = this.ctx, canvas = this.canvas, that = this;
    /* 清空雪花 */
    ctx.clearRect(0, 0, canvas.width, canvas.height);
    for (var e = 0; e < maxFlake; e++) {
        flakes[e].update();
        flakes[e].render(ctx);
    }
    /*  一帧一帧的画 */
    this.loop = requestAnimationFrame(function () {
        drawSnow.apply(that);
    });
}

/* 调用及控制方法 */
var snow = new snowFall({maxFlake: 60});
snow.start();

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

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

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

# 雪花特效
snow:
  enable: true

添加樱花飘落效果

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

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

function Sakura(x, y, s, r, fn) {
    this.x = x;
    this.y = y;
    this.s = s;
    this.r = r;
    this.fn = fn;
}

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

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

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

window.onresize = function () {
    var canvasSnow = document.getElementById('canvas_snow');
}
img.onload = function () {
    startSakura();
}

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

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

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

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

# 樱花特效
sakura:
  enable: true

28. 鼠标点击爱心效果

鼠标点击样式1

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

!function(e,t,a){function r(){for(var e=0;e<n.length;e++)n[e].alpha<=0?(t.body.removeChild(n[e].el),n.splice(e,1)):(n[e].y--,n[e].scale+=.004,n[e].alpha-=.013,n[e].el.style.cssText="left:"+n[e].x+"px;top:"+n[e].y+"px;opacity:"+n[e].alpha+";transform:scale("+n[e].scale+","+n[e].scale+") rotate(45deg);background:"+n[e].color+";z-index:99999");requestAnimationFrame(r)}var n=[];e.requestAnimationFrame=e.requestAnimationFrame||e.webkitRequestAnimationFrame||e.mozRequestAnimationFrame||e.oRequestAnimationFrame||e.msRequestAnimationFrame||function(e){setTimeout(e,1e3/60)},function(e){var a=t.createElement("style");a.type="text/css";try{a.appendChild(t.createTextNode(e))}catch(t){a.styleSheet.cssText=e}t.getElementsByTagName("head")[0].appendChild(a)}(".heart{width: 10px;height: 10px;position: fixed;background: #f00;transform: rotate(45deg);-webkit-transform: rotate(45deg);-moz-transform: rotate(45deg);}.heart:after,.heart:before{content: '';width: inherit;height: inherit;background: inherit;border-radius: 50%;-webkit-border-radius: 50%;-moz-border-radius: 50%;position: fixed;}.heart:after{top: -5px;}.heart:before{left: -5px;}"),function(){var a="function"==typeof e.onclick&&e.onclick;e.onclick=function(e){a&&a(),function(e){var a=t.createElement("div");a.className="heart",n.push({el:a,x:e.clientX-5,y:e.clientY-5,scale:1,alpha:1,color:"rgb("+~~(255*Math.random())+","+~~(255*Math.random())+","+~~(255*Math.random())+")"}),t.body.appendChild(a)}(e)}}(),r()}(window,document);

鼠标点击样式2

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

/* 鼠标点击文字特效 */
var a_idx = 0;
jQuery(document).ready(function ($) {
    $("body").click(function (e) {
        // var a = new Array("❤富强❤","❤民主❤","❤文明❤","❤和谐❤","❤自由❤","❤平等❤","❤公正❤","❤法治❤","❤爱国❤","❤敬业❤","❤诚信❤","❤友善❤");
        var a = new Array("富强", "民主", "文明", "和谐", "自由", "平等", "公正", "法治", "爱国", "敬业", "诚信", "友善");
        var $i = $("<span></span>").text(a[a_idx]);
        a_idx = (a_idx + 1) % a.length;
        var x = e.pageX,
            y = e.pageY;
        $i.css({
            "z-index": 
           999999999999999999999999999999999999999999999999999999999999999999999,
            "top": y - 20,
            "left": x,
            "position": "absolute",
            "font-weight": "bold",
            "color": "rgb(" + ~~(255 * Math.random()) + "," + ~~(255 * Math.random()) 
           + "," + ~~(255 * Math.random()) + ")"
        });
        $("body").append($i);
        $i.animate({
                "top": y - 180,
                "opacity": 0
            },
            1500,
            function () {
                $i.remove();
            });
    });
});

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

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

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

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

29. 修改博客文章模板

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

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

30. 在线编辑hexo博客

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

首先安装hexo-admin插件

npm install --save hexo-admin

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

编辑后还可以快速部署发布.

hexo-admin在线写博客

31. 站点统计不显示问题

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

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

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

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

/*文章链接超长部分隐藏*/
.reprint__type {
    display: inline-block;
    width: 100%;
    overflow: hidden;
}

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

背景静态彩带

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

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

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

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

背景动态彩带

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

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

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

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

背景canvas

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

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

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

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

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

34. 提取相册(壁纸)

新建相册(壁纸)文件

hexo new page wallpaper

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

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

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

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

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

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

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

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

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

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

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

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

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

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

35. 添加相册列表

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

参考友链的卡片列表

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

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

添加清单-相册菜单

这里要修改几个文件:

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

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

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

hexo new page "galleries"

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

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

添加相册配置文件

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

[
  {
    "name": "博客背景图",
    "url": "/gallery2",
    "cover": "https://www.bing.com/th?id=OHR.Mpumalanga_ZH-CN9666962271_tmb.jpg&rf=",
    "description": "博客背景图",
    "photos": [
      "https://www.bing.com/th?id=OHR.Mpumalanga_ZH-CN9666962271_tmb.jpg&rf=",
    ]
  },
  {
    "name": "彭于晏",
    "url": "/gallery0",
    "cover": "https://uploadfile.bizhizu.cn/2016/0106/20160106033828391.jpg",
    "description": "彭于晏写生",
    "photos": [
      "http://i.52desktop.cn:81/upimg/allimg/20191204/2019124151645578778013.jpg",
      "http://i.52desktop.cn:81/upimg/allimg/20191204/2019124151645687778016.jpg"
    ]
  },
  {
    "name": "刘德华",
    "url": "/gallery1",
    "cover": "https://tu1.whhost.net/uploads/20181029/18/1540809870-NgSCnhWkcJ.jpg",
    "description": "刘德华写生",
    "photos": [
      "https://uploadfile.bizhizu.cn/2015/0306/20150306103233272.jpg",
      "https://www.beihaiting.com/uploads/allimg/150401/10723-150401195426203.jpg"
    ]
  }
]

添加相册布局文件

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

<link rel="stylesheet" href="<%- theme.jsDelivr.url %><%- url_for(theme.libs.css.gallery) %>">

<%- partial('_partial/bg-cover') %>

<main class="content">
    <div class="container">
        <div class="title center-align" data-aos="zoom-in-up">
            <i class="fas fa-camera"></i>&nbsp;&nbsp;<%- __('galleries') %>
        </div>

        <% if (site.data && site.data.galleries) { %>
            <% var galleries = site.data.galleries; %>
            <div class="gallery-wrapper row">
                <% for (var i = 0, len = galleries.length; i < len; i++) { %>
                    <% var gallery = galleries[i]; %>
                    <div class="col s6 m4 l4 xl3 gallery-box">
                        <a href="./<%- gallery.url %>" 
                           class="gallery-item" data-aos="zoom-in-up">
                            <div class="gallery-cover-box"
                                 style="background-image: url(<%- gallery.cover %>);">
                            </div>
                            <p class="gallery-name" style="font-size: 22px; 
                                                           font-family: '华文行楷'">
                                <%- gallery.name %>
                            </p>
                        </a>
                    </div>

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

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

<%- partial('_partial/bg-cover') %>
<%
var galleries = site.data.galleries;
var pageTitle = page.title;
var currentGallery = getCurrentGallery(galleries, pageTitle)
var photos = currentGallery.photos;

function getCurrentGallery(galleries, pageTitle) {
    for (let i = 0; i < galleries.length; i++) {
        if (galleries[i]['name'] == pageTitle) {
            return galleries[i];
        }
    }
}
/***/
%>
<div id="myGallery" class="my-gallery">
    <div class="title center-align" data-aos="zoom-in-up">
        <p class="gallery-name">
            <b style="font-size: 35px;"><%- currentGallery.name %></b>
        </p>
    </div>

    <div class="row">
        <% if (photos) { %>
            <% Object.keys(photos).forEach(function(photo) { %>
                <div class="photo col s12 m6 l4" data-aos="fade-up">
                    <div class="img-item" data-src="<%- photos[photo] %>">
                        <img src="<%- theme.jsDelivr.url %><%- url_for(photos[photo]) 
                                  %>" class="responsive-img">
                    </div>
                </div>
            <% }); %>
        <% } %>
    </div>
</div>

<script>
    $(function () {
        let animateClass = 'animated pulse';
        $('#myGallery .photo').hover(function () {
            $(this).addClass(animateClass);
        }, function () {
            $(this).removeClass(animateClass);
        });
    });
</script>

添加相册列表样式

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

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

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

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

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

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

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

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

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

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

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

.photo-item .photo-name {
    font-size: 14px;
    line-height: 30px;
    text-align: center;
    margin-top: 10px;
    margin-bottom: 10px;
    border-top: 1px solid #dddddd;
}

/*适配移动端布局*/
@media only screen and (max-width: 601px) {
    .waterfall {
        column-count: 2;
        column-gap: 1em;
    }
}

相册列表效果

相册列表效果

相册展示效果

相册展示效果

相册密码设置

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

<% if (theme.verifyPassword.enable) { %>
    <script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.crypto) %>"></script>
    <script>
        (function() {
            /*pwd是博客中配置的sha256加密后的密码*/
            let pwd = '<%- page.password %>';
            if (pwd && pwd.length > 0) {
                if (pwd !== CryptoJS.SHA256(prompt('请输入访问本相册的密
')).toString(CryptoJS.enc.Hex)) {
                    alert('密码错误!');
                    location.href = '<%- url_for("/galleries")  %>';
                }
            }
        })();
    </script>
<% } %>

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

image-20210908213414259

36. 修改导航栏不透明

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

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

37. 添加快捷导航

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

hexo new page navigate

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

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

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

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

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

<div class="navi-height bg-cover pd-header">
    <div class="link-box container">
        <div class="baidu baidu-2 large-screen">
            <form name="f" action="https://www.baidu.com/" target="_blank">
                <div id="Select-2">
                    <div class="Select-box-2" id="baidu">
                        <ul style="height:46px">
                            <li class="this_s">百 度</li>
                            <li class="bing_s">必 应</li>
                            <li class="google_s">谷 歌</li>
                            <li class="baidu_s">百 度</li>
                        </ul>
                    </div>
                    <input name="wd" id="kw-2" maxlength="100" 
                           autocomplete="off" type="text">
                </div>
                <div class="qingkong" id="qingkong" title="清 · 空" 
                     style="display:block">x</div>
                <input value="搜 索" id="su-2" type="submit"/>
                <ul class="keylist"></ul>
            </form>
        </div>

        <div class="row tags-posts">
            <div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up">
                <div class="card">
                    <div class="jj-list-tit">编程 · 学习</div>
                    <ul class="jj-list-con">
                        <li>
                            <a href="https://www.oschina.net/" class="link-3" 
                               target="_blank">开源中国</a>
                        </li>
                        <li>
                            <a href="https://htmldog.com/" class="link-3" 
                               target="_blank">HTML狗</a>
                        </li>
                        <li>
                            <a href="https://www.icourse163.org/" class="link-3" 
                               target="_blank">中国大学慕课</a>
                        </li>
                        <li>
                            <a href="https://www.imooc.com/" class="link-3" 
                               target="_blank">慕课网</a>
                        </li>
                        <li>
                            <a href="http://www.wxapp-union.com/" class="link-3" 
                               target="_blank">小程序</a>
                        </li>
                        <li>
                            <a href="https://www.runoob.com/" class="link-3" 
                               target="_blank">菜鸟教程</a>
                        </li>
                        <li>
                            <a href="https://blog.51cto.com/" class="link-3" 
                               target="_blank">51CTO</a>
                        </li>
                        <li>
                            <a href="https://www.shiyanlou.com/library/" 
                               class="link-3" target="_blank">实验楼</a>
                        </li>
                        <li>
                            <a href="/posts/2d1a17c5.html" class="link-3" 
                               target="_blank">个人收藏页</a>
                        </li>
                    </ul>
                </div>
            </div>
            <div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up">
                <div class="card">
                    <div class="jj-list-tit">社区 · Code</div>
                    <ul class="jj-list-con">
                        <li>
                            <a href="https://www.zhangxiaocai.com/contact/" 
                               class="link-3" target="_blank">留言
                            </a>
                        </li>
                        <li>
                            <a href="https://github.com/" class="link-3" 
                               target="_blank">GitHub</a>
                        </li>
                        <li>
                            <a href="https://coding.net/" class="link-3" 
                               target="_blank">Coding</a>
                        </li>
                        <li>
                            <a href="https://juejin.im/" class="link-3" 
                               target="_blank">掘金</a>
                        </li>
                        <li>
                            <a href="https://gitee.com/" class="link-3" 
                               target="_blank">码云</a>
                        </li>
                        <li>
                            <a href="https://www.csdn.net/" class="link-3" 
                               target="_blank">CSDN</a>
                        </li>
                        <li>
                            <a href="https://www.jianshu.com/" class="link-3" 
                               target="_blank">简书</a>
                        </li>
                        <li>
                            <a href="https://segmentfault.com/" class="link-3" 
                               target="_blank">思否</a>
                        </li>
                        <li>
                            <a href="https://cloud.tencent.com/developer/" 
                               class="link-3" target="_blank">云+社区
                            </a>
                        </li>
                    </ul>
                </div>
            </div>
            <div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up">
                <div class="card">
                    <div class="jj-list-tit">实用 · 工具</div>
                    <ul class="jj-list-con">
                        <li>
                            <a href="https://mdnice.com/" class="link-3"
                               target="_blank">Nice编辑器</a>
                        </li>
                        <li>
                            <a href="https://translate.google.cn/" class="link-3"
                               target="_blank">谷歌翻译</a>
                        </li>
                        <li>
                            <a href="https://www.uupoop.com/" class="link-3"
                               target="_blank">在线PS</a>
                        </li>
                        <li>
                            <a href="https://www.processon.com/" class="link-3" 
                               target="_blank">思维导图</a>
                        </li>
                        <li>
                            <a href="https://wallhaven.cc/" class="link-3" 
                               target="_blank">超清壁纸</a>
                        </li>
                        <li>
                            <a href="https://cli.im/" class="link-3" 
                               target="_blank">二维码</a>
                        </li>
                        <li>
                            <a href="http://www.yinfans.me/" class="link-3" 
                               target="_blank">音范思</a>
                        </li>
                        <li>
                            <a href="https://www.extfans.com" class="link-3" 
                               target="_blank">谷歌插件</a>
                        </li>
                        <li>
                            <a href="https://my.openwrite.cn/" class="link-3" 
                               target="_blank">OW分发</a>
                        </li>
                    </ul>
                </div>
            </div>
            <div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up">
                <div class="card">
                    <div class="jj-list-tit">娱乐 · 影视</div>
                    <ul class="jj-list-con">
                        <li>
                            <a href="https://www.jd.com/" class="link-3" 
                               target="_blank">京东</a>
                        </li>
                        <li>
                            <a href="https://www.taobao.com/" class="link-3" 
                               target="_blank">淘宝</a>
                        </li>
                        <li>
                            <a href="https://www.tmall.com/" class="link-3" 
                               target="_blank">天猫</a>
                        </li>
                        <li>
                            <a href="https://v.qq.com/" class="link-3" 
                               target="_blank">腾讯视频</a>
                        </li>
                        <li>
                            <a href="http://www.iqiyi.com/" class="link-3" 
                               target="_blank">爱奇艺</a>
                        </li>
                        <li>
                            <a href="https://www.bilibili.com/" class="link-3"
                               target="_blank">哔哩哔哩</a>
                        </li>
                        <li>
                            <a href="https://music.163.com/" class="link-3" 
                               target="_blank">网易云音乐</a>
                        </li>
                        <li>
                            <a href="https://y.qq.com/" class="link-3" 
                               target="_blank">QQ音乐</a>
                        </li>
                        <li>
                            <a href="http://www.kugou.com/" class="link-3" 
                               target="_blank">酷狗音乐</a>
                        </li>
                    </ul>
                </div>
            </div>
            <div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up">
                <div class="card">
                    <div class="jj-list-tit">资讯 · 趋势</div>
                    <ul class="jj-list-con">
                        <li>
                            <a href="https://www.huxiu.com/" class="link-3" 
                               target="_blank">虎嗅</a>
                        </li>
                        <li>
                            <a href="https://insights.stackoverflow.com/" 
                               class="link-3" target="_blank">技术调查
                            </a>
                        </li>
                        <li>
                            <a href="http://www.asciiworld.com/" class="link-3" 
                               target="_blank">摸鱼</a>
                        </li>
                        <li>
                            <a href="https://sspai.com/" class="link-3"
                               target="_blank">少数派</a>
                        </li>
                        <li>
                            <a href="https://zh.wikihow.com/" class="link-3"
                               target="_blank">WikeHom</a>
                        </li>
                        <li>
                            <a href="https://www.awesomes.cn/rank?sort=hot"
                               class="link-3" target="_blank">
                                前端趋势
                            </a>
                        </li>
                        <li>
                            <a href="https://github-trending.com/" class="link-3"
                               target="_blank">GitHub趋势</a>
                        </li>
                        <li>
                            <a href="https://www.tiobe.com/" class="link-3" 
                               target="_blank">编程趋势</a>
                        </li>
                        <li>
                            <a href="https://trends.google.com/" class="link-3" 
                               target="_blank">Google趋势</a>
                        </li>
                    </ul>
                </div>
            </div>
            <div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up">
                <div class="card">
                    <div class="jj-list-tit">搜索 · 其他</div>
                    <ul class="jj-list-con">
                        <li>
                            <a href="https://ac.scmor.com/" class="link-3" 
                               target="_blank">谷歌镜像</a>
                        </li>
                        <li>
                            <a href="http://www.pansoso.com/" class="link-3" 
                               target="_blank">网盘搜索</a>
                        </li>
                        <li>
                            <a href="http://music.ifkdy.com/" class="link-3" 
                               target="_blank">音乐搜索</a>
                        </li>
                        <li>
                            <a href="https://www.dytt8.net/" class="link-3" 
                               target="_blank">电影天堂</a>
                        </li>
                        <li>
                            <a href="https://carbon.now.sh/" class="link-3"
                               target="_blank">代码图片</a>
                        </li>
                        <li>
                            <a href="https://www.zhipin.com/" class="link-3" 
                               target="_blank">Boos</a>
                        </li>
                        <li>
                            <a href="https://fontawesome.dashgame.com/" 
                               class="link-3" target="_blank">图标库</a>
                        </li>
                        <li>
                            <a href="https://www.qvdv.com/tools/qvdv-guid.html"
                               class="link-3" target="_blank">
                                在线工具
                            </a>
                        </li>
                        <li>
                            <a href="http://zhongguose.com/"
                               class="link-3" target="_blank">中国色</a>
                        </li>
                    </ul>
                </div>
            </div>
        </div>
        <script>
            $(".Select-box ul").hover(function () {
                $(this).css("height", "auto")
            }, function () {
                $(this).css("height", "40px")
            }), $(".Select-box-2 ul").hover(function () {
                $(this).css("height", "auto")
            }, function () {
                $(this).css("height", "46px")
            }), $(".Select-box li").click(function () {
                var t = $(this).attr("class"), s = $(this).html();
                "baidu_s" == t && (t = "https://www.baidu.com/s", _name = "wd"), "google_s" == t && (t = "https://www.google.com/search", _name = "q"), "bing_s" == t && (t = "https://www.bing.com/search", _name = "q"), $(".baidu form").attr("action", t), $(".this_s").html(s), $("#kw").attr("name", _name), $(".Select-box ul").css("height", "40px")
            }), $(".Select-box-2 li").click(function () {
                var t = $(this).attr("class"), s = $(this).html();
                "baidu_s" == t && (t = "https://www.baidu.com/s", _name = "wd"), "google_s" == t && (t = "https://www.google.com/search", _name = "q"), "bing_s" == t && (t = "https://www.bing.com/search", _name = "q"), $(".baidu form").attr("action", t), $(".this_s").html(s), $("#kw-2").attr("name", _name), $(".Select-box-2 ul").css("height", "48px")
            });

            $("#qingkong").click(function () {
                $("input[ name='wd' ] ").val("");
            });

            let timer;
            function backgroundImgRandom() {
                clearInterval(timer);
                timer = setInterval(function () {
                   // [0-9)
                   $("body").css("background-image", 
                   "url(<%- theme.background.url[parseInt(Math.random() * 9)] %>");
                }, 5000);
            }
            window.onload = backgroundImgRandom;
        </script>
    </div>
</div>
<style>
    * {
        margin: 0;
        padding: 0;
        font-family: consolas, hl, "微软雅黑"
    }

    dd, dl, dt, form, h1, h2, h3, h4, h5, h6, li, p, ul {
        margin: 0;
        padding: 0;
        font-size: 14px;
        font-weight: 400
    }

    img {
        border-style: none
    }

    li {
        list-style: none;
        float: left
    }

    a {
        text-decoration: none
    }

    .card {
        background-color: rgba(25, 240, 229, 0);
        width: 96%;
        margin-left: 2%
    }

    .baidu {
        float: left;
        margin-left: 100px
    }

    .baidu form {
        position: relative
    }

    .Select-box ul {
        height: 40px;
        position: absolute;
        left: -1px;
        top: 0;
        z-index: 9999;
        background: #FFF;
        border: 1px solid #ccc;
        border-top: none;
        overflow: hidden
    }

    .Select-box li {
        width: 60px;
        line-height: 40px;
        font-size: 14px;
        color: #484848;
        border: 0;
        cursor: pointer
    }

    .Select-box li:hover {
        background: #3385ff;
        color: #FFF
    }

    .Select-box .this_s {
        color: #317ef3
    }

    .Select-box .this_s:hover {
        background: #FFF;
        color: #317ef3
    }

    .qingkong {
        position: absolute;
        right: 120px;
        top: 12px;
        width: 18px;
        height: 18px;
        background: rgba(0, 0, 0, .1);
        border-radius: 18px;
        line-height: 16px;
        color: #666;
        cursor: pointer;
        text-align: center;
        font-size: 14px;
        display: none;
        color: #881509;
    }

    .qingkong:hover {
        background: rgba(0, 0, 0, .2)
    }

    .qingkong:active {
        background: rgba(0, 0, 0, .3)
    }

    .baidu-2 {
        width: 100%;
        height: 110px;
        margin: 0 auto;
        background: 0 0;
        padding-top: 50px
    }

    .baidu-2 form {
        width: 520px;
        margin: 0 auto
    }

    .baidu-2 input {
        padding: 13px 8px;
        opacity: .9;
        font-size: 15px
    }

    #Select-2 {
        float: left
    }

    .Select-box-2 {
        text-align: center;
        float: left;
        position: relative
    }

    .Select-box-2 ul {
        height: 46px;
        position: absolute;
        left: 0;
        top: 1px;
        z-index: 9999;
        background: rgba(255, 255, 255, .9);
        border: 1px solid #ccc;
        border-top: none;
        overflow: hidden
    }

    .Select-box-2 li {
        width: 60px;
        line-height: 46px;
        font-size: 15px;
        color: #484848;
        border: 0;
        cursor: pointer
    }

    .Select-box-2 li:hover {
        background: #3385ff;
        color: #FFF
    }

    .Select-box-2 .this_s {
        color: #317ef3
    }

    .Select-box-2 .this_s:hover {
        background: 0 0;
        color: #317ef3
    }

    #kw-2 {
        width: 335px;
        outline: 0;
        border: 1px solid #ccc;
        background: rgba(255, 255, 255, .2);
        color: #000;
        padding-left: 70px;
        font-weight: 700
    }

    #su-2 {
        width: 90px;
        background: #4e6ef2;
        border: none;
        border-top: #3385ff 1px solid;
        border-bottom: 1px solid #2d78f4;
        color: #FFF;
        cursor: pointer;
        outline: 0
    }

    #su-2:hover {
        background: #00f;
        border-bottom: 1px solid #00f
    }

    #su-2:active {
        background: #00f;
        box-shadow: inset 1px 1px 3px #00f;
        -webkit-box-shadow: inset 1px 1px 3px #00f
    }

    #su-3 {
        width: 90px;
        background: #00f;
        border: none;
        border-top: #3385ff 1px solid;
        border-bottom: 1px solid #2d78f4;
        color: #FFF;
        cursor: pointer;
        outline: 0
    }

    .jj-list-tit {
        font-size: 16px;
        line-height: 25px;
        color: #fff;
        width: 100%;
        padding-left: 38.5%
    }

    .jj-list-con {
        overflow: hidden;
        margin: 0 auto
    }

    .jj-list-con li {
        width: 31.333%;
        margin: 1%
    }

    .link-3 {
        display: block;
        background: rgba(0, 0, 0, .35);
        color: #FFF;
        font-size: 13px;
        text-align: center;
        line-height: 35px;
        padding: 4px 0;
        border-radius: 2px;
        transition: all .2s
    }

    .link-3:hover {
        background: rgba(0, 0, 0, .45);
        font-size: 15px;
        font-weight: 700
    }

    @media only screen and (max-width: 584px) {
        .navi-height {
            height: 1300px
        }

        .link-box {
            margin-top: 5%
        }

        .large-screen {
            display: none
        }
    }

    @media only screen and (min-width: 584px) and (max-width: 993px) {
        .navi-height {
            height: 800px
        }

        .link-box {
            margin-top: 5%
        }

        .large-screen {
            display: none
        }
    }

    @media only screen and (min-width: 993px) {
        .navi-height {
            position: absolute;
            width: 100%;
            height: 100%
        }
    }

    .page-footer {
        display: none
    }
</style>

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

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

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

38. 添加音乐导航页

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

hexo new page musics

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

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

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

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

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

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

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

<link rel="stylesheet" href="<%- theme.jsDelivr.url %><%- url_for(theme.libs.css.aplayer) %>">
<style>
    .aplayer .aplayer-lrc p {
        <%if(theme.mymusic.hideLrc){%>
        display: none;
        <%}%>
        font-size: 12px;
        font-weight: 700;
        line-height: 16px !important;
    }

    .aplayer .aplayer-lrc p.aplayer-lrc-current {
        <%if(theme.mymusic.hideLrc){%>
        display: none;
        <%}%>
        font-size: 15px;
        color: <%- theme.mymusic.theme %>;
    }

    <%if(theme.mymusic.autoHide){%>
    .aplayer.aplayer-fixed.aplayer-narrow .aplayer-body {
        left: -66px !important;
    }

    .aplayer.aplayer-fixed.aplayer-narrow .aplayer-body:hover {
        left: 0px !important;
    }

    <%}%>
</style>
<div class="<% if(!theme.mymusic.fixed) { %>music-player<% } %>">
    <% if (!theme.mymusic.fixed && theme.mymusic.title.enable) { %>
        <div class="title center-align">
            <i class="fas fa-music"></i>&nbsp;&nbsp;<%- theme.mymusic.title.show %>
        </div>
    <% } %>
    <div class="row">
        <meting-js class="col l8 offset-l2 m10 offset-m1 s12"
                   server="<%- theme.mymusic.server %>"
                   type="<%- theme.mymusic.type %>"
                   id="<%- theme.mymusic.id %>"
                   fixed='<%- theme.mymusic.fixed ? 'true' : 'false' %>'
                   autoplay='<%- theme.mymusic.autoplay === true %>'
                   theme='<%- theme.mymusic.theme %>'
                   loop='<%- theme.mymusic.loop %>'
                   order='<%- theme.mymusic.order %>'
                   preload='<%- theme.mymusic.preload %>'
                   volume='<%- Number(theme.mymusic.volume) %>'
                   list-folded='<%- theme.mymusic.listFolded ? 'true' : 'false' %>'
        >
        </meting-js>
    </div>
</div>

<script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.aplayer) %>"></script>
<script src="https://cdn.jsdelivr.net/npm/meting@2/dist/Meting.min.js"></script>

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

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

39. 生成电影卡片

安装hexo-tag-mtime插件

npm install hexo-tag-mtime --save

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

hexo new page movies

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

---
title: 电影推荐
date: 2021-08-25 19:56:04
type: movies
---
## 精彩电影推荐
###  怒火·重案
{% mtime 263501 %}
###  再见,少年
{% mtime 259370 %}
### 流浪地球
{% mtime 218707 %}

id可以在时光网 https://www.mtime.com 相应的电影网址获取

40. 添加夜间模式切换

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

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

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

// 深色模式按钮设置
if (localStorage.getItem('dark') === '1') {
    document.body.classList.add('dark');
} else if (new Date().getHours() >= 22 || new Date().getHours() < 7) {
    /*定时开启暗色模式<默认晚22点至早6点默认开启>*/
    // document.body.classList.add('dark');
    // $("#nightMode").removeClass("fa-moon-o").addClass("fa-lightbulb");
} else if (matchMedia('(prefers-color-scheme: dark)').matches) {
    document.body.classList.add('dark');
}

// 深色模式设置
function switchNightMode() {
    var body = document.body;
    if (body.classList.contains('dark')) {
        document.body.classList.remove('dark');
        localStorage.setItem('dark', '0');
        $('#nightMode').removeClass("fa-lightbulb").addClass("fa-moon-o");
        return;
    } else {
        document.body.classList.add('dark');
        localStorage.setItem('dark', '1');
        $('#nightMode').removeClass("fa-moon-o").addClass("fa-lightbulb");
        return;
    }
}

/*提醒开启夜间模式功能*/
setTimeout(
    function () {
        if ((new Date().getHours() >= 19 || new Date().getHours() < 7) && !$('body').hasClass('DarkMode')) {
            let toastHTML = '<span style="color:#97b8b2;border-radius: 10px;>'
                + '<i class="fa fa-bell" aria-hidden="true"></i>晚上使用深色模式阅读更好哦。(゚▽゚)/</span>'
            M.toast({html: toastHTML})
        }
    }, 2200);

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

/******夜间模式切换样式 start*******/
/* 深色模式按钮设置 */
#sma {
    background: #000;
    width: 38px;
    height: 38px;
    display: block;
    position: fixed;
    border-radius: 50%;
    right: 15px;
    bottom: 170px;
    padding-top: 15px;
    margin-bottom: 0;
    z-index: 998;
    cursor: pointer;
}

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

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

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

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

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

/* 背景颜色变灰色 */
body.dark .card,
body.dark .block-with-text:after {
    background-color: #282c34;
}

/* 背景颜色变黑色 */
body.dark,
body.dark .v[data-class=v] .vcount,
body.dark #rewardModal .modal-content,
body.dark .modal,
body.dark header .side-nav,
body.dark header .side-nav .menu-list .m-nav-show {
    background-color: #12121c;
   /**因为我的背景图导致部分页面无法全部切换成深色背景, 需要取消背景图片**/
    background-image: url(#); 
}

/* 改变透明度 */
body.dark .aplayer {
    background: #2f3742 !important;
}

body.dark img, body.dark strong {
    filter: brightness(.7);
}
/******夜间模式切换样式 end*******/

41. 添加valine评论功能

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

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

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

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

<style>
    .valine-card {
        margin: 1.5rem auto;
    }

    .valine-card .card-content {
        padding: 20px 20px 5px 20px;
    }

    #vcomments textarea {
        box-sizing: border-box;
        background: url("<%- url_for(theme.valine.background) %>") 100% 100% no-repeat;
    }

    #vcomments p {
        margin: 2px 2px 10px;
        font-size: 1.05rem;
        line-height: 1.78rem;
    }

    #vcomments blockquote p {
        text-indent: 0.2rem;
    }

    #vcomments a {
        padding: 0 2px;
        color: #4cbf30;
        font-weight: 500;
        text-decoration: none;
    }

    #vcomments img {
        max-width: 100%;
        height: auto;
        cursor: pointer;
    }

    #vcomments ol li {
        list-style-type: decimal;
    }

    #vcomments ol,
    ul {
        display: block;
        padding-left: 2em;
        word-spacing: 0.05rem;
    }

    #vcomments ul li,
    ol li {
        display: list-item;
        line-height: 1.8rem;
        font-size: 1rem;
    }

    #vcomments ul li {
        list-style-type: disc;
    }

    #vcomments ul ul li {
        list-style-type: circle;
    }

    #vcomments table, th, td {
        padding: 12px 13px;
        border: 1px solid #dfe2e5;
    }

    #vcomments table, th, td {
        border: 0;
    }

    table tr:nth-child(2n), thead {
        background-color: #fafafa;
    }

    #vcomments table th {
        background-color: #f2f2f2;
        min-width: 80px;
    }

    #vcomments table td {
        min-width: 80px;
    }

    #vcomments h1 {
        font-size: 1.85rem;
        font-weight: bold;
        line-height: 2.2rem;
    }

    #vcomments h2 {
        font-size: 1.65rem;
        font-weight: bold;
        line-height: 1.9rem;
    }

    #vcomments h3 {
        font-size: 1.45rem;
        font-weight: bold;
        line-height: 1.7rem;
    }

    #vcomments h4 {
        font-size: 1.25rem;
        font-weight: bold;
        line-height: 1.5rem;
    }

    #vcomments h5 {
        font-size: 1.1rem;
        font-weight: bold;
        line-height: 1.4rem;
    }

    #vcomments h6 {
        font-size: 1rem;
        line-height: 1.3rem;
    }

    #vcomments p {
        font-size: 1rem;
        line-height: 1.5rem;
    }

    #vcomments hr {
        margin: 12px 0;
        border: 0;
        border-top: 1px solid #ccc;
    }

    #vcomments blockquote {
        margin: 15px 0;
        border-left: 5px solid #42b983;
        padding: 1rem 0.8rem 0.3rem 0.8rem;
        color: #666;
        background-color: rgba(66, 185, 131, .1);
    }

    #vcomments pre {
        font-family: monospace, monospace;
        padding: 1.2em;
        margin: .5em 0;
        background: #272822;
        overflow: auto;
        border-radius: 0.3em;
        tab-size: 4;
    }

    #vcomments code {
        font-family: monospace, monospace;
        padding: 1px 3px;
        font-size: 0.92rem;
        color: #e96900;
        background-color: #f8f8f8;
        border-radius: 2px;
    }

    #vcomments pre code {
        font-family: monospace, monospace;
        padding: 0;
        color: #e8eaf6;
        background-color: #272822;
    }

    #vcomments pre[class*="language-"] {
        padding: 1.2em;
        margin: .5em 0;
    }

    #vcomments code[class*="language-"],
    pre[class*="language-"] {
        color: #e8eaf6;
    }

    #vcomments [type="checkbox"]:not(:checked), [type="checkbox"]:checked {
        position: inherit;
        margin-left: -1.3rem;
        margin-right: 0.4rem;
        margin-top: -1px;
        vertical-align: middle;
        left: unset;
        visibility: visible;
    }

    #vcomments b,
    strong {
        font-weight: bold;
    }

    #vcomments dfn {
        font-style: italic;
    }

    #vcomments small {
        font-size: 85%;
    }

    #vcomments cite {
        font-style: normal;
    }

    #vcomments mark {
        background-color: #fcf8e3;
        padding: .2em;
    }

    #vcomments table, th, td {
        padding: 12px 13px;
        border: 1px solid #dfe2e5;
    }

    table tr:nth-child(2n), thead {
        background-color: #fafafa;
    }

    #vcomments table th {
        background-color: #f2f2f2;
        min-width: 80px;
    }

    #vcomments table td {
        min-width: 80px;
    }

    #vcomments [type="checkbox"]:not(:checked), [type="checkbox"]:checked {
        position: inherit;
        margin-left: -1.3rem;
        margin-right: 0.4rem;
        margin-top: -1px;
        vertical-align: middle;
        left: unset;
        visibility: visible;
    }

    .v[data-class="v"] .vwrap .vheader .vinput {
        width: 32%;
        border-bottom: 1px dashed #dedede;
    }
</style>

<div class="card valine-card" data-aos="fade-up">
    <div class="comment_headling"
         style="font-size: 20px; font-weight: 700; position: relative; padding-left: 20px; top: 15px; padding-bottom: 5px;">
        <i class="fas fa-comments fa-fw" aria-hidden="true"></i>
        <span>评论</span>
    </div>
    <div id="vcomments" class="card-content" style="display: grid">
    </div>
</div>

<script src="<%- theme.jsDelivr.url %><%- url_for('/libs/valine/av-min.js') %>"></script>
<script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.valine) %>"></script>
<script>
    let metaPlaceholder = <%- JSON.stringify(theme.valine.metaPlaceholder) %> ;
    //这里要换行
    new Valine({
        el: '#vcomments',
        appId: '<%- theme.valine.appId %>',
        appKey: '<%- theme.valine.appKey %>',
        notify: '<%- theme.valine.notify %>' === 'true',
        verify: '<%- theme.valine.verify %>' === 'true',
        visitor: '<%- theme.valine.visitor %>' === 'true',
        avatar: '<%- theme.valine.avatar %>',
        pageSize: '<%- theme.valine.pageSize %>',
        lang: '<%- theme.valine.lang %>',
        placeholder: '<%= theme.valine.placeholder %>',

        meta: <%- '["' + theme.valine.guest_info.join('", "') + '"]' %>,
        recordIP: '<%- theme.valine.recordIP %>' === 'true',
        enableQQ: '<%- theme.valine.avatar %>',
        requiredFields: <%- '["' + theme.valine.guest_info.join('", "') + '"]' %>,
        master: <%- '["' + theme.valine.master.join('", "') + '"]' %>,
        friends: <%- '["' + theme.valine.friends.join('", "') + '"]' %>,
        tagMeta: <%- '["' + theme.valine.tagMeta.join('", "') + '"]' %>,
        metaPlaceholder: metaPlaceholder,

    });

    document.body.addEventListener('click', function (e) {
        if (e.target.classList.contains('vsubmit')) {
            const email = document.querySelector('input[type=email]');
            const nick = document.querySelector('input[name=nick]');
            const reg = /^[A-Za-z0-9_-\u4e00-\u9fa5]+@[a-zA-Z0-9_-]+(\.[a-zA-Z0-9_-]+)+$/;
            if (!email.value || !nick.value || !reg.test(email.value)) {
                const str = `<div class="valert txt-center"><div class="vtext">请填写正确的昵称和邮箱!</div></div>`;
                const vmark = document.querySelector('.vmark');
                vmark.innerHTML = str;
                vmark.style.display = 'block';

                e.stopPropagation();

                setTimeout(function () {
                    vmark.style.display = 'none';
                    vmark.innerHTML = '';
                }, 2500);
            }
        }
    }, true);
</script>

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

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

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

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

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

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

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

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

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

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

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

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

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

42. 添加artitalk说说功能

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

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

首先, 下载artitalk源码

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

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

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

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

找到css在最后一行添加:

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

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

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

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

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

  js:
    jquery: /libs/jquery/jquery.min.js
    materialize: /libs/materialize/materialize.min.js # 1.0.0
    artitalk: /libs/artitalk/artitalk.min.js  # 最后一行添加

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

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

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

<style type="text/css">
    /* don't remove. */
    .about-cover {
        height: 75vh;
    }
</style>

<%- partial('_partial/bg-cover') %>

<main class="content">
    <% if (theme.artitalk && theme.artitalk.enable) { %>
        <%- partial('_widget/artitalk') %>
    <% } %>
</main>

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

<style type="text/css">

    #artitalk_main .cbp_tmtimeline > li .cbp_tmlabel::after {
        right: 100%;
        border: solid transparent;
        z-index: -1;
        content: " ";
        height: 0;
        width: 0;
        position: absolute;
        pointer-events: none;
        border-right-color:  #0bb7fbd6 ;
        border-width: 10px;
        top: 4px;
    }

    #pubShuo {

        margin-right: 5px;
    }

    #operare_artitalk .shuoshuo_input_log {
        outline-style: none;
        margin-top: 5px;
        border: 1px solid #ccc;
        border-radius: 6px;
        padding: 8px 16px;
        font-size: 12px;
        background-color: transparent;

        color: #0bb7fbd6;
        width: 70%;
        height: 28px;
        margin-left: 10px;
    }

    #artitalk_main {

        margin-top: 5px;
        margin-left: 5%;
        margin-right: 5%;

    }

    #lazy {
        margin-top: 40px;
    }

</style>

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

<article id="articles11" class="container  chip-container">
    <div class="row ">

        <div class=" card">
            <div class="card-content">
                <div class="tag-title center-align">
                    <i class="fas fa-pen-alt"></i> 说说
                </div>
                <div id="artitalk_main"></div>
            </div>
        </div>

    </div>
</article>
<script>
    new Artitalk({
        appId: "<%= theme.artitalk.appId %>",
        appKey: "<%= theme.artitalk.appKey %>",
        <% if (theme.artitalk.serverURL) { %>
        serverURL: "<%= theme.artitalk.serverURL %>",
        <% } %>
        <% if (theme.artitalk.lang) { %>
        lang: "<%= theme.artitalk.lang %>",
        <% } %>
        <% if (theme.artitalk.pageSize) { %>
        pageSize: "<%= theme.artitalk.pageSize %>",
        <% } %>
        <% if (theme.artitalk.shuoPla) { %>
        shuoPla: "<%= theme.artitalk.shuoPla %>",
        <% } %>
        <% if (theme.artitalk.avatarPla) { %>
        avatarPla: "<%= theme.artitalk.avatarPla %>",
        <% } %>
        <% if (theme.artitalk.motion == 0) { %>
        motion: 0,
        <% } else { %>
        motion: 1,
        <% } %>
        <% if (theme.artitalk.bgImg) { %>
        bgImg: "<%= theme.artitalk.bgImg %>",
        <% } %>
        <% if (theme.artitalk.color1) { %>
        color1: "<%= theme.artitalk.color1 %>",
        <% } %>
        <% if (theme.artitalk.color2) { %>
        color2: "<%= theme.artitalk.color2 %>",
        <% } %>
        <% if (theme.artitalk.color3) { %>
        color3: "<%= theme.artitalk.color3 %>",
        <% } %>
        <% if (theme.artitalk.cssUrl) { %>
        cssUrl: "<%= theme.artitalk.cssUrl %>",
        <% } %>
        atEmoji: {
            baiyan: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/baiyan.png",
            bishi: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/bishi.png",
            bizui: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/bizui.png",
            chan: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/chan.png",
            daku: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/daku.png",
            dalao: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/dalao.png",
            dalian: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/dalian.png",
            dianzan: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/dianzan.png",
            doge: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/doge.png",
            facai: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/facai.png",
            fadai: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/fadai.png",
            fanu: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/fanu.png",
        },
    })
</script>

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

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

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

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

hexo new page "artitalk"

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

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

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

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

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

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

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

 <!--时间轴区域块-->
    <div id="cd-timeline" class="container" style="display: none;">
        <% page.posts.each(function(post) { %>
            <div class="cd-timeline-block">

                <%# year. %>
                <% if (date(post.date, 'YYYY') != year) { %>
                    <% year = date(post.date, 'YYYY'); %>
                    <div class="cd-timeline-img year" data-aos="zoom-in-up">
                        <a href="<%- url_for('/archives/' + year) %>"><%- year %></a>
                    </div>
                <% } %>

                <%# month. %>
                <% if (date(post.date, 'YYYY-MM') != month) { %>
                    <%
                        month = date(post.date, 'YYYY-MM');
                        var m = date(post.date, 'MM')
                    %>
                    <div class="cd-timeline-img month" data-aos="zoom-in-up">
                        <a href="<%- url_for('/archives/' + year + '/' + m) %>">
                           <%- m %></a>
                    </div>
                <% } %>

                <%# every day posts. %>
                <div class="cd-timeline-img day" data-aos="zoom-in-up">
                    <span><%- date(post.date, 'YYYY-MM-DD').substring(8, 10) %></span>
                </div>
                <article class="cd-timeline-content" data-aos="fade-up">
                    <div class="article col s12 m6">
                        <div class="card">
                            <a href="<%- url_for(post.path) %>">
                                <div class="card-image">
                                    <% if (post.img) { %>
                                        <img src="<%- url_for(post.img) %>" 
                                             class="responsive-img"
                                             alt="<%= post.title %>">
                                    <% } else { %>
                                        <%
                                            featureimg =                                           featureImages[Math.abs(hashCode(post.title) 
                                           % featureImages.length)];
                                        %>
                                     <img src="<%- theme.jsDelivr.url %>
                                                  <%- url_for(featureimg) %>"
                                      class="responsive-img" alt="<%= post.title
                                                                  %>">
                                    <% } %>
                                    <span class="card-title"><%= post.title %>
                                       </span>
                                </div>
                            </a>
                            <div class="card-content article-content">
                                <div class="summary block-with-text">
                                    <% if (post.summary && post.summary.length > 0)
                                       { %>
                                        <%- post.summary %>
                                    <% } else { %>
                                        <%- strip_html(post.content).substring(0, 
                                            120) %>
                                    <% } %>
                                </div>
                                <div class="publish-info">
                                <span class="publish-date">
                                    <i class="far fa-clock fa-fw icon-date"></i>
                                   <%= date(post.date, config.date_format) %>
                                </span>
                                    <span class="publish-author">
                                        <% if (post.categories && 
                                           post.categories.length > 0) { %>
                                            <i class="fas fa-bookmark fa-fw 
                                                      icon-category"></i>
                                            <% post.categories.forEach(category => { 
                                               %>
                                                <a href="<%- url_for(category.path)
                                                         %>" class="post-category">
                                    <%- category.name %>
                                    </a>
                                            <% }); %>
                                        <% } else if (post.author && 
                                           post.author.length > 0) { %>
                                            <i class="fas fa-user fa-fw"></i>
                                            <%- post.author %>
                                        <% } else { %>
                                            <i class="fas fa-user fa-fw"></i>
                                            <%- config.author %>
                                        <% } %>
                                    </span>
                                </div>
                            </div>

                            <% if (post.tags && post.tags.length) { %>
                                <div class="card-action article-tags">
                                    <% post.tags.forEach(tag => { %>
                                        <a href="<%- url_for(tag.path) %>">
                                           <span
                                                    class="chip bg-color">
                                              <%= tag.name %></span></a>
                                    <% }); %>
                                </div>
                            <% } %>
                        </div>
                    </div>
                </article>
            </div>
        <% }); %>
    </div>

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

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

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

实现切换效果的js代码:

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

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

切换到时间列表的代码:

<div id="cd-table" class="container archive-container">
        <% page.posts.each(function(post) { %>
            <div class="card">
                <div class="card-content">
                    <div class="archive">
                        <%# year. %>
                        <% if (date(post.date, 'YYYY') != year) { %>
                            <% year = date(post.date, 'YYYY'); %>
                            <h4 class="archive-year" id="<%- year %>"><%- year %>
                            </h4>
                        <% } %>

                        <div class="articles">
                            <div class="article content>">
                                <div class="article-sort-post">
                                    <div class="article-sort-item_title">
                                        <a href="<%- url_for(post.path) %>">
                                        <h5>
                                        <i class="fa fa-clock" 
                                          style="font-size: 1rem;cursor: pointer;">														 </i>
                                         <time class="is-text-small" 																			datetime="2021-08-20T20:20:00.000Z"
                                         itemprop="datePublished" style="color: 																#ff542d;">
                                          <%- date(post.date, 'YYYY-MM-DD') %>
                                           </time>
                                            </h5>
                                        </a>
                                        <h6 class="is-6">
                                            <a href="<%- url_for(post.path) %>"></a>
                                            <a href="<%- url_for(post.path) %>">
                                               <%= post.title %></a>
                                        </h6>
                                    </div>
                                </div>
                            </div>

                        </div>
                    </div>
                </div>
            </div>
        <% }); %>
</div>

查看我的效果:

归档时间列表效果

44. 域名解析

为自己的博客申请自定义域名, 我在腾讯云平台购买的域名, 在阿里云平台或其他平台购买都可以. 进入腾讯云域名控制台 , 就会看到我注册的域名:

image-20210914203719272

然后点击解析进入域名解析控制台, 添加记录, 看我的解析配置:

image-20210914204014844

域名解析主要配置项的含义是:

主机记录 :

  • @ 表示直接解析主域名 crystalblog.xyz
  • www 表示将域名解析为 www.crystalblog.xyz

记录类型:

记录值:

获取gitHub服务地址

hexo博客配置CNAME, 在hexo博客的source目录下,创建一个名为CNAME的文件(注意要大写),内容写之前购买的域名。下面是我引用的博友的图片作为参考:

hexo博客CNAME配置

注:如果不做这一步,每次hexo d部署到github上以后,对应仓库的域名设置里的域名会被重置回原来的username.github.io,你需要再次设置成你的域名,很麻烦。

hexo博客根目录下的_config.yml文件中配置域名:

# URL
## Set your site url here. For example, if you use GitHub Page, set url as 'https://username.github.io/project'
## 自定义域名, 目前是部署到gitHub上面解析自定义域名的, gitee个人版不支持自定义域名
url: https://www.crystalblog.xyz
## gitee
#url: https://wang-qz.gitee.io/crystal-blog
## github
#url: https://wang-qz.github.io/

最后进入gitHub部署的博客仓库设置域名, 下面是我引用的博友的图片作为参考:

hexo博客GitHub设置域名

另外,在你绑定了新域名之后,gitHub原来的默认域名https://wang-qz.github.io/并没有失效,而是会自动跳转到你的新域名https://www.crystalblog.xyz

最后附上hexo域名绑定参考资料https://zhuanlan.zhihu.com/p/338299590

45. 文件压缩

压缩静态文件可以提高静态博客的页面加载速度, 下面提供两种文件压缩方式.

hexo-neat压缩参考资料: https://blog.csdn.net/weixin_41287260/article/details/99687257

gulp压缩参考资料: https://blog.csdn.net/ganzhilin520/article/details/79052512

我使用的是hexo-neat方式, 使用简单.

46. SEO优化

SEO(Search Engine Optimization):汉译为搜索引擎优化。是一种方式:利用搜索引擎的规则提高网站在有关搜索引擎内的自然排名。 ——百度百科

参考资料1: https://www.zyskys.com/posts/60945

参考资料2: https://blog.csdn.net/lzy98/article/details/81140704

47. Gitee+PicGo图床

markdown写博客时粘贴图片都是自动生成的本地图片链接, 当部署到服务器上时, 无法读取到我们本地环境的图片地址, 网上解决方案比较多且推荐的是采用图床, 就是将本地图片上传到网络端的图片服务器中, 然后在写博客时引用网络地址, 读取网络图片的方式展示. 推荐的图床:

SM.MS , 七牛图床 , 路过图床, 去不图床 , 晚风图床

也可以使用gitHub或gitee做图床服务器, 由于gitHub访问速度慢, 我采用的是Gitee+PicGo搭建的图床服务器.

搭建细节就不陈述了, 下面给出参考资料:

Typora 使用 gitee 作为图床

Gitee + Typora,论如何搭建你的免费图床

图床方案之(Gitee+PicGo)

Front-matter

Front-matter 选项详解

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

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

注意:

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

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

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

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

最全示例

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

使用说明

  1. xxxx
  2. xxxx
  3. xxxx

参与贡献

  1. Fork 本仓库
  2. 新建 Feat_xxx 分支
  3. 提交代码
  4. 新建 Pull Request

特技

  1. 使用 Readme_XXX.md 来支持不同的语言,例如 Readme_en.md, Readme_zh.md
  2. Gitee 官方博客 blog.gitee.com
  3. 你可以 https://gitee.com/explore 这个地址来了解 Gitee 上的优秀开源项目
  4. GVP 全称是 Gitee 最有价值开源项目,是综合评定出的优秀开源项目
  5. Gitee 官方提供的使用手册 https://gitee.com/help
  6. Gitee 封面人物是一档用来展示 Gitee 会员风采的栏目 https://gitee.com/gitee-stars/
Apache License Version 2.0, January 2004 http://www.apache.org/licenses/ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 1. Definitions. "License" shall mean the terms and conditions for use, reproduction, and distribution as defined by Sections 1 through 9 of this document. "Licensor" shall mean the copyright owner or entity authorized by the copyright owner that is granting the License. "Legal Entity" shall mean the union of the acting entity and all other entities that control, are controlled by, or are under common control with that entity. For the purposes of this definition, "control" means (i) the power, direct or indirect, to cause the direction or management of such entity, whether by contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the outstanding shares, or (iii) beneficial ownership of such entity. "You" (or "Your") shall mean an individual or Legal Entity exercising permissions granted by this License. "Source" form shall mean the preferred form for making modifications, including but not limited to software source code, documentation source, and configuration files. "Object" form shall mean any form resulting from mechanical transformation or translation of a Source form, including but not limited to compiled object code, generated documentation, and conversions to other media types. "Work" shall mean the work of authorship, whether in Source or Object form, made available under the License, as indicated by a copyright notice that is included in or attached to the work (an example is provided in the Appendix below). "Derivative Works" shall mean any work, whether in Source or Object form, that is based on (or derived from) the Work and for which the editorial revisions, annotations, elaborations, or other modifications represent, as a whole, an original work of authorship. For the purposes of this License, Derivative Works shall not include works that remain separable from, or merely link (or bind by name) to the interfaces of, the Work and Derivative Works thereof. "Contribution" shall mean any work of authorship, including the original version of the Work and any modifications or additions to that Work or Derivative Works thereof, that is intentionally submitted to Licensor for inclusion in the Work by the copyright owner or by an individual or Legal Entity authorized to submit on behalf of the copyright owner. For the purposes of this definition, "submitted" means any form of electronic, verbal, or written communication sent to the Licensor or its representatives, including but not limited to communication on electronic mailing lists, source code control systems, and issue tracking systems that are managed by, or on behalf of, the Licensor for the purpose of discussing and improving the Work, but excluding communication that is conspicuously marked or otherwise designated in writing by the copyright owner as "Not a Contribution." "Contributor" shall mean Licensor and any individual or Legal Entity on behalf of whom a Contribution has been received by Licensor and subsequently incorporated within the Work. 2. Grant of Copyright License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable copyright license to reproduce, prepare Derivative Works of, publicly display, publicly perform, sublicense, and distribute the Work and such Derivative Works in Source or Object form. 3. Grant of Patent License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this section) patent license to make, have made, use, offer to sell, sell, import, and otherwise transfer the Work, where such license applies only to those patent claims licensable by such Contributor that are necessarily infringed by their Contribution(s) alone or by combination of their Contribution(s) with the Work to which such Contribution(s) was submitted. If You institute patent litigation against any entity (including a cross-claim or counterclaim in a lawsuit) alleging that the Work or a Contribution incorporated within the Work constitutes direct or contributory patent infringement, then any patent licenses granted to You under this License for that Work shall terminate as of the date such litigation is filed. 4. Redistribution. You may reproduce and distribute copies of the Work or Derivative Works thereof in any medium, with or without modifications, and in Source or Object form, provided that You meet the following conditions: (a) You must give any other recipients of the Work or Derivative Works a copy of this License; and (b) You must cause any modified files to carry prominent notices stating that You changed the files; and (c) You must retain, in the Source form of any Derivative Works that You distribute, all copyright, patent, trademark, and attribution notices from the Source form of the Work, excluding those notices that do not pertain to any part of the Derivative Works; and (d) If the Work includes a "NOTICE" text file as part of its distribution, then any Derivative Works that You distribute must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be construed as modifying the License. You may add Your own copyright statement to Your modifications and may provide additional or different license terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated in this License. 5. Submission of Contributions. Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the Licensor shall be under the terms and conditions of this License, without any additional terms or conditions. Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement you may have executed with Licensor regarding such Contributions. 6. Trademarks. This License does not grant permission to use the trade names, trademarks, service marks, or product names of the Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing the content of the NOTICE file. 7. Disclaimer of Warranty. Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides its Contributions) on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or redistributing the Work and assume any risks associated with Your exercise of permissions under this License. 8. Limitation of Liability. In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even if such Contributor has been advised of the possibility of such damages. 9. Accepting Warranty or Additional Liability. While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional liability. END OF TERMS AND CONDITIONS APPENDIX: How to apply the Apache License to your work. To apply the Apache License to your work, attach the following boilerplate notice, with the fields enclosed by brackets "[]" replaced with your own identifying information. (Don't include the brackets!) The text should be enclosed in the appropriate comment syntax for the file format. We also recommend that a file or class name and description of purpose be included on the same "printed page" as the copyright notice for easier identification within third-party archives. Copyright [yyyy] [name of copyright owner] Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.

简介

hexo博客源码 展开 收起
HTML 等 4 种语言
Apache-2.0
取消

发行版

暂无发行版

贡献者

全部

近期动态

加载更多
不能加载更多了
1
https://gitee.com/wang-qz/hexo-blog-crystal.git
git@gitee.com:wang-qz/hexo-blog-crystal.git
wang-qz
hexo-blog-crystal
hexo-blog-crystal
master

搜索帮助

53164aa7 5694891 3bd8fe86 5694891