折腾 | Hugo-Luna主题装修指南

新年新气象,换了新主题 Hugo Luna !

最先看到这个主题,是才开始折腾博客的时候,是使用Hugo MEME主题,然后通过友链间接着访问,看到了小鱼的博客是使用Hugo MEME主题的,当时就很喜欢了。

本次博客装修也是基本参考了她的装修日记

Thanks for all the fish

先列出小鱼装修日记中已经提及的修改。

  • 主题安装
  • 全站字数统计
  • 把文末的 Licences 全部删掉
  • Waline 评论区配置
  • 表情包大小
  • 首页卡片
  • 卡片中的 Summary 变量改为 Description
  • 阅读时间改为字数统计
  • 文章题图
  • 页尾运行时间

以上的内容优先参考小鱼的内容。

修改文章链接显示

permalinks:
  # 文章链接的格式
  posts: /posts/:slug
  pages: /:slug.html # about.html

修改archetypes\default.md生成文章的默认模板为:

title: "{{ replace .Name "-" " " | title }}"
date: {{ .Date }}
slug: {{ substr (md5 (printf "%s%s" .Date (replace .TranslationBaseName "-" " " | title))) 4 8 }}
draft: false
featured_image: cover.jpg
tags:
  - 默认

新建文章时,可以随机生成slug,替代原来的XXX.html后缀。

原来使用hexo 的arrblink标签值可以直接替换为slug

标签链接修正

分别修改\layouts\partials\article\components\info.html\layouts\partials\article-list\info.html.

将其中的<a href="{{- "/tags/" | relLangURL -}}{{- . | urlize -}}" title="{{- . -}}"

替换为<a href="{{- "/tags/" | relLangURL -}}{{- . | urlize -}}{{- "/" -}}" title="{{- . -}}"

这样部署到OSS之类中,不会出现tags标签因为/123,的跳转错误。

Cover,Album

在Luna中想要给文章增加封面,以及使用shortcode的Gallery时,注意需要新建一个文件夹,将文档名改为index.md,然后直接引用本地图片,而不是使用图床图片。

否则无法渲染,会出现错误。

使用upx-upyun

在使用hexo时,有hexo-deployer-upyun-purge,配合一个.bat脚本完成博客部署到upyun。

搜索hugo没有直接发现相关的插件,最终找到了upx,配合一个.bat脚本实现博客部署到upyun中。

本着能白嫖就白嫖的思路,一定要使用!比单纯放在Github上要快很多,就是需要域名备案。

基本思路就是

  1. hugo命令生成public文件夹,注意每次需要新生成。
  2. upx命令upx.exe sync D:\Code\Hugo-Luna\public同步public文件夹。

upyun参考文档.

// upx 基本使用
// 登录方式一
upx-windows-amd64-v0.1.4.exe login
// 依次输入服务名、操作员、操作员的密码
ServiceName: <bucket>
Operator: <operator>
Password: <operator_password>
    
// 上传文件
upx-windows-amd64-v0.1.4.exe put <本地路径> <云存储路径>    

// 增量文件同步,加 -v,显示增量文件同步过程信息
upx-windows-amd64-v0.1.4.exe sync <本地目录> <云存储目录> -v
    
//.bat脚本上传
hugo
cd upx
upx.exe sync D:\Code\Hugo-Luna\public
pause    

豆瓣标签

API不能直接用了,找机会更新其他方法。

layouts\shortcodes里新建douban.html,并贴入以下code:

