crystalBlog
Part I hexo blog building and theme optimization (I)
Theme optimization II
21. Website log settings
Under the subject directory_ In the config.yml configuration file:
# Configure website favicon and website LOGO ## local #favicon: /favicon.png #logo: /medias/logo.png # The CDN I use here can also use local files favicon: https://cdn.jsdelivr.net/gh/guixinchn/image/blog/favicon.png logo: https://cdn.jsdelivr.net/gh/guixinchn/image/blog/logo.png
Picture resources are under \ themes \ hexo theme matrix \ source \ medias in the theme directory. You can also use external chain pictures
22. Website dynamic title line
Implementation method: introduce JS file, create funnyTitle.js under / source/js / under the theme file, and add the following code:
var OriginTitle = document.title; var titleTime; document.addEventListener('visibilitychange', function () { if (document.hidden) { $('[rel="icon"]').attr('href', "https://cdn.jsdelivr.net/gh/guixinchn/image/blog/favicon.png"); document.title = 'I believe you will come back!'; clearTimeout(titleTime); } else { $('[rel="icon"]').attr('href', "https://cdn.jsdelivr.net/gh/guixinchn/image/blog/favicon.png"); document.title = 'Ha ha, I knew it!' + OriginTitle; titleTime = setTimeout(function () { document.title = OriginTitle; }, 2000); } });
Then import it in / layout/layout.ejs under the theme directory
<script src="<%- theme.jsDelivr.url %><%- url_for('/js/funnyTitle.js') %>"></script>
23. Add a resume to the about page
Open the / layout/about.ejs file in the theme directory and add the following code:
<div class="card-content article-card-content"> <div class="title center-align" data-aos="zoom-in-up"> <i class="fa fa-address-book"></i> <%- __('curriculum vitae') %> </div> <div id="articleContent" data-aos="fade-up"> <%- page.content %> </div> </div>
In the / layout/about.ejs file in the theme directory, the relevant information about the profile in the following code is from the theme_ Get from config.yml configuration file
<div class="profile center-align"> <div class="avatar"> <img src="<%- theme.jsDelivr.url %><%- url_for(theme.profile.avatar) %>" alt="<%- config.author %>" class="circle responsive-img avatar-img"> </div> <div class="author"> <div class="post-statis hide-on-large-only" data-aos="zoom-in-right"> <%- partial('_partial/post-statis') %> </div> <div class="title"><%- config.author %></div> <div class="career"><%- theme.profile.career %></div> <div class="social-link hide-on-large-only" data-aos="zoom-in-left"> <%- partial('_partial/social-link') %> </div> </div> </div>
Under the subject directory_ The profile information in the config.yml configuration file can be modified
# profile in about page, including avatars, career, and personal introductions. # Configure personal information in the about page, including avatar, occupation and personal introduction profile: avatar: /medias/avatar.jpg career: Software Engineer introduction: If you wish to succeed, you should use persistence as your good friend, experience as your reference, prudence as your brother and hope as your sentry.
24. Page 404
The original topic does not have a 404 page. First, create a 404.md in the / source / directory under the topic directory, as follows:
--- title: 404 date: 2017-07-19 16:41:10 type: "404" layout: "404" description: "The page you visited was taken away by aliens :(" ---
Then create a new / layout/404.ejs file in the theme directory, as follows:
<style type="text/css"> /* don't remove. */ .about-cover { height: 90.2vh; } </style> <div class="bg-cover pd-header about-cover"> <div class="container"> <div class="row"> <div class="col s10 offset-s1 m8 offset-m2 l8 offset-l2"> <div class="brand"> <div class="title center-align"> 404 </div> <div class="description center-align"> <%= page.description %> </div> </div> </div> </div> </div> </div> <% if (theme.banner.enable) { %> <script> // Switch banner image every day var bannerUrl = "<%- theme.jsDelivr.url %><%- url_for('/medias/banner/') %>" + new Date().getDay() + '.jpg'; $('.bg-cover').css('background-image', 'url(' + bannerUrl + ')'); </script> <% } else { %> <script> $('.bg-cover').css('background-image', 'url(<%- theme.jsDelivr.url %><%- url_for('/medias/banner/0.jpg') %>)'); </script> <% } %>
25. Website footer modification
Modify / layout under the theme directory as needed/_ Partial / footer.ejs file, you can set site visits, number of visitors, word count, site running time, website filing and other information
<footer class="page-footer bg-color"> <% if (theme.music.enable && theme.music.fixed) { %> <%- partial('_widget/music') %> <% } %> <div class="container row center-align" style="margin-bottom: <% if (theme.time.enable) { %>15<% } else { %>0<% } %>px !important;"> <div class="col s12 m8 l8 copy-right"> Copyright © <% if (theme.time.year !== new Date().getFullYear()) { %> <span id="year"><%- theme.time.year %>-<%- new Date().getFullYear() %></span> <% } else { %> <span id="year"><%- theme.time.year %></span> <% } %> <i class="fa fa-heart" style="color: #ff71a8"></i> <a href="<%- url_for('/about') %>" target="_blank"><%- config.author %></a> | Powered by <a href="https://hexo.io/" target="_blank">Hexo</a> | <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"> <i class="fas fa-chart-area"></i> <%- __('siteTotalWords') %>: <span class="white-color"><%= totalcount(site) %></span> <span/> <% } %> <% let socialClass = '' %> <% if (theme.busuanziStatistics && theme.busuanziStatistics.enable) { %> <% socialClass = 'social-statis' %> <% } %> <% if (theme.busuanziStatistics && theme.busuanziStatistics.totalTraffic) { %> <span id="busuanzi_container_site_pv3" style="display: inline"> | <i class="far fa-eye"></i> <%- __('siteTotalVisits') %>: <span id="busuanzi_value_site_pv" class="white-color"><%= totalcount(site) %></span> </span> <% } %> <% if (theme.busuanziStatistics && theme.busuanziStatistics.totalNumberOfvisitors) { %> <span id="busuanzi_container_site_uv3" style="display: inline"> | <i class="fas fa-users"></i> <%- __('siteTotalVisitors') %>: <span id="busuanzi_value_site_uv" class="white-color"><%= totalcount(site) %></span> </span> <% } %> <br> <!-- Running days reminder. --> <% if (theme.time.enable) { %> <span id="sitetime"> Loading ...</span> <script> var calcSiteTime = function () { var seconds = 1000; var minutes = seconds * 60; var hours = minutes * 60; var days = hours * 24; var years = days * 365; var today = new Date(); var startYear = "<%- theme.time.year %>"; var startMonth = "<%- theme.time.month %>"; var startDate = "<%- theme.time.date %>"; var startHour = "<%- theme.time.hour %>"; var startMinute = "<%- theme.time.minute %>"; var startSecond = "<%- theme.time.second %>"; var todayYear = today.getFullYear(); var todayMonth = today.getMonth() + 1; var todayDate = today.getDate(); var todayHour = today.getHours(); var todayMinute = today.getMinutes(); var todaySecond = today.getSeconds(); var t1 = Date.UTC(startYear, startMonth, startDate, startHour, startMinute, startSecond); var t2 = Date.UTC(todayYear, todayMonth, todayDate, todayHour, todayMinute, todaySecond); var diff = t2 - t1; var diffYears = Math.floor(diff / years); var diffDays = Math.floor((diff / days) - diffYears * 365); var diffHours = Math.floor((diff / hours) - diffYears * 365 * 24 - diffDays * 24); var diffMinutes = Math.floor((diff / minutes) - diffYears * 365 * 24 * 60 - diffDays * 24 * 60 - diffHours * 60); var diffSeconds = Math.floor((diff / seconds) - diffYears * 365 * 24 * 60 * 60 - diffDays * 24 * 60 * 60 - diffHours * 60 * 60 - diffMinutes * 60); // Distinguish whether there is a year var language = '<%- config.language %>'; if (startYear === String(todayYear)) { document.getElementById("year").innerHTML = todayYear; var daysTip = 'This site has been running for ' + diffDays + ' days'; if (language === 'zh-CN') { daysTip = 'This station has been running ' + diffDays + ' day'; } else if (language === 'zh-HK') { daysTip = 'This station has been running ' + diffDays + ' day'; } document.getElementById("sitetime").innerHTML = daysTip; } else { document.getElementById("year").innerHTML = startYear + " - " + todayYear; var yearsAndDaysTip = 'This site has been running for ' + diffYears + ' years and ' + diffDays + ' days'; if (language === 'zh-CN') { yearsAndDaysTip = 'This station has been running ' + diffYears + ' year ' + diffDays + ' day ' + diffHours + ' hour ' + diffMinutes + ' minute ' + diffSeconds + ' second'; } else if (language === 'zh-HK') { yearsAndDaysTip = 'This station has been running ' + diffYears + ' year ' + diffDays + ' day'; } document.getElementById("sitetime").innerHTML = yearsAndDaysTip; } } var timer = setInterval(calcSiteTime); // calcSiteTime(); </script> <% } %> | <% if (theme.icp.enable) { %> <span id="icp"> <img src="<%- theme.jsDelivr.url %><%- url_for('/medias/icp.png') %>" style="vertical-align: text-bottom;"/> <a href="<%- url_for(theme.icp.url) %>" target="_blank"><%= theme.icp.text %></a> </span> <% } %> </div> <div class="col s12 m4 l4 social-link <%- socialClass %>"> <%- partial('_partial/social-link') %> </div> </div> </footer>
You can also add Baidu garlic statistics
/ layout found/_ Partial / footer.ejs file, modify the corresponding style to
<!--Total number of visitors--> <% if (theme.busuanziStatistics && theme.busuanziStatistics.totalNumberOfvisitors) { %> <span id="busuanzi_container_site_uv" style="display: inline"> | <i class="fas fa-users"></i> <%- __('siteTotalVisitors') %>: <span id="busuanzi_value_site_uv" class="white-color"></span> </span> <% } %> <!--Finally add--> <script> let _hmt = _hmt || []; (function () { var hm = document.createElement("script"); hm.src = "https://hm.baidu.com/hm.js?147475454185ebcf440a27cc35e793ef"; var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(hm, s); })(); </script>
26. Add anime characters
Install plug-ins hexo-helper-live2d
npm install --save hexo-helper-live2d
Install and download the animated character library. There are many animated characters. You can query information online. Several are recommended below
npm install --save live2d-widget-model-shizuku #Desk girl npm install --save live2d-widget-model-hibiki #domineering lady npm install --save live2d-widget-model-wanko #Dog npm install --save live2d-widget-model-haruto #Navy girl npm install --save live2d-widget-model-miku #lolita
Blog root_ config.yml file configuration:
## Add animation live2d module NPM install -- save hexo helper live2d ## Download animated character library npm install live2d-widget-model-z16 -D live2d: enable: true scriptFrom: local # default pluginRootPath: live2dw/ # The root directory (relative path) of the plug-in on the site pluginJsPath: lib/ # Script file relative to plug-in root directory path pluginModelPath: assets/ # Model file relative to plug-in root directory path tagMode: false # Label mode, do you want to replace only live2d tag labels instead of inserting them into all pages debug: false # Debug, whether to output logs on the console model: use: live2d-widget-model-miku display: position: right #Animation position width: 150 height: 190 # Position configuration. This is in the middle of the left sidebar hOffset: 50 # Adjust horizontal position vOffset: -5 # Adjust vertical position mobile: show: false # Show on mobile device scale: 0.5 # Zoom on mobile devices react: opacityDefault: 0.7 opacityOnHover: 0.8
27. Snowflake and cherry blossom effect
Add snowflake falling effect
Add a new / source/js/snow.js file in the theme directory. Add the following contents:
/*Style one*/ (function ($) { $.fn.snow = function (options) { var $flake = $('<div id="snowbox" />').css({ 'position': 'absolute', 'z-index': '9999', 'top': '-50px' }).html('❄'), documentHeight = $(document).height(), documentWidth = $(document).width(), defaults = { minSize: 10, maxSize: 20, newOn: 1000, flakeColor: "#AFDAEF" /* Here you can define the color of snowflakes. If you want white, you can change it to #FFFFFF */ }, options = $.extend({}, defaults, options); var interval = setInterval(function () { var startPositionLeft = Math.random() * documentWidth - 100, startOpacity = 0.5 + Math.random(), sizeFlake = options.minSize + Math.random() * options.maxSize, endPositionTop = documentHeight - 200, endPositionLeft = startPositionLeft - 500 + Math.random() * 500, durationFall = documentHeight * 10 + Math.random() * 5000; $flake.clone().appendTo('body').css({ left: startPositionLeft, opacity: startOpacity, 'font-size': sizeFlake, color: options.flakeColor }).animate({ top: endPositionTop, left: endPositionLeft, opacity: 0.2 }, durationFall, 'linear', function () { $(this).remove() }); }, options.newOn); }; })(jQuery); $(function () { $.fn.snow({ minSize: 5, /* Define minimum snowflake size */ maxSize: 50,/* Define maximum snowflake size */ newOn: 300 /* Define the density. The smaller the number, the more dense it is */ }); }); /*Style II*/ /* Control snow */ function snowFall(snow) { /* Configurable properties */ snow = snow || {}; this.maxFlake = snow.maxFlake || 200; /* Maximum number of slices */ this.flakeSize = snow.flakeSize || 10; /* Snowflake shape */ this.fallSpeed = snow.fallSpeed || 1; /* Falling speed */ } /* Compatible writing */ requestAnimationFrame = window.requestAnimationFrame || window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || window.msRequestAnimationFrame || window.oRequestAnimationFrame || function (callback) { setTimeout(callback, 1000 / 60); }; cancelAnimationFrame = window.cancelAnimationFrame || window.mozCancelAnimationFrame || window.webkitCancelAnimationFrame || window.msCancelAnimationFrame || window.oCancelAnimationFrame; /* It began to snow */ snowFall.prototype.start = function () { /* Create canvas */ snowCanvas.apply(this); /* Create a snowflake shape */ createFlakes.apply(this); /* Painting snow */ drawSnow.apply(this) } /* Create canvas */ function snowCanvas() { /* Add Dom node */ var snowcanvas = document.createElement("canvas"); snowcanvas.id = "snowfall"; snowcanvas.width = window.innerWidth; snowcanvas.height = document.body.clientHeight; snowcanvas.setAttribute("style", "position:absolute; top: 0; left: 0; z-index: 1; pointer-events: none;"); document.getElementsByTagName("body")[0].appendChild(snowcanvas); this.canvas = snowcanvas; this.ctx = snowcanvas.getContext("2d"); /* Processing of window size change */ window.onresize = function () { snowcanvas.width = window.innerWidth; /* snowcanvas.height = window.innerHeight */ } } /* Snow moving object */ function flakeMove(canvasWidth, canvasHeight, flakeSize, fallSpeed) { this.x = Math.floor(Math.random() * canvasWidth); /* x coordinate */ this.y = Math.floor(Math.random() * canvasHeight); /* y coordinate */ this.size = Math.random() * flakeSize + 2; /* shape */ this.maxSize = flakeSize; /* Maximum shape */ this.speed = Math.random() * 1 + fallSpeed; /* Falling speed */ this.fallSpeed = fallSpeed; /* Falling speed */ this.velY = this.speed; /* Y Directional velocity */ this.velX = 0; /* X Directional velocity */ this.stepSize = Math.random() / 30; /* step */ this.step = 0 /* Number of steps */ } flakeMove.prototype.update = function () { var x = this.x, y = this.y; /* Swing left and right (cosine) */ this.velX *= 0.98; if (this.velY <= this.speed) { this.velY = this.speed } this.velX += Math.cos(this.step += .05) * this.stepSize; this.y += this.velY; this.x += this.velX; /* Handling of flying out of boundary */ if (this.x >= canvas.width || this.x <= 0 || this.y >= canvas.height || this.y <= 0) { this.reset(canvas.width, canvas.height) } }; /* Fly out of the boundary - place the top and continue to fall */ flakeMove.prototype.reset = function (width, height) { this.x = Math.floor(Math.random() * width); this.y = 0; this.size = Math.random() * this.maxSize + 2; this.speed = Math.random() * 1 + this.fallSpeed; this.velY = this.speed; this.velX = 0; }; // Render snowflake - random shape (snowflake color can be modified here!!!) flakeMove.prototype.render = function (ctx) { var snowFlake = ctx.createRadialGradient(this.x, this.y, 0, this.x, this.y, this.size); snowFlake.addColorStop(0, "rgba(255, 255, 255, 0.9)"); /* Here is the snowflake color. The default is white */ snowFlake.addColorStop(.5, "rgba(255, 255, 255, 0.5)"); /* If you want to change to another color, please check it yourself */ snowFlake.addColorStop(1, "rgba(255, 255, 255, 0)"); /* Find the hexadecimal RGB color code. */ ctx.save(); ctx.fillStyle = snowFlake; ctx.beginPath(); ctx.arc(this.x, this.y, this.size, 0, Math.PI * 2); ctx.fill(); ctx.restore(); }; /* Create snowflakes - define shapes */ function createFlakes() { var maxFlake = this.maxFlake, flakes = this.flakes = [], canvas = this.canvas; for (var i = 0; i < maxFlake; i++) { flakes.push(new flakeMove(canvas.width, canvas.height, this.flakeSize, this.fallSpeed)) } } /* Painting snow */ function drawSnow() { var maxFlake = this.maxFlake, flakes = this.flakes; ctx = this.ctx, canvas = this.canvas, that = this; /* Empty snowflakes */ ctx.clearRect(0, 0, canvas.width, canvas.height); for (var e = 0; e < maxFlake; e++) { flakes[e].update(); flakes[e].render(ctx); } /* Frame by frame */ this.loop = requestAnimationFrame(function () { drawSnow.apply(that); }); } /* Call and control method */ var snow = new snowFall({maxFlake: 60}); snow.start();
Add the following code in / layout/layout.ejs under the Theme Directory:
<!-- Snow effect --> <% if (theme.snow.enable) { %> <script src="<%- theme.jsDelivr.url %><%- url_for('/js/snow.js' ) %>"></script> <% } %>
Under the topic directory_ Configuration in config.yml:
# Snow effect snow: enable: true
Add Cherry Blossom falling effect
Add a new / source/js/sakura.js file in the theme directory. Add the following contents:
var stop, staticx; var img = new Image(); img.src = ""; function Sakura(x, y, s, r, fn) { this.x = x; this.y = y; this.s = s; this.r = r; this.fn = fn; } Sakura.prototype.draw = function (cxt) { cxt.save(); var xc = 40 * this.s / 4; cxt.translate(this.x, this.y); cxt.rotate(this.r); cxt.drawImage(img, 0, 0, 40 * this.s, 40 * this.s) cxt.restore(); } Sakura.prototype.update = function () { this.x = this.fn.x(this.x, this.y); this.y = this.fn.y(this.y, this.y); this.r = this.fn.r(this.r); if (this.x > window.innerWidth || this.x < 0 || this.y > window.innerHeight || this.y < 0) { this.r = getRandom('fnr'); if (Math.random() > 0.4) { this.x = getRandom('x'); this.y = 0; this.s = getRandom('s'); this.r = getRandom('r'); } else { this.x = window.innerWidth; this.y = getRandom('y'); this.s = getRandom('s'); this.r = getRandom('r'); } } } SakuraList = function () { this.list = []; } SakuraList.prototype.push = function (sakura) { this.list.push(sakura); } SakuraList.prototype.update = function () { for (var i = 0, len = this.list.length; i < len; i++) { this.list[i].update(); } } SakuraList.prototype.draw = function (cxt) { for (var i = 0, len = this.list.length; i < len; i++) { this.list[i].draw(cxt); } } SakuraList.prototype.get = function (i) { return this.list[i]; } SakuraList.prototype.size = function () { return this.list.length; } function getRandom(option) { var ret, random; switch (option) { case 'x': ret = Math.random() * window.innerWidth; break; case 'y': ret = Math.random() * window.innerHeight; break; case 's': ret = Math.random(); break; case 'r': ret = Math.random() * 6; break; case 'fnx': random = -0.5 + Math.random() * 1; ret = function (x, y) { return x + 0.5 * random - 1.7; }; break; case 'fny': random = 1.5 + Math.random() * 0.7 ret = function (x, y) { return y + random; }; break; case 'fnr': random = Math.random() * 0.03; ret = function (r) { return r + random; }; break; } return ret; } function startSakura() { requestAnimationFrame = window.requestAnimationFrame || window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || window.msRequestAnimationFrame || window.oRequestAnimationFrame; var canvas = document.createElement('canvas'), cxt; staticx = true; canvas.height = window.innerHeight; canvas.width = window.innerWidth; canvas.setAttribute('style', 'position: fixed;left: 0;top: 0;pointer-events: none;'); canvas.setAttribute('id', 'canvas_sakura'); document.getElementsByTagName('body')[0].appendChild(canvas); cxt = canvas.getContext('2d'); var sakuraList = new SakuraList(); for (var i = 0; i < 50; i++) { var sakura, randomX, randomY, randomS, randomR, randomFnx, randomFny; randomX = getRandom('x'); randomY = getRandom('y'); randomR = getRandom('r'); randomS = getRandom('s'); randomFnx = getRandom('fnx'); randomFny = getRandom('fny'); randomFnR = getRandom('fnr'); sakura = new Sakura(randomX, randomY, randomS, randomR, { x: randomFnx, y: randomFny, r: randomFnR }); sakura.draw(cxt); sakuraList.push(sakura); } stop = requestAnimationFrame(function () { cxt.clearRect(0, 0, canvas.width, canvas.height); sakuraList.update(); sakuraList.draw(cxt); stop = requestAnimationFrame(arguments.callee); }) } window.onresize = function () { var canvasSnow = document.getElementById('canvas_snow'); } img.onload = function () { startSakura(); } function stopp() { if (staticx) { var child = document.getElementById("canvas_sakura"); child.parentNode.removeChild(child); window.cancelAnimationFrame(stop); staticx = false; } else { startSakura(); } }
Add the following code in / layout/layout.ejs under the Theme Directory:
<!-- Cherry Blossom effect --> <% if (theme.sakura.enable) { %> <script src="<%- theme.jsDelivr.url %><%- url_for('/js/sakura.js') %>"></script> <% } %>
Under the topic directory_ Configuration in config.yml:
# Cherry Blossom effect sakura: enable: true
28. Mouse click effect
Mouse click style 1
In the / source/libs/others/clicklove.js file under the theme directory, provide the code for mouse click effect
!function(e,t,a){function r(){for(var e=0;e<n.length;e++)n[e].alpha<=0?(t.body.removeChild(n[e].el),n.splice(e,1)):(n[e].y--,n[e].scale+=.004,n[e].alpha-=.013,n[e].el.style.cssText="left:"+n[e].x+"px;top:"+n[e].y+"px;opacity:"+n[e].alpha+";transform:scale("+n[e].scale+","+n[e].scale+") rotate(45deg);background:"+n[e].color+";z-index:99999");requestAnimationFrame(r)}var n=[];e.requestAnimationFrame=e.requestAnimationFrame||e.webkitRequestAnimationFrame||e.mozRequestAnimationFrame||e.oRequestAnimationFrame||e.msRequestAnimationFrame||function(e){setTimeout(e,1e3/60)},function(e){var a=t.createElement("style");a.type="text/css";try{a.appendChild(t.createTextNode(e))}catch(t){a.styleSheet.cssText=e}t.getElementsByTagName("head")[0].appendChild(a)}(".heart{width: 10px;height: 10px;position: fixed;background: #f00;transform: rotate(45deg);-webkit-transform: rotate(45deg);-moz-transform: rotate(45deg);}.heart:after,.heart:before{content: '';width: inherit;height: inherit;background: inherit;border-radius: 50%;-webkit-border-radius: 50%;-moz-border-radius: 50%;position: fixed;}.heart:after{top: -5px;}.heart:before{left: -5px;}"),function(){var a="function"==typeof e.onclick&&e.onclick;e.onclick=function(e){a&&a(),function(e){var a=t.createElement("div");a.className="heart",n.push({el:a,x:e.clientX-5,y:e.clientY-5,scale:1,alpha:1,color:"rgb("+~~(255*Math.random())+","+~~(255*Math.random())+","+~~(255*Math.random())+")"}),t.body.appendChild(a)}(e)}}(),r()}(window,document);
Mouse click style 2
In the / source/js/wenzi.js file under the theme directory, provide the code for mouse click effect
/* Mouse click text effect */ var a_idx = 0; jQuery(document).ready(function ($) { $("body").click(function (e) { // var a = new Array(" ❤ Prosperity and strength ❤","❤ democratic ❤","❤ civilization ❤","❤ harmonious ❤","❤ free ❤","❤ equality ❤","❤ fair ❤","❤ rule by law ❤","❤ Patriotic ❤","❤ be dedicated ❤","❤ sincerity ❤","❤ friendly ❤"); var a = new Array("Prosperity and strength", "democratic", "civilization", "harmonious", "free", "equality", "fair", "rule by law", "Patriotic", "be dedicated", "sincerity", "friendly"); var $i = $("<span></span>").text(a[a_idx]); a_idx = (a_idx + 1) % a.length; var x = e.pageX, y = e.pageY; $i.css({ "z-index": 999999999999999999999999999999999999999999999999999999999999999999999, "top": y - 20, "left": x, "position": "absolute", "font-weight": "bold", "color": "rgb(" + ~~(255 * Math.random()) + "," + ~~(255 * Math.random()) + "," + ~~(255 * Math.random()) + ")" }); $("body").append($i); $i.animate({ "top": y - 180, "opacity": 0 }, 1500, function () { $i.remove(); }); }); });
Add the following code in / layout/layout.ejs under the Theme Directory:
<!-- Mouse click effect --> <% if (theme.wenzi.enable) { %> <script src="<%- theme.jsDelivr.url %><%- url_for('/js/wenzi.js') %>"></script> <% } %>
Configure in _config.yml under the topic Directory:
# Mouse click effect wenzi: enable: true
29. Modify blog post template
For the convenience of creating a new article, we can modify the article template. You can modify / Scaffolding / post.md to the following code:
title: {{ title }} date: {{ date }} author: img: cover: false coverImg: top: false toc: true mathjax: false password: summary: keywords: tags: categories:
30. Edit hexo blog online
When editing articles, hexo's native mode is inconvenient. The official website provides a plug-in hexo admin interface and a markdown editor
First install the hexo admin plug-in
npm install --save hexo-admin
Then start hexo s to access http://127.0.0.1:4000/crystalBlog/admin You can edit blog posts conveniently and quickly
After editing, you can also deploy and publish quickly. However, you prefer to write markdown syntax on typera. If you deploy it on your own server, you can use this function, but you still can't use it on gitee or github
31. Site statistics do not show problems
Sometimes it is slow to request busuanzi data, and then the number of views and visitors will be hidden. It may be the default. Add the following code in matrix.css to keep it displayed all the time
#busuanzi_container_site_pv, #busuanzi_value_site_pv, #busuanzi_container_site_uv { display: inline !important; }
32. The article link is too long
Add the following code in / source / CSS / material.css:
/*Article link super long part hidden*/ .reprint__type { display: inline-block; width: 100%; overflow: hidden; }
33. Static / dynamic ribbon, background canvas
Background static ribbon
The configuration item of static ribbon in _config.yml in the theme directory of / layout/layout.ejs file in the Theme Directory:
<!--Background still ribbon--> <% if (theme.ribbon.enable) { %> <% var ribbonSrc = theme.ribbon.clickChange ? theme.libs.js.ribbon : theme.libs.js.ribbonRefresh; %> <script type="text/javascript" size="<%- theme.ribbon.size %>" alpha='<%- theme.ribbon.alpha %>' zIndex="<%- theme.ribbon.zIndex %>" src="<%- theme.jsDelivr.url %><%- url_for(ribbonSrc) %>" async="async"></script> <% } %>
Configuration items of static ribbon in _config.yml under the Theme Directory:
# Background still ribbon ribbon: enable: false # Change to true to open the background static ribbon size: 150 # RIBBON size, default: 90 alpha: 0.6 # Ribbon transparency (0 ~ 1), default: 0.6 zIndex: -1 # The background z-index attribute and css attribute are used to control the position of the layer. The default is: - 1 clickChange: false # Set whether to change the ribbon every time you click
Background dynamic ribbon
/ layout/layout.ejs file in the theme directory_ Configuration items of dynamic ribbon in config.yml:
<!--Background dynamic ribbon--> <% if (theme.ribbon_dynamic.enable) { %> <script type="text/javascript" src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.ribbon_dynamic) %>" async="async"></script> <% } %>
Subject directory_ Configuration items of dynamic ribbon in config.yml:
# Background dynamic ribbon ribbon_dynamic: enable: true # Change to true to open the background dynamic ribbon
Background canvas
/ layout/layout.ejs file in the theme directory_ Configuration item of canvas in config.yml:
<!--Background still ribbon--> <% if (theme.ribbon.enable) { %> <% var ribbonSrc = theme.ribbon.clickChange ? theme.libs.js.ribbon : theme.libs.js.ribbonRefresh; %> <script type="text/javascript" size="<%- theme.ribbon.size %>" alpha='<%- theme.ribbon.alpha %>' zIndex="<%- theme.ribbon.zIndex %>" src="<%- theme.jsDelivr.url %><%- url_for(ribbonSrc) %>" async="async"></script> <% } %>
Subject directory_ Configuration item of canvas in config.yml:
#Background canvas nest canvas_nest: enable: true color: 0,0,255 # Line color, default: '0,0,0'; The three numbers are (R,G,B). Pay attention to the use of and segmentation pointColor: 0,0,255 # Intersection color, default: '0,0,0'; The three numbers are (R,G,B). Pay attention to the use of and segmentation opacity: 0.7 # Line transparency (0 ~ 1), default: 0.5 zIndex: -1 # The background z-index attribute and css attribute are used to control the position of the layer. The default is: - 1 count: 99 # Total number of lines, default: 99
/ layout/layout.ejs in the theme directory is a global layout file. You can add custom effects yourself in the same way as above Snowflakes fall.
34. Extract photo album (wallpaper)
New album (wallpaper) file
hexo new page wallpaper
Modify the under the topic directory_ config.yml file, mine is extracted to the list - Album navigation
Lists: ##detailed list url: / icon: fas fa-list children: - name: music url: /musics icon: fas fa-music - name: film url: /movies icon: fas fa-film - name: read url: /books icon: fas fa-book - name: wallpaper url: /wallpaper icon: fas fa-image
Modify the site / galleries/index.md file
--- title: wallpaper date: 2019-02-04 21:35:22 layout: wallpaper ---
Create a new / layout/wallpaper.ejs file under the theme directory, and add the following contents:
<style type="text/css"> /* don't remove. */ .about-cover { height: 75vh; } </style> <%- partial('_partial/bg-cover') %> <!--The following is the code to extract the photo album function of the blog--> <main class="content"> <% if (theme.myGallery && theme.myGallery.enable) { %> <%- partial('_widget/my-gallery') %> <% } %> </main> <% if (page.total > 1) { %> <%- partial('_partial/paging') %> <% } %>
At the same time, note the following parts of the / layout/about.ejs file in the topic Directory:
<!--gallery Function migration to[detailed list-album]Navigation Division--> <!--<% if (theme.myGallery && theme.myGallery.enable) { %> <%- partial('_widget/my-gallery') %> <% } %>-->
The picture configuration path read from the album is under the theme path_ In config.yml file:
# Configure my album pictures on the about page. If you don't need this information, you can set it inactive or delete it myGallery: enable: true data: - /medias/featureimages/0.jpg - /medias/featureimages/1.jpg - /medias/featureimages/2.jpg
Modify the album layout, find the / source / CSS / material.css file, and modify the following parts:
.my-gallery { margin: 4.5rem auto 1rem; padding: 0 1.2rem; /*This is the display width, the front is the page width, and the back is the picture width*/ max-width: 1100px; /*position: relative;*/ } .my-gallery .photo { margin: .5rem 0; /*This is the space between the top and bottom lines*/ /*position: relative; overflow: hidden;*/ } .my-gallery .photo img { width: 100%; height: 200px; /*Limit the height and keep the peers at the same height, otherwise it will be very chaotic*/ border-radius: 10px; cursor: pointer; }
35. Add album list
For the album list, you can refer to the friendship link interface of the blog and store the friendship link information in / source/_ In the data / friends.json file, and then hexo will render the friend chain list according to the structure in the friends.ejs template file. The effect is as follows:
The principle is actually three a tags, which contain avatar, address and other information. Click and jump to the corresponding address. Then we can also customize the configuration file (galleries.json) and template file (galleries.ejs) of an album list, and then hexo reads the configuration file to automatically generate the album list interface. If this can be optimized, To add an album later, you only need to maintain the album information in the configuration file galleries.json. (add, delete and modify)
Let's just do it. Let's start now
Add list - Album menu
Here are several files to modify:
Configuration files under the topic directory_ config.yml. Don't confuse it with the file with the same name in the root directory of the site. Add the following code under menu:
menu: Lists: ##detailed list url: /crystal-blog icon: fas fa-list children: # Other menus are omitted here - name: wallpaper url: /wallpaper icon: fas fa-image - name: album url: /galleries icon: fas fa-camera
Create a new galleries directory under the site root directory source, and then create index.md under this directory to generate the index.html file
hexo new page "galleries"
Modify the / galleries/index.md file to specify the layout interface:
--- title: My album date: 2021-08-25 19:56:35 type: galleries layout: galleries ---
Add album profile
Create a new / source under the topic directory/_ Data / galleries.json file, add the following album configuration content according to the custom Convention (I maintain three albums):
[ { "name": "Blog background", "url": "/gallery2", "cover": "https://www.bing.com/th?id=OHR.Mpumalanga_ZH-CN9666962271_tmb.jpg&rf=", "description": "Blog background", "photos": [ "https://www.bing.com/th?id=OHR.Mpumalanga_ZH-CN9666962271_tmb.jpg&rf=", ] }, { "name": "Peng Yuyan", "url": "/gallery0", "cover": "https://uploadfile.bizhizu.cn/2016/0106/20160106033828391.jpg", "description": "Peng Yuyan sketching", "photos": [ "http://i.52desktop.cn:81/upimg/allimg/20191204/2019124151645578778013.jpg", "http://i.52desktop.cn:81/upimg/allimg/20191204/2019124151645687778016.jpg" ] }, { "name": "Lau Andy", "url": "/gallery1", "cover": "https://tu1.whhost.net/uploads/20181029/18/1540809870-NgSCnhWkcJ.jpg", "description": "Andy Lau sketching", "photos": [ "https://uploadfile.bizhizu.cn/2015/0306/20150306103233272.jpg", "https://www.beihaiting.com/uploads/allimg/150401/10723-150401195426203.jpg" ] } ]
Add album layout file
Create a new / layout/galleries.ejs template file in the theme directory. Refer to the friends.ejs file. The modified content is as follows:
<link rel="stylesheet" href="<%- theme.jsDelivr.url %><%- url_for(theme.libs.css.gallery) %>"> <%- partial('_partial/bg-cover') %> <main class="content"> <div class="container"> <div class="title center-align" data-aos="zoom-in-up"> <i class="fas fa-camera"></i> <%- __('galleries') %> </div> <% if (site.data && site.data.galleries) { %> <% var galleries = site.data.galleries; %> <div class="gallery-wrapper row"> <% for (var i = 0, len = galleries.length; i < len; i++) { %> <% var gallery = galleries[i]; %> <div class="col s6 m4 l4 xl3 gallery-box"> <a href="./<%- gallery.url %>" class="gallery-item" data-aos="zoom-in-up"> <div class="gallery-cover-box" style="background-image: url(<%- gallery.cover %>);"> </div> <p class="gallery-name" style="font-size: 22px; font-family: 'Chinese block letters'"> <%- gallery.name %> </p> </a> </div> <% } %> </div> <% } %> </div> </main>
Each album in the album list is a < a > < / a > label. Clicking a single album will jump to the album picture display page and create / layout/gallery.ejs under the theme directory. Here we can refer to the layout and rendering method of the wallpaper my-gallery.ejs file. The optimized code is as follows:
<%- partial('_partial/bg-cover') %> <% var galleries = site.data.galleries; var pageTitle = page.title; var currentGallery = getCurrentGallery(galleries, pageTitle) var photos = currentGallery.photos; function getCurrentGallery(galleries, pageTitle) { for (let i = 0; i < galleries.length; i++) { if (galleries[i]['name'] == pageTitle) { return galleries[i]; } } } /***/ %> <div id="myGallery" class="my-gallery"> <div class="title center-align" data-aos="zoom-in-up"> <p class="gallery-name"> <b style="font-size: 35px;"><%- currentGallery.name %></b> </p> </div> <div class="row"> <% if (photos) { %> <% Object.keys(photos).forEach(function(photo) { %> <div class="photo col s12 m6 l4" data-aos="fade-up"> <div class="img-item" data-src="<%- photos[photo] %>"> <img src="<%- theme.jsDelivr.url %><%- url_for(photos[photo]) %>" class="responsive-img"> </div> </div> <% }); %> <% } %> </div> </div> <script> $(function () { let animateClass = 'animated pulse'; $('#myGallery .photo').hover(function () { $(this).addClass(animateClass); }, function () { $(this).removeClass(animateClass); }); }); </script>
Add album list style
/The layout/galleries.ejs template file refers to the gallery.css style file, so you need to add a new / source/css/gallery.css file in the theme directory. The contents are as follows:
.gallery-wrapper { padding-top: 30px; } .gallery-wrapper .gallery-box { padding: 5px !important; } .gallery-wrapper .gallery-item { display: block; overflow: hidden; background-color: #fff; padding: 5px; padding-bottom: 0; position: relative; -moz-box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.22); -webkit-box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.22); box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.22); } .gallery-cover-box { width: 100%; padding-top: 60%; text-align: center; overflow: hidden; position: relative; background: center center no-repeat; -webkit-background-size: cover; background-size: cover; } .gallery-cover-box .gallery-cover-img { display: inline-block; width: 100%; position: absolute; left: 50%; top: 50%; transform: translate(-50%, -50%); } .gallery-item .gallery-name { font-size: 14px; line-height: 24px; text-align: center; color: #666; margin: 0; } .waterfall { column-count: 3; column-gap: 1em; } .photo-wrapper { padding-top: 20px; } .photo-item { display: block; padding: 10px; padding-bottom: 0; margin-bottom: 14px; font-size: 0; -moz-page-break-inside: avoid; -webkit-column-break-inside: avoid; break-inside: avoid; background: white; -moz-box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.22); -webkit-box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.22); box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.22); } .photo-item img { width: 100%; } .photo-item .photo-name { font-size: 14px; line-height: 30px; text-align: center; margin-top: 10px; margin-bottom: 10px; border-top: 1px solid #dddddd; } /*Adapt mobile terminal layout*/ @media only screen and (max-width: 601px) { .waterfall { column-count: 2; column-gap: 1em; } }
Album list effect
Album display effect
Album password settings
To set the password for the album, you can refer to the blog post password. Add the following code in the / layout/gallery.ejs file under the Theme Directory:
<% if (theme.verifyPassword.enable) { %> <script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.crypto) %>"></script> <script> (function() { /*pwd Is the encrypted password of sha256 configured in the blog*/ let pwd = '<%- page.password %>'; if (pwd && pwd.length > 0) { if (pwd !== CryptoJS.SHA256(prompt('Please enter a password to access this album code')).toString(CryptoJS.enc.Hex)) { alert('Wrong password!'); location.href = '<%- url_for("/galleries") %>'; } } })(); </script> <% } %>
Then set the password in the / source/galleries/gallery3/index.md file in the gallery3 album directory where the password needs to be set. The password needs to be encrypted with SHA256
36. Modify navigation bar opacity
The transparent navigation bar often causes me reading difficulties. You can set not to use the transparent navigation bar and specify good-looking colors. Find the / source / CSS / matrix.css file in the theme directory and modify the following part:
header .nav-transparent { background-color: transparent !important; /*background-color: #000B3F;*/ /*Modify navigation bar opacity #16103f #4cbf30 #7371BC*/ background-image: none; box-shadow: none; }
37. Add shortcut navigation
\ source in the blog root directory\_ Create a new shortcut navigation in posts \ navigation \ index.md Directory:
hexo new page navigate
Modify the under the topic directory_ config.yml file, add shortcut navigation menu:
## Quick navigation menu: // ... other menus are omitted here Navigate: url: /navigate icon: fas fa-location-arrow
Modify the site / navigator / index.md file
--- title: Quick navigation date: 2021-08-29 16:25:05 layout: navigate ---
Create a new / layout / navigation.ejs file under the theme directory, and add the following contents:
<div class="navi-height bg-cover pd-header"> <div class="link-box container"> <div class="baidu baidu-2 large-screen"> <form name="f" action="https://www.baidu.com/" target="_blank"> <div id="Select-2"> <div class="Select-box-2" id="baidu"> <ul style="height:46px"> <li class="this_s">Baidu</li> <li class="bing_s">Bing</li> <li class="google_s">Google</li> <li class="baidu_s">Baidu</li> </ul> </div> <input name="wd" id="kw-2" maxlength="100" autocomplete="off" type="text"> </div> <div class="qingkong" id="qingkong" title="clear · empty" style="display:block">x</div> <input value="Search" id="su-2" type="submit"/> <ul class="keylist"></ul> </form> </div> <div class="row tags-posts"> <div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up"> <div class="card"> <div class="jj-list-tit">programming · study</div> <ul class="jj-list-con"> <li> <a href="https://www.oschina.net/" class="link-3" target="_blank">Open source China</a> </li> <li> <a href="https://htmldog.com/" class="link-3" target="_blank">HTML dog</a> </li> <li> <a href="https://www.icourse163.org/" class="link-3" target="_blank">Chinese University Course</a> </li> <li> <a href="https://www.imooc.com/" class="link-3" target="_blank">Muke network</a> </li> <li> <a href="http://www.wxapp-union.com/" class="link-3" target="_blank">Applet</a> </li> <li> <a href="https://www.runoob.com/" class="link-3" target="_blank">Rookie tutorial</a> </li> <li> <a href="https://blog.51cto.com/" class="link-3" target="_blank">51CTO</a> </li> <li> <a href="https://www.shiyanlou.com/library/" class="link-3" target="_blank">Laboratory Building</a> </li> <li> <a href="/posts/2d1a17c5.html" class="link-3" target="_blank">Personal collection page</a> </li> </ul> </div> </div> <div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up"> <div class="card"> <div class="jj-list-tit">community · Code</div> <ul class="jj-list-con"> <li> <a href="https://www.zhangxiaocai.com/contact/" class="link-3" target="_blank">Leaving a message. </a> </li> <li> <a href="https://github.com/" class="link-3" target="_blank">GitHub</a> </li> <li> <a href="https://coding.net/" class="link-3" target="_blank">Coding</a> </li> <li> <a href="https://juejin.im/" class="link-3" target="_blank">Nuggets</a> </li> <li> <a href="https://gitee.com/" class="link-3" target="_blank">Code cloud</a> </li> <li> <a href="https://www.csdn.net/" class="link-3" target="_blank">CSDN</a> </li> <li> <a href="https://www.jianshu.com/" class="link-3" target="_blank">Jian Shu</a> </li> <li> <a href="https://segmentfault.com/" class="link-3" target="_blank">Si no</a> </li> <li> <a href="https://cloud.tencent.com/developer/" class="link-3" target="_blank">cloud+community </a> </li> </ul> </div> </div> <div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up"> <div class="card"> <div class="jj-list-tit">practical · tool</div> <ul class="jj-list-con"> <li> <a href="https://mdnice.com/" class="link-3" target="_blank">Nice editor </a> </li> <li> <a href="https://translate.google.cn/" class="link-3" target="_blank">Google translation</a> </li> <li> <a href="https://www.uupoop.com/" class="link-3" target="_blank">on-line PS</a> </li> <li> <a href="https://www.processon.com/" class="link-3" target="_blank">Mind map</a> </li> <li> <a href="https://wallhaven.cc/" class="link-3" target="_blank">Ultra clear wallpaper</a> </li> <li> <a href="https://cli.im/" class="link-3" target="_blank">QR code</a> </li> <li> <a href="http://www.yinfans.me/" class="link-3" target="_blank">Yin Fansi</a> </li> <li> <a href="https://www.extfans.com" class="link-3" target="_blank">Google plugin</a> </li> <li> <a href="https://my.openwrite.cn/" class="link-3" target="_blank">OW distribute</a> </li> </ul> </div> </div> <div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up"> <div class="card"> <div class="jj-list-tit">entertainment · Movies</div> <ul class="jj-list-con"> <li> <a href="https://www.jd.com/" class="link-3" target="_blank">JD.COM</a> </li> <li> <a href="https://www.taobao.com/" class="link-3" target="_blank">TaoBao</a> </li> <li> <a href="https://www.tmall.com/" class="link-3" target="_blank">Tmall</a> </li> <li> <a href="https://v.qq.com/" class="link-3" target="_blank">Tencent video</a> </li> <li> <a href="http://www.iqiyi.com/" class="link-3" target="_blank">Iqiyi</a> </li> <li> <a href="https://www.bilibili.com/" class="link-3" target="_blank">Bilibili</a> </li> <li> <a href="https://music.163.com/" class="link-3" target="_blank">NetEase cloud music</a> </li> <li> <a href="https://y.qq.com/" class="link-3" target="_blank">QQ music</a> </li> <li> <a href="http://www.kugou.com/" class="link-3" target="_blank">Cool dog music</a> </li> </ul> </div> </div> <div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up"> <div class="card"> <div class="jj-list-tit">real-time info · trend</div> <ul class="jj-list-con"> <li> <a href="https://www.huxiu.com/" class="link-3" target="_blank">Tiger smell</a> </li> <li> <a href="https://insights.stackoverflow.com/" class="link-3" target="_blank">Technical investigation </a> </li> <li> <a href="http://www.asciiworld.com/" class="link-3" target="_blank">loaf on a job</a> </li> <li> <a href="https://sspai.com/" class="link-3" target="_blank">Minority </a> </li> <li> <a href="https://zh.wikihow.com/" class="link-3" target="_blank">WikeHom</a> </li> <li> <a href="https://www.awesomes.cn/rank?sort=hot" class="link-3" target="_blank"> Front end trend </a> </li> <li> <a href="https://github-trending.com/" class="link-3" target="_blank">GitHub trend</a> </li> <li> <a href="https://www.tiobe.com/" class="link-3" target="_blank">Programming trends</a> </li> <li> <a href="https://trends.google.com/" class="link-3" target="_blank">Google trend</a> </li> </ul> </div> </div> <div class="col s12 m6 l4 friend-div" data-aos="zoom-in-up"> <div class="card"> <div class="jj-list-tit">search · other</div> <ul class="jj-list-con"> <li> <a href="https://ac.scmor.com/" class="link-3" target="_blank">Google Image</a> </li> <li> <a href="http://www.pansoso.com/" class="link-3" target="_blank">Network disk search</a> </li> <li> <a href="http://music.ifkdy.com/" class="link-3" target="_blank">Music search</a> </li> <li> <a href="https://www.dytt8.net/" class="link-3" target="_blank">Movie paradise</a> </li> <li> <a href="https://carbon.now.sh/" class="link-3" target="_blank">Code picture</a> </li> <li> <a href="https://www.zhipin.com/" class="link-3" target="_blank">Boos</a> </li> <li> <a href="https://fontawesome.dashgame.com/" class="link-3" target="_blank">Icon Library</a> </li> <li> <a href="https://www.qvdv.com/tools/qvdv-guid.html" class="link-3" target="_blank"> Online tools </a> </li> <li> <a href="http://zhongguose.com/" class="link-3" target="_blank">Chinese color</a> </li> </ul> </div> </div> </div> <script> $(".Select-box ul").hover(function () { $(this).css("height", "auto") }, function () { $(this).css("height", "40px") }), $(".Select-box-2 ul").hover(function () { $(this).css("height", "auto") }, function () { $(this).css("height", "46px") }), $(".Select-box li").click(function () { var t = $(this).attr("class"), s = $(this).html(); "baidu_s" == t && (t = "https://www.baidu.com/s", _name = "wd"), "google_s" == t && (t = "https://www.google.com/search", _name = "q"), "bing_s" == t && (t = "https://www.bing.com/search", _name = "q"), $(".baidu form").attr("action", t), $(".this_s").html(s), $("#kw").attr("name", _name), $(".Select-box ul").css("height", "40px") }), $(".Select-box-2 li").click(function () { var t = $(this).attr("class"), s = $(this).html(); "baidu_s" == t && (t = "https://www.baidu.com/s", _name = "wd"), "google_s" == t && (t = "https://www.google.com/search", _name = "q"), "bing_s" == t && (t = "https://www.bing.com/search", _name = "q"), $(".baidu form").attr("action", t), $(".this_s").html(s), $("#kw-2").attr("name", _name), $(".Select-box-2 ul").css("height", "48px") }); $("#qingkong").click(function () { $("input[ name='wd' ] ").val(""); }); let timer; function backgroundImgRandom() { clearInterval(timer); timer = setInterval(function () { // [0-9) $("body").css("background-image", "url(<%- theme.background.url[parseInt(Math.random() * 9)] %>"); }, 5000); } window.onload = backgroundImgRandom; </script> </div> </div> <style> * { margin: 0; padding: 0; font-family: consolas, hl, "Microsoft YaHei " } dd, dl, dt, form, h1, h2, h3, h4, h5, h6, li, p, ul { margin: 0; padding: 0; font-size: 14px; font-weight: 400 } img { border-style: none } li { list-style: none; float: left } a { text-decoration: none } .card { background-color: rgba(25, 240, 229, 0); width: 96%; margin-left: 2% } .baidu { float: left; margin-left: 100px } .baidu form { position: relative } .Select-box ul { height: 40px; position: absolute; left: -1px; top: 0; z-index: 9999; background: #FFF; border: 1px solid #ccc; border-top: none; overflow: hidden } .Select-box li { width: 60px; line-height: 40px; font-size: 14px; color: #484848; border: 0; cursor: pointer } .Select-box li:hover { background: #3385ff; color: #FFF } .Select-box .this_s { color: #317ef3 } .Select-box .this_s:hover { background: #FFF; color: #317ef3 } .qingkong { position: absolute; right: 120px; top: 12px; width: 18px; height: 18px; background: rgba(0, 0, 0, .1); border-radius: 18px; line-height: 16px; color: #666; cursor: pointer; text-align: center; font-size: 14px; display: none; color: #881509; } .qingkong:hover { background: rgba(0, 0, 0, .2) } .qingkong:active { background: rgba(0, 0, 0, .3) } .baidu-2 { width: 100%; height: 110px; margin: 0 auto; background: 0 0; padding-top: 50px } .baidu-2 form { width: 520px; margin: 0 auto } .baidu-2 input { padding: 13px 8px; opacity: .9; font-size: 15px } #Select-2 { float: left } .Select-box-2 { text-align: center; float: left; position: relative } .Select-box-2 ul { height: 46px; position: absolute; left: 0; top: 1px; z-index: 9999; background: rgba(255, 255, 255, .9); border: 1px solid #ccc; border-top: none; overflow: hidden } .Select-box-2 li { width: 60px; line-height: 46px; font-size: 15px; color: #484848; border: 0; cursor: pointer } .Select-box-2 li:hover { background: #3385ff; color: #FFF } .Select-box-2 .this_s { color: #317ef3 } .Select-box-2 .this_s:hover { background: 0 0; color: #317ef3 } #kw-2 { width: 335px; outline: 0; border: 1px solid #ccc; background: rgba(255, 255, 255, .2); color: #000; padding-left: 70px; font-weight: 700 } #su-2 { width: 90px; background: #4e6ef2; border: none; border-top: #3385ff 1px solid; border-bottom: 1px solid #2d78f4; color: #FFF; cursor: pointer; outline: 0 } #su-2:hover { background: #00f; border-bottom: 1px solid #00f } #su-2:active { background: #00f; box-shadow: inset 1px 1px 3px #00f; -webkit-box-shadow: inset 1px 1px 3px #00f } #su-3 { width: 90px; background: #00f; border: none; border-top: #3385ff 1px solid; border-bottom: 1px solid #2d78f4; color: #FFF; cursor: pointer; outline: 0 } .jj-list-tit { font-size: 16px; line-height: 25px; color: #fff; width: 100%; padding-left: 38.5% } .jj-list-con { overflow: hidden; margin: 0 auto } .jj-list-con li { width: 31.333%; margin: 1% } .link-3 { display: block; background: rgba(0, 0, 0, .35); color: #FFF; font-size: 13px; text-align: center; line-height: 35px; padding: 4px 0; border-radius: 2px; transition: all .2s } .link-3:hover { background: rgba(0, 0, 0, .45); font-size: 15px; font-weight: 700 } @media only screen and (max-width: 584px) { .navi-height { height: 1300px } .link-box { margin-top: 5% } .large-screen { display: none } } @media only screen and (min-width: 584px) and (max-width: 993px) { .navi-height { height: 800px } .link-box { margin-top: 5% } .large-screen { display: none } } @media only screen and (min-width: 993px) { .navi-height { position: absolute; width: 100%; height: 100% } } .page-footer { display: none } </style>
/ layout under Theme Directory/_ Add Chinese name of shortcut navigation menu and mobile client file / layout to partial / navigation.ejs file/_ Add as partial / mobile-nav.ejs
<% var menuMap = new Map(); menuMap.set("Index", "home page"); menuMap.set("Tags", "label"); menuMap.set("Categories", "classification"); menuMap.set("Archives", "file"); menuMap.set("About", "about"); menuMap.set("Contact", "Message Board"); menuMap.set("Friends", "Links"); menuMap.set("Lists", "detailed list"); menuMap.set("Navigate", "Quick navigation"); var configRoot = config.root configRoot = (configRoot === null || configRoot === undefined || configRoot === '/') ? '' : configRoot; %>
38. Add music navigation page
Add \ source under the blog root directory\_ Posts \ music \ index.md file
hexo new page musics
Under the subject directory_ Add the [list - music] menu to the config.yml file:
menu: // Other menu items are omitted here Lists: ##detailed list url: / icon: fas fa-list children: - name: music url: /musics icon: fas fa-music
Modify the music file / music / index.md to specify the layout configuration:
--- title: musics date: 2021-08-25 19:55:53 layout: musics ---
Add the music layout file / layout / music.ejs under the theme directory, and add the following contents:
<style type="text/css"> /* don't remove. */ .about-cover { height: 75vh; } </style> <%- partial('_partial/bg-cover') %> <main class="content"> <% if (theme.mymusic.enable) { %> <%- partial('_widget/mymusic') %> <% } %> </main>
Add a custom music player file / layout in the theme directory/_ Widget / mymusic.ejs, add the following:
<link rel="stylesheet" href="<%- theme.jsDelivr.url %><%- url_for(theme.libs.css.aplayer) %>"> <style> .aplayer .aplayer-lrc p { <%if(theme.mymusic.hideLrc){%> display: none; <%}%> font-size: 12px; font-weight: 700; line-height: 16px !important; } .aplayer .aplayer-lrc p.aplayer-lrc-current { <%if(theme.mymusic.hideLrc){%> display: none; <%}%> font-size: 15px; color: <%- theme.mymusic.theme %>; } <%if(theme.mymusic.autoHide){%> .aplayer.aplayer-fixed.aplayer-narrow .aplayer-body { left: -66px !important; } .aplayer.aplayer-fixed.aplayer-narrow .aplayer-body:hover { left: 0px !important; } <%}%> </style> <div class="<% if(!theme.mymusic.fixed) { %>music-player<% } %>"> <% if (!theme.mymusic.fixed && theme.mymusic.title.enable) { %> <div class="title center-align"> <i class="fas fa-music"></i> <%- theme.mymusic.title.show %> </div> <% } %> <div class="row"> <meting-js class="col l8 offset-l2 m10 offset-m1 s12" server="<%- theme.mymusic.server %>" type="<%- theme.mymusic.type %>" id="<%- theme.mymusic.id %>" fixed='<%- theme.mymusic.fixed ? 'true' : 'false' %>' autoplay='<%- theme.mymusic.autoplay === true %>' theme='<%- theme.mymusic.theme %>' loop='<%- theme.mymusic.loop %>' order='<%- theme.mymusic.order %>' preload='<%- theme.mymusic.preload %>' volume='<%- Number(theme.mymusic.volume) %>' list-folded='<%- theme.mymusic.listFolded ? 'true' : 'false' %>' > </meting-js> </div> </div> <script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.aplayer) %>"></script> <script src="https://cdn.jsdelivr.net/npm/meting@2/dist/Meting.min.js"></script>
Configuration file under subject directory_ config.yml add custom music player configuration:
# Customize music components and display them on the page you need mymusic: enable: true title: #Non bottom suction mode is effective enable: true show: Relaxed moment autoHide: true # hide automaticaly server: netease #require music platform: netease, tencent, kugou, xiami, baidu type: playlist #require song, playlist, album, search, artist id: 4965675848 #require song id / playlist id (useful playlist 4965675848 2888085740)/ album id / search keyword fixed: false # Turn on the bottom suction mode. The true player will be on the side of the site, and click it to appear autoplay: false # Auto play theme: '#42b983' loop: 'all' # Audio cycle playback, optional values: 'all', 'one', 'none' order: 'random' # Audio cycle sequence, optional values: 'list', 'random' preload: 'auto' # Preload, optional values: 'none', 'metadata', 'auto' volume: 0.7 # Default volume. Please note that the player will remember the user's settings. After the user manually sets the volume, the default volume will become invalid listFolded: false # List default collapse hideLrc: false # Hide lyrics
39. Generate movie cards
Install the hexo tag mtime plug-in
npm install hexo-tag-mtime --save
Add a movie navigation page, and generate the / source/movies/index.md file in the blog root directory:
hexo new page movies
/Add the following content to the source/movies/index.md file:
--- title: Film recommendation date: 2021-08-25 19:56:04 type: movies --- ## Wonderful film recommendation ### Anger · serious case {% mtime 263501 %} ### Bye, boy {% mtime 259370 %} ### Wandering the earth {% mtime 218707 %} id Can be on time net https://www.mtime.com the corresponding movie website
40. Add night mode switching
Add the night mode switch button in the / layout/layout.ejs file under the Theme Directory:
<!-- Switch night/Day mode button --> <a onclick="switchNightMode()" id="sma"> <i class="fa fa-moon-o" id="nightMode" aria-hidden="true"></i> </a>
Add JS code to the / source / JS / material.js file in the topic Directory:
// Dark mode button settings if (localStorage.getItem('dark') === '1') { document.body.classList.add('dark'); } else if (new Date().getHours() >= 22 || new Date().getHours() < 7) { /*Regularly turn on dark mode < defau lt on from 22:00 p.m. to 6:00 a.m. >*/ // document.body.classList.add('dark'); // $("#nightMode").removeClass("fa-moon-o").addClass("fa-lightbulb"); } else if (matchMedia('(prefers-color-scheme: dark)').matches) { document.body.classList.add('dark'); } // Dark mode settings function switchNightMode() { var body = document.body; if (body.classList.contains('dark')) { document.body.classList.remove('dark'); localStorage.setItem('dark', '0'); $('#nightMode').removeClass("fa-lightbulb").addClass("fa-moon-o"); return; } else { document.body.classList.add('dark'); localStorage.setItem('dark', '1'); $('#nightMode').removeClass("fa-moon-o").addClass("fa-lightbulb"); return; } } /*Remind to turn on the night mode function*/ setTimeout( function () { if ((new Date().getHours() >= 19 || new Date().getHours() < 7) && !$('body').hasClass('DarkMode')) { let toastHTML = '<span style="color:#97b8b2;border-radius: 10px;>' + '<i class="fa fa-bell" aria-hidden="true"></i>It's better to read in dark mode at night.(゚▽゚)/</span>' M.toast({html: toastHTML}) } }, 2200);
Add the style of night mode switching in the / source / CSS / material.css file under the Theme Directory:
/******Night mode switching style start*******/ /* Dark mode button settings */ #sma { background: #000; width: 38px; height: 38px; display: block; position: fixed; border-radius: 50%; right: 15px; bottom: 170px; padding-top: 15px; margin-bottom: 0; z-index: 998; cursor: pointer; } #sma .fa-moon-o { position: absolute; right: 8px; bottom: 8px; font-size: 1.48rem !important; } #sma .fa-lightbulb { position: absolute; right: 13px; bottom: 8px; font-size: 1.5rem !important; } .fa-moon-o:before { content: "\f186"; } .fa-comments:before { content: "\f086"; } /* Dark mode settings */ /* The font color turns grayish white */ body.dark .fas, body.dark .title, body.dark .row .text, body.dark article .article-content .summary, body.dark .card .card-image .card-title, body.dark .fa-moon-o:before, body.dark .fa-lightbulb:before, body.dark article .article-tags .chip, body.dark .chip-container .tag-title, body.dark div.jqcloud a, body.dark .friends-container .tag-title, body.dark .frind-ship .title h1, body.dark .card .card-content p, body.dark .v[data-class=v] .vcount, body.dark .v[data-class=v] .vcount .vnum, body.dark pre code, body.dark h1, body.dark h2, body.dark h3, body.dark h4, body.dark h5, body.dark h6, body.dark li, body.dark p, body.dark header .side-nav .mobile-head .logo-name, body.dark header .side-nav .mobile-head .logo-desc, body.dark header .side-nav .menu-list a, body.dark .bg-cover .post-title, body.dark a { color: rgba(255, 255, 255, 0.6); } /* The background color turns gray */ body.dark .card, body.dark .block-with-text:after { background-color: #282c34; } /* The background color turns black */ body.dark, body.dark .v[data-class=v] .vcount, body.dark #rewardModal .modal-content, body.dark .modal, body.dark header .side-nav, body.dark header .side-nav .menu-list .m-nav-show { background-color: #12121c; /**Some pages cannot be switched to dark background because of my background image. You need to cancel the background image**/ background-image: url(#); } /* Change transparency */ body.dark .aplayer { background: #2f3742 !important; } body.dark img, body.dark strong { filter: brightness(.7); } /******Night mode switching style end*******/
41. Add valine comment function
Using the valine comment function, you can use leanCloud International Store comment data. The specific application ID and KEY tutorials are as follows:
Text tutorial: https://cndrew.cn/2020/04/10/hexo-shuoshuo/
Station B video: https://www.bilibili.com/video/BV16A411b7UF
Create / layout under the theme directory/_ Partial / valine.ejs file, add the following:
<style> .valine-card { margin: 1.5rem auto; } .valine-card .card-content { padding: 20px 20px 5px 20px; } #vcomments textarea { box-sizing: border-box; background: url("<%- url_for(theme.valine.background) %>") 100% 100% no-repeat; } #vcomments p { margin: 2px 2px 10px; font-size: 1.05rem; line-height: 1.78rem; } #vcomments blockquote p { text-indent: 0.2rem; } #vcomments a { padding: 0 2px; color: #4cbf30; font-weight: 500; text-decoration: none; } #vcomments img { max-width: 100%; height: auto; cursor: pointer; } #vcomments ol li { list-style-type: decimal; } #vcomments ol, ul { display: block; padding-left: 2em; word-spacing: 0.05rem; } #vcomments ul li, ol li { display: list-item; line-height: 1.8rem; font-size: 1rem; } #vcomments ul li { list-style-type: disc; } #vcomments ul ul li { list-style-type: circle; } #vcomments table, th, td { padding: 12px 13px; border: 1px solid #dfe2e5; } #vcomments table, th, td { border: 0; } table tr:nth-child(2n), thead { background-color: #fafafa; } #vcomments table th { background-color: #f2f2f2; min-width: 80px; } #vcomments table td { min-width: 80px; } #vcomments h1 { font-size: 1.85rem; font-weight: bold; line-height: 2.2rem; } #vcomments h2 { font-size: 1.65rem; font-weight: bold; line-height: 1.9rem; } #vcomments h3 { font-size: 1.45rem; font-weight: bold; line-height: 1.7rem; } #vcomments h4 { font-size: 1.25rem; font-weight: bold; line-height: 1.5rem; } #vcomments h5 { font-size: 1.1rem; font-weight: bold; line-height: 1.4rem; } #vcomments h6 { font-size: 1rem; line-height: 1.3rem; } #vcomments p { font-size: 1rem; line-height: 1.5rem; } #vcomments hr { margin: 12px 0; border: 0; border-top: 1px solid #ccc; } #vcomments blockquote { margin: 15px 0; border-left: 5px solid #42b983; padding: 1rem 0.8rem 0.3rem 0.8rem; color: #666; background-color: rgba(66, 185, 131, .1); } #vcomments pre { font-family: monospace, monospace; padding: 1.2em; margin: .5em 0; background: #272822; overflow: auto; border-radius: 0.3em; tab-size: 4; } #vcomments code { font-family: monospace, monospace; padding: 1px 3px; font-size: 0.92rem; color: #e96900; background-color: #f8f8f8; border-radius: 2px; } #vcomments pre code { font-family: monospace, monospace; padding: 0; color: #e8eaf6; background-color: #272822; } #vcomments pre[class*="language-"] { padding: 1.2em; margin: .5em 0; } #vcomments code[class*="language-"], pre[class*="language-"] { color: #e8eaf6; } #vcomments [type="checkbox"]:not(:checked), [type="checkbox"]:checked { position: inherit; margin-left: -1.3rem; margin-right: 0.4rem; margin-top: -1px; vertical-align: middle; left: unset; visibility: visible; } #vcomments b, strong { font-weight: bold; } #vcomments dfn { font-style: italic; } #vcomments small { font-size: 85%; } #vcomments cite { font-style: normal; } #vcomments mark { background-color: #fcf8e3; padding: .2em; } #vcomments table, th, td { padding: 12px 13px; border: 1px solid #dfe2e5; } table tr:nth-child(2n), thead { background-color: #fafafa; } #vcomments table th { background-color: #f2f2f2; min-width: 80px; } #vcomments table td { min-width: 80px; } #vcomments [type="checkbox"]:not(:checked), [type="checkbox"]:checked { position: inherit; margin-left: -1.3rem; margin-right: 0.4rem; margin-top: -1px; vertical-align: middle; left: unset; visibility: visible; } .v[data-class="v"] .vwrap .vheader .vinput { width: 32%; border-bottom: 1px dashed #dedede; } </style> <div class="card valine-card" data-aos="fade-up"> <div class="comment_headling" style="font-size: 20px; font-weight: 700; position: relative; padding-left: 20px; top: 15px; padding-bottom: 5px;"> <i class="fas fa-comments fa-fw" aria-hidden="true"></i> <span>comment</span> </div> <div id="vcomments" class="card-content" style="display: grid"> </div> </div> <script src="<%- theme.jsDelivr.url %><%- url_for('/libs/valine/av-min.js') %>"></script> <script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.valine) %>"></script> <script> let metaPlaceholder = <%- JSON.stringify(theme.valine.metaPlaceholder) %> ; //There's a new line here new Valine({ el: '#vcomments', appId: '<%- theme.valine.appId %>', appKey: '<%- theme.valine.appKey %>', notify: '<%- theme.valine.notify %>' === 'true', verify: '<%- theme.valine.verify %>' === 'true', visitor: '<%- theme.valine.visitor %>' === 'true', avatar: '<%- theme.valine.avatar %>', pageSize: '<%- theme.valine.pageSize %>', lang: '<%- theme.valine.lang %>', placeholder: '<%= theme.valine.placeholder %>', meta: <%- '["' + theme.valine.guest_info.join('", "') + '"]' %>, recordIP: '<%- theme.valine.recordIP %>' === 'true', enableQQ: '<%- theme.valine.avatar %>', requiredFields: <%- '["' + theme.valine.guest_info.join('", "') + '"]' %>, master: <%- '["' + theme.valine.master.join('", "') + '"]' %>, friends: <%- '["' + theme.valine.friends.join('", "') + '"]' %>, tagMeta: <%- '["' + theme.valine.tagMeta.join('", "') + '"]' %>, metaPlaceholder: metaPlaceholder, }); document.body.addEventListener('click', function (e) { if (e.target.classList.contains('vsubmit')) { const email = document.querySelector('input[type=email]'); const nick = document.querySelector('input[name=nick]'); const reg = /^[A-Za-z0-9_-\u4e00-\u9fa5]+@[a-zA-Z0-9_-]+(\.[a-zA-Z0-9_-]+)+$/; if (!email.value || !nick.value || !reg.test(email.value)) { const str = `<div class="valert txt-center"><div class="vtext">Please fill in the correct nickname and email!</div></div>`; const vmark = document.querySelector('.vmark'); vmark.innerHTML = str; vmark.style.display = 'block'; e.stopPropagation(); setTimeout(function () { vmark.style.display = 'none'; vmark.innerHTML = ''; }, 2500); } } }, true); </script>
Then, valine.ejs is referenced on the page that needs comment function. My has added comment function to the message board and blog posts, namely / layout/contact.ejs and / layout under the theme directory/_ Add the following to the partial / post-detail.ejs file:
<% if (theme.valine && theme.valine.enable) { %> <%- partial('_partial/valine') %> <% } %>
Then under the topic directory_ Enable valine comment function in config.yml file, and add some custom attributes used in valine.ejs
# The configuration of Valine comment module is not activated by default. If you want to use it, please activate the configuration item and set appId and appKey valine: enable: true appId: NiVMV7aR8ipVDy9EywBYtLwI-MdYXbMMI appKey: P7BGA1hMeBAFbHYz3dKY8doJ notify: false # Enable email reminder( https://valine.js.org/notify.html) verify: true # Enable anti spam verification visitor: true avatar: monsterid # Default avatar display mode, little monster # 'mm' # Gravatar style : mm/identicon/monsterid/wavatar/retro/hide pageSize: 10 placeholder: 'fill in QQ You can comment on No. 1. Come on~' # Comment Box placeholder /medias/comment_bg.png #Background map background: https://cdn.jsdelivr.net/gh/drew233/cdn/20200409110727.webp coolpushkey: # New attribute wang-qz comment_count: true enableQQ: true # Forced QQ recordIP: true requiredFields: # Required information, default [nickname, email, link] - nick - mail guest_info: # The input items displayed in the comment box are [nick, mail, link] by default. Some options can be set not to be filled in - nick - mail - link master: - e4de1f6da602d22e423d6dfb24611b6b # md5 encrypted blogger mailbox wang-qz@foxmail.com metaPlaceholder: # Background text of the input box nick: nickname/QQ number(Required) mail: mailbox(Required) link: website(https://) lang: zh-CN tagMeta: # The text to display in the comment label - Blogger - buddy - visitor friends: # md5 encrypted small partner mailbox - 410739a5ad278251b305dab085768c57 # 1848276756@qq.com - 410739a5ad278251b305dab085768c57 - 410739a5ad278251b305dab085768c57
In addition, valine.min.js needs to be upgraded. The version I use is valine-1.4:
<script src="https://cdn.jsdelivr.net/gh/small-rose/small-rose.github.io/libs/valine/Valine.min.js"></script>
Because the above ` ` valine.ejs file refers to / lib/js/valine/valine.min.js', you can also download it to replace the JS file
You can also clone the following version from gitHub, and then replace the local version with its valine.min.js
https://github.com/LuckyZmj/LuckyBlog/blob/master/themes/matery/source/libs/valine/Valine.min.js
valine.ejs file refers to valine.min.js, which needs to be in the topic directory_ Search the keyword libs in the config.yml file, find JS, reference the last line of the configuration in JS, and then add:
valine: /libs/valine/Valine.min.js
In this way, the / lib/js/valine/valine.min.js file can be found by the following reference method
<script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.valine) %>"></script>
42. Add arttalk function
reference material: Add ArtiTalk talk module to the theme of matrix in Hexo blog
See for specific use Arttalk official documentation , published data can be stored in LeanCloud , the above comment function has provided relevant reference materials
First, download Arttalk source code
git clone git@github.com:ArtitalkJS/Artitalk.git
Create a new / source / LIBS / arttalk folder under the theme directory and find the folder you just downloaded Arttalk source code , enter the dist directory, where there are two folders: css and js, and then perform the following operations:
Copy arttalk.min.css under / arttalk / dist / CSS / to the topic directory / source / LIBS / arttalk;
Copy arttalk.min.js under / artalk / dist / JS / to the topic directory / source / LIBS / arttalk;
To keep consistent with the introduction style of the theme, modify the theme configuration_ config.yml file, search for the keyword libs,
Find css and add on the last line:
artitalk: /libs/artitalk/artitalk.min.css
Find js and add in the last line:
artitalk: /libs/artitalk/artitalk.min.js
I'll list the final effects. I won't list all of them because there are many. Just know the addition of the last line. Pay attention to alignment, as follows:
libs: css: fontAwesome: /libs/awesome/css/all.css # V5.11.1 materialize: /libs/materialize/materialize.min.css # 1.0.0 artitalk: /libs/artitalk/artitalk.min.css # Add last line js: jquery: /libs/jquery/jquery.min.js materialize: /libs/materialize/materialize.min.js # 1.0.0 artitalk: /libs/artitalk/artitalk.min.js # Add last line
Find / layout under the theme directory/_ Partial / head.ejs, introduce css in the header:
<link rel="stylesheet" type="text/css" href="<%- theme.jsDelivr.url %><%- url_for(theme.libs.css.artitalk) %>">
Module preparation, create a new / layout / arttalk.ejs file in the theme directory, and add the following contents:
<style type="text/css"> /* don't remove. */ .about-cover { height: 75vh; } </style> <%- partial('_partial/bg-cover') %> <main class="content"> <% if (theme.artitalk && theme.artitalk.enable) { %> <%- partial('_widget/artitalk') %> <% } %> </main>
Create a new / layout under the theme directory/_ Widget / arttalk.ejs file, which has been introduced in the above code. The contents of the file are as follows:
<style type="text/css"> #artitalk_main .cbp_tmtimeline > li .cbp_tmlabel::after { right: 100%; border: solid transparent; z-index: -1; content: " "; height: 0; width: 0; position: absolute; pointer-events: none; border-right-color: #0bb7fbd6 ; border-width: 10px; top: 4px; } #pubShuo { margin-right: 5px; } #operare_artitalk .shuoshuo_input_log { outline-style: none; margin-top: 5px; border: 1px solid #ccc; border-radius: 6px; padding: 8px 16px; font-size: 12px; background-color: transparent; color: #0bb7fbd6; width: 70%; height: 28px; margin-left: 10px; } #artitalk_main { margin-top: 5px; margin-left: 5%; margin-right: 5%; } #lazy { margin-top: 40px; } </style> <script src="<%- theme.jsDelivr.url %><%- url_for(theme.libs.js.artitalk) %>"></script> <article id="articles11" class="container chip-container"> <div class="row "> <div class=" card"> <div class="card-content"> <div class="tag-title center-align"> <i class="fas fa-pen-alt"></i> say something </div> <div id="artitalk_main"></div> </div> </div> </div> </article> <script> new Artitalk({ appId: "<%= theme.artitalk.appId %>", appKey: "<%= theme.artitalk.appKey %>", <% if (theme.artitalk.serverURL) { %> serverURL: "<%= theme.artitalk.serverURL %>", <% } %> <% if (theme.artitalk.lang) { %> lang: "<%= theme.artitalk.lang %>", <% } %> <% if (theme.artitalk.pageSize) { %> pageSize: "<%= theme.artitalk.pageSize %>", <% } %> <% if (theme.artitalk.shuoPla) { %> shuoPla: "<%= theme.artitalk.shuoPla %>", <% } %> <% if (theme.artitalk.avatarPla) { %> avatarPla: "<%= theme.artitalk.avatarPla %>", <% } %> <% if (theme.artitalk.motion == 0) { %> motion: 0, <% } else { %> motion: 1, <% } %> <% if (theme.artitalk.bgImg) { %> bgImg: "<%= theme.artitalk.bgImg %>", <% } %> <% if (theme.artitalk.color1) { %> color1: "<%= theme.artitalk.color1 %>", <% } %> <% if (theme.artitalk.color2) { %> color2: "<%= theme.artitalk.color2 %>", <% } %> <% if (theme.artitalk.color3) { %> color3: "<%= theme.artitalk.color3 %>", <% } %> <% if (theme.artitalk.cssUrl) { %> cssUrl: "<%= theme.artitalk.cssUrl %>", <% } %> atEmoji: { baiyan: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/baiyan.png", bishi: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/bishi.png", bizui: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/bizui.png", chan: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/chan.png", daku: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/daku.png", dalao: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/dalao.png", dalian: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/dalian.png", dianzan: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/dianzan.png", doge: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/doge.png", facai: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/facai.png", fadai: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/fadai.png", fanu: "https://cdn.jsdelivr.net/gh/Artitalk/Artitalk-emoji/fanu.png", }, }) </script>
Under the topic directory_ Add the following configuration to the config.yml file:
# Tell me https://artitalk.js.org/ artitalk: enable: true appId: NiVMV7aR8ipVDy9EywBYtLwI-MdYXbMMI # App appId of leancloud appKey: P7BGA1hMeBAFbHYz3dKY8doJ # Application appKey of leancloud serverURL: # #The security domain name bound to leancloud does not need to be filled in if the international version is used lang: zh # Language setting: zh is Chinese, en is English and es is Spanish. The default is Chinese pageSize: 6 # Quantity per page shuoPla: 'Only the prince can release it' # Placeholder in the input box of the edit table motion: 1 #Switch for loading animation, 1 is on, 0 is off, and the default is on #Tell me about the url of the background picture of the input box bgImg: https://cdn.jsdelivr.net/gh/drew233/cdn/20200409110727.webp avatarPla: https://Cdn.jsdelivr.net/gh/small-rose/small-rose.github.io/media/avatar.jpg # placeholder for input box of custom avatar url color1: linear-gradient(45deg, rgb(109, 208, 242) 15%, rgb(245, 154, 190) 85%) #Talk about background color 1 & button color 1 color2: linear-gradient(45deg, rgb(109, 208, 242) 15%, rgb(245, 154, 190) 85%) #Talk about background color 2 & button color 2 color3: black #Talk about font color
For more configuration items, please refer to the official website: Configuration item description
Finally, you can create the page manually or execute the hexo command
hexo new page "artitalk"
An arttalk folder will be generated in the source directory of hexo. Modify the index.md in it:
--- title: artitalk date: 2021-09-03 20:31:58 type: artitalk layout: artitalk ---
Under the subject directory_ config.yml file configuration menu:
menu: ##about About: url: /crystal-blog icon: fas fa-rocket children: - name: About me url: /about icon: fas fa-user-circle - name: artitalk url: /artitalk icon: fas fa-pen-alt
Color reference: https://colordrop.io/
43. Switching of adding time list to archive timeline
The archived layout file is / layout/archive.ejs under the theme directory. The original logic is to display all blog posts in the form of cards. The specific code is as follows:
<!--Timeline area block--> <div id="cd-timeline" class="container" style="display: none;"> <% page.posts.each(function(post) { %> <div class="cd-timeline-block"> <%# year. %> <% if (date(post.date, 'YYYY') != year) { %> <% year = date(post.date, 'YYYY'); %> <div class="cd-timeline-img year" data-aos="zoom-in-up"> <a href="<%- url_for('/archives/' + year) %>"><%- year %></a> </div> <% } %> <%# month. %> <% if (date(post.date, 'YYYY-MM') != month) { %> <% month = date(post.date, 'YYYY-MM'); var m = date(post.date, 'MM') %> <div class="cd-timeline-img month" data-aos="zoom-in-up"> <a href="<%- url_for('/archives/' + year + '/' + m) %>"> <%- m %></a> </div> <% } %> <%# every day posts. %> <div class="cd-timeline-img day" data-aos="zoom-in-up"> <span><%- date(post.date, 'YYYY-MM-DD').substring(8, 10) %></span> </div> <article class="cd-timeline-content" data-aos="fade-up"> <div class="article col s12 m6"> <div class="card"> <a href="<%- url_for(post.path) %>"> <div class="card-image"> <% if (post.img) { %> <img src="<%- url_for(post.img) %>" class="responsive-img" alt="<%= post.title %>"> <% } else { %> <% featureimg = featureImages[Math.abs(hashCode(post.title) % featureImages.length)]; %> <img src="<%- theme.jsDelivr.url %> <%- url_for(featureimg) %>" class="responsive-img" alt="<%= post.title %>"> <% } %> <span class="card-title"><%= post.title %> </span> </div> </a> <div class="card-content article-content"> <div class="summary block-with-text"> <% if (post.summary && post.summary.length > 0) { %> <%- post.summary %> <% } else { %> <%- strip_html(post.content).substring(0, 120) %> <% } %> </div> <div class="publish-info"> <span class="publish-date"> <i class="far fa-clock fa-fw icon-date"></i> <%= date(post.date, config.date_format) %> </span> <span class="publish-author"> <% if (post.categories && post.categories.length > 0) { %> <i class="fas fa-bookmark fa-fw icon-category"></i> <% post.categories.forEach(category => { %> <a href="<%- url_for(category.path) %>" class="post-category"> <%- category.name %> </a> <% }); %> <% } else if (post.author && post.author.length > 0) { %> <i class="fas fa-user fa-fw"></i> <%- post.author %> <% } else { %> <i class="fas fa-user fa-fw"></i> <%- config.author %> <% } %> </span> </div> </div> <% if (post.tags && post.tags.length) { %> <div class="card-action article-tags"> <% post.tags.forEach(tag => { %> <a href="<%- url_for(tag.path) %>"> <span class="chip bg-color"> <%= tag.name %></span></a> <% }); %> </div> <% } %> </div> </div> </article> </div> <% }); %> </div>
I see that the archiving of some blogs on the Internet is displayed in the form of time list, so I can have a better viewing experience for all blog articles. So I copy the blog effect code of the online boss through the debug mode, and then modify it by myself to achieve the desired effect. The following operation
First add the switching buttons of the time list and timeline, and add the following code in the / layout/archive.ejs file under the topic Directory:
<div class="container"> <div class="card"> <div class="card-content"> <div class="tag-chips"> <span onclick="showTable()" id="sp-table" class="chip center-align waves-effect waves-light default" data-tagname="Time list" style="background: linear-gradient(to right, rgb(76, 191, 48) 0%, rgb(15, 157, 88) 100%); color: rgb(255, 255, 255);">Time list </span> <span onclick="showTime()" id="sp-timeline" class="chip center-align waves-effect waves-light default" data-tagname="time axis" style="background: rgb(249, 235, 234); color: rgba(0, 0, 0, 0.6);">time axis </span> </div> </div> </div> </div>
js code to achieve switching effect:
<script> function showTime() { $("#cd-timeline").show(); $("#cd-table").hide(); $("#sp-timeline").css('background', 'linear-gradient(to right, #4cbf30 0%, #0f9d58 100%)'); $("#sp-timeline").css('color', '#fff'); $("#sp-table").css('background', '#F9EBEA') $("#sp-table").css('color', 'rgba(0,0,0,0.6)'); } function showTable() { $("#cd-timeline").hide(); $("#cd-table").show(); $("#sp-table").css('background', 'linear-gradient(to right, #4cbf30 0%, #0f9d58 100%)'); $("#sp-table").css('color', '#fff'); $("#sp-timeline").css('background', '#F9EBEA') $("#sp-timeline").css('color', 'rgba(0,0,0,0.6)'); } </script>
Code to switch to time list:
<div id="cd-table" class="container archive-container"> <% page.posts.each(function(post) { %> <div class="card"> <div class="card-content"> <div class="archive"> <%# year. %> <% if (date(post.date, 'YYYY') != year) { %> <% year = date(post.date, 'YYYY'); %> <h4 class="archive-year" id="<%- year %>"><%- year %>year </h4> <% } %> <div class="articles"> <div class="article content>"> <div class="article-sort-post"> <div class="article-sort-item_title"> <a href="<%- url_for(post.path) %>"> <h5> <i class="fa fa-clock" style="font-size: 1rem;cursor: pointer;"> </i> <time class="is-text-small" datetime="2021-08-20T20:20:00.000Z" itemprop="datePublished" style="color: #ff542d;"> <%- date(post.date, 'YYYY-MM-DD') %> </time> </h5> </a> <h6 class="is-6"> <a href="<%- url_for(post.path) %>"></a> <a href="<%- url_for(post.path) %>"> <%= post.title %></a> </h6> </div> </div> </div> </div> </div> </div> </div> <% }); %> </div>
View my effects:
Front-matter
Detailed explanation of front matter options
All contents in the front matter option are not required. However, it is still recommended to fill in at least the values of title and date.
configuration option | Default value | describe |
---|---|---|
title | Markdown's file title | This option is highly recommended for article title |
date | Date and time when the file was created | It is strongly recommended to fill in this option for publishing time, and it is best to ensure global uniqueness |
author | Root_ author in config.yml | Article author |
img | A value in featureImages | Article feature map |
top | true | Recommended articles (whether the article is at the top or not). If the top value is true, it will be used as the recommended article on the home page |
cover | false | Indicates whether the article needs to be added to the rotation cover of the home page |
coverImg | nothing | Indicates the image path to be displayed on the rotation cover of the article on the home page. If not, the characteristic image of the article will be used by default |
password | nothing | The article reading password. If you want to set the reading verification password for the article, you can set the value of password, which must be the password encrypted with SHA256 to prevent others from seeing through. If the verifyPassword option is activated in config.yml of the topic |
toc | true | Whether TOC is enabled or not, you can turn off the function of TOC separately for an article. The premise is that the TOC option is activated in the config.yml of the topic |
mathjax | false | Whether mathematical formula support is enabled, whether mathjax is enabled in this article, and it needs to be in the_ The config.yml file also needs to be opened |
summary | nothing | Article summary, a customized article summary content. If this attribute has a value, the article card summary will display this text, otherwise the program will automatically intercept the content of the article |
tags | nothing | Article tags. An article can have multiple tags |
categories | nothing | Article classification. The classification of this topic represents a macro classification. Only one classification is recommended for an article |
keywords | Article title | Article keywords, required for SEO |
reprintPolicy | cc_by | The article reprint rule can be CC_ by, cc_ by_ nd, cc_ by_ sa, cc_ by_ nc, cc_ by_ nc_ nd, cc_ by_ nc_ One of SA, cc0, noreprint or pay |
be careful:
-
If the img attribute is not filled in, the article feature map will take the remainder according to the hashcode value of the article title, and then select the corresponding feature picture in the topic, so as to make all articles have their own features.
-
The value of date should ensure that each article is unique as far as possible, because Gitalk and Gitment identification IDS in this topic are uniquely identified by the value of date.
-
If you want to set the function of reading verification password for the article, you should not only set the value of password encrypted with SHA256 in front matter, but also in the_ Configuration is activated in config.yml. Some online SHA256 encrypted addresses are available: Open source China online tools,chahuo,Webmaster Tools .
-
You can specify reprint policy in the front mate of the article md file to configure reprint rules for a single article
Most complete example
--- title: be based on Hexo of hexo-theme-matery Theme build blog and optimize date: 2019-10-03 14:25:00 author: Alont img: /source/images/xxx.jpg top: true cover: true coverImg: /images/1.jpg password: 8d969eef6ecad3c29a3a629280e686cf0c3f5d5a86aff3ca12020c923adc6c92 toc: false mathjax: false summary: This is your customized article summary content. If this attribute has a value, the article card summary will display this text, otherwise the program will automatically intercept part of the article as a summary categories: tool tags: - blog - hexo ---