<!DOCTYPE HTML>
<html lang="en">
<head>
    <title></title>
    <style>
        .post-preview {
            max-width: 780px;
            height: 200px;
            margin: 1em auto;
            position: relative;
            display: flex;
            /*background: #eee;*/
            background: var(--entry);
            border-radius: 15px;
            box-shadow: 0 2px 4px rgba(0, 0, 0, .25), 0 0 2px rgba(0, 0, 0, .25);
        }

        .dark .post-preview {
            /*background: #383838;*/
            background: var(--entry);
            box-shadow: 0 2px 4px rgba(0, 0, 0, .5), 0 0 2px rgba(0, 0, 0, .5);
        }

        .post-preview--meta {
            width: 80%;
            padding: 23px;
            overflow: hidden;
        }

        .post-preview--middle {
            line-height: 28px;
        }

        .post-preview--title {
            font-size: 20px;
            margin: 0 !important;
        }

        .post-preview--title a {
            text-decoration: none;
        }

        .post-preview--date {
            font-size: 12px;
            color: #999;
        }

        .post-preview--excerpt {
            font-size: 14px;
            line-height: 1.825;
        }

        .post-preview--excerpt p {
            display: hidden;
            margin: 0;
			line-height: 1.825;
        }

        .post-preview--image {
            height: 200px !important;
            width: 25% !important;
            float: right;
            border-radius: 0 15px 15px 0;
			margin-top: 0rem !important;
        }

        .post-preview img {
            margin: unset;
            width: 20%;
            border-radius: 0 15px 15px 0;
        }

        @media (max-width: 550px) {
            .post-preview {
                width: 95%;
            }

            .post-preview--excerpt {
                display: none;
            }

            .post-preview--middle {
                line-height: 19px;
            }
        }

        .rating {
            display: block;
            line-height: 15px;
        }

        .rating-star {
            display: inline-block;
            width: 75px;
            height: 15px;
            background-repeat: no-repeat;
            background-image: url();
            overflow: hidden;
        }

        .allstar10 {
            background-position: 0px 0px;
        }

        .allstar9 {
            background-position: 0px -15px;
        }

        .allstar8 {
            background-position: 0px -30px;
        }

        .allstar7 {
            background-position: 0px -45px;
        }

        .allstar6 {
            background-position: 0px -60px;
        }

        .allstar5 {
            background-position: 0px -75px;
        }

        .allstar4 {
            background-position: 0px -90px;
        }

        .allstar3 {
            background-position: 0px -105px;
        }

        .allstar2 {
            background-position: 0px -120px;
        }

        .allstar1 {
            background-position: 0px -135px;
        }

        .allstar0 {
            background-position: 0px -150px;
        }


        .rating-average {
            color: #777;
            display: inline-block;
            font-size: 13px;
            margin-left: 10px;
        }
    </style>
    <script src="https://code.jquery.com/jquery-1.12.4.min.js"
            integrity="sha256-ZosEbRLbNQzLpnKIkEdrPv7lOy9C27hHQ+Xp8a4MxAQ=" crossorigin="anonymous"></script>
    <script type="text/javascript">
        $(document).ready(function () {
            $('.douban_item').each(function () {
                var _this = $(this);
                var strs = _this.attr('urlstring').toString();
                var db_reg = /^https\:\/\/(movie|book)\.douban\.com\/subject\/([0-9]+)\/?/;
                if (db_reg.test(strs)) {
                    var db_type = strs.replace(db_reg, "$1");
                    var db_id = strs.replace(db_reg, "$2").toString();
                    var db_api = "https://douban.edui.fun/";
                    if (db_type === 'movie') {
                        var ls_item = 'movie' + db_id;
                        var url = db_api + "movies/" + db_id;
                        if (localStorage.getItem(ls_item) == null || localStorage.getItem(ls_item) === 'undefined') {
                            $.ajax({
                                url: url, type: 'GET', dataType: "json", success: function (data) {
                                    localStorage.setItem(ls_item, JSON.stringify(data));
                                    movieShow(_this, ls_item, strs)
                                }
                            })
                        } else {
                            movieShow(_this, ls_item, strs)
                        }
                    } else if (db_type === 'book') {
                        var ls_item = 'book' + db_id;
                        var url = db_api + "v2/book/id/" + db_id;
                        if (localStorage.getItem(ls_item) == null || localStorage.getItem(ls_item) === 'undefined') {
                            $.ajax({
                                url: url, type: 'GET', dataType: 'json', success: function (data) {
                                    localStorage.setItem('book' + db_id, JSON.stringify(data));
                                    bookShow(_this, ls_item, strs)
                                }
                            })
                        } else {
                            bookShow(_this, ls_item, strs)
                        }
                    }
                }
            });
        });

        function movieShow(_this, ls_item, str) {
            var storage = localStorage.getItem(ls_item);
            var data = JSON.parse(storage);
            var db_star = Math.ceil(data.rating);
            $("<div class='post-preview'><div class='post-preview--meta'><div class='post-preview--middle'><div class='post-preview--title'><a target='_blank' style='box-shadow: none; font-weight: bolder;' href='" + str + "'>" + data.name + "</a></div><div class='rating'><div class='rating-star allstar" + db_star + "'></div><div class='rating-average'>" + data.rating + "</div></div><time class='post-preview--date'>导演:" + data.director + " / 类型:" + data.genre + " / " + data.year + "</time><section style='max-height:75px;overflow:hidden;' class='post-preview--excerpt'>" + data.intro.replace(/\s*/g, "") + "</section></div></div><img referrerpolicy='no-referrer' loading='lazy' class='post-preview--image' src=" + data.img + "></div>").replaceAll(_this)
        }

        function bookShow(_this, ls_item, str) {
            var storage = localStorage.getItem(ls_item);
            var data = JSON.parse(storage);
            var db_star = Math.ceil(data.rating.average);
            $("<div class='post-preview'><div class='post-preview--meta'><div class='post-preview--middle'><div class='post-preview--title'><a target='_blank' style='box-shadow: none; font-weight: bolder;' href='" + str + "'>" + data.title + "</a></div><div class='rating'><div class='rating-star allstar" + db_star + "'></div><div class='rating-average'>" + data.rating.average + "</div></div><time class='post-preview--date'>作者:" + data.author + " / 出版:"+ data.pubdate +" / "+ data.publisher +" </time><section style='max-height:75px;overflow:hidden;' class='post-preview--excerpt'>" + data.summary.replace(/\s*/g, "") + "</section></div></div><img referrerpolicy='no-referrer' loading='lazy' class='post-preview--image' src=" + data.images.medium + "></div>").replaceAll(_this)
        }
    </script>
</head>
<body>
<div class="douban_show">
    <div id="db{{ .Get "src" }}" urlstring="{{ .Get "src" }}" class="douban_item post-preview"></div>
</div>
</body>
</html>

CSS调整为适合Hugo-Luna的,可能适用到你自己主题需要再改下,可参考源文件