Compare commits

...

610 Commits
2.2.5 ... 5.5.4

285 changed files with 18628 additions and 9496 deletions

13
.github/FUNDING.yml vendored Normal file
View File

@@ -0,0 +1,13 @@
# These are supported funding model platforms
github: # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2]
patreon: # Replace with a single Patreon username
open_collective: # Replace with a single Open Collective username
ko_fi: # Replace with a single Ko-fi username
tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel
community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry
liberapay: # Replace with a single Liberapay username
issuehunt: # Replace with a single IssueHunt username
otechie: # Replace with a single Otechie username
lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry
custom: ['https://buy.stripe.com/3cs6rP6YA91sbbG5kk'] # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2']

View File

@@ -1,45 +0,0 @@
<!--
IMPORTANT: Please follow the template to create a new issue.
重要:請依照該模板來提交。
-->
## I want to create a new issue <!-- 我想要建立一個新的issue -->
<!-- Check all with "x" especially FAQ & Documentation!! (使用 "x" 選擇) -->
<!-- 請確認是否都已經翻閱過如下的資料, 尤其是安裝文檔!! -->
- [] Yes, I have read [Hexo Docs page](https://hexo.io/docs/), especially [Templates](https://hexo.io/docs/templates.html), [Variables](https://hexo.io/docs/variables.html), [Helpers](https://hexo.io/docs/helpers.html) and [Troubleshooting](https://hexo.io/docs/troubleshooting.html).
- [] Yes, I have read [Hexo-theme-butterfly Documentation](https://jerryc.me/posts/21cfbf15/).
- [] And yes, I already searched for current [issues](https://github.com/jerryc127/hexo-theme-butterfly/issues?utf8=%E2%9C%93&q=is%3Aissue) and this did not help me.
## Butterfly Information
<!-- Butterfly的版本 -->
<!-- 檢視主題的package.json -->
**Butterfly Version:**
<!-- Windows/macOS/Linux/Android/iOS -->
**Platform:**
<!-- Chrome/Safari/FireFox/.. -->
**Browser:**
## Expected behavior <!-- (預期行為) -->
## Actual behavior <!-- (實際行為) -->
<!-- Please give me the screenshots to locate the issue -->
<!-- 請儘量提供截圖來定位問題 -->
## Steps to reproduce the behavior <!-- (重現步驟) -->
## Feature Request
<!-- If you have any ideas of theme-butterfly, please write down here and we can have a discussion. -->
<!-- 如果你有任何關於Butterfly的功能方面的想法可以在這個部分裡寫下來我們一起討論 -->
---
<!--
Love hexo-theme-butterfly? Please consider starring the repo to support it!
喜歡 hexo-theme-butterfly嗎 考慮一下給它點個star來支援它吧
-->

83
.github/ISSUE_TEMPLATE/bug_report.yml vendored Normal file
View File

@@ -0,0 +1,83 @@
name: Bug report
description: Create a report to help us improve
title: '[Bug]: '
body:
- type: markdown
attributes:
value: |
重要:請依照該模板來提交
Important: Please follow the template to create a new issue
- type: input
id: butterfly-ver
attributes:
label: 使用的 Butterfly 版本? | What version of Butterfly are you using?
description: 檢視主題的 package.json | Check the theme's package.json
validations:
required: true
- type: dropdown
id: modify
attributes:
label: 是否修改過主題文件? | Has the theme files been modified?
options:
- 是 (Yes)
- 否 (No)
validations:
required: true
- type: dropdown
id: browser
attributes:
label: 使用的瀏覽器? | What browser are you using?
options:
- Chrome
- Edge
- Safari
- Opera
- Other
validations:
required: true
- type: dropdown
id: platform
attributes:
label: 使用的系統? | What operating system are you using?
options:
- Windows
- macOS
- Linux
- Android
- iOS
- Other
validations:
required: true
- type: textarea
id: dependencies
attributes:
label: 依賴插件 | Package dependencies information
description: 在 Hexo 根目錄下執行 `npm ls --depth 0` | Run `npm ls --depth 0` in Hexo root directory
render: Text
validations:
required: true
- type: textarea
id: description
attributes:
label: 問題描述 | Describe the bug
description: 請描述你的問題現象 | A clear and concise description of what the bug is.
placeholder: 請儘量提供截圖來定位問題 | If applicable, add screenshots to help explain your problem
value:
validations:
required: true
- type: input
id: website
attributes:
label: 出現問題的網站 | Website with the issue
description: 請提供可復現問題的網站地址 | Please provide a website URL where the problem can be reproduced.
placeholder: 請填寫具體的網址,不要填寫 localhost 網站 | Please provide a specific URL, do not use localhost.
validations:
required: true

18
.github/ISSUE_TEMPLATE/config.yml vendored Normal file
View File

@@ -0,0 +1,18 @@
blank_issues_enabled: false
contact_links:
- name: Questions about Butterfly
url: https://github.com/jerryc127/hexo-theme-butterfly/discussions
about: 一些使用問題請到 Discussion 詢問。 Please ask questions in Discussion.
- name: Butterfly Q&A
url: https://butterfly.js.org/posts/98d20436/
about: Butterfly Q&A
- name: Telegram
url: https://t.me/bu2fly
about: 'Official Telegram Group'
- name: QQ 群
url: https://jq.qq.com/?_wv=1027&k=KU9105XR
about: '群號 1070540070'

View File

@@ -0,0 +1,14 @@
name: Feature request
description: Suggest an idea for this project
title: '[Feature]: '
body:
- type: textarea
id: feature-request
attributes:
label: 想要的功能 | What feature do you want?
description: 請描述你需要的新功能 | A clear and concise description of what the feature is.
placeholder:
value:
validations:
require: true

21
.github/stale.yml vendored
View File

@@ -1,21 +0,0 @@
# Number of days of inactivity before an issue becomes stale
daysUntilStale: 30
# Number of days of inactivity before a stale issue is closed
daysUntilClose: 7
# Issues with these labels will never be considered stale
exemptLabels:
- pinned
- security
- bug
- enhancement
- Solved
- documentation
# Label to use when marking an issue as stale
staleLabel: wontfix
# Comment to post when marking an issue as stale. Set to `false` to disable
markComment: >
This issue has been automatically marked as stale because it has not had
recent activity. It will be closed if no further activity occurs. Thank you
for your contributions.
# Comment to post when closing a stale issue. Set to `false` to disable
closeComment: false

19
.github/workflows/publish.yml vendored Normal file
View File

@@ -0,0 +1,19 @@
name: npm publish
on:
release:
types: [created]
jobs:
build:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
# Setup .npmrc file to publish to npm
- uses: actions/setup-node@v1
with:
node-version: '12.x'
registry-url: 'https://registry.npmjs.org'
- run: npm install
- run: npm publish
env:
NODE_AUTH_TOKEN: ${{ secrets.NPM_TOKEN }}

19
.github/workflows/stale.yml vendored Normal file
View File

@@ -0,0 +1,19 @@
name: 'Close stale issues and PRs'
on:
schedule:
- cron: '30 1 * * *'
jobs:
stale:
runs-on: ubuntu-latest
steps:
- uses: actions/stale@v5
with:
days-before-issue-stale: 30
days-before-pr-stale: -1
days-before-close: 7
stale-issue-message: 'This issue has been automatically marked as stale because it has not had recent activity. It will be closed if no further activity occurs. Thank you for your contributions.'
close-pr-message: 'This issue has not seen any activity since it was marked stale. Closing.'
stale-issue-label: 'Stale'
exempt-issue-labels: 'pinned,bug,enhancement,documentation,Plan'
operations-per-run: 1000

2
.gitignore vendored Normal file
View File

@@ -0,0 +1,2 @@
.DS_Store
node_modules/

210
LICENSE
View File

@@ -1,8 +1,202 @@
MIT License
Copyright (c) <year> <copyright holders> Apache License
Version 2.0, January 2004
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: http://www.apache.org/licenses/
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 1. Definitions.
"License" shall mean the terms and conditions for use, reproduction,
and distribution as defined by Sections 1 through 9 of this document.
"Licensor" shall mean the copyright owner or entity authorized by
the copyright owner that is granting the License.
"Legal Entity" shall mean the union of the acting entity and all
other entities that control, are controlled by, or are under common
control with that entity. For the purposes of this definition,
"control" means (i) the power, direct or indirect, to cause the
direction or management of such entity, whether by contract or
otherwise, or (ii) ownership of fifty percent (50%) or more of the
outstanding shares, or (iii) beneficial ownership of such entity.
"You" (or "Your") shall mean an individual or Legal Entity
exercising permissions granted by this License.
"Source" form shall mean the preferred form for making modifications,
including but not limited to software source code, documentation
source, and configuration files.
"Object" form shall mean any form resulting from mechanical
transformation or translation of a Source form, including but
not limited to compiled object code, generated documentation,
and conversions to other media types.
"Work" shall mean the work of authorship, whether in Source or
Object form, made available under the License, as indicated by a
copyright notice that is included in or attached to the work
(an example is provided in the Appendix below).
"Derivative Works" shall mean any work, whether in Source or Object
form, that is based on (or derived from) the Work and for which the
editorial revisions, annotations, elaborations, or other modifications
represent, as a whole, an original work of authorship. For the purposes
of this License, Derivative Works shall not include works that remain
separable from, or merely link (or bind by name) to the interfaces of,
the Work and Derivative Works thereof.
"Contribution" shall mean any work of authorship, including
the original version of the Work and any modifications or additions
to that Work or Derivative Works thereof, that is intentionally
submitted to Licensor for inclusion in the Work by the copyright owner
or by an individual or Legal Entity authorized to submit on behalf of
the copyright owner. For the purposes of this definition, "submitted"
means any form of electronic, verbal, or written communication sent
to the Licensor or its representatives, including but not limited to
communication on electronic mailing lists, source code control systems,
and issue tracking systems that are managed by, or on behalf of, the
Licensor for the purpose of discussing and improving the Work, but
excluding communication that is conspicuously marked or otherwise
designated in writing by the copyright owner as "Not a Contribution."
"Contributor" shall mean Licensor and any individual or Legal Entity
on behalf of whom a Contribution has been received by Licensor and
subsequently incorporated within the Work.
2. Grant of Copyright License. Subject to the terms and conditions of
this License, each Contributor hereby grants to You a perpetual,
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
copyright license to reproduce, prepare Derivative Works of,
publicly display, publicly perform, sublicense, and distribute the
Work and such Derivative Works in Source or Object form.
3. Grant of Patent License. Subject to the terms and conditions of
this License, each Contributor hereby grants to You a perpetual,
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
(except as stated in this section) patent license to make, have made,
use, offer to sell, sell, import, and otherwise transfer the Work,
where such license applies only to those patent claims licensable
by such Contributor that are necessarily infringed by their
Contribution(s) alone or by combination of their Contribution(s)
with the Work to which such Contribution(s) was submitted. If You
institute patent litigation against any entity (including a
cross-claim or counterclaim in a lawsuit) alleging that the Work
or a Contribution incorporated within the Work constitutes direct
or contributory patent infringement, then any patent licenses
granted to You under this License for that Work shall terminate
as of the date such litigation is filed.
4. Redistribution. You may reproduce and distribute copies of the
Work or Derivative Works thereof in any medium, with or without
modifications, and in Source or Object form, provided that You
meet the following conditions:
(a) You must give any other recipients of the Work or
Derivative Works a copy of this License; and
(b) You must cause any modified files to carry prominent notices
stating that You changed the files; and
(c) You must retain, in the Source form of any Derivative Works
that You distribute, all copyright, patent, trademark, and
attribution notices from the Source form of the Work,
excluding those notices that do not pertain to any part of
the Derivative Works; and
(d) If the Work includes a "NOTICE" text file as part of its
distribution, then any Derivative Works that You distribute must
include a readable copy of the attribution notices contained
within such NOTICE file, excluding those notices that do not
pertain to any part of the Derivative Works, in at least one
of the following places: within a NOTICE text file distributed
as part of the Derivative Works; within the Source form or
documentation, if provided along with the Derivative Works; or,
within a display generated by the Derivative Works, if and
wherever such third-party notices normally appear. The contents
of the NOTICE file are for informational purposes only and
do not modify the License. You may add Your own attribution
notices within Derivative Works that You distribute, alongside
or as an addendum to the NOTICE text from the Work, provided
that such additional attribution notices cannot be construed
as modifying the License.
You may add Your own copyright statement to Your modifications and
may provide additional or different license terms and conditions
for use, reproduction, or distribution of Your modifications, or
for any such Derivative Works as a whole, provided Your use,
reproduction, and distribution of the Work otherwise complies with
the conditions stated in this License.
5. Submission of Contributions. Unless You explicitly state otherwise,
any Contribution intentionally submitted for inclusion in the Work
by You to the Licensor shall be under the terms and conditions of
this License, without any additional terms or conditions.
Notwithstanding the above, nothing herein shall supersede or modify
the terms of any separate license agreement you may have executed
with Licensor regarding such Contributions.
6. Trademarks. This License does not grant permission to use the trade
names, trademarks, service marks, or product names of the Licensor,
except as required for reasonable and customary use in describing the
origin of the Work and reproducing the content of the NOTICE file.
7. Disclaimer of Warranty. Unless required by applicable law or
agreed to in writing, Licensor provides the Work (and each
Contributor provides its Contributions) on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
implied, including, without limitation, any warranties or conditions
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
PARTICULAR PURPOSE. You are solely responsible for determining the
appropriateness of using or redistributing the Work and assume any
risks associated with Your exercise of permissions under this License.
8. Limitation of Liability. In no event and under no legal theory,
whether in tort (including negligence), contract, or otherwise,
unless required by applicable law (such as deliberate and grossly
negligent acts) or agreed to in writing, shall any Contributor be
liable to You for damages, including any direct, indirect, special,
incidental, or consequential damages of any character arising as a
result of this License or out of the use or inability to use the
Work (including but not limited to damages for loss of goodwill,
work stoppage, computer failure or malfunction, or any and all
other commercial damages or losses), even if such Contributor
has been advised of the possibility of such damages.
9. Accepting Warranty or Additional Liability. While redistributing
the Work or Derivative Works thereof, You may choose to offer,
and charge a fee for, acceptance of support, warranty, indemnity,
or other liability obligations and/or rights consistent with this
License. However, in accepting such obligations, You may act only
on Your own behalf and on Your sole responsibility, not on behalf
of any other Contributor, and only if You agree to indemnify,
defend, and hold each Contributor harmless for any liability
incurred by, or claims asserted against, such Contributor by reason
of your accepting any such warranty or additional liability.
END OF TERMS AND CONDITIONS
APPENDIX: How to apply the Apache License to your work.
To apply the Apache License to your work, attach the following
boilerplate notice, with the fields enclosed by brackets "[]"
replaced with your own identifying information. (Don't include
the brackets!) The text should be enclosed in the appropriate
comment syntax for the file format. We also recommend that a
file or class name and description of purpose be included on the
same "printed page" as the copyright notice for easier
identification within third-party archives.
Copyright [yyyy] [name of copyright owner]
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.

197
README.md
View File

@@ -1,48 +1,193 @@
<div align="right">
<a title="中文" href="/README_CN.md">中文</a>
</div>
<div align="center">
<img src="./source/img/butterfly-icon.png" width="150" height="150" alt="Butterfly Logo" />
# hexo-theme-butterfly # hexo-theme-butterfly
<a href="https://github.com/jerryc127/hexo-theme-butterfly/releases"><img alt="Version" src="https://img.shields.io/badge/release-2.2.5-blue"/></a> A modern, elegant and feature-rich theme for Hexo
<a href="https://jerryc.me"><img alt="Author" src="https://img.shields.io/badge/author-JerryC-blur"/></a>
<a href="https://hexo.io"><img alt="Hexo" src="https://img.shields.io/badge/hexo-4.0+-0e83c"/></a>
<a href="https://nodejs.org/"><img alt="node.js" src="https://img.shields.io/badge/node.js-8.0+-blur"/></a>
Demo: https://demo.jerryc.me/ ![master version](https://img.shields.io/github/package-json/v/jerryc127/hexo-theme-butterfly/master?color=%231ab1ad&label=master)
![dev version](https://img.shields.io/github/package-json/v/jerryc127/hexo-theme-butterfly/dev?label=dev)
![npm version](https://img.shields.io/npm/v/hexo-theme-butterfly?color=%09%23bf00ff)
![hexo version](https://img.shields.io/badge/hexo-5.3.0+-0e83cd)
![license](https://img.shields.io/github/license/jerryc127/hexo-theme-butterfly?color=FF5531)
![GitHub stars](https://img.shields.io/github/stars/jerryc127/hexo-theme-butterfly?style=social)
JerryC: https://jerryc.me/ 📢 **Demo**: [Butterfly Official](https://butterfly.js.org/) | [CrazyWong's Blog](https://blog.crazywong.com/)
Based on [hexo-theme-melody](https://github.com/Molunerfinn/hexo-theme-melody) theme. 📖 **Documentation**: [English Docs](https://butterfly.js.org/en/posts/butterfly-docs-en-get-started/) | [中文文档](https://butterfly.js.org/posts/21cfbf15/)
## Installation ![Butterfly Theme Preview](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/theme-butterfly-readme.png)
Stable branch: </div>
``` ---
git clone -b master https://github.com/jerryc127/hexo-theme-butterfly.git themes/Butterfly
## 🚀 Quick Start
### 💾 Installation
#### Method 1: Git Installation (Recommended)
> 💡 **Tip**: If GitHub access is slow in mainland China, you can use the [Gitee Mirror](https://gitee.com/immyw/hexo-theme-butterfly.git)
Execute in your Hexo blog root directory:
```bash
# Install stable version (recommended)
git clone -b master https://github.com/jerryc127/hexo-theme-butterfly.git themes/butterfly
``` ```
Dev branch: ```bash
# Install development version (early access to new features)
``` git clone -b dev https://github.com/jerryc127/hexo-theme-butterfly.git themes/butterfly
git clone -b dev https://github.com/jerryc127/hexo-theme-butterfly.git themes/Butterfly
``` ```
## Configuration #### Method 2: NPM Installation
Set theme in the hexo work folder's root config file `_config.yml`: > ⚠️ **Note**: NPM installation only supports Hexo 5.0.0 and above
> theme: Butterfly ```bash
npm install hexo-theme-butterfly
```
If you don't have pug & stylus renderer, try this: ### ⚙️ Theme Configuration
> npm install hexo-renderer-pug hexo-renderer-stylus 1. **Enable Theme**: Modify your Hexo configuration file `_config.yml`:
## Documentation ```yaml
theme: butterfly
```
Find in [hexo-theme-butterfly docs](https://docs.jerryc.me) and [JerryC](https://jerryc.me/posts/21cfbf15). it supports `zh-TW` 2. **Install Dependencies**: If you haven't installed pug and stylus renderers, please run:
## Screenshots ```bash
npm install hexo-renderer-pug hexo-renderer-stylus --save
```
![image](https://user-images.githubusercontent.com/16351105/58887365-1272f780-8718-11e9-9329-3292c6ba20d4.png) ## ✨ Theme Features
![](https://user-images.githubusercontent.com/16351105/58887457-3cc4b500-8718-11e9-9417-2bdea603c92e.png) ### 🎨 Design Style
- [x] **Card-based Design** - Modern card-style layout
- [x] **Rounded/Square Design** - Customizable border styles
- [x] **Responsive Design** - Perfect adaptation to all screen sizes
- [x] **Two-column Layout** - Optimized reading experience
- [x] **Dark Mode** - Eye-friendly night mode
![](https://user-images.githubusercontent.com/16351105/69338594-7d03f980-0c9e-11ea-8b64-7f165e6508e2.png) ### 📝 Content Features
- [x] **Multi-level Menu** - Support for secondary navigation menus
- [x] **Reading Mode** - Focused article reading experience
- [x] **TOC Navigation** - Desktop and mobile TOC support
- [x] **Word Count** - Display article word count and reading time
- [x] **Related Articles** - Smart recommendation of related content
- [x] **Outdated Reminder** - Automatic article update status alerts
- [x] **Traditional/Simplified Chinese** - Support for Traditional and Simplified Chinese switching
- [x] **Tag Plugins** - Rich tag plugin support
### 🔍 Search & Navigation
- [x] **Multiple Search Options** - Algolia Search / Local Search / Docsearch
- [x] **Built-in 404** - Beautiful 404 error page
- [x] **Pjax Support** - Smooth page transition experience
### 🎨 Code Display
- [x] **Syntax Highlighting** - Built-in multiple themes (darker/pale night/light/ocean)
- [x] **Code Features** - Language display/fold expand/copy button/auto-wrap
- [x] **Math Formulas** - Support for Mathjax and Katex
### 💬 Social Interaction
- [x] **Multiple Comment Systems** - Disqus/Gitalk/Valine/Waline/Twikoo/Giscus/Artalk etc.
- [x] **Dual Comments Support** - Enable two comment systems simultaneously
- [x] **Share Features** - Sharejs/Addtoany sharing components
- [x] **Live Chat** - Chatra/Tidio/Crisp instant messaging
### 📊 Analytics & Statistics
- [x] **Visit Statistics** - Busuanzi counter
- [x] **Site Analytics** - Google Analytics/Baidu Analytics/Cloudflare Analytics/Microsoft Clarity/Umami
- [x] **Webmaster Verification** - Major search engine verification
- [x] **Ad Support** - Google AdSense/custom ad slots
### 🎪 Visual Effects
- [x] **Typing Effects** - activate_power_mode animations
- [x] **Background Effects** - Static ribbons/dynamic ribbons/floating ribbons/Canvas Nest
- [x] **Mouse Effects** - Fireworks/hearts/text click effects
- [x] **Loading Animations** - Preloader and pace.js progress bars
- [x] **Image Effects** - Medium Zoom/Fancybox image lightbox
- [x] **Lazy Loading** - Image lazy loading optimization
### 🛠️ Advanced Features
- [x] **PWA Support** - Progressive Web App
- [x] **Copy Protection** - Disable text copying/copyright info append
- [x] **Theme Customization** - Custom site color schemes
- [x] **Chart Support** - Mermaid flowcharts/Chart.js data charts
- [x] **Music Notation** - ABCJS music notation support
- [x] **Music Player** - APlayer/Meting music playback
- [x] **Article Series** - Series article organization
- [x] **Instantpage** - Page preloading acceleration
- [x] **Snackbar** - Elegant notification messages
## 🤝 Contributors
Thanks to all the developers who have contributed to the Butterfly theme!
[![Contributors](https://contrib.rocks/image?repo=jerryc127/hexo-theme-butterfly)](https://github.com/jerryc127/hexo-theme-butterfly/graphs/contributors)
## 📸 Screenshots
<div align="center">
![Theme Demo](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-1.jpg)
![Theme Demo](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-2.jpg)
![Theme Demo](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-3.jpg)
![Theme Demo](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-4.jpg)
</div>
## ⭐ Star History
[![Star History Chart](https://api.star-history.com/svg?repos=jerryc127/hexo-theme-butterfly&type=Date)](https://star-history.com/#jerryc127/hexo-theme-butterfly&Date)
## 🤝 Building a Better Theme Together
We believe **the power of open source comes from everyone's participation**! Whether you're a developer, designer, or user, you can contribute to the development of the Butterfly theme.
### 💬 Get Help & Support
- 🐛 **Found a bug?** → [GitHub Issues](https://github.com/jerryc127/hexo-theme-butterfly/issues) - Let's solve it together!
- 💡 **Have ideas?** → [GitHub Discussions](https://github.com/jerryc127/hexo-theme-butterfly/discussions) - Share your creative ideas!
- 📚 **Learning to use?** → [Official Documentation](https://butterfly.js.org/) - Detailed usage guide
- 💬 **Real-time discussion?** → [Telegram Group](https://t.me/bu2fly) - Chat with community members
### 🎯 Contributing
Want to make Butterfly better? We welcome any form of contribution:
- **🔧 Code Contributions** - Fix bugs, add new features, optimize performance
- **📝 Documentation** - Improve docs, translate content, write tutorials
- **🎨 Design Suggestions** - UI/UX improvements, theme colors, icon design
- **🧪 Testing & Feedback** - Test new features, report issues, provide user experience
- **💰 Financial Support** - [Sponsor the Project](https://buy.stripe.com/3cs6rP6YA91sbbG5kk) - Support long-term development
## 📄 License
This project is licensed under the [Apache 2.0](LICENSE) License.
## 🙏 Acknowledgments
This theme is developed based on [hexo-theme-melody](https://github.com/Molunerfinn/hexo-theme-melody). Thanks to the original author for their excellent work that provided inspiration and foundation!
Thanks to all friends who have contributed to the development of the Butterfly theme. Your support has made this theme continuously improve and progress.
---
<div align="center">
**✨ If this theme helps you, please give us a ⭐ Star! ✨**
</div>

View File

@@ -1,46 +1,193 @@
# hexo-theme-butterfly <div align="right">
<a title="English" href="/README.md">English</a>
<a href="https://github.com/jerryc127/hexo-theme-butterfly/releases"><img alt="Version" src="https://img.shields.io/badge/release-2.2.5-blue"/></a> </div>
<a href="https://jerryc.me"><img alt="Author" src="https://img.shields.io/badge/author-JerryC-blur"/></a>
<a href="https://hexo.io"><img alt="Hexo" src="https://img.shields.io/badge/hexo-4.0+-0e83c"/></a> <div align="center">
<a href="https://nodejs.org/"><img alt="node.js" src="https://img.shields.io/badge/node.js-8.0+-blur"/></a>
<img src="./source/img/butterfly-icon.png" width="150" height="150" alt="Butterfly Logo" />
Demo: https://demo.jerryc.me/
# hexo-theme-butterfly
JerryC: https://jerryc.me/
一個適用於 Hexo 的現代化、美觀且功能豐富的主題
一款基於[hexo-theme-melody](https://github.com/Molunerfinn/hexo-theme-melody)修改的主題 ![master version](https://img.shields.io/github/package-json/v/jerryc127/hexo-theme-butterfly/master?color=%231ab1ad&label=master)
![dev version](https://img.shields.io/github/package-json/v/jerryc127/hexo-theme-butterfly/dev?label=dev)
## 安裝 ![npm version](https://img.shields.io/npm/v/hexo-theme-butterfly?color=%09%23bf00ff)
![hexo version](https://img.shields.io/badge/hexo-5.3.0+-0e83cd)
在你的博客根目錄裡 ![license](https://img.shields.io/github/license/jerryc127/hexo-theme-butterfly?color=FF5531)
![GitHub stars](https://img.shields.io/github/stars/jerryc127/hexo-theme-butterfly?style=social)
```
git clone -b master https://github.com/jerryc127/hexo-theme-butterfly.git themes/Butterfly 📢 **在線預覽**: [Butterfly 官方](https://butterfly.js.org/) | [CrazyWong 博客](https://blog.crazywong.com/)
```
📖 **完整文檔**: [中文文檔](https://butterfly.js.org/posts/21cfbf15/) | [English Docs](https://butterfly.js.org/en/posts/butterfly-docs-en-get-started/)
如果想要安裝比較新的dev分支可以
![Butterfly 主題預覽](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/theme-butterfly-readme.png)
```
git clone -b dev https://github.com/jerryc127/hexo-theme-butterfly.git themes/Butterfly </div>
```
---
## 應用主題
修改hexo配置文件`_config.yml`,把主題改為`Butterfly` ## 🚀 快速開始
``` ### 💾 安裝方式
theme: Butterfly
``` #### 方式一Git 安裝(推薦)
>如果你沒有pug以及stylus的渲染器請下載安裝 npm install hexo-renderer-pug hexo-renderer-stylus --save or yarn add hexo-renderer-pug hexo-renderer-stylus > 💡 **提示**: 如果您在中國大陸訪問 GitHub 速度較慢,可以使用 [Gitee 鏡像](https://gitee.com/immyw/hexo-theme-butterfly.git)
## 文檔 在您的 Hexo 博客根目錄下執行:
可查看[hexo-theme-butterfly docs](https://docs.jerryc.me) 和 [JerryC](https://jerryc.me/posts/21cfbf15) ```bash
# 安裝穩定版本(推薦)
## 截圖 git clone -b master https://github.com/jerryc127/hexo-theme-butterfly.git themes/butterfly
![image](https://user-images.githubusercontent.com/16351105/58887365-1272f780-8718-11e9-9329-3292c6ba20d4.png) ```
![image](https://user-images.githubusercontent.com/16351105/58887457-3cc4b500-8718-11e9-9417-2bdea603c92e.png)
```bash
![](https://user-images.githubusercontent.com/16351105/69338594-7d03f980-0c9e-11ea-8b64-7f165e6508e2.png) # 安裝開發版本(搶先體驗新功能)
git clone -b dev https://github.com/jerryc127/hexo-theme-butterfly.git themes/butterfly
```
#### 方式二NPM 安裝
> ⚠️ **注意**: NPM 安裝方式僅支援 Hexo 5.0.0 及以上版本
```bash
npm install hexo-theme-butterfly
```
### ⚙️ 主題配置
1. **啟用主題**: 修改您的 Hexo 配置檔案 `_config.yml`
```yaml
theme: butterfly
```
2. **安裝依賴**: 如果您尚未安裝 pug 和 stylus 渲染器,請執行:
```bash
npm install hexo-renderer-pug hexo-renderer-stylus --save
```
## ✨ 主題特色
### 🎨 設計風格
- [x] **卡片化設計** - 現代化的卡片式佈局
- [x] **圓角/直角設計** - 支援自訂邊框樣式
- [x] **響應式設計** - 完美適配各種螢幕尺寸
- [x] **雙欄佈局** - 優化的閱讀體驗
- [x] **深色模式** - 護眼的夜間模式
### 📝 內容功能
- [x] **多級選單** - 支援二級導航選單
- [x] **閱讀模式** - 專注的文章閱讀體驗
- [x] **目錄導航** - 電腦和手機雙端支援 TOC
- [x] **字數統計** - 顯示文章字數和閱讀時間
- [x] **相關文章** - 智能推薦相關內容
- [x] **過期提醒** - 自動提示文章更新狀態
- [x] **簡繁轉換** - 支援繁體中文和簡體中文切換
- [x] **標籤外掛** - 豐富的標籤外掛支持
### 🔍 搜尋與導航
- [x] **多種搜尋** - Algolia 搜尋 / 本地搜尋 / Docsearch
- [x] **內建 404** - 美觀的 404 錯誤頁面
- [x] **Pjax 支援** - 流暢的頁面切換體驗
### 🎨 程式碼展示
- [x] **語法高亮** - 內建多種主題darker/pale night/light/ocean
- [x] **程式碼功能** - 語言顯示/摺疊展開/複製按鈕/自動換行
- [x] **數學公式** - 支援 Mathjax 和 Katex
### 💬 社交互動
- [x] **多元評論系統** - Disqus/Gitalk/Valine/Waline/Twikoo/Giscus/Artalk 等
- [x] **雙評論支援** - 可同時啟用兩套評論系統
- [x] **分享功能** - Sharejs/Addtoany 分享套件
- [x] **線上客服** - Chatra/Tidio/Crisp 即時聊天
### 📊 數據分析
- [x] **訪問統計** - 不蒜子計數器
- [x] **網站分析** - Google Analytics/百度統計/Cloudflare Analytics/Microsoft Clarity/Umami
- [x] **站長驗證** - 各大搜尋引擎驗證
- [x] **廣告支援** - Google AdSense/自訂廣告位
### 🎪 視覺效果
- [x] **打字特效** - activate_power_mode 動畫
- [x] **背景特效** - 靜態彩帶/動態彩帶/飄帶效果/Canvas Nest
- [x] **滑鼠特效** - 煙花/愛心/文字點擊效果
- [x] **載入動畫** - Preloader 和 pace.js 進度條
- [x] **圖片效果** - Medium Zoom/Fancybox 圖片燈箱
- [x] **懶載入** - 圖片延遲載入優化
### 🛠️ 進階功能
- [x] **PWA 支援** - 漸進式網頁應用
- [x] **複製保護** - 可關閉文字複製/版權資訊追加
- [x] **主題定製** - 自訂網站配色方案
- [x] **圖表支援** - Mermaid 流程圖/Chart.js 數據圖表
- [x] **音樂符號** - ABCJS 音樂記譜法支援
- [x] **音樂播放器** - APlayer/Meting 音樂播放功能
- [x] **系列文章** - 系列文章組織功能
- [x] **Instantpage** - 頁面預載入加速
- [x] **Snackbar** - 優雅的提示訊息
## 🤝 貢獻者
感謝所有為 Butterfly 主題做出貢獻的開發者們!
[![Contributors](https://contrib.rocks/image?repo=jerryc127/hexo-theme-butterfly)](https://github.com/jerryc127/hexo-theme-butterfly/graphs/contributors)
## 📸 主題截圖
<div align="center">
![主題展示](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-1.jpg)
![主題展示](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-2.jpg)
![主題展示](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-3.jpg)
![主題展示](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-4.jpg)
</div>
## ⭐ Star 趨勢
[![Star History Chart](https://api.star-history.com/svg?repos=jerryc127/hexo-theme-butterfly&type=Date)](https://star-history.com/#jerryc127/hexo-theme-butterfly&Date)
## 🤝 一起構建更美好的主題
我們相信,**開源的力量來自於每一個人的參與**!無論您是開發者、設計師還是用戶,都可以為 Butterfly 主題的發展貢獻力量。
### 💬 獲取幫助與支援
- 🐛 **發現問題?** → [GitHub Issues](https://github.com/jerryc127/hexo-theme-butterfly/issues) - 讓我們一起解決!
- 💡 **有好想法?** → [GitHub Discussions](https://github.com/jerryc127/hexo-theme-butterfly/discussions) - 分享您的創意想法!
- 📚 **學習使用?** → [官方文檔](https://butterfly.js.org/) - 詳細的使用指南
- 💬 **即時討論?** → [Telegram 群組](https://t.me/bu2fly) - 與社群成員實時交流
### 🎯 參與貢獻
想要讓 Butterfly 變得更好嗎?我們歡迎您的任何形式的貢獻:
- **🔧 代碼貢獻** - 修復 Bug、添加新功能、優化性能
- **📝 文檔完善** - 改進文檔、翻譯內容、撰寫教程
- **🎨 設計建議** - UI/UX 改進、主題配色、圖示設計
- **🧪 測試反饋** - 測試新功能、回報問題、提供使用體驗
- **💰 資金支援** - [贊助項目](https://buy.stripe.com/3cs6rP6YA91sbbG5kk) - 支持長期發展
## 📄 授權條款
本專案採用 [Apache 2.0](LICENSE) 授權條款。
## 🙏 致敬與感謝
本主題基於 [hexo-theme-melody](https://github.com/Molunerfinn/hexo-theme-melody) 進行開發,感謝原作者的精彩創作為我們提供了靈感與基礎!
感謝所有為 Butterfly 主題發展做出貢獻的朋友們,是你們的支持讓這個主題能夠不斷完善與進步。
---
<div align="center">
**✨ 如果這個主題對您有幫助,請給我們一個 ⭐ Star✨**
</div>

File diff suppressed because it is too large Load Diff

View File

@@ -1,106 +1,123 @@
sidebar: footer:
catalog: Catalog framework: Framework
have_read: You've read theme: Theme
footer: copy:
driven: Driven success: Copy Successful
theme: Theme error: Copy Failed
noSupport: Browser Not Supported
copy:
success: Copy successfully page:
error: Copy error articles: All Articles
noSupport: The browser does not support tag: Tag
category: Category
page: archives: Archives
articles: Articles
tag: Tag card_post_count: comments
category: Category
archives: Archives no_title: Untitled
sticky: Sticky post:
no_title: No title created: Created
updated: Updated
post: wordcount: Word Count
created: Created min2read: Reading Time
updated: Updated min2read_unit: mins
wordcount: Word count page_pv: Post Views
min2read: "Reading time: %s min" comments: Comments
page_pv: Post View copyright:
comments: Comments author: Author
copyright: link: Link
author: Author copyright_notice: Copyright Notice
link: Link copyright_content: 'All articles on this blog are licensed under <a href="%s">%s</a> unless otherwise stated.'
copyright_notice: Copyright Notice recommend: Related Articles
copyright_content: 'All articles in this blog are licensed under <a href="%s">%s</a> unless stating additionally.' edit: Edit
recommend: Recommend back_to_home: Back to Home
search: Search search:
algolia_search: title: Search
input_placeholder: Search for Posts load_data: Loading Database
hits_empty: "We didn't find any results for the search: ${query}." input_placeholder: Search for Posts
hits_stats: "${hits} results found in ${time} ms" algolia_search:
hits_empty: 'No results found for: ${query}'
local_search: hits_stats: '${hits} results found in ${time} ms'
label: Local search local_search:
input_placeholder: Search for Posts hits_empty: 'No results found for: ${query}'
hits_empty: "We didn't find any results for the search: ${query}" hits_stats: '${hits} articles found'
powered_by: Powered by
pagination:
pagination: prev: Previous
prev: Previous Post next: Next
next: Next Post page_info: 'Page ${current} of ${total}'
comment: Comment comment: Comments
aside: aside:
articles: Articles articles: Articles
tags: Tags tags: Tags
categories: Categories categories: Categories
card_announcement: Announcement card_announcement: Announcement
card_categories: Categories card_categories: Categories
card_tags: Tags card_tags: Tags
card_archives: Archives card_archives: Archives
card_recent_post: Recent Post card_recent_post: Recent Posts
card_bookmark: Add to bookmark card_webinfo:
card_webinfo: headline: Website Info
headline: Info article_name: Article Count
article_name: Article runtime:
runtime_name: Run time name: Runtime
site_wordcount: Total Count unit: days
site_uv_name: UV last_push_date:
site_pv_name: PV name: Last Update
site_wordcount: Total Word Count
donate: Donate site_uv_name: Unique Visitors
share: Share site_pv_name: Page Views
bookmark: more_button: View More
title: Bookmark card_newest_comments:
headline: Latest Comments
rightside: loading_text: Loading...
readmode_title: Read Mode error: Unable to retrieve comments, please check the configuration
font_plus_title: Increase font size zero: No comments
font_minus_title: Decrease font size image: Image
translate_title: Traditional Chinese and Simplified Chinese Conversion link: Link
night_mode_title: Dark Mode code: Code
back_to_top: Back to top card_toc: Contents
toc: Table of Contents card_post_series: Post Series
scroll_to_comment: Scroll to comment
setting: Setting date_suffix:
just: Just now
runtime_unit: days min: minutes ago
hour: hours ago
copy_copyright: day: days ago
author: Author month: months ago
link: Link
source: Source donate: Sponsor
info: Copyright is owned by the author. For commercial reprints, please contact the author for authorization. For non-commercial reprints, please indicate the source. share: Share
Snackbar: rightside:
bookmark: readmode_title: Reading Mode
message_prev: Press translate_title: Toggle Between Traditional and Simplified Chinese
message_next: to bookmark this page night_mode_title: Toggle Between Light and Dark Mode
chs_to_cht: Traditional Chinese Activated Manually back_to_top: Back to Top
cht_to_chs: Simplified Chinese Activated Manually toc: Table of Contents
day_to_night: Dark Mode Activated Manually scroll_to_comment: Scroll to Comments
night_to_day: Light Mode Activated Manually setting: Settings
aside: Toggle Between Single-column and Double-column
error_title: Page not found chat: Chat
copy_copyright:
author: Author
link: Link
source: Source
info: Copyright belongs to the author. For commercial use, please contact the author for authorization. For non-commercial use, please indicate the source.
Snackbar:
chs_to_cht: You have switched to Traditional Chinese
cht_to_chs: You have switched to Simplified Chinese
day_to_night: You have switched to Dark Mode
night_to_day: You have switched to Light Mode
loading: Loading...
load_more: Load More
error404: Page Not Found

View File

@@ -1,106 +1,123 @@
sidebar: footer:
catalog: Catalog framework: Framework
have_read: You've read theme: Theme
footer: copy:
driven: Driven success: Copy Successful
theme: Theme error: Copy Failed
noSupport: Browser Not Supported
copy:
success: Copy successfully page:
error: Copy error articles: All Articles
noSupport: The browser does not support tag: Tag
category: Category
page: archives: Archives
articles: Articles
tag: Tag card_post_count: comments
category: Category
archives: Archives no_title: Untitled
sticky: Sticky post:
no_title: No title created: Created
updated: Updated
post: wordcount: Word Count
created: Created min2read: Reading Time
updated: Updated min2read_unit: mins
wordcount: Word count page_pv: Post Views
min2read: "Reading time: %s min" comments: Comments
page_pv: Post View copyright:
comments: Comments author: Author
copyright: link: Link
author: Author copyright_notice: Copyright Notice
link: Link copyright_content: 'All articles on this blog are licensed under <a href="%s">%s</a> unless otherwise stated.'
copyright_notice: Copyright Notice recommend: Related Articles
copyright_content: 'All articles in this blog are licensed under <a href="%s">%s</a> unless stating additionally.' edit: Edit
recommend: Recommend back_to_home: Back to Home
search: Search search:
algolia_search: title: Search
input_placeholder: Search for Posts load_data: Loading Database
hits_empty: "We didn't find any results for the search: ${query}." input_placeholder: Search for Posts
hits_stats: "${hits} results found in ${time} ms" algolia_search:
hits_empty: 'No results found for: ${query}'
local_search: hits_stats: '${hits} results found in ${time} ms'
label: Local search local_search:
input_placeholder: Search for Posts hits_empty: 'No results found for: ${query}'
hits_empty: "We didn't find any results for the search: ${query}" hits_stats: '${hits} articles found'
powered_by: Powered by
pagination:
pagination: prev: Previous
prev: Previous Post next: Next
next: Next Post page_info: 'Page ${current} of ${total}'
comment: Comment comment: Comments
aside: aside:
articles: Articles articles: Articles
tags: Tags tags: Tags
categories: Categories categories: Categories
card_announcement: Announcement card_announcement: Announcement
card_categories: Categories card_categories: Categories
card_tags: Tags card_tags: Tags
card_archives: Archives card_archives: Archives
card_recent_post: Recent Post card_recent_post: Recent Posts
card_bookmark: Add to bookmark card_webinfo:
card_webinfo: headline: Website Info
headline: Info article_name: Article Count
article_name: Article runtime:
runtime_name: Run time name: Runtime
site_wordcount: Total Count unit: days
site_uv_name: UV last_push_date:
site_pv_name: PV name: Last Update
site_wordcount: Total Word Count
donate: Donate site_uv_name: Unique Visitors
share: Share site_pv_name: Page Views
bookmark: more_button: View More
title: Bookmark card_newest_comments:
headline: Latest Comments
rightside: loading_text: Loading...
readmode_title: Read Mode error: Unable to retrieve comments, please check the configuration
font_plus_title: Increase font size zero: No comments
font_minus_title: Decrease font size image: Image
translate_title: Traditional Chinese and Simplified Chinese Conversion link: Link
night_mode_title: Dark Mode code: Code
back_to_top: Back to top card_toc: Contents
toc: Table of Contents card_post_series: Post Series
scroll_to_comment: Scroll to comment
setting: Setting date_suffix:
just: Just now
runtime_unit: days min: minutes ago
hour: hours ago
copy_copyright: day: days ago
author: Author month: months ago
link: Link
source: Source donate: Sponsor
info: Copyright is owned by the author. For commercial reprints, please contact the author for authorization. For non-commercial reprints, please indicate the source. share: Share
Snackbar: rightside:
bookmark: readmode_title: Reading Mode
message_prev: Press translate_title: Toggle Between Traditional and Simplified Chinese
message_next: to bookmark this page night_mode_title: Toggle Between Light and Dark Mode
chs_to_cht: Traditional Chinese Activated Manually back_to_top: Back to Top
cht_to_chs: Simplified Chinese Activated Manually toc: Table of Contents
day_to_night: Dark Mode Activated Manually scroll_to_comment: Scroll to Comments
night_to_day: Light Mode Activated Manually setting: Settings
aside: Toggle Between Single-column and Double-column
error_title: Page not found chat: Chat
copy_copyright:
author: Author
link: Link
source: Source
info: Copyright belongs to the author. For commercial use, please contact the author for authorization. For non-commercial use, please indicate the source.
Snackbar:
chs_to_cht: You have switched to Traditional Chinese
cht_to_chs: You have switched to Simplified Chinese
day_to_night: You have switched to Dark Mode
night_to_day: You have switched to Light Mode
loading: Loading...
load_more: Load More
error404: Page Not Found

123
languages/ja.yml Normal file
View File

@@ -0,0 +1,123 @@
footer:
framework: フレームワーク
theme: テーマ
copy:
success: コピー成功
error: コピー失敗
noSupport: ブラウザが対応していません
page:
articles: 記事一覧
tag: タグ
category: カテゴリ
archives: アーカイブ
card_post_count: コメント数
no_title: タイトルなし
post:
created: 作成日
updated: 更新日
wordcount: 総文字数
min2read: 読む時間
min2read_unit:
page_pv: 閲覧数
comments: コメント数
copyright:
author: 著者
link: リンク
copyright_notice: 著作権表示
copyright_content: 'このブログのすべての記事は、<a href="%s">%s</a> ライセンスの下で提供されており、特に明記されていない限り、すべての権利を留保します。転載時には出典を明記してください: <a href="%s">%s</a>。'
recommend: 関連記事
edit: 編集
back_to_home: ホームに戻る
search:
title: 検索
load_data: データベースを読み込んでいます
input_placeholder: 記事を検索
algolia_search:
hits_empty: '${query} の検索結果が見つかりませんでした。'
hits_stats: '${hits} 件の結果が ${time}ms で見つかりました'
local_search:
hits_empty: '${query} の検索結果が見つかりませんでした。'
hits_stats: '${hits} 件の記事が見つかりました'
pagination:
prev: 前へ
next: 次へ
page_info: '${current} ページ / 合計 ${total} ページ'
comment: コメント
aside:
articles: 記事
tags: タグ
categories: カテゴリ
card_announcement: お知らせ
card_categories: カテゴリ
card_tags: タグ
card_archives: アーカイブ
card_recent_post: 最近の記事
card_webinfo:
headline: サイト情報
article_name: 記事数
runtime:
name: 稼働時間
unit:
last_push_date:
name: 最終更新日
site_wordcount: 総文字数
site_uv_name: ユーザー数
site_pv_name: ページビュー数
more_button: もっと見る
card_newest_comments:
headline: 最新コメント
loading_text: ローディング中...
error: コメントを取得できませんでした。設定を確認してください。
zero: コメントがありません
image: 画像
link: リンク
code: コード
card_toc: 目次
card_post_series: シリーズ記事
date_suffix:
just: たった今
min: 分前
hour: 時間前
day: 日前
month: ヶ月前
donate: 寄付
share: 共有
rightside:
readmode_title: 読書モード
translate_title: 簡体字と繁体字の切り替え
night_mode_title: ライトモード/ダークモード切り替え
back_to_top: トップに戻る
toc: 目次
scroll_to_comment: コメントへ移動
setting: 設定
aside: シングルカラムとダブルカラムの切り替え
chat: チャット
copy_copyright:
author: 著者
link: リンク
source: ソース
info: 著作権は著者に帰属します。商業的利用の場合は著者に連絡して許可を得てください。非商業的利用の場合は出典を明記してください。
Snackbar:
chs_to_cht: 繁体字に切り替えました
cht_to_chs: 簡体字に切り替えました
day_to_night: ダークモードに切り替えました
night_to_day: ライトモードに切り替えました
loading: ローディング中...
load_more: もっと見る
error404: ページが見つかりません

123
languages/ko.yml Normal file
View File

@@ -0,0 +1,123 @@
footer:
framework: 프레임워크
theme: 테마
copy:
success: 복사 성공
error: 복사 실패
noSupport: 브라우저가 지원되지 않음
page:
articles: 모든 글
tag: 태그
category: 카테고리
archives: 아카이브
card_post_count: 댓글 수
no_title: 제목 없음
post:
created: 작성일
updated: 수정일
wordcount: 총 글자 수
min2read: 읽기 시간
min2read_unit:
page_pv: 조회수
comments: 댓글
copyright:
author: 작성자
link: 링크
copyright_notice: 저작권 고지
copyright_content: '이 블로그의 모든 글은 <a href="%s">%s</a> 라이선스를 따르며, 별도로 명시되지 않는 한 모든 권리를 보유합니다. 재배포 시 출처를 명시해 주세요: <a href="%s">%s</a>.'
recommend: 관련 글
edit: 편집
back_to_home: 홈으로 돌아가기
search:
title: 검색
load_data: 데이터베이스 로드 중
input_placeholder: 글 검색
algolia_search:
hits_empty: '${query}에 대한 결과를 찾을 수 없습니다.'
hits_stats: '${hits}개의 결과를 ${time}ms 만에 찾음'
local_search:
hits_empty: '${query}에 대한 결과를 찾을 수 없습니다.'
hits_stats: '${hits}개의 글을 찾음'
pagination:
prev: 이전
next: 다음
page_info: '${current} 페이지 / 총 ${total} 페이지'
comment: 댓글
aside:
articles:
tags: 태그
categories: 카테고리
card_announcement: 공지
card_categories: 카테고리
card_tags: 태그
card_archives: 아카이브
card_recent_post: 최근 글
card_webinfo:
headline: 사이트 정보
article_name: 글 수
runtime:
name: 운영 시간
unit:
last_push_date:
name: 마지막 업데이트
site_wordcount: 총 글자 수
site_uv_name: 방문자 수
site_pv_name: 총 조회수
more_button: 더 보기
card_newest_comments:
headline: 최신 댓글
loading_text: 로딩 중...
error: 댓글을 가져올 수 없습니다. 설정을 확인해 주세요.
zero: 댓글 없음
image: 이미지
link: 링크
code: 코드
card_toc: 목차
card_post_series: 시리즈 글
date_suffix:
just: 방금
min: 분 전
hour: 시간 전
day: 일 전
month: 달 전
donate: 후원
share: 공유
rightside:
readmode_title: 읽기 모드
translate_title: 번체와 간체 전환
night_mode_title: 라이트/다크 모드 전환
back_to_top: 맨 위로
toc: 목차
scroll_to_comment: 댓글로 이동
setting: 설정
aside: 단일/이중 열 전환
chat: 채팅
copy_copyright:
author: 작성자
link: 링크
source: 출처
info: 저작권은 작성자에게 있습니다. 상업적 사용을 위해서는 작성자의 허가를 받아야 하며, 비상업적 사용 시에는 출처를 명시해 주세요.
Snackbar:
chs_to_cht: 번체로 전환되었습니다.
cht_to_chs: 간체로 전환되었습니다.
day_to_night: 다크 모드로 전환되었습니다.
night_to_day: 라이트 모드로 전환되었습니다.
loading: 로딩 중...
load_more: 더 보기
error404: 페이지를 찾을 수 없습니다.

View File

@@ -1,108 +1,124 @@
sidebar: footer:
catalog: 目录 framework: 框架
have_read: 你已经读了 theme: 主题
footer: copy:
driven: 驱动 success: 复制成功
theme: 主题 error: 复制失败
noSupport: 浏览器不支持
copy:
success: 复制成功 page:
error: 复制错误 articles: 全部文章
noSupport: 浏览器不支持 tag: 标签
category: 分类
page: archives: 归档
articles: 文章总览
tag: 标签 card_post_count: 条评论
category: 分类
archives: 归档 no_title: 无标题
sticky: 置顶 post:
no_title: 无题 created: 发表于
updated: 更新于
post: wordcount: 总字数
created: 发表于 min2read: 阅读时长
updated: 更新于 min2read_unit: 分钟
wordcount: 字数总计 page_pv: 浏览量
min2read: "阅读时长: %s 分钟" comments: 评论数
page_pv: 阅读量 copyright:
comments: 评论数 author: 文章作者
copyright: link: 文章链接
author: 文章作者 copyright_notice: 版权声明
link: 文章链接 copyright_content: '本博客所有文章除特别声明外,均采用
copyright_notice: 版权声明 <a href="%s" target="_blank">%s</a> 许可协议。转载请注明来源 <a href="%s" target="_blank">%s</a>'
copyright_content: '本博客所有文章除特别声明外,均采用 recommend: 相关推荐
<a href="%s" target="_blank">%s</a> 许可协议。转载请注明来自 <a href="%s" target="_blank">%s</a>' edit: 编辑
recommend: 相关推荐 back_to_home: 返回首页
search: 搜索 search:
algolia_search: title: 搜索
input_placeholder: 搜索文章 load_data: 数据加载中
hits_empty: "找不到您查询的内容:${query}" input_placeholder: 搜索文章
hits_stats: "找到 ${hits} 条结果,用时 ${time} 毫秒" algolia_search:
hits_empty: '未找到符合您查询的内容:${query}'
local_search: hits_stats: '找到 ${hits} 条结果,耗时 ${time} 毫秒'
label: 本地搜索 local_search:
input_placeholder: 搜索文章 hits_empty: '未找到符合您查询的内容:${query}'
hits_empty: "找不到您查询的内容:${query}" hits_stats: '共找到 ${hits} 篇文章'
powered: "提供支持"
by: pagination:
prev: 上一篇
pagination: next: 下一篇
prev: 上一篇 page_info: '第 ${current} 页 / 共 ${total} 页'
next: 下一篇
comment: 评论
comment: 评论
aside:
aside: articles: 文章
articles: 文章 tags: 标签
tags: 标签 categories: 分类
categories: 分类 card_announcement: 公告
card_announcement: 公告 card_categories: 分类
card_categories: 分类 card_tags: 标签
card_tags: 标签 card_archives: 归档
card_archives: 归档 card_recent_post: 最新文章
card_recent_post: 最新文章 card_webinfo:
card_bookmark: 加入书签 headline: 网站信息
card_webinfo: article_name: 文章数目
headline: 网站资讯 runtime:
article_name: 文章数目 name: 运行时间
runtime_name: 已运行时间 unit:
site_wordcount: 本站总字数 last_push_date:
site_uv_name: 本站访客数 name: 最后更新时间
site_pv_name: 本站总访问量 site_wordcount: 本站总字数
site_uv_name: 本站访客数
donate: 打赏 site_pv_name: 本站总浏览量
share: 分享 more_button: 查看更多
bookmark: card_newest_comments:
title: 添加书签 headline: 最新评论
loading_text: 加载中...
rightside: error: 无法获取评论,请确认相关配置是否正确
readmode_title: 阅读模式 zero: 暂无评论
font_plus_title: 放大字体 image: 图片
font_minus_title: 缩小字体 link: 链接
translate_title: 简繁转换 code: 代码
night_mode_title: 夜间模式 card_toc: 目录
back_to_top: 回到顶部 card_post_series: 系列文章
toc: 目录
scroll_to_comment: 直达评论 date_suffix:
setting: 设置 just: 刚刚
min: 分钟前
runtime_unit: hour: 小时前
day: 天前
copy_copyright: month: 个月前
author: 作者
link: 链接 donate: 赞助
source: 来源 share: 分享
info: 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
rightside:
Snackbar: readmode_title: 阅读模式
bookmark: translate_title: 简繁转换
message_prev: night_mode_title: 日间和夜间模式切换
message_next: 键将本页加入书签 back_to_top: 回到顶部
chs_to_cht: 你已切换为繁体 toc: 目录
cht_to_chs: 你已切换为简体 scroll_to_comment: 前往评论
day_to_night: 你已切换为深色模式 setting: 设置
night_to_day: 你已切换为浅色模式 aside: 单栏和双栏切换
chat: 聊天
error_title: 页面没有找到
copy_copyright:
author: 作者
link: 链接
source: 来源
info: 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
Snackbar:
chs_to_cht: 已切换为繁体中文
cht_to_chs: 已切换为简体中文
day_to_night: 已切换为深色模式
night_to_day: 已切换为浅色模式
loading: 加载中...
load_more: 加载更多
error404: 页面未找到

123
languages/zh-HK.yml Normal file
View File

@@ -0,0 +1,123 @@
footer:
framework: 框架
theme: 主題
copy:
success: 複製成功
error: 複製失敗
noSupport: 瀏覽器不支援
page:
articles: 全部文章
tag: 標籤
category: 分類
archives: 歸檔
card_post_count: 條評論
no_title: 無標題
post:
created: 發表於
updated: 更新於
wordcount: 字數統計
min2read: 閱讀時間
min2read_unit: 分鐘
page_pv: 瀏覽量
comments: 評論數
copyright:
author: 文章作者
link: 文章連結
copyright_notice: 版權聲明
copyright_content: '除特別聲明外,本博客所有文章均採用<a href="%s">%s</a> 授權協議。轉載請註明出處:<a href="%s">%s</a>。'
recommend: 相關文章
edit: 編輯
back_to_home: 返回首頁
search:
title: 搜尋
load_data: 正在加載數據庫
input_placeholder: 搜尋文章
algolia_search:
hits_empty: '未找到相關內容:${query}'
hits_stats: '找到 ${hits} 條結果,耗時 ${time} 毫秒'
local_search:
hits_empty: '未找到相關內容:${query}'
hits_stats: '找到 ${hits} 篇文章'
pagination:
prev: 上一頁
next: 下一頁
page_info: '第 ${current} 頁 / 共 ${total} 頁'
comment: 評論
aside:
articles: 文章
tags: 標籤
categories: 分類
card_announcement: 公告
card_categories: 分類
card_tags: 標籤
card_archives: 歸檔
card_recent_post: 最新文章
card_webinfo:
headline: 網站資訊
article_name: 文章數目
runtime:
name: 運行時間
unit:
last_push_date:
name: 最後更新時間
site_wordcount: 總字數
site_uv_name: 訪客數
site_pv_name: 總瀏覽量
more_button: 查看更多
card_newest_comments:
headline: 最新評論
loading_text: 正在加載...
error: 無法取得評論,請確認配置是否正確
zero: 暫無評論
image: 圖片
link: 連結
code: 代碼
card_toc: 目錄
card_post_series: 系列文章
date_suffix:
just: 剛剛
min: 分鐘前
hour: 小時前
day: 天前
month: 個月前
donate: 贊助
share: 分享
rightside:
readmode_title: 閱讀模式
translate_title: 簡繁轉換
night_mode_title: 切換日夜模式
back_to_top: 回到頂部
toc: 目錄
scroll_to_comment: 前往評論
setting: 設定
aside: 單欄與雙欄切換
chat: 聊天
copy_copyright:
author: 作者
link: 連結
source: 來源
info: 版權屬於作者所有。商業用途請聯絡作者獲得授權,非商業用途請註明出處。
Snackbar:
chs_to_cht: 已切換為繁體中文
cht_to_chs: 已切換為簡體中文
day_to_night: 已切換為深色模式
night_to_day: 已切換為淺色模式
loading: 正在加載...
load_more: 加載更多
error404: 未找到頁面

View File

@@ -1,110 +1,123 @@
sidebar: footer:
catalog: 目錄 framework: 框架
have_read: 你已經讀了 theme: 主題
footer: copy:
driven: Power by success: 複製成功
theme: Theme error: 複製失敗
noSupport: 瀏覽器不支援
copy:
success: 複製成功 page:
error: 複製錯誤 articles: 所有文章
noSupport: 瀏覽器不支持 tag: 標籤
category: 分類
page: archives: 歸檔
articles: 文章總覽
tag: 標籤 card_post_count: 則評論
category: 分類
archives: 歸檔 no_title: 無標題
sticky: 置頂 post:
no_title: 無題 created: 發表於
updated: 更新於
post: wordcount: 總字數
created: 發表於 min2read: 閱讀時間
updated: 更新於 min2read_unit: 分鐘
wordcount: 字數總計 page_pv: 瀏覽量
min2read: "閱讀時長: %s 分鐘" comments: 評論數
page_pv: 閱讀量 copyright:
comments: 評論數 author: 文章作者
copyright: link: 文章連結
author: 文章作者 copyright_notice: 版權聲明
link: 文章鏈接 copyright_content: '本部落格所有文章除特別聲明外,均採用<a href="%s" target="_blank">%s</a> 授權協議。轉載請註明來源 <a href="%s" target="_blank">%s</a>'
copyright_notice: 版權聲明 recommend: 相關推薦
copyright_content: '本博客所有文章除特別聲明外,均採用 edit: 編輯
<a href="%s" target="_blank">%s</a> 許可協議。轉載請註明來自 <a href="%s" target="_blank">%s</a>' back_to_home: 返回首頁
recommend: 相關推薦
search:
search: title:
algolia_search: load_data: 資料載入中
input_placeholder: 文章 input_placeholder: 文章
hits_empty: "找不到您查詢的內容:${query}" algolia_search:
hits_stats: "找到 ${hits} 條結果,用時 ${time} 毫秒" hits_empty: '找不到符合您查詢的內容:${query}'
hits_stats: '找到 ${hits} 筆結果,耗時 ${time} 毫秒'
local_search: local_search:
label: 本地搜索 hits_empty: '找不到符合您查詢的內容:${query}'
input_placeholder: 搜索文章 hits_stats: '共找到 ${hits} 篇文章'
hits_empty: "找不到您查詢的內容:${query}"
powered: "提供支持" pagination:
by: prev: 上一篇
next: 下一篇
pagination: page_info: '第 ${current} 頁 / 共 ${total} 頁'
prev: 上一篇
next: 下一篇 comment: 評論
comment: 評論 aside:
articles: 文章
aside: tags: 標籤
articles: 文章 categories: 分類
tags: 標籤 card_announcement: 公告
categories: 分類 card_categories: 分類
card_announcement: 公告 card_tags: 標籤
card_categories: 分類 card_archives: 歸檔
card_tags: 標籤 card_recent_post: 最新文章
card_archives: 歸檔 card_webinfo:
card_recent_post: 最新文章 headline: 網站資訊
card_bookmark: 加入書籤 article_name: 文章數量
card_webinfo: runtime:
headline: 網站資訊 name: 運行時間
article_name: 文章數目 unit:
runtime_name: 已運行時間 last_push_date:
site_wordcount: 本站總字數 name: 最後更新時間
site_uv_name: 本站訪客 site_wordcount: 總字
site_pv_name: 本站總訪問量 site_uv_name: 訪客數
site_pv_name: 總瀏覽量
donate: 打賞 more_button: 檢視更多
share: 分享 card_newest_comments:
bookmark: headline: 最新評論
title: 添加書籤 loading_text: 載入中...
error: 無法獲取評論,請確認相關配置是否正確
rightside: zero: 尚無評論
readmode_title: 閱讀模式 image: 圖片
font_plus_title: 放大字體 link: 連結
font_minus_title: 縮小字體 code: 程式碼
translate_title: 簡繁轉換 card_toc: 目錄
night_mode_title: 夜間模式 card_post_series: 系列文章
back_to_top: 回到頂部
toc: 目錄 date_suffix:
scroll_to_comment: 直達評論 just: 剛剛
setting: 設置 min: 分鐘前
hour: 小時前
runtime_unit: day:
month: 個月前
copy_copyright:
author: 作者 donate: 贊助
link: 鏈接 share: 分享
source: 來源
info: 著作權歸作者所有。商業轉載請聯繫作者獲得授權,非商業轉載請註明出處。 rightside:
readmode_title: 閱讀模式
Snackbar: translate_title: 繁簡轉換
bookmark: night_mode_title: 日夜模式切換
message_prev: back_to_top: 回到頂端
message_next: 鍵將本頁加入書籤 toc: 目錄
chs_to_cht: 你已切換為繁體 scroll_to_comment: 前往評論
cht_to_chs: 你已切換為簡體 setting: 設定
day_to_night: 你已切換為深色模式 aside: 單欄和雙欄切換
night_to_day: 你已切換為淺色模式 chat: 聊天
error_title: 頁面沒有找到 copy_copyright:
author: 作者
link: 連結
source: 來源
info: 著作權歸作者所有。如需商業轉載,請聯絡作者獲得授權,非商業轉載請註明出處。
Snackbar:
chs_to_cht: 已切換為繁體中文
cht_to_chs: 已切換為簡體中文
day_to_night: 已切換為深色模式
night_to_day: 已切換為淺色模式
loading: 載入中...
load_more: 載入更多
error404: 找不到頁面

View File

@@ -1,23 +0,0 @@
- var top_img = theme.error_404.background || theme.default_top_img
- var bg_img = `background-image: url(${top_img})`
doctype html
html(lang=config.language data-theme=theme.display_mode)
head
include includes/head.pug
body
if theme.fireworks && theme.fireworks.enable
canvas.fireworks
include includes/mobile-sidebar/index.pug
nav#nav.error-no-found(style=bg_img)
include includes/header/header.pug
#error_info.is-center
h1#error_title= '404'
#error_subtitle= theme.error_404.subtitle
include includes/rightside.pug
include includes/search/index.pug
each item in theme.CDN_USE.js
script(src=url_for(item))
include includes/additional-js.pug

View File

@@ -1,8 +1,8 @@
extends includes/layout.pug extends includes/layout.pug
block content block content
include ./includes/mixins/article-sort.pug include ./includes/mixins/article-sort.pug
#archive #archive
.article-sort-title= _p('page.articles') + ' - ' + site.posts.length .article-sort-title= `${_p('page.articles')} - ${getArchiveLength()}`
+articleSort(page.posts) +articleSort(page.posts)
include includes/pagination.pug include includes/pagination.pug

View File

@@ -1,14 +1,12 @@
extends includes/layout.pug extends includes/layout.pug
block content block content
if theme.category_ui == 'index' if theme.category_ui == 'index'
include ./includes/mixins/UI.pug include ./includes/mixins/indexPostUI.pug
#recent-posts.recent-posts.category_ui +indexPostUI
+UI_NEW(page.posts) else
include includes/pagination.pug include ./includes/mixins/article-sort.pug
else #category
include ./includes/mixins/article-sort.pug .article-sort-title= _p('page.category') + ' - ' + page.category
#category +articleSort(page.posts)
.article-sort-title= _p('page.category') + ' - ' + page.category include includes/pagination.pug
+articleSort(page.posts)
include includes/pagination.pug

View File

@@ -1,20 +0,0 @@
#page
.flink#article-container
each i in site.data.link
h2= i.class_name
.post-cards
.md-links
each item in i.link_list
.md-links-item
a(href=item.link title=item.name target="_blank")
if theme.lazyload.enable
img.lazyload(data-src=item.avatar onerror=`this.onerror=null;this.src='` + url_for(theme.lodding_bg.flink) + `'` alt=item.name )
else
img(src=item.avatar onerror=`this.onerror=null;this.src='` + url_for(theme.lodding_bg.flink) + `'` alt=item.name )
.md-links-title= item.name
.md-links-des= item.descr
!= page.content
if page.comments !== false
include includes/comments/index.pug

View File

@@ -1,65 +1,61 @@
if theme.translate && theme.translate.enable div
script(src=url_for(theme.CDN.translate)) script(src=url_for(theme.asset.utils))
script(src=url_for(theme.asset.main))
if (theme.medium_zoom && theme.medium_zoom.enable)
script(src=url_for(theme.CDN.medium_zoom)) if theme.translate.enable
script(src=url_for(theme.asset.translate))
if (theme.fancybox && theme.fancybox.enable)
script(src=url_for(theme.CDN.fancybox)) if theme.lightbox
script(src=url_for(theme.asset[theme.lightbox]))
include ./math/index.pug
if theme.instantpage
if theme.fireworks && theme.fireworks.enable script(src=url_for(theme.asset.instantpage), type='module')
script(src=url_for(theme.CDN.anime))
script(src=url_for(theme.CDN.fireworks)) if theme.lazyload.enable && !theme.lazyload.native
script(src=url_for(theme.asset.lazyload))
if (theme.snackbar && theme.snackbar.enable)
script(src=url_for(theme.CDN.snackbar)) if theme.snackbar.enable
script(src=url_for(theme.asset.snackbar))
if (theme.canvas_ribbon && theme.canvas_ribbon.enable)
include ./third-party/canvas-ribbon.pug .js-pjax
if needLoadCountJs
if (theme.canvas_ribbon_piao && theme.canvas_ribbon_piao.enable) != partial("includes/third-party/card-post-count/index", {}, { cache: true })
include ./third-party/canvas-ribbon-piao.pug
if loadSubJs
if (theme.canvas_nest && theme.canvas_nest.enable) include ./third-party/subtitle.pug
include ./third-party/canvas-nest.pug
include ./third-party/math/index.pug
if theme.activate_power_mode.enable include ./third-party/abcjs/index.pug
script(src=url_for(theme.CDN.activate_power_mode))
script. if commentsJsLoad
POWERMODE.colorful = #{theme.activate_power_mode.colorful}; include ./third-party/comments/js.pug
POWERMODE.shake = #{theme.activate_power_mode.shake};
document.body.addEventListener('input', POWERMODE); != partial("includes/third-party/prismjs", {}, { cache: true })
if theme.busuanzi.site_uv || theme.busuanzi.site_pv || theme.busuanzi.page_pv if theme.aside.enable && theme.aside.card_newest_comments.enable
script(async src=url_for(theme.CDN.busuanzi)) if theme.pjax.enable || (globalPageType !== 'post' && page.aside !== false)
!= partial("includes/third-party/newest-comments/index", {}, { cache: true })
if theme.instantpage.enable
script(src=url_for(theme.CDN.instantpage) type="module") != fragment_cache('injectBottom', function(){return injectHtml(theme.inject.bottom)})
if theme.lazyload.enable != partial("includes/third-party/effect", {}, { cache: true })
script(src=url_for(theme.CDN.lazyload) async="") != partial("includes/third-party/chat/index", {}, { cache: true })
//- 鼠標特效 if theme.aplayerInject && theme.aplayerInject.enable
if theme.click_heart && theme.click_heart.enable if theme.pjax.enable || theme.aplayerInject.per_page || page.aplayer
script(src=url_for(theme.CDN.click_heart)) include ./third-party/aplayer.pug
if theme.ClickShowText && theme.ClickShowText.enable if theme.pjax.enable
script(src=url_for(theme.CDN.ClickShowText)) != partial("includes/third-party/pjax", {}, { cache: true })
if theme.pangu && theme.pangu.enable if theme.umami_analytics.enable
include ./third-party/pangu.pug != partial("includes/third-party/umami_analytics", {}, { cache: true })
//- search if theme.busuanzi.site_uv || theme.busuanzi.site_pv || theme.busuanzi.page_pv
if theme.algolia_search && theme.algolia_search.enable script(async data-pjax src=theme.asset.busuanzi ? url_for(theme.asset.busuanzi) : '//busuanzi.ibruce.info/busuanzi/2.3/busuanzi.pure.mini.js')
script(src=url_for(theme.CDN.algolia_js))
else if theme.local_search && theme.local_search.enable != partial('includes/third-party/search/index', {}, { cache: true })
script(src=url_for(theme.CDN.local_search))
if theme.google_tag_manager && theme.google_tag_manager.tag_id
//- mermaid noscript
if theme.mermaid.enable iframe(src=`${theme.google_tag_manager.domain ? theme.google_tag_manager.domain : 'https://www.googletagmanager.com'}/ns.html?id=${theme.google_tag_manager.tag_id}` height="0" width="0" style="display:none;visibility:hidden")
include ./math/mermaid.pug
if is_home()
include ./head/subtitle.pug

View File

@@ -1,24 +0,0 @@
#disqus_thread
script.
var disqus_config = function () {
this.page.url = '!{ page.permalink }';
this.page.identifier = '!{ page.path }';
this.page.title = '!{ page.title }';
};
(function() {
var d = document, s = d.createElement('script');
s.src = 'https://!{theme.disqus.shortname}.disqus.com/embed.js';
s.setAttribute('data-timestamp', +new Date());
(d.head || d.body).appendChild(s);
})();
if is_post() && theme.disqus.count
script.
function getDisqusCount() {
var d = document, s = d.createElement('script');
s.src = 'https://!{theme.disqus.shortname}.disqus.com/count.js';
s.id = 'dsq-count-scr';
(d.head || d.body).appendChild(s);
}
window.addEventListener('load', getDisqusCount, false);

View File

@@ -1,24 +0,0 @@
#disqus_thread
script(src=url_for(theme.CDN.disqusjs))
script.
var dsqjs = new DisqusJS({
shortname: '!{theme.disqusjs.shortname}',
siteName: "!{theme.disqusjs.siteName}",
identifier: '!{ page.path }',
url: '!{ page.permalink }',
title: '!{ page.title }',
api: '!{theme.disqusjs.api}',
apikey: '!{theme.disqusjs.apikey}',
admin: '!{theme.disqusjs.admin}',
adminLabel: '!{theme.disqusjs.adminLabel}'
});
if is_post() && theme.disqusjs.count
script.
function getDisqusCount() {
var d = document, s = d.createElement('script');
s.src = 'https://!{theme.disqusjs.shortname}.disqus.com/count.js';
s.id = 'dsq-count-scr';
(d.head || d.body).appendChild(s);
}
window.addEventListener('load', getDisqusCount, false);

View File

@@ -1,27 +0,0 @@
#gitalk-container
script.
var gitalk = new Gitalk({
clientID: '!{theme.gitalk.client_id}',
clientSecret: '!{theme.gitalk.client_secret}',
repo: '!{theme.gitalk.repo}',
owner: '!{theme.gitalk.owner}',
admin: ['!{theme.gitalk.admin}'],
id: md5(decodeURI(location.pathname)),
language: '!{theme.gitalk.language}',
perPage: !{theme.gitalk.perPage},
distractionFreeMode: !{theme.gitalk.distractionFreeMode},
pagerDirection: '!{theme.gitalk.pagerDirection}',
createIssueManually: !{theme.gitalk.createIssueManually},
updateCountCallback: commentCount
})
gitalk.render('gitalk-container')
function commentCount(n){
try {
document.getElementsByClassName('gitalk-comment-count')[0].innerHTML= n
} catch (e) {
return false
}
}

View File

@@ -1,26 +0,0 @@
- var d = theme.disqus && theme.disqus.enable
- var dj = theme.disqusjs && theme.disqusjs.enable
- var l = theme.laibili && theme.laibili.enable
- var gt = theme.gitalk && theme.gitalk.enable
- var v = theme.valine && theme.valine.enable
- var u = theme.utterances && theme.utterances.enable
- var isComment = d || dj || l || gt || v || u
if isComment
hr
#post-comment
.comment_headling
i.fa.fa-comments.fa-fw(aria-hidden="true")
span= ' ' + _p('comment')
if d
include ./disqus.pug
else if dj
include ./disqusjs.pug
else if l
include ./laibili.pug
else if gt
include ./gitalk.pug
else if v
include ./valine.pug
else if u
include ./utterances.pug

View File

@@ -1,10 +0,0 @@
#lv-container(data-id="city" data-uid=theme.laibili.uid)
script.
(function(d, s) {
var j, e = d.getElementsByTagName(s)[0];
if (typeof LivereTower === 'function') { return; }
j = d.createElement(s);
j.src = 'https://cdn-city.livere.com/js/embed.dist.js';
j.async = true;
e.parentNode.insertBefore(j, e);
})(document, 'script');

View File

@@ -1,21 +0,0 @@
script#utterances_comment(src=theme.CDN.utterances
repo=theme.utterances.repo
issue-term=theme.utterances.issue_term
theme= theme.display_mode === 'dark' ? theme.utterances.dark_theme : theme.utterances.light_theme
crossorigin="anonymous"
async)
if theme.darkmode.enable
script.
var themeNow = document.documentElement.getAttribute('data-theme') === 'dark' ? '#{theme.utterances.dark_theme}' : '#{theme.utterances.light_theme}'
document.getElementById('utterances_comment').setAttribute('theme',themeNow)
function utterancesTheme () {
var theme = document.documentElement.getAttribute('data-theme') === 'dark' ? '#{theme.utterances.dark_theme}' : '#{theme.utterances.light_theme}'
const message = {
type: 'set-theme',
theme: theme
};
const iframe = document.querySelector('.utterances-frame');
iframe.contentWindow.postMessage(message, 'https://utteranc.es');
}

View File

@@ -1,23 +0,0 @@
#vcomment.vcomment
script(src=url_for(theme.CDN.valine))
script.
var GUEST_INFO = ['nick','mail','link'];
var guest_info = '#{ theme.valine.guest_info }'.split(',').filter(function(item){
return GUEST_INFO.indexOf(item) > -1
});
guest_info = guest_info.length == 0 ? GUEST_INFO :guest_info;
window.valine = new Valine({
el:'#vcomment',
notify: #{theme.valine.notify},
verify: #{theme.valine.verify},
appId: '#{theme.valine.appId}',
appKey: '#{theme.valine.appKey}',
placeholder: '#{theme.valine.placeholder}',
avatar: '#{theme.valine.avatar}',
meta: guest_info,
pageSize: '#{theme.valine.pageSize}',
lang: '#{theme.valine.lang}',
recordIP: #{theme.valine.recordIP},
serverURLs: '#{theme.valine.serverURLs}'
});

View File

@@ -1,24 +1,39 @@
#footer-wrap - const { nav, owner, copyright, custom_text } = theme.footer
- var now = new Date()
- var nowYear = now.getFullYear() if nav
if theme.since && theme.since != nowYear .footer-flex
.copyright!= `&copy;${theme.since} - ${nowYear} By ${config.author}` for block in nav
else .footer-flex-items(style=`${ block.width ? 'flex-grow:' + block.width : '' }`)
.copyright!= `&copy;${nowYear} By ${config.author}` for blockItem in block.content
if theme.footer_copyright.enable .footer-flex-item
.framework-info .footer-flex-title= blockItem.title
span= _p('footer.driven') + ' ' .footer-flex-content
a(href='https://hexo.io') for subitem in blockItem.item
span Hexo if subitem.html
span.footer-separator | div!= subitem.html
span= _p('footer.theme') + ' ' else if subitem.url
a(href='https://github.com/jerryc127/hexo-theme-butterfly') a(href=url_for(subitem.url), target='_blank' title=subitem.title)= subitem.title
span Butterfly else if subitem.title
if theme.footer_custom_text div!= subitem.title
.footer_custom_text!=`${theme.footer_custom_text}` .footer-other
if theme.ICP.enable .footer-copyright
.icp if owner.enable
a(href=theme.ICP.url) - const currentYear = new Date().getFullYear()
if theme.ICP.icon - const sinceYear = owner.since
img.icp-icon(src=url_for(theme.ICP.icon)) span.copyright
span=theme.ICP.text if sinceYear && sinceYear != currentYear
!= `&copy;&nbsp;${sinceYear} - ${currentYear} By ${config.author}`
else
!= `&copy;&nbsp;${currentYear} By ${config.author}`
if copyright.enable
- const v = copyright.version ? getVersion() : false
span.framework-info
if owner.enable && nav
span.footer-separator |
span= _p('footer.framework') + ' '
a(href='https://hexo.io')= `Hexo${ v ? ' ' + v.hexo : '' }`
span.footer-separator |
span= _p('footer.theme') + ' '
a(href='https://github.com/jerryc127/hexo-theme-butterfly')= `Butterfly${ v ? ' ' + v.theme : '' }`
if theme.footer.custom_text
.footer_custom_text!= theme.footer.custom_text

View File

@@ -1,105 +1,77 @@
- var pageTitle - var pageTitle
- if (is_archive()) pageTitle = _p('page.archives') - globalPageType === 'archive' ? page.title = findArchivesTitle(page, theme.menu, date) : ''
- else if (is_tag()) pageTitle = _p('page.tag') + ': ' + page.tag case globalPageType
- else if (is_category()) pageTitle = _p('page.category') + ': ' + page.category when 'tag'
- else if (is_month()) pageTitle += ': ' + page.month + '/' + page.year - pageTitle = _p('page.tag') + ': ' + page.tag
- else if (is_year()) pageTitle += ': ' + page.year when 'category'
- else if (is_current('/404.html', [strict])) pageTitle = _p('error_title') - pageTitle = _p('page.category') + ': ' + page.category
- else pageTitle = page.title || config.title || '' when '404'
- pageTitle = _p('error404')
- var isSubtitle = config.subtitle ? ' - ' + config.subtitle : '' default
- var tabTitle = is_home() || !pageTitle ? config.title + isSubtitle : pageTitle + ' | ' + config.title - pageTitle = page.title || config.title || ''
- var pageDescription = page.description || page.title || config.description
- var pageKeywords - var isSubtitle = config.subtitle ? ' - ' + config.subtitle : ''
- if (page.tags && page.tags.data) pageKeywords = page.tags.data.map(function(tag) {return tag.name;}).join(',') - var tabTitle = globalPageType === 'home' || !pageTitle ? config.title + isSubtitle : pageTitle + ' | ' + config.title
- else pageKeywords = Array.isArray(config.keywords) ? (config.keywords).join(','): ([]).join(',') || config.keywords - var pageAuthor = config.email ? config.author + ',' + config.email : config.author
- var pageAuthor = config.email ? config.author + ',' + config.email : config.author - var pageCopyright = config.copyright || config.author
- var pageCopyright = config.copyright || config.author - var themeColorLight = theme.theme_color && theme.theme_color.enable && theme.theme_color.meta_theme_color_light || '#ffffff'
- var without_html = url.replace('index.html', '') - var themeColorDark = theme.theme_color && theme.theme_color.enable && theme.theme_color.meta_theme_color_dark || '#0d0d0d'
- var themeColor = theme.display_mode === 'dark' ? themeColorDark : themeColorLight
meta(charset='UTF-8')
meta(http-equiv="X-UA-Compatible" content="IE=edge") meta(charset='UTF-8')
meta(name="viewport" content="width=device-width,initial-scale=1") meta(http-equiv="X-UA-Compatible" content="IE=edge")
title= tabTitle meta(name="viewport" content="width=device-width, initial-scale=1.0,viewport-fit=cover")
meta(name="description" content=pageDescription) title= tabTitle
if pageKeywords meta(name="author" content=pageAuthor)
meta(name="keywords" content=pageKeywords) meta(name="copyright" content=pageCopyright)
meta(name="author" content=pageAuthor) meta(name ="format-detection" content="telephone=no")
meta(name="copyright" content=pageCopyright) meta(name="theme-color" content=themeColor)
meta(name ="format-detection" content="telephone=no")
!=favicon_tag(theme.favicon || config.favicon) //- Open_Graph
include ./head/Open_Graph.pug
if theme.disable_baidu_transformation
meta(http-equiv="Cache-Control" content="no-transform") //- Structured Data
meta(http-equiv="Cache-Control" content="no-siteapp") include ./head/structured_data.pug
if page.hide == true !=favicon_tag(theme.favicon || config.favicon)
meta(name="robots" content="noindex") link(rel="canonical" href=urlNoIndex(null,config.pretty_urls.trailing_index,config.pretty_urls.trailing_html))
if theme.douban //- 預解析
if theme.douban.meta && (is_current('/movies/', [strict]) || is_current('/books/', [strict]) || is_current('/games/', [strict])) !=partial('includes/head/preconnect', {}, {cache: true})
meta(name="referrer" content="no-referrer")
//- 網站驗證
//- 預解析 !=partial('includes/head/site_verification', {}, {cache: true})
!=partial('includes/head/dns_prefetch', {}, {cache:theme.fragment_cache})
//- PWA
//- 網站驗證 if (theme.pwa && theme.pwa.enable)
!=partial('includes/head/site_verification', {}, {cache:theme.fragment_cache}) !=partial('includes/head/pwa', {}, {cache: true})
//- Open_Graph //- main css
include ./head/Open_Graph.pug link(rel='stylesheet', href=url_for(theme.asset.main_css))
link(rel='stylesheet', href=url_for(theme.asset.fontawesome))
//- PWA
if (theme.pwa && theme.pwa.enable) if (theme.snackbar && theme.snackbar.enable)
!=partial('includes/head/pwa', {}, {cache:theme.fragment_cache}) link(rel='stylesheet', href=url_for(theme.asset.snackbar_css) media="print" onload="this.media='all'")
script(src=url_for(theme.CDN.js_cookies)) if theme.lightbox === 'fancybox'
link(rel='stylesheet' href=url_for(theme.asset.fancybox_css) media="print" onload="this.media='all'")
if theme.darkmode.enable
!=partial('includes/head/darkmode', {}, {cache:theme.fragment_cache}) !=fragment_cache('injectHeadJs', function(){return inject_head_js()})
each item in theme.CDN_USE.css //- google_adsense
link(rel='stylesheet', href=url_for(item)) !=partial('includes/head/google_adsense', {}, {cache: true})
if theme.fontawesome_v5 && theme.fontawesome_v5.enable //- analytics
link(rel='stylesheet', href=url_for(theme.CDN.fontawesome_v5)) !=partial('includes/head/analytics', {}, {cache: true})
else
link(rel='stylesheet', href=url_for(theme.CDN.fontawesome_v4)) //- font
if theme.blog_title_font && theme.blog_title_font.font_link
if (theme.fancybox && theme.fancybox.enable) link(rel='stylesheet' href=url_for(theme.blog_title_font.font_link) media="print" onload="this.media='all'")
link(rel='stylesheet', href=url_for(theme.CDN.fancybox_css))
//- global config
if (theme.snackbar && theme.snackbar.enable) !=partial('includes/head/config', {}, {cache: true})
link(rel='stylesheet', href=url_for(theme.CDN.snackbar_css))
include ./head/config_site.pug
if theme.canonical
link(rel="canonical" href=without_html) !=fragment_cache('injectHead', function(){return injectHtml(theme.inject.head)})
if is_post()
if(page.prev)
link(rel="prev" title=page.prev.title href=url_for(page.prev.permalink) )
if(page.next)
link(rel="next" title=page.next.title href=url_for(page.next.permalink) )
if theme.algolia_search.enable
link(rel="stylesheet" type="text/css" href=url_for(theme.CDN.algolia_search_css))
script(src=url_for(theme.CDN.algolia_search) defer)
//- comment css
include ./head/comment.pug
//- google_adsense
!=partial('includes/head/google_adsense', {}, {cache:theme.fragment_cache})
//- analytics
!=partial('includes/head/analytics', {}, {cache:theme.fragment_cache})
if theme.blog_title_font.font_link
link(rel='stylesheet', href=url_for(theme.blog_title_font.font_link))
//- global config
!=partial('includes/head/config', {}, {cache:theme.fragment_cache})
include ./head/config_site.pug
include ./head/noscript.pug

View File

@@ -1,17 +1,16 @@
- var contentType = is_post() ? 'article' : 'website' if theme.Open_Graph_meta.enable
-
//- twitter meta const coverVal = page.cover_type === 'img' ? page.cover : theme.avatar.img
if theme.twitter_meta let ogOption = Object.assign({
meta(name="twitter:card" content="summary") type: globalPageType === 'post' ? 'article' : 'website',
meta(name="twitter:title" content=pageTitle) image: coverVal ? full_url_for(coverVal) : '',
meta(name="twitter:description" content=pageDescription) fb_admins: theme.facebook_comments.user_id || '',
meta(name="twitter:image" content=full_url_for(page.cover || theme.avatar.img)) fb_app_id: theme.facebook_comments.app_id || '',
}, theme.Open_Graph_meta.option)
//- Open_Graph -
if theme.Open_Graph_meta != open_graph(ogOption)
meta(property="og:type" content=contentType) else
meta(property="og:title" content=pageTitle) - const description = page.description || page.content || page.title || config.description
meta(property="og:url" content=without_html) if description
meta(property="og:site_name" content=config.title) meta(name="description" content=truncate(description, 150))
meta(property="og:description" content=pageDescription)
meta(property="og:image" content=full_url_for(page.cover || theme.avatar.img))

View File

@@ -1,22 +1,45 @@
if theme.baidu_analytics if theme.baidu_analytics
script. script.
var _hmt = _hmt || []; var _hmt = _hmt || [];
(function() { (function() {
var hm = document.createElement("script"); var hm = document.createElement("script");
hm.src = "https://hm.baidu.com/hm.js?!{theme.baidu_analytics}"; hm.src = "https://hm.baidu.com/hm.js?!{theme.baidu_analytics}";
var s = document.getElementsByTagName("script")[0]; var s = document.getElementsByTagName("script")[0];
s.parentNode.insertBefore(hm, s); s.parentNode.insertBefore(hm, s);
})(); })();
btf.addGlobalFn('pjaxComplete', () => {
if theme.google_analytics _hmt.push(['_trackPageview',window.location.pathname])
script. }, 'baidu_analytics')
(function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){
(i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o), if theme.google_analytics
m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m) script(async src=`https://www.googletagmanager.com/gtag/js?id=${theme.google_analytics}`)
})(window,document,'script','https://www.google-analytics.com/analytics.js','ga'); script.
window.dataLayer = window.dataLayer || []
ga('create', '!{theme.google_analytics}', 'auto'); function gtag(){dataLayer.push(arguments)}
ga('send', 'pageview'); gtag('js', new Date())
gtag('config', '!{theme.google_analytics}')
if theme.tencent_analytics btf.addGlobalFn('pjaxComplete', () => {
script(src=`https://tajs.qq.com/stats?sId=${theme.tencent_analytics}` charset="UTF-8") gtag('config', '!{theme.google_analytics}', {'page_path': window.location.pathname})
}, 'google_analytics')
if theme.cloudflare_analytics
script(defer data-pjax src='https://static.cloudflareinsights.com/beacon.min.js' data-cf-beacon=`{"token": "${theme.cloudflare_analytics}"}`)
if theme.microsoft_clarity
script.
(function(c,l,a,r,i,t,y){
c[a]=c[a]||function(){(c[a].q=c[a].q||[]).push(arguments)};
t=l.createElement(r);t.async=1;t.src="https://www.clarity.ms/tag/"+i;
y=l.getElementsByTagName(r)[0];y.parentNode.insertBefore(t,y);
})(window, document, "clarity", "script", "!{theme.microsoft_clarity}");
if (theme.google_tag_manager && theme.google_tag_manager.tag_id)
script.
(function(w,d,s,l,i){w[l]=w[l]||[];w[l].push({'gtm.start':
new Date().getTime(),event:'gtm.js'});var f=d.getElementsByTagName(s)[0],
j=d.createElement(s),dl=l!='dataLayer'?'&l='+l:'';j.async=true;j.src=
"!{theme.google_tag_manager.domain ? theme.google_tag_manager.domain : 'https://www.googletagmanager.com'}/gtm.js?id="+i+dl;f.parentNode.insertBefore(j,f);
})(window,document,'script','dataLayer','!{theme.google_tag_manager.tag_id}');
btf.addGlobalFn('pjaxComplete', () => {
dataLayer.push({'event': 'pjaxComplete', 'page_title': document.title, 'page_location': location.href, 'page_path': window.location.pathname})
}, 'google_tag_manager')

View File

@@ -1,8 +0,0 @@
if (theme.disqusjs && theme.disqusjs.enable && page.comments !== false && !is_tag() && !is_category() && !is_archive() && !is_home())
link(rel="stylesheet" type="text/css" href=url_for(theme.CDN.disqusjs_css))
if (theme.gitalk && theme.gitalk.enable && page.comments !== false && !is_tag() && !is_category() && !is_archive() && !is_home())
link(rel="stylesheet" type="text/css" href=url_for(theme.CDN.gitalk_css))
script(src=url_for(theme.CDN.gitalk))
script(src=url_for(theme.CDN.blueimp_md5))

View File

@@ -1,108 +1,125 @@
- -
var algolia = 'undefined'; let algolia = 'undefined'
var env = process.env; if (theme.search.use === 'algolia_search') {
if (theme.algolia_search.enable) { const { ALGOLIA_APP_ID, ALGOLIA_API_KEY, ALGOLIA_INDEX_NAME } = process.env
algolia = JSON.stringify({ const { appId, applicationID, apiKey, indexName } = config.algolia
appId: env.ALGOLIA_APP_ID || config.algolia.appId || config.algolia.applicationID, algolia = JSON.stringify({
apiKey: env.ALGOLIA_API_KEY || config.algolia.apiKey, appId: ALGOLIA_APP_ID || appId || applicationID,
indexName: env.ALGOLIA_INDEX_NAME || config.algolia.indexName, apiKey: ALGOLIA_API_KEY || apiKey,
hits: theme.algolia_search.hits, indexName: ALGOLIA_INDEX_NAME || indexName,
// search languages hitsPerPage: theme.search.algolia_search.hitsPerPage,
languages: { // search languages
input_placeholder: _p("algolia_search.input_placeholder"), languages: {
hits_empty: _p("algolia_search.hits_empty"), hits_empty: _p("search.algolia_search.hits_empty"),
hits_stats: _p("algolia_search.hits_stats") hits_stats: _p("search.algolia_search.hits_stats"),
} }
}) })
} }
var localSearch = 'undefined'; let localSearch = 'undefined'
if (theme.local_search && theme.local_search.enable) { if (theme.search.use === 'local_search') {
localSearch = JSON.stringify({ const { CDN, preload, top_n_per_article, pagination, unescape } = theme.search.local_search
path: config.search.path, localSearch = JSON.stringify({
languages: { path: CDN || config.root + config.search.path,
// search languages preload,
hits_empty: _p("local_search.hits_empty") top_n_per_article,
} unescape,
}) pagination: {
} enable: pagination.enable,
var translate = 'undefined'; hitsPerPage: pagination.hitsPerPage
if (theme.translate && theme.translate.enable){ },
translate = JSON.stringify({ languages: {
defaultEncoding: theme.translate.defaultEncoding, // search languages
translateDelay: theme.translate.translateDelay, hits_empty: _p("search.local_search.hits_empty"),
cookieDomain: theme.translate.cookieDomain, hits_stats: _p("search.local_search.hits_stats"),
msgToTraditionalChinese: theme.translate.msgToTraditionalChinese, }
msgToSimplifiedChinese: theme.translate.msgToSimplifiedChinese })
}) }
}
let translate = 'undefined'
var copyright = 'undefined' ; if (theme.translate && theme.translate.enable){
if (theme.copy.enable && theme.copy.copyright){ translate = JSON.stringify({
copyright = JSON.stringify({ defaultEncoding: theme.translate.defaultEncoding,
languages: { translateDelay: theme.translate.translateDelay,
author: _p("copy_copyright.author") + ': ' + config.author, msgToTraditionalChinese: theme.translate.msgToTraditionalChinese,
link: _p("copy_copyright.link") + ': ', msgToSimplifiedChinese: theme.translate.msgToSimplifiedChinese
source: _p("copy_copyright.source") + ': ' + config.title, })
info: _p("copy_copyright.info") }
}
}) let copyright = 'undefined'
} if (theme.copy.enable && theme.copy.copyright.enable){
copyright = JSON.stringify({
var ClickShowText = 'undefined'; limitCount: theme.copy.copyright.limit_count,
if (theme.ClickShowText && theme.ClickShowText.enable) { languages: {
ClickShowText = JSON.stringify({ author: _p("copy_copyright.author") + ': ' + config.author,
text: theme.ClickShowText.text.join(","), link: _p("copy_copyright.link") + ': ',
fontSize: theme.ClickShowText.fontSize source: _p("copy_copyright.source") + ': ' + config.title,
}) info: _p("copy_copyright.info")
} }
})
}
var Snackbar = 'undefined';
if (theme.snackbar && theme.snackbar.enable) { let Snackbar = 'undefined'
Snackbar = JSON.stringify({ if (theme.snackbar && theme.snackbar.enable) {
bookmark: { Snackbar = JSON.stringify({
message_prev: _p("Snackbar.bookmark.message_prev"), chs_to_cht: _p("Snackbar.chs_to_cht"),
message_next: _p("Snackbar.bookmark.message_next") cht_to_chs: _p("Snackbar.cht_to_chs"),
}, day_to_night: _p("Snackbar.day_to_night"),
chs_to_cht: _p("Snackbar.chs_to_cht"), night_to_day: _p("Snackbar.night_to_day"),
cht_to_chs: _p("Snackbar.cht_to_chs"), bgLight: theme.snackbar.bg_light,
day_to_night: _p("Snackbar.day_to_night"), bgDark: theme.snackbar.bg_dark,
night_to_day: _p("Snackbar.night_to_day"), position: theme.snackbar.position,
bgLight: theme.snackbar.bg_light, })
bgDark: theme.snackbar.bg_dark, }
position: theme.snackbar.position,
}) let highlightProvider = config.syntax_highlighter || (config.highlight.enable ? 'highlight.js' : config.prismjs.enable ? 'prismjs' : null)
} const { copy, language, height_limit, fullpage, macStyle, shrink } = theme.code_blocks
let highlight = JSON.stringify({
script. plugin: highlightProvider,
var GLOBAL_CONFIG = { highlightCopy: copy,
root: '!{config.root}', highlightLang: language,
algolia: !{algolia}, highlightHeightLimit: height_limit,
localSearch: !{localSearch}, highlightFullpage: fullpage,
translate: !{translate}, highlightMacStyle: macStyle
copy: { })
success: '!{_p("copy.success")}',
error: '!{_p("copy.error")}', script.
noSupport: '!{_p("copy.noSupport")}' const GLOBAL_CONFIG = {
}, root: '!{config.root}',
bookmark: { algolia: !{algolia},
message_prev: '!{_p("Snackbar.bookmark.message_prev")}', localSearch: !{localSearch},
message_next: '!{_p("Snackbar.bookmark.message_next")}' translate: !{translate},
}, highlight: !{highlight},
runtime_unit: '!{_p("runtime_unit")}', copy: {
runtime: !{theme.runtimeshow.enable}, success: '!{_p("copy.success")}',
copyright: !{copyright}, error: '!{_p("copy.error")}',
ClickShowText: !{ClickShowText}, noSupport: '!{_p("copy.noSupport")}'
medium_zoom: !{theme.medium_zoom.enable}, },
fancybox: !{theme.fancybox.enable}, relativeDate: {
Snackbar: !{Snackbar}, homepage: !{theme.post_meta.page.date_format === 'relative'},
baiduPush: !{theme.baidu_push && theme.baidu_push.enable}, post: !{theme.post_meta.post.date_format === 'relative'}
highlightCopy: !{theme.highlight_copy}, },
highlightLang: !{theme.highlight_lang}, runtime: '!{theme.aside.card_webinfo.runtime_date ? _p("aside.card_webinfo.runtime.unit") : ""}',
highlightShrink: '!{theme.highlight_shrink}', dateSuffix: {
isFontAwesomeV5: !{theme.fontawesome_v5 && theme.fontawesome_v5.enable}, just: '!{_p("date_suffix.just")}',
isPhotoFigcaption: !{theme.photofigcaption} min: '!{_p("date_suffix.min")}',
hour: '!{_p("date_suffix.hour")}',
} day: '!{_p("date_suffix.day")}',
month: '!{_p("date_suffix.month")}'
},
copyright: !{copyright},
lightbox: '!{ theme.lightbox || 'null' }',
Snackbar: !{Snackbar},
infinitegrid: {
js: '!{url_for(theme.asset.egjs_infinitegrid)}',
buttonText: '!{_p("load_more")}'
},
isPhotoFigcaption: !{theme.photofigcaption},
islazyloadPlugin: !{theme.lazyload.enable && !theme.lazyload.native},
isAnchor: !{theme.anchor.auto_update || false},
percent: {
toc: !{theme.toc.scroll_percent},
rightside: !{theme.rightside_scroll_percent},
},
autoDarkmode: !{theme.darkmode.enable && theme.darkmode.autoChangeMode === 1}
}

View File

@@ -1,7 +1,25 @@
script. -
var GLOBAL_CONFIG_SITE = { const titleVal = pageTitle.replace(/'/ig,"\\'")
isPost: !{is_post()},
isHome: !{is_home()}, let isHighlightShrink
isSidebar: !{is_post() && page.toc !== false && theme.toc.enable && (toc(page.content) !== '' || page.encrypt == true )} if (theme.code_blocks.shrink == 'none') isHighlightShrink = 'undefined'
} else if (typeof page.highlight_shrink == 'boolean') isHighlightShrink = page.highlight_shrink
else isHighlightShrink = theme.code_blocks.shrink
var showToc = false
if (theme.aside.enable && page.aside !== false) {
let tocEnable = false
if (globalPageType === 'post' && theme.toc.post) tocEnable = true
else if (globalPageType === 'page' && theme.toc.page) tocEnable = true
const pageToc = typeof page.toc === 'boolean' ? page.toc : tocEnable
showToc = pageToc && (toc(page.content) !== '' || page.encrypt === true)
}
-
script#config-diff.
var GLOBAL_CONFIG_SITE = {
title: '!{titleVal}',
isHighlightShrink: !{isHighlightShrink},
isToc: !{showToc},
pageType: '!{page.type == 'shuoshuo' ? 'shuoshuo' : globalPageType}'
}

View File

@@ -1,47 +0,0 @@
script.
var autoChangeMode = '#{theme.darkmode.autoChangeMode}'
var t = Cookies.get("theme")
if (autoChangeMode == '1'){
var isDarkMode = window.matchMedia("(prefers-color-scheme: dark)").matches
var isLightMode = window.matchMedia("(prefers-color-scheme: light)").matches
var isNotSpecified = window.matchMedia("(prefers-color-scheme: no-preference)").matches
var hasNoSupport = !isDarkMode && !isLightMode && !isNotSpecified
if (t === undefined){
if (isLightMode) activateLightMode()
else if (isDarkMode) activateDarkMode()
else if (isNotSpecified || hasNoSupport){
console.log('You specified no preference for a color scheme or your browser does not support it. I Schedule dark mode during night time.')
var now = new Date()
var hour = now.getHours()
var isNight = hour < 6 || hour >= 18
isNight ? activateDarkMode() : activateLightMode()
}
} else if (t == 'light') activateLightMode()
else activateDarkMode()
} else if (autoChangeMode == '2'){
now = new Date();
hour = now.getHours();
isNight = hour < 6 || hour >= 18
if(t === undefined) isNight? activateDarkMode() : activateLightMode()
else if (t === 'light') activateLightMode()
else activateDarkMode()
} else {
if ( t == 'dark' ) activateDarkMode()
else if ( t == 'light') activateLightMode()
}
function activateDarkMode(){
document.documentElement.setAttribute('data-theme', 'dark')
if (document.querySelector('meta[name="theme-color"]') !== null){
document.querySelector('meta[name="theme-color"]').setAttribute('content','#000')
}
}
function activateLightMode(){
document.documentElement.setAttribute('data-theme', 'light')
if (document.querySelector('meta[name="theme-color"]') !== null){
document.querySelector('meta[name="theme-color"]').setAttribute('content','#fff')
}
}

View File

@@ -1,16 +0,0 @@
link(rel="preconnect" href="//cdn.jsdelivr.net")
if theme.google_analytics
link(rel="preconnect" href="https://www.google-analytics.com" crossorigin)
if theme.baidu_analytics
link(rel="preconnect" href="https://hm.baidu.com")
if theme.tencent_analytics
link(rel="preconnect" href="http://ta.qq.com")
if theme.blog_title_font.font_link
link(rel="preconnect" href="https://fonts.googleapis.com" crossorigin)
if theme.busuanzi.site_uv || theme.busuanzi.site_pv || theme.busuanzi.page_pv
link(rel="preconnect" href="//busuanzi.ibruce.info")

View File

@@ -1,7 +1,9 @@
if (theme.google_adsense && theme.google_adsense.enable) if (theme.google_adsense && theme.google_adsense.enable)
script(async src=theme.google_adsense.js) script(async src=theme.google_adsense.js)
script.
(adsbygoogle = window.adsbygoogle || []).push({ if theme.google_adsense.auto_ads
google_ad_client: '!{theme.google_adsense.client}', script.
enable_page_level_ads: '!{theme.google_adsense.enable_page_level_ads}' (adsbygoogle = window.adsbygoogle || []).push({
}); google_ad_client: '!{theme.google_adsense.client}',
enable_page_level_ads: '!{theme.google_adsense.enable_page_level_ads}'
});

View File

@@ -1,9 +0,0 @@
noscript.
<style>
#page-header {
opacity: 1
}
.justified-gallery img{
opacity: 1
}
</style>

View File

@@ -0,0 +1,35 @@
-
const { internal_provider, third_party_provider, custom_format } = theme.CDN
const providers = {
'jsdelivr': '//cdn.jsdelivr.net',
'cdnjs': '//cdnjs.cloudflare.com',
'unpkg': '//unpkg.com',
'custom': custom_format && custom_format.match(/^((https?:)?(\/\/[^/]+)|([^/]+))(\/|$)/)[1]
}
-
if internal_provider === third_party_provider && internal_provider !== 'local'
link(rel="preconnect" href=providers[internal_provider])
else
if internal_provider !== 'local'
link(rel="preconnect" href=providers[internal_provider])
if third_party_provider !== 'local'
link(rel="preconnect" href=providers[third_party_provider])
if theme.google_analytics
link(rel="preconnect" href="//www.google-analytics.com" crossorigin='')
if theme.baidu_analytics
link(rel="preconnect" href="//hm.baidu.com")
if theme.cloudflare_analytics
link(rel="preconnect" href="//static.cloudflareinsights.com")
if theme.microsoft_clarity
link(rel="preconnect" href="//www.clarity.ms")
if theme.blog_title_font && theme.blog_title_font.font_link && theme.blog_title_font.font_link.indexOf('//fonts.googleapis.com') != -1
link(rel="preconnect" href="//fonts.googleapis.com" crossorigin='')
if !theme.asset.busuanzi && (theme.busuanzi.site_uv || theme.busuanzi.site_pv || theme.busuanzi.page_pv)
link(rel="preconnect" href="//busuanzi.ibruce.info")

View File

@@ -1,13 +1,13 @@
link(rel="manifest" href=theme.pwa.manifest) - const { manifest, theme_color, apple_touch_icon, favicon_32_32, favicon_16_16, mask_icon } = theme.pwa
if(theme.pwa.theme_color)
meta(name="theme-color" content=theme.pwa.theme_color) link(rel="manifest" href=url_for(manifest))
if(theme.pwa.theme_color) if theme_color
meta(name="msapplication-TileColor" content=theme.pwa.theme_color) meta(name="msapplication-TileColor" content=theme_color)
if(theme.pwa.apple_touch_icon) if apple_touch_icon
link(rel="apple-touch-icon" sizes="180x180" href=theme.pwa.apple_touch_icon) link(rel="apple-touch-icon" sizes="180x180" href=url_for(apple_touch_icon))
if(theme.pwa.favicon_32_32) if favicon_32_32
link(rel="icon" type="image/png" sizes="32x32" href=theme.pwa.favicon_32_32) link(rel="icon" type="image/png" sizes="32x32" href=url_for(favicon_32_32))
if(theme.pwa.favicon_16_16) if favicon_16_16
link(rel="icon" type="image/png" sizes="16x16" href=theme.pwa.favicon_16_16) link(rel="icon" type="image/png" sizes="16x16" href=url_for(favicon_16_16))
if(theme.pwa.mask_icon) if mask_icon
link(rel="mask-icon" href=theme.pwa.mask_icon color="#5bbad5") link(rel="mask-icon" href=url_for(mask_icon) color="#5bbad5")

View File

@@ -1,11 +1,3 @@
if theme.google_site_verification if theme.site_verification
meta(name="google-site-verification" content=theme.google_site_verification) each item in theme.site_verification
meta(name=item.name content=item.content)
if theme.bing_site_verification
meta(name="msvalidate.01" content=theme.bing_site_verification)
if theme.baidu_site_verification
meta(name="baidu-site-verification" content=theme.baidu_site_verification)
if theme.qihu_site_verification
meta(name="360-site-verification" content=theme.qihu_site_verification)

View File

@@ -0,0 +1,67 @@
if theme.structured_data
if page.layout === 'post'
-
// https://developers.google.com/search/docs/appearance/structured-data/article
const title = page.title
const url = page.permalink
const imageVal = page.cover_type === 'img' ? page.cover : theme.avatar.img
const image = imageVal ? full_url_for(imageVal) : ''
const datePublished = page.date.toISOString()
const dateModified = (page.updated || page.date).toISOString()
const author = page.copyright_author || config.author
const authorHrefVal = page.copyright_author_href || theme.post_copyright.author_href || config.url
const authorHref = full_url_for(authorHrefVal)
const jsonLd = {
"@context": "https://schema.org",
"@type": "BlogPosting",
"headline": title,
"url": url,
"image": image,
"datePublished": datePublished,
"dateModified": dateModified,
"author": [{
"@type": "Person",
"name": author,
"url": authorHref
}]
}
jsonLdScript = JSON.stringify(jsonLd, null, 2)
-
else if is_home() && (!page.current || page.current === 1)
-
// https://developers.google.com/search/docs/appearance/site-names#website
const baseUrl = config.url;
const currentPath = url_for('/');
const isRootOrSubdomain = currentPath.split('/').filter(Boolean).length === 0;
if (isRootOrSubdomain) {
const domain = new URL(config.url).hostname;
const alternateNames = theme.structured_data.alternate_name || [];
if (config.subtitle) {
alternateNames.push(config.subtitle);
}
if (domain) {
alternateNames.push(domain);
}
const jsonLd = {
"@context": "https://schema.org",
"@type": "WebSite",
"name": config.title,
"alternateName": alternateNames,
"url": full_url_for('/'),
}
jsonLdScript = JSON.stringify(jsonLd, null, 2)
}
-
script(type="application/ld+json").
!{jsonLdScript}

View File

@@ -1,116 +0,0 @@
if theme.subtitle.enable
- var source = theme.subtitle.source
- var subtitleEffect = theme.subtitle.effect
if subtitleEffect
script(src=url_for(theme.CDN.typed))
if source == '1'
script.
var subtitleEffect = !{subtitleEffect}
fetch('https://api.ooopn.com/ciba/api.php',)
.then(function (res){
return res.json();
})
.then(function (data) {
if (subtitleEffect){
var sub = '!{theme.subtitle.sub}'.length == 0 ? new Array() : '!{theme.subtitle.sub}'.split(",");
var both = sub.unshift(data['ciba-en'],data.ciba)
var typed = new Typed("#subtitle", {
strings: sub,
startDelay: 300,
typeSpeed: 150,
loop: !{theme.subtitle.loop},
backSpeed: 50
});
}else{
document.getElementById("subtitle").innerHTML = data['ciba-en']
}
})
.catch(function (err) {
console.error(err);
})
else if source == '2'
script.
var subtitleEffect = !{subtitleEffect}
fetch('https://v1.hitokoto.cn')
.then(function (res){
return res.json();
})
.then(function (data) {
if (subtitleEffect){
var from = '出自 ' + data.from
var sub = '!{theme.subtitle.sub}'.length == 0 ? new Array() : '!{theme.subtitle.sub}'.split(",");
var both = sub.unshift(data.hitokoto,from)
var typed = new Typed("#subtitle", {
strings: sub,
startDelay: 300,
typeSpeed: 150,
loop: !{theme.subtitle.loop},
backSpeed: 50
});
}else{
document.getElementById("subtitle").innerHTML = data.hitokoto
}
})
.catch(function (err) {
console.error(err);
})
else if source == "3"
script(type="text/javascript" src="http://yijuzhan.com/api/word.php?m=js")
script.
var subtitleEffect = !{subtitleEffect}
var con = str[0];
if (subtitleEffect){
var from = "出自 " + str[1];
var sub = '!{theme.subtitle.sub}'.length == 0 ? new Array() : '!{theme.subtitle.sub}'.split(",");
var both = sub.unshift(con,from)
var typed = new Typed("#subtitle", {
strings: sub,
startDelay: 300,
typeSpeed: 150,
loop: !{theme.subtitle.loop},
backSpeed: 50
});
}else{
document.getElementById("subtitle").innerHTML = con
}
else if source == '4'
script(type="text/javascript" src="https://sdk.jinrishici.com/v2/browser/jinrishici.js" charset="utf-8")
script.
var subtitleEffect = !{subtitleEffect}
jinrishici.load(function(result) {
if (subtitleEffect){
var sub = '!{theme.subtitle.sub}'.length == 0 ? new Array() : '!{theme.subtitle.sub}'.split(",");
var content = result.data.content;
var both = sub.unshift(content)
var typed = new Typed("#subtitle", {
strings: sub,
startDelay: 300,
typeSpeed: 150,
loop: !{theme.subtitle.loop},
backSpeed: 50
});
}else{
document.getElementById("subtitle").innerHTML = result.data.content
}
})
else
- var subtitle = theme.subtitle.sub[0]
script.
var subtitleEffect = !{subtitleEffect}
if (subtitleEffect){
var typed = new Typed("#subtitle", {
strings: '!{theme.subtitle.sub}'.split(","),
startDelay: 300,
typeSpeed: 150,
loop: !{theme.subtitle.loop},
backSpeed: 50
})
}else{
document.getElementById("subtitle").innerHTML = '!{subtitle}'
}

View File

@@ -1,18 +0,0 @@
#page-header
span#blog_name.pull_left
a#site-name.blog_title(href=url_for('/')) #[=config.title]
span.pull_right.menus
if (theme.algolia_search.enable || theme.local_search && theme.local_search.enable)
#search_button
a.site-page.social-icon.search
i.fa.fa-search.fa-fw
span=' '+_p('search')
.menus_items
include ./menu_item.pug
span.toggle-menu.close
a.site-page
i.fa.fa-bars.fa-fw(aria-hidden="true")

View File

@@ -1,43 +1,52 @@
if is_home() -
- var top_img = theme.index_img || theme.default_top_img const returnTopImg = img => img !== false ? img || theme.default_top_img : false
else if is_post() const isFixedClass = theme.nav.fixed ? ' fixed' : ''
- var top_img = page.top_img || page.cover || page.randomcover || theme.default_top_img var top_img = false
else if is_archive() let headerClassName = 'not-top-img'
- var top_img = theme.archive_img || theme.default_top_img var bg_img = ''
else if is_tag()
- var top_img = theme.tag_img || theme.default_top_img if !theme.disable_top_img && page.top_img !== false
else if is_category() case globalPageType
- var top_img = theme.category_img || theme.default_top_img when 'post'
else - top_img = page.top_img || page.cover || theme.default_top_img
- var top_img = page.top_img || theme.default_top_img when 'page'
- top_img = page.top_img || theme.default_top_img
if theme.douban when 'tag'
if is_current('/movies/', [strict]) - top_img = theme.tag_per_img && theme.tag_per_img[page.tag] || returnTopImg(theme.tag_img)
- var top_img = theme.douban.movies_img || theme.default_top_img when 'category'
else if is_current('/books/', [strict]) - top_img = theme.category_per_img && theme.category_per_img[page.category] || returnTopImg(theme.category_img)
- var top_img = theme.douban.books_img || theme.default_top_img when 'home'
else if is_current('/games/', [strict]) - top_img = returnTopImg(theme.index_img)
- var top_img = theme.douban.games_img || theme.default_top_img when 'archive'
- top_img = returnTopImg(theme.archive_img)
- var bg_img = top_img && top_img !== true ? `background-image: url(${top_img})` : '' default
- var site_title = is_archive() ? _p('page.archives') : page.title || page.tag || page.category || config.title - top_img = page.top_img || theme.default_top_img
- var isHomeClass = is_home() ? 'full_page' : 'not-index-bg'
- is_post() ? isHomeClass = 'post-bg' : isHomeClass if top_img !== false
- bg_img = getBgPath(top_img)
#nav(class=isHomeClass style=bg_img) - headerClassName = globalPageType === 'home' ? 'full_page' : globalPageType === 'post' ? 'post-bg' : 'not-home-page'
!=partial('includes/header/header', {}, {cache:theme.fragment_cache})
if is_home() header#page-header(class=`${headerClassName + isFixedClass}` style=bg_img)
#site-info include ./nav.pug
h1#site_title=site_title if top_img !== false
#site_subtitle if globalPageType === 'post'
span#subtitle include ./post-info.pug
if(theme.social) else if globalPageType === 'home'
#site_social_icons #site-info
!=partial('includes/header/social', {}, {cache:theme.fragment_cache}) h1#site-title=config.title
#scroll_down if theme.subtitle.enable
i.fa.fa-angle-down.scroll-down-effects - var loadSubJs = true
else if is_post() #site-subtitle
include ./post-info.pug span#subtitle
else if theme.social
#page_site-info #site_social_icons
h1#site_title=site_title !=partial('includes/header/social', {}, {cache: true})
#scroll-down
i.fas.fa-angle-down.scroll-down-effects
else
#page-site-info
h1#site-title=page.title || page.tag || page.category
else
//- improve seo
if globalPageType !== 'post'
h1.title-seo=page.title || page.tag || page.category || config.title

View File

@@ -1,18 +1,27 @@
each value, label in theme.menu if theme.menu
if !Array.isArray(value) .menus_items
.menus_item each value, label in theme.menu
a.site-page(href=url_for(trim(value.split('||')[0]))) if typeof value !== 'object'
i.fa-fw(class=trim(value.split('||')[1])) .menus_item
span=' '+label - const [link, icon] = value.split('||').map(part => trim(part))
else a.site-page(href=url_for(link))
.menus_item if icon
a.site-page i.fa-fw(class=icon)
i.fa-fw(class=trim(label.split('||')[1]) aria-hidden="true") span= ' ' + label
span=' '+ trim(label.split('||')[0]) else
i.fa.fa-chevron-down.menus-expand(aria-hidden="true") .menus_item
ul.menus_item_child - const [groupLabel, groupIcon, groupClass] = label.split('||').map(part => trim(part))
each i in value - const hideClass = groupClass === 'hide' ? 'hide' : ''
li span.site-page.group(class=hideClass)
a.site-page(href=url_for(trim(i.split('||')[1]))) if groupIcon
i.fa-fw(class=trim(i.split('||')[2])) i.fa-fw(class=groupIcon)
span=' '+trim(i.split('||')[0]) span= ' ' + groupLabel
i.fas.fa-chevron-down
ul.menus_item_child
each val, lab in value
- const [childLink, childIcon] = val.split('||').map(part => trim(part))
li
a.site-page.child(href=url_for(childLink))
if childIcon
i.fa-fw(class=childIcon)
span= ' ' + lab

View File

@@ -0,0 +1,26 @@
nav#nav
span#blog-info
a.nav-site-title(href=url_for('/'))
if theme.nav.logo
img.site-icon(src=url_for(theme.nav.logo) alt='Logo')
if theme.nav.display_title
span.site-name=config.title
if globalPageType === 'post' && theme.nav.display_post_title
a.nav-page-title(href=url_for('/'))
span.site-name=(page.title || config.title)
span.site-name
i.fa-solid.fa-circle-arrow-left
span= ' ' + _p('post.back_to_home')
#menus
if theme.search.use
#search-button
span.site-page.social-icon.search
i.fas.fa-search.fa-fw
span= ' ' + _p('search.title')
if theme.menu
!= partial('includes/header/menu_item', {}, {cache: true})
#toggle-menu
span.site-page
i.fas.fa-bars.fa-fw

View File

@@ -1,84 +1,149 @@
#post-info - let comments = theme.comments
#post-title #post-info
.posttitle= page.title || _p('no_title') h1.post-title= page.title || _p('no_title')
if theme.post_edit.enable
#post-meta a.post-edit-link(href=theme.post_edit.url + page.source title=_p('post.edit') target="_blank")
.meta-firstline i.fas.fa-pencil-alt
if (theme.post_meta.post.date_type)
if (theme.post_meta.post.date_type === 'both') #post-meta
time.post-meta__date .meta-firstline
span.post-meta__date-created(title= _p('post.created')+' '+full_date(page.date)) if theme.post_meta.post.date_type
i.fa.fa-calendar(aria-hidden="true") span.post-meta-date
=' '+_p('post.created')+' '+date(page.date, config.date_format) if theme.post_meta.post.date_type === 'both'
span.post-meta__separator | i.far.fa-calendar-alt.fa-fw.post-meta-icon
span.post-meta__date-updated(title= _p('post.updated')+' '+full_date(page.updated)) span.post-meta-label= _p('post.created')
i.fa.fa-history(aria-hidden="true") time.post-meta-date-created(datetime=date_xml(page.date) title=_p('post.created') + ' ' + full_date(page.date))= date(page.date, config.date_format)
=' '+_p('post.updated')+' '+date(page.updated, config.date_format) span.post-meta-separator |
else i.fas.fa-history.fa-fw.post-meta-icon
- var data_type_update = theme.post_meta.post.date_type === 'updated' span.post-meta-label= _p('post.updated')
- var date_type = data_type_update ? 'updated' : 'date' time.post-meta-date-updated(datetime=date_xml(page.updated) title=_p('post.updated') + ' ' + full_date(page.updated))= date(page.updated, config.date_format)
- var date_icon = data_type_update ? 'fa-history' :'fa-calendar' else
- var data_info = data_type_update ? _p('post.updated') : _p('post.created') - let data_type_update = theme.post_meta.post.date_type === 'updated'
time.post-meta__date(title=data_info + ' ' + full_date(page[date_type])) - let date_type = data_type_update ? 'updated' : 'date'
i.fa(class=date_icon aria-hidden="true") - let date_icon = data_type_update ? 'fas fa-history' : 'far fa-calendar-alt'
=' ' + data_info + ' ' + date(page[date_type], config.date_format) - let date_title = data_type_update ? _p('post.updated') : _p('post.created')
i.fa-fw.post-meta-icon(class=date_icon)
if (theme.post_meta.post.categories && page.categories.data.length > 0) span.post-meta-label= date_title
span.post-meta__categories time(datetime=date_xml(page[date_type]) title=date_title + ' ' + full_date(page[date_type]))= date(page[date_type], config.date_format)
if (theme.post_meta.post.date_type) if theme.post_meta.post.categories && page.categories.data.length > 0
span.post-meta__separator | span.post-meta-categories
if theme.post_meta.post.date_type
each item, index in page.categories.data span.post-meta-separator |
i.fa.fa-inbox.post-meta__icon(aria-hidden="true") each item, index in page.categories.data
a(href=url_for(item.path)).post-meta__categories #[=item.name] i.fas.fa-inbox.fa-fw.post-meta-icon
if (index < page.categories.data.length - 1) a(href=url_for(item.path)).post-meta-categories #[=item.name]
i.fa.fa-angle-right.post-meta__separator(aria-hidden="true") if index < page.categories.data.length - 1
i.fas.fa-angle-right.post-meta-separator
.meta-secondline
- let postWordcount = theme.wordcount.enable && (theme.wordcount.post_wordcount || theme.wordcount.min2read) .meta-secondline
if (postWordcount) - let postWordcount = theme.wordcount.enable && (theme.wordcount.post_wordcount || theme.wordcount.min2read)
span.post-meta-wordcount if postWordcount
if theme.wordcount.post_wordcount span.post-meta-separator |
- var wordCountIcon = theme.fontawesome_v5 && theme.fontawesome_v5.enable ? 'far fa-file-word' : 'fa fa-file-word-o' span.post-meta-wordcount
i.post-meta__icon(class=wordCountIcon aria-hidden="true") if theme.wordcount.post_wordcount
span= _p('post.wordcount') + ':' i.far.fa-file-word.fa-fw.post-meta-icon
span.word-count= wordcount(page.content) span.post-meta-label= _p('post.wordcount') + ':'
if theme.wordcount.min2read span.word-count= wordcount(page.content)
span.post-meta__separator | if theme.wordcount.min2read
if theme.wordcount.min2read span.post-meta-separator |
- var readCountIcon = theme.fontawesome_v5 && theme.fontawesome_v5.enable ? 'far fa-clock' : 'fa fa-clock-o' if theme.wordcount.min2read
i.post-meta__icon(class=readCountIcon aria-hidden="true") i.far.fa-clock.fa-fw.post-meta-icon
span= _p('post.min2read', min2read(page.content, {cn: 350, en: 160})) span.post-meta-label= _p('post.min2read') + ':'
span= min2read(page.content, {cn: 350, en: 160}) + _p('post.min2read_unit')
.meta-thirdline
span.post-meta-pv-cv //- for pv and count
if (postWordcount && theme.busuanzi.page_pv) mixin pvBlock(parent_id, parent_class, parent_title)
span.post-meta__separator | span.post-meta-separator |
if theme.busuanzi.page_pv span(class=parent_class id=parent_id data-flag-title=parent_title)
i.fa.fa-eye.post-meta__icon(aria-hidden="true") i.far.fa-eye.fa-fw.post-meta-icon
span=_p('post.page_pv') + ':' span.post-meta-label= _p('post.page_pv') + ':'
span#busuanzi_value_page_pv if block
block
span.post-meta-commentcount
- var commentCount = theme.fontawesome_v5 && theme.fontawesome_v5.enable ? 'far fa-comments' : 'fa fa-comment-o' mixin otherPV()
if ((theme.disqus && theme.disqus.enable && theme.disqus.count && page.comments !== false) || (theme.disqusjs && theme.disqusjs.enable && theme.disqusjs.count && page.comments !== false)) if theme.umami_analytics.enable && theme.umami_analytics.UV_PV.page_pv
if (theme.busuanzi && theme.busuanzi.page_pv) +pvBlock('', '', '')
span.post-meta__separator | span#umamiPV(data-path=url_for(page.path))
i.post-meta__icon(class=commentCount aria-hidden="true") i.fa-solid.fa-spinner.fa-spin
span= _p('post.comments') + ':' else if theme.busuanzi.page_pv
span.disqus-comment-count.comment-count +pvBlock('', 'post-meta-pv-cv', '')
a(href=full_url_for(page.path) + '#disqus_thread') span#busuanzi_value_page_pv
else if (theme.valine && theme.valine.enable && theme.valine.count && page.comments !== false) i.fa-solid.fa-spinner.fa-spin
if (theme.busuanzi && theme.busuanzi.page_pv)
span.post-meta__separator | - const commentUse = comments.use && comments.use[0]
i.post-meta__icon(class=commentCount aria-hidden="true") if page.comments !== false && commentUse && !comments.lazyload
span= _p('post.comments') + ':' if commentUse === 'Valine' && theme.valine.visitor
a(href=url_for(page.path) + '#post-comment' itemprop="discussionUrl") +pvBlock(url_for(page.path), 'leancloud_visitors', page.title)
span.valine-comment-count.comment-count(data-xid=url_for(page.path) itemprop="commentCount") span.leancloud-visitors-count
else if (theme.gitalk && theme.gitalk.enable && theme.gitalk.count && page.comments !== false) i.fa-solid.fa-spinner.fa-spin
if (theme.busuanzi && theme.busuanzi.page_pv) else if commentUse === 'Waline' && theme.waline.pageview
span.post-meta__separator | +pvBlock('', '', '')
i.post-meta__icon(class=commentCount aria-hidden="true") span.waline-pageview-count(data-path=url_for(page.path))
span= _p('post.comments') + ':' i.fa-solid.fa-spinner.fa-spin
a(href=url_for(page.path) + '#post-comment') else if commentUse === 'Twikoo' && theme.twikoo.visitor
span.gitalk-comment-count.comment-count +pvBlock('', '', '')
span#twikoo_visitors
i.fa-solid.fa-spinner.fa-spin
else if commentUse === 'Artalk' && theme.artalk.visitor
+pvBlock('', '', '')
span#ArtalkPV
i.fa-solid.fa-spinner.fa-spin
else
+otherPV()
else
+otherPV()
if comments.count && !comments.lazyload && page.comments !== false && comments.use
- var whichCount = comments.use[0]
mixin countBlock
span.post-meta-separator |
span.post-meta-commentcount
i.far.fa-comments.fa-fw.post-meta-icon
span.post-meta-label= _p('post.comments') + ':'
if block
block
case whichCount
when 'Disqus'
+countBlock
a.disqus-comment-count(href=full_url_for(page.path) + '#post-comment')
i.fa-solid.fa-spinner.fa-spin
when 'Disqusjs'
+countBlock
a.disqusjs-comment-count(href=full_url_for(page.path) + '#post-comment')
i.fa-solid.fa-spinner.fa-spin
when 'Valine'
+countBlock
a(href=url_for(page.path) + '#post-comment' itemprop="discussionUrl")
span.valine-comment-count(data-xid=url_for(page.path) itemprop="commentCount")
i.fa-solid.fa-spinner.fa-spin
when 'Waline'
+countBlock
a(href=url_for(page.path) + '#post-comment')
span.waline-comment-count(data-path=url_for(page.path))
i.fa-solid.fa-spinner.fa-spin
when 'Gitalk'
+countBlock
a(href=url_for(page.path) + '#post-comment')
span.gitalk-comment-count
i.fa-solid.fa-spinner.fa-spin
when 'Twikoo'
+countBlock
a(href=url_for(page.path) + '#post-comment')
span#twikoo-count
i.fa-solid.fa-spinner.fa-spin
when 'Facebook Comments'
+countBlock
a(href=url_for(page.path) + '#post-comment')
span.fb-comments-count(data-href=urlNoIndex())
when 'Remark42'
+countBlock
a(href=url_for(page.path) + '#post-comment')
span.remark42__counter(data-url=urlNoIndex())
i.fa-solid.fa-spinner.fa-spin
when 'Artalk'
+countBlock
a(href=url_for(page.path) + '#post-comment')
span#ArtalkCount
i.fa-solid.fa-spinner.fa-spin

View File

@@ -1,4 +1,8 @@
each url, icon in theme.social each url, icon in theme.social
a.social-icon(href=trim(url.split('||')[0]) target="_blank" -
title=url.split('||')[1] === undefined ? '' : trim(url.split('||')[1])) const [link, title, color] = url.split('||').map(i => trim(i))
i(class=icon aria-hidden="true") const href = url_for(link)
const iconStyle = color ? `color: ${color.replace(/[\'\"]/g, '')};` : ''
const iconTitle = title || ''
a.social-icon(href=href target="_blank" title=iconTitle)
i(class=icon style=iconStyle)

View File

@@ -1,50 +1,59 @@
- var hideAside = !theme.aside.enable || page.aside === false ? 'hide-aside' : '' - var globalPageType = getPageType(page, is_home)
- var htmlClassHideAside = theme.aside.enable && theme.aside.hide ? 'hide-aside' : ''
doctype html - page.aside = globalPageType === 'archive' ? theme.aside.display.archive: globalPageType === 'category' ? theme.aside.display.category : globalPageType === 'tag' ? theme.aside.display.tag : page.aside
html(lang=config.language data-theme=theme.display_mode) - var hideAside = !theme.aside.enable || page.aside === false ? 'hide-aside' : ''
head - var pageType = globalPageType === 'post' ? 'post' : 'page'
include ./head.pug - pageType = page.type ? pageType + ' type-' + page.type : pageType
body
if theme.fireworks && theme.fireworks.enable doctype html
canvas.fireworks html(lang=config.language data-theme=theme.display_mode class=htmlClassHideAside)
head
!=partial('includes/mobile-sidebar/index', {}, {cache:theme.fragment_cache}) include ./head.pug
body
include ./sidebar.pug !=partial('includes/loading/index', {}, {cache: true})
#body-wrap if theme.background
if theme.background if !Array.isArray(theme.background)
- var is_photo = theme.background.substring(3,0) === 'url' ? 'photo':'color' #web_bg.bg-animation(style=getBgPath(theme.background))
#web_bg(data-type=is_photo) else
include ./header/index.pug #web_bg.bg-animation
- const bgStyleArr = theme.background.map(getBgPath)
if (!is_post()) script.
main#content-inner.layout_page(class=hideAside) (() => {
if body const arr = !{JSON.stringify(bgStyleArr)}
div!= body const webBgDiv = document.getElementById('web_bg')
else
block content const setRandomBg = () => {
include widget/index.pug webBgDiv.style = arr[Math.floor(Math.random() * arr.length)]
else requestAnimationFrame(() => webBgDiv.classList.add('bg-animation'))
main#content-inner.layout_post }
if body
div!= body document.addEventListener('pjax:send', () => {
else webBgDiv.style = ''
block content webBgDiv.classList.remove('bg-animation')
})
- var footer_bg = theme.footer_bg == false ? '' : bg_img
- var is_bg = theme.footer_bg == false ? 'color' : 'photo' document.addEventListener('pjax:complete', setRandomBg)
footer#footer(style=footer_bg data-type=is_bg) document.addEventListener('DOMContentLoaded', setRandomBg)
!=partial('includes/footer', {}, {cache:theme.fragment_cache}) })()
include ./rightside.pug !=partial('includes/sidebar', {}, {cache: true})
!=partial('includes/search/index', {}, {cache:theme.fragment_cache}) #body-wrap(class=pageType)
include ./header/index.pug
each item in theme.CDN_USE.js
script(src=url_for(item)) main#content-inner.layout(class=hideAside)
if body
include ./additional-js.pug div!= body
else
block content
if theme.aside.enable && page.aside !== false
include widget/index.pug
- const footerBg = theme.footer_img
- const footer_bg = footerBg ? footerBg === true ? bg_img : getBgPath(footerBg) : ''
footer#footer(style=footer_bg)
!=partial('includes/footer', {}, {cache: true})
include ./rightside.pug
include ./additional-js.pug

View File

@@ -0,0 +1,42 @@
#loading-box
.loading-left-bg
.loading-right-bg
.spinner-box
.configure-border-1
.configure-core
.configure-border-2
.configure-core
.loading-word= _p('loading')
script.
(()=>{
const $loadingBox = document.getElementById('loading-box')
const $body = document.body
const preloader = {
endLoading: () => {
if ($loadingBox.classList.contains('loaded')) return
$body.style.overflow = ''
$loadingBox.classList.add('loaded')
},
initLoading: () => {
$body.style.overflow = 'hidden'
$loadingBox.classList.remove('loaded')
}
}
preloader.initLoading()
if (document.readyState === 'complete') {
preloader.endLoading()
} else {
window.addEventListener('load', preloader.endLoading)
document.addEventListener('DOMContentLoaded', preloader.endLoading)
// Add timeout protection: force end after 7 seconds
setTimeout(preloader.endLoading, 7000)
}
if (!{theme.pjax && theme.pjax.enable}) {
btf.addGlobalFn('pjaxSend', preloader.initLoading, 'preloader_init')
btf.addGlobalFn('pjaxComplete', preloader.endLoading, 'preloader_end')
}
})()

View File

@@ -0,0 +1,5 @@
if theme.preloader.enable
if theme.preloader.source === 1
include ./fullpage-loading.pug
else
include ./pace.pug

View File

@@ -0,0 +1,12 @@
script.
window.paceOptions = {
restartOnPushState: false
}
btf.addGlobalFn('pjaxSend', () => {
Pace.restart()
}, 'pace_restart')
link(rel="stylesheet", href=url_for(theme.preloader.pace_css_url || theme.asset.pace_default_css))
script(src=url_for(theme.asset.pace_js))

View File

@@ -1,15 +0,0 @@
if theme.mathjax && theme.mathjax.enable
if theme.mathjax.per_page
if(!is_tag() && !is_category() && !is_archive() && !is_home())
include ./mathjax.pug
else
if page.mathjax
include ./mathjax.pug
if theme.katex && theme.katex.enable
if theme.katex.per_page
if(!is_tag() && !is_category() && !is_archive() && !is_home())
include ./katex.pug
else
if page.katex
include ./katex.pug

View File

@@ -1,7 +0,0 @@
link(rel="stylesheet" type="text/css" href=theme.CDN.katex)
script(src=url_for(theme.CDN.katex_copytex))
link(rel="stylesheet" type="text/css" href=theme.CDN.katex_copytex_css)
script.
$(function () {
$('span.katex-display').wrap('<div class="katex-wrap"></div>')
})

View File

@@ -1,27 +0,0 @@
script(type="text/x-mathjax-config").
MathJax.Hub.Config({
tex2jax: {
inlineMath: [ ['$','$'], ["\\(","\\)"] ],
processEscapes: true,
skipTags: ['script', 'noscript', 'style', 'textarea', 'pre', 'code']
},
CommonHTML: {
linebreaks: { automatic: true, width: "90% container" }
},
"HTML-CSS": {
linebreaks: { automatic: true, width: "90% container" }
},
"SVG": {
linebreaks: { automatic: true, width: "90% container" }
}
});
script(type="text/x-mathjax-config").
MathJax.Hub.Queue(function() {
var all = MathJax.Hub.getAllJax(), i;
for (i=0; i < all.length; i += 1) {
all[i].SourceElement().parentNode.className += ' has-jax';
}
});
script(src=theme.CDN.mathjax)

View File

@@ -1,8 +0,0 @@
script.
if (document.getElementsByClassName('mermaid').length) {
loadScript('!{theme.CDN.mermaid}',function () {
mermaid.initialize({
theme: '!{theme.mermaid.theme}',
})
})
}

View File

@@ -1,65 +0,0 @@
mixin UI_NEW(posts)
- posts.each(function(article,index){
.recent-post-item
- var link = article.link || article.path
- var post_cover = article.cover
- var title = article.title || _p('no_title')
- var leftOrRight = index%2 == 0 ? 'left_radius' : 'right_radius'
- var no_cover = article.cover === false || !theme.cover.index_enable ? 'no-cover' : ''
if post_cover && theme.cover.index_enable
.post_cover(class=leftOrRight)
a(href=url_for(link) title=title)
if theme.lazyload.enable
img.post_bg.lazyload(data-src=`${post_cover}` alt=title onerror=`this.onerror=null;this.src='`+ url_for(theme.lodding_bg.post_page) + `'`)
else
img.post_bg(src=`${post_cover}` alt=title onerror=`this.onerror=null;this.src='`+ url_for(theme.lodding_bg.post_page) + `'`)
.recent-post-info(class=no_cover)
a.article-title(href=url_for(link) title=title)= title
.article-meta-wrap
if (theme.post_meta.page.date_type)
if (theme.post_meta.page.date_type === 'both')
time.post-meta__date
span.post-meta__date-created(title=_p('post.created') + ' ' + full_date(article.date))
i.fa.fa-calendar(aria-hidden="true")
=date(article.date, config.date_format)
span.article-meta__separator |
span.post-meta__date-updated(title=_p('post.updated') + ' ' + full_date(article.updated))
i.fa.fa-history(aria-hidden="true")
=date(article.updated, config.date_format)
else
- var data_type_updated = theme.post_meta.page.date_type === 'updated'
- var date_type = data_type_updated ? 'updated' : 'date'
- var date_icon = data_type_updated ? 'fa-history' :'fa-calendar'
- var date_title = data_type_updated ? _p('post.updated') : _p('post.created')
time.post-meta__date(title=date_title + ' ' + full_date(article[date_type]))
i.fa(class=date_icon aria-hidden="true")
=date(article[date_type], config.date_format)
if (theme.post_meta.page.categories && article.categories.data.length > 0)
span.article-meta
span.article-meta__separator |
each item, index in article.categories.data
i.fa.fa-inbox.article-meta__icon(aria-hidden="true")
a(href=url_for(item.path)).article-meta__categories #[=item.name]
if (index < article.categories.data.length - 1)
i.fa.fa-angle-right(aria-hidden="true")
if (theme.post_meta.page.tags && article.tags.data.length > 0)
span.article-meta.tags
span.article-meta__separator |
each item, index in article.tags.data
i.fa.fa-tag.article-meta__icon(aria-hidden="true")
a(href=url_for(item.path)).article-meta__tags #[=item.name]
if (index < article.tags.data.length - 1)
span.article-meta__link -
if theme.auto_excerpt && theme.auto_excerpt.enable
- const content = strip_html(article.content)
- let expert = content.substring(0, theme.auto_excerpt.length)
- content.length > theme.auto_excerpt.length ? expert += ' ...' : ''
.content!= expert
else
.content!= article.description
if theme.ad && theme.ad.index
if (index + 1) % 3 == 0
.recent-post-item.ad_height!=theme.ad.index
- })

View File

@@ -1,20 +1,23 @@
mixin articleSort(posts) mixin articleSort(posts)
.article-sort .article-sort
- var year - let year
- posts.each(function (article) { - posts.forEach(article => {
- var tempYear = date(article.date, 'YYYY') - const tempYear = date(article.date, 'YYYY')
- var no_cover = article.cover === false || !theme.cover.archives_enable ? 'no-article-cover' : '' - const noCoverClass = article.cover === false || !theme.cover.archives_enable ? 'no-article-cover' : ''
if tempYear !== year - const title = article.title || _p('no_title')
- year = tempYear if tempYear !== year
.article-sort-item.year= year - year = tempYear
.article-sort-item(class=no_cover) .article-sort-item.year= year
if article.cover && theme.cover.archives_enable .article-sort-item(class=noCoverClass)
.article-sort-img if article.cover && theme.cover.archives_enable
a.article-sort-item__img(href=url_for(article.path)) a.article-sort-item-img(href=url_for(article.path) title=title)
img(src=article.cover alt=article.title || 'No Title' onerror=`this.onerror=null;this.src='`+ url_for(theme.lodding_bg.post_page) + `'`) if article.cover_type === 'img'
.article-sort-post img(src=url_for(article.cover) alt=title onerror=`this.onerror=null;this.src='${url_for(theme.error_img.post_page)}'`)
a.article-sort-item__post(href=url_for(article.path)) else
i.fa.fa-clock-o(aria-hidden="true") div(style=`background: ${article.cover}`)
time.article-sort-item__time(title=_p('post.created') + ' ' + full_date(article.date))= date(article.date, config.date_format) .article-sort-item-info
.article-sort-item__title= article.title || 'No Title' .article-sort-item-time
i.far.fa-calendar-alt
time.post-meta-date-created(datetime=date_xml(article.date) title=_p('post.created') + ' ' + full_date(article.date))= date(article.date, config.date_format)
a.article-sort-item-title(href=url_for(article.path) title=title)= title
- }) - })

View File

@@ -0,0 +1,119 @@
mixin indexPostUI()
- const indexLayout = theme.index_layout
- const masonryLayoutClass = [6, 7].includes(indexLayout) ? 'masonry' : ''
#recent-posts.recent-posts.nc(class=masonryLayoutClass)
.recent-post-items
each article, index in page.posts.data
.recent-post-item
- const link = article.link || article.path
- const title = article.title || _p('no_title')
- const leftOrRight = indexLayout === 3 ? (index % 2 === 0 ? 'left' : 'right') : (indexLayout === 2 ? 'right' : '')
- const postCover = article.cover
- const noCover = article.cover === false || !theme.cover.index_enable ? 'no-cover' : ''
if postCover && theme.cover.index_enable
.post_cover(class=leftOrRight)
a(href=url_for(link) title=title)
if article.cover_type === 'img'
img.post-bg(src=url_for(postCover) onerror=`this.onerror=null;this.src='${url_for(theme.error_img.post_page)}'` alt=title)
else
div.post-bg(style=`background: ${postCover}`)
.recent-post-info(class=noCover)
a.article-title(href=url_for(link) title=title)
if globalPageType === 'home' && (article.top || article.sticky > 0)
i.fas.fa-thumbtack.sticky
= title
.article-meta-wrap
if theme.post_meta.page.date_type
span.post-meta-date
if theme.post_meta.page.date_type === 'both'
i.far.fa-calendar-alt
span.article-meta-label=_p('post.created')
time.post-meta-date-created(datetime=date_xml(article.date) title=_p('post.created') + ' ' + full_date(article.date))= date(article.date, config.date_format)
span.article-meta-separator |
i.fas.fa-history
span.article-meta-label=_p('post.updated')
time.post-meta-date-updated(datetime=date_xml(article.updated) title=_p('post.updated') + ' ' + full_date(article.updated))= date(article.updated, config.date_format)
else
- const isUpdatedType = theme.post_meta.page.date_type === 'updated'
- const dateType = isUpdatedType ? 'updated' : 'date'
- const dateIcon = isUpdatedType ? 'fas fa-history' : 'far fa-calendar-alt'
- const dateTitle = isUpdatedType ? _p('post.updated') : _p('post.created')
i(class=dateIcon)
span.article-meta-label= dateTitle
time(datetime=date_xml(article[dateType]) title=dateTitle + ' ' + full_date(article[dateType]))= date(article[dateType], config.date_format)
if theme.post_meta.page.categories && article.categories.data.length > 0
span.article-meta
span.article-meta-separator |
each item, index in article.categories.data
i.fas.fa-inbox
a(href=url_for(item.path)).article-meta__categories #[=item.name]
if index < article.categories.data.length - 1
i.fas.fa-angle-right.article-meta-link
if theme.post_meta.page.tags && article.tags.length > 0
span.article-meta.tags
span.article-meta-separator |
each item, index in article.tags.data
i.fas.fa-tag
a(href=url_for(item.path)).article-meta__tags #[=item.name]
if index < article.tags.data.length - 1
span.article-meta-link #[='•']
mixin countBlockInIndex
- needLoadCountJs = true
span.article-meta
span.article-meta-separator |
i.fas.fa-comments
if block
block
span.article-meta-label= ' ' + _p('card_post_count')
if theme.comments.card_post_count && theme.comments.use
- const commentSystem = theme.comments.use[0]
- const commentLink = url_for(link) + '#post-comment'
case commentSystem
when 'Disqus'
when 'Disqusjs'
+countBlockInIndex
a.disqus-count(href=full_url_for(link) + '#post-comment')
i.fa-solid.fa-spinner.fa-spin
when 'Valine'
+countBlockInIndex
a(href=commentLink)
span.valine-comment-count(data-xid=url_for(link))
i.fa-solid.fa-spinner.fa-spin
when 'Waline'
+countBlockInIndex
a(href=commentLink)
span.waline-comment-count(data-path=url_for(link))
i.fa-solid.fa-spinner.fa-spin
when 'Twikoo'
+countBlockInIndex
a.twikoo-count(href=commentLink)
i.fa-solid.fa-spinner.fa-spin
when 'Facebook Comments'
+countBlockInIndex
a(href=commentLink)
span.fb-comments-count(data-href=urlNoIndex(article.permalink))
when 'Remark42'
+countBlockInIndex
a(href=commentLink)
span.remark42__counter(data-url=urlNoIndex(article.permalink))
i.fa-solid.fa-spinner.fa-spin
when 'Artalk'
+countBlockInIndex
a(href=commentLink)
span.artalk-count(data-page-key=url_for(link))
i.fa-solid.fa-spinner.fa-spin
//- Display the article introduction on homepage
- const content = postDesc(article)
if content
.content!=content
if theme.ad && theme.ad.index
if (index + 1) % 3 === 0
.recent-post-item.ads-wrap!= theme.ad.index
include ../pagination.pug

View File

@@ -1,3 +0,0 @@
#mobile-sidebar
#menu_mask
include ../mobile-sidebar/mobile-menus.pug

View File

@@ -1,27 +0,0 @@
#mobile-sidebar-menus
.mobile_author_icon
img.avatar-img(src=url_for(theme.avatar.img) onerror=`onerror=null;src='${theme.lodding_bg.flink}'` alt="avatar")
.mobile_post_data
if site.posts.length
.mobile_data_item.is-center
.mobile_data_link
a(href=url_for(config.archive_dir) + '/')
.headline= _p('aside.articles')
.length_num= site.posts.length
if site.tags.length
.mobile_data_item.is-center
.mobile_data_link
a(href=url_for(config.tag_dir) + '/' )
.headline= _p('aside.tags')
.length_num= site.tags.length
if site.categories.length
.mobile_data_item.is-center
.mobile_data_link
a(href=url_for(config.category_dir) + '/')
.headline= _p('aside.categories')
.length_num= site.categories.length
hr
.menus_items
include ../header/menu_item.pug

View File

@@ -0,0 +1,8 @@
- var top_img_404 = theme.error_404.background || theme.default_top_img
.error-content
.error-img
img(src=url_for(top_img_404) alt='Page not found')
.error-info
h1.error_title= '404'
.error_subtitle= theme.error_404.subtitle || _p('error404')

View File

@@ -0,0 +1 @@
.category-lists!= list_categories()

View File

@@ -0,0 +1,2 @@
#article-container.container
!= page.content

View File

@@ -0,0 +1,82 @@
#article-container.container
.flink
- let { content, random, flink_url } = page
- let pageContent = content
if flink_url || random
- const linkData = flink_url ? false : site.data.link || false
script.
(()=>{
const replaceSymbol = (str) => {
return str.replace(/[\p{P}\p{S}]/gu, "-")
}
let result = ""
const add = (str) => {
for(let i = 0; i < str.length; i++){
const replaceClassName = replaceSymbol(str[i].class_name)
const className = str[i].class_name ? `<h2 id="${replaceClassName}"><a href="#${replaceClassName}" class="headerlink" title="${str[i].class_name}"></a>${str[i].class_name}</h2>` : ""
const classDesc = str[i].class_desc ? `<div class="flink-desc">${str[i].class_desc}</div>` : ""
let listResult = ""
const lists = str[i].link_list
if (!{random === true}) {
lists.sort(() => Math.random() - 0.5)
}
for(let j = 0; j < lists.length; j++){
listResult += `
<div class="flink-list-item">
<a href="${lists[j].link}" title="${lists[j].name}" target="_blank">
<div class="flink-item-icon">
<img class="no-lightbox" src="${lists[j].avatar}" onerror='this.onerror=null;this.src="!{url_for(theme.error_img.flink)}"' alt="${lists[j].name}" />
</div>
<div class="flink-item-name">${lists[j].name}</div>
<div class="flink-item-desc" title="${lists[j].descr}">${lists[j].descr}</div>
</a>
</div>`
}
result += `${className}${classDesc} <div class="flink-list">${listResult}</div>`
}
document.querySelector(".flink").insertAdjacentHTML("afterbegin", result)
window.lazyLoadInstance && window.lazyLoadInstance.update()
}
const linkData = !{JSON.stringify(linkData)}
if (!{Boolean(flink_url)}) {
fetch("!{url_for(flink_url)}")
.then(response => response.json())
.then(add)
} else if (linkData) {
add(linkData)
}
})()
else
if site.data.link
- let result = ""
each i in site.data.link
- let className = i.class_name ? markdown(`## ${i.class_name}`) : ""
- let classDesc = i.class_desc ? `<div class="flink-desc">${i.class_desc}</div>` : ""
- let listResult = ""
each j in i.link_list
-
listResult += `
<div class="flink-list-item">
<a href="${j.link}" title="${j.name}" target="_blank">
<div class="flink-item-icon">
<img class="no-lightbox" src="${j.avatar}" onerror='this.onerror=null;this.src="${url_for(theme.error_img.flink)}"' alt="${j.name}" />
</div>
<div class="flink-item-name">${j.name}</div>
<div class="flink-item-desc" title="${j.descr}">${j.descr}</div>
</a>
</div>`
-
- result += `${className}${classDesc} <div class="flink-list">${listResult}</div>`
- pageContent = result + pageContent
!= pageContent

View File

@@ -0,0 +1,332 @@
//- - author:
//- avatar:
//- date:
//- content:
//- tags:
//- - tag1
//- - tag2
- page.toc = false
#article-container
if page.shuoshuo_url || (site.data.shuoshuo && site.data.shuoshuo.length)
if page.comments !== false && theme.comments.use
- commentsJsLoad = true
script.
(() => {
const commentDiv = `!{partial('includes/third-party/comments/index', {}, {cache: true})}`
const runDestroy = (shuoshuoComment) => {
if (!shuoshuoComment) return
for (const [key, fn] of Object.entries(shuoshuoComment)) {
if (key.startsWith('destroy')) fn()
}
}
window.addCommentToShuoshuo = e => {
const btn = e.target.closest('.shuoshuo-comment-btn')
if (!btn) return
const ele = btn.closest('.container').nextElementSibling
const { shuoshuoComment } = window
const isInclude = ele.classList.contains('no-comment')
runDestroy(shuoshuoComment)
if (isInclude) {
ele.classList.remove('no-comment')
ele.innerHTML = commentDiv
const key = `${location.pathname.replace(/\/$/, '')}?key=${ele.getAttribute('data-key')}`
btf.switchComments(ele, key)
shuoshuoComment.loadComment && shuoshuoComment.loadComment(ele, key)
}
}
})()
- const localDate = page.shuoshuo_url ? [] : shuoshuoFN(site.data.shuoshuo, page)
if !page.shuoshuo_url
script(type='application/json' id='shuoshuo-data')!= safeJSON(localDate)
- const { enable, native, placeholder, field } = theme.lazyload
script.
(() => {
const limitConfig = !{ JSON.stringify(page.limit || {}) }
const sortDataByDate = data => data.sort((a, b) => new Date(b.date) - new Date(a.date))
const filterDataByLimit = (data, limit) => {
if (!limit || !limit.type) return data
if (limit.type === 'num') return data.slice(0, limit.value)
if (limit.type === 'date') {
const limitDate = new Date(limit.value)
return data.filter(item => new Date(item.date) >= limitDate)
}
return data
};
const formatToTimeZone = (date) => {
const fullDate = date.length === 10 ? `${date} 00:00:00` : date
const visitorTimeZone = '#{config.timezone}' || Intl.DateTimeFormat().resolvedOptions().timeZone
const options = {
timeZone: visitorTimeZone,
year: 'numeric',
month: '2-digit',
day: '2-digit',
hour: '2-digit',
minute: '2-digit',
second: '2-digit',
hour12: false
}
const [day, month, year, hour, minute, second] = new Intl.DateTimeFormat('en-GB', options)
.format(new Date(fullDate))
.match(/\d+/g)
return `${year}-${month}-${day} ${hour}:${minute}:${second}`
}
const addLazyload = str => {
const config = {
enable: !{Boolean(enable)},
native: !{Boolean(native)},
field: '!{field}',
placeholder: '!{url_for(placeholder)}',
}
if (!config.enable || config.field !== 'site') return str
const parser = new DOMParser()
const doc = parser.parseFromString(str, 'text/html')
const images = doc.querySelectorAll('img')
images.forEach(img => {
if (config.native) {
img.setAttribute('loading', 'lazy')
} else {
const src = img.getAttribute('src')
img.setAttribute('data-lazy-src', src)
if (config.placeholder) {
img.setAttribute('src', config.placeholder)
} else {
img.removeAttribute('src')
}
}
})
return doc.body.innerHTML
}
let currentPage = 1
const itemsPerPage = 8
let totalPages = 0
let data = []
let inputEventsAttached = false // Flag to mark if input event listeners have been added
const renderData = (dataSlice) => {
const content = dataSlice.map(item => {
const formattedDate = formatToTimeZone(item.date)
const tags = item.tags && item.tags.map(tag => `<span class="shuoshuo-tag">${tag}</span>`).join('') || ''
const commentButton = item.key && !{commentsJsLoad}
? `<div class="shuoshuo-comment-btn" onclick="addCommentToShuoshuo(event)">
<i class="fa-solid fa-comments"></i>
</div>`
: ''
const commentContainer = item.key
? `<div class="shuoshuo-comment no-comment" data-key="${item.key}"></div>`
: ''
return `
<div class="shuoshuo-item">
<div class="container">
<div class="shuoshuo-item-header">
<div class="shuoshuo-avatar">
<img class="no-lightbox" src="${item.avatar || '!{url_for(theme.avatar.img)}'}">
</div>
<div class="shuoshuo-info">
<div class="shuoshuo-author">${item.author || '!{config.author}'}</div>
<time class="shuoshuo-date" title="${formattedDate}">
${btf.diffDate(formattedDate, true)}
</time>
</div>
</div>
<div class="shuoshuo-content">${addLazyload(item.content)}</div>
<div class="shuoshuo-footer ${tags ? 'flex-between' : 'flex-end'}">
${tags ? `<div class="shuoshuo-tags">${tags}</div>` : ''}
${commentButton}
</div>
</div>
${commentContainer}
</div>`
}).join('')
const container = document.getElementById('article-container')
container.innerHTML = content
window.lazyLoadInstance && window.lazyLoadInstance.update()
btf.loadLightbox(document.querySelectorAll('#article-container img:not(.no-lightbox)'))
}
const renderNavigation = () => {
const container = document.getElementById('article-container')
const existingNav = container.nextElementSibling
if (existingNav && existingNav.classList.contains('shuoshuo-navigation')) {
existingNav.remove()
}
const pageInfoTemplate = '#{__('pagination.page_info')}'
const pageInfoText = pageInfoTemplate
.replace(/\$\{current}/g, currentPage)
.replace(/\$\{total}/g, totalPages)
const navHtml = `
<div class="shuoshuo-navigation">
<button onclick="window.shuoshuoPrevPage()" ${currentPage === 1 ? 'disabled' : ''}><i class="fa-solid fa-chevron-left"></i></button>
<span class="shuoshuo-page-info">${pageInfoText}</span>
<input type="number" class="shuoshuo-page-input" min="1" max="${totalPages}" placeholder="${currentPage}" onkeydown="window.shuoshuoHandleKeyDown(event)">
<button onclick="window.shuoshuoNextPage()" ${currentPage === totalPages ? 'disabled' : ''}><i class="fa-solid fa-chevron-right"></i></button>
</div>
`
container.insertAdjacentHTML('afterend', navHtml)
// Add input validation event listeners (only once)
if (!inputEventsAttached) {
setTimeout(() => {
const input = document.querySelector('.shuoshuo-page-input')
if (input) {
// Clear placeholder when clicking the input box
input.addEventListener('focus', (event) => {
event.target.placeholder = ''
})
// Restore placeholder if no content when losing focus
input.addEventListener('blur', (event) => {
if (!event.target.value.trim()) {
event.target.placeholder = currentPage
}
})
input.addEventListener('input', (event) => {
const value = parseInt(event.target.value) || 0
let wasInvalid = false
if (value > totalPages) {
event.target.value = totalPages
wasInvalid = true
} else if (value < 1 && event.target.value !== '') {
event.target.value = 1
wasInvalid = true
}
// If value is corrected, show red and shake effect
if (wasInvalid) {
event.target.classList.add('invalid')
setTimeout(() => {
event.target.classList.remove('invalid')
}, 500)
}
})
inputEventsAttached = true // Mark that event listeners have been added
}
}, 0)
}
}
const renderPage = (page) => {
const start = (page - 1) * itemsPerPage
const end = start + itemsPerPage
const pageData = data.slice(start, end)
renderData(pageData)
renderNavigation()
}
window.shuoshuoPrevPage = () => {
if (currentPage > 1) {
currentPage--
renderPage(currentPage)
}
}
window.shuoshuoNextPage = () => {
if (currentPage < totalPages) {
currentPage++
renderPage(currentPage)
}
}
window.shuoshuoGoToPage = (page) => {
if (typeof page === 'number') {
// Directly jump to the specified page
if (page >= 1 && page <= totalPages && page !== currentPage) {
currentPage = page
renderPage(currentPage)
}
} else {
// Get page from input box
const input = document.querySelector('.shuoshuo-page-input')
const inputValue = input.value.trim()
const inputPage = inputValue === '' ? currentPage : parseInt(inputValue)
if (inputPage >= 1 && inputPage <= totalPages && inputPage !== currentPage) {
currentPage = inputPage
renderPage(currentPage)
} else if (inputValue === '') {
// If input box is empty, re-render current page (update placeholder)
renderPage(currentPage)
}
}
}
window.shuoshuoHandleKeyDown = (event) => {
const input = event.target
const value = input.value + event.key
// Allow delete, arrow keys, backspace, etc.
if (event.key === 'Enter' || event.key === 'Backspace' || event.key === 'Delete' ||
event.key === 'ArrowLeft' || event.key === 'ArrowRight' ||
event.key === 'Tab' || event.ctrlKey || event.metaKey) {
if (event.key === 'Enter') {
window.shuoshuoGoToPage()
}
return
}
// Only allow numbers
if (!/^\d$/.test(event.key)) {
event.preventDefault()
return
}
// Check if the value after input exceeds the range
const newValue = parseInt(value) || 0
if (newValue > totalPages || (value.length > 1 && newValue === 0)) {
event.preventDefault()
// Add red and shake effect
input.classList.add('invalid')
setTimeout(() => {
input.classList.remove('invalid')
}, 500)
}
}
const loadShuoshuo = async () => {
try {
let originData = []
if (!{Boolean(page.shuoshuo_url)}) {
const response = await fetch('!{url_for(page.shuoshuo_url)}')
originData = await response.json()
} else {
const dataElement = document.getElementById('shuoshuo-data')
originData = dataElement ? JSON.parse(dataElement.textContent) : []
}
data = filterDataByLimit(sortDataByDate(originData), limitConfig)
totalPages = Math.ceil(data.length / itemsPerPage)
renderPage(currentPage)
} catch (error) {
console.error(error)
}
};
window.pjax ? loadShuoshuo() : window.addEventListener('load', loadShuoshuo)
})()

View File

@@ -0,0 +1,2 @@
.tag-cloud-list.text-center
!=cloudTags({source: site.tags, orderby: page.orderby || 'random', order: page.order || 1, minfontsize: 1.2, maxfontsize: 1.5, limit: 0, unit: 'em', custom_colors: page.custom_colors})

View File

@@ -1,39 +1,38 @@
- if page.total !== 1
var options = { -
prev_text: '<i class="fa fa-fw fa-chevron-left" aria-hidden="true"></i>', var options = {
next_text: '<i class="fa fa-fw fa-chevron-right" aria-hidden="true"></i>', prev_text: '<i class="fas fa-chevron-left fa-fw"></i>',
mid_size: 1, next_text: '<i class="fas fa-chevron-right fa-fw"></i>',
escape: false mid_size: 1,
} escape: false
}
if(!is_post())
nav#pagination if globalPageType === 'post'
div.pagination - let paginationOrder = theme.post_pagination === 2 ? { prev: page.prev, next: page.next } : { prev: page.next, next: page.prev }
!=paginator(options)
else nav#pagination.pagination-post
nav#pagination.pagination_post each direction, key in paginationOrder
if(page.prev) if direction
- var hasPageNext = page.prev ? 'pull_left' : 'pull-full' - const getPostDesc = direction.postDesc || postDesc(direction)
.prev-post(class=hasPageNext) - let className = key === 'prev' ? (paginationOrder.next ? '' : 'full-width') : (paginationOrder.prev ? '' : 'full-width')
- var pagination_cover = page.prev.cover === false ? page.prev.randomcover : page.prev.cover - className = getPostDesc ? className : className + ' no-desc'
a(href=url_for(page.prev.path))
if theme.lazyload.enable a.pagination-related(class=className href=url_for(direction.path) title=direction.title)
img.prev_cover.lazyload(data-src=`${pagination_cover}` onerror=`onerror=null;src='${theme.lodding_bg.post_page}'` ) if direction.cover_type === 'img'
else img.cover(src=url_for(direction.pagination_cover || direction.cover) onerror=`onerror=null;src='${url_for(theme.error_img.post_page)}'` alt=`cover of ${key === 'prev' ? 'previous' : 'next'} post`)
img.prev_cover(src=`${pagination_cover}` onerror=`onerror=null;src='${theme.lodding_bg.post_page}'` ) else
.pagination-info .cover(style=`background: ${direction.cover || 'var(--default-bg-color)'}`)
.label=_p('pagination.prev')
.prev_info=page.prev.title .info(class=key === 'prev' ? '' : 'text-right')
.info-1
if(page.next) .info-item-1=_p(`pagination.${key}`)
- var hasPagePrev = page.prev ? 'pull_right' : 'pull-full' .info-item-2!=direction.title
- var pagination_cover = page.next.cover == false ? page.next.randomcover : page.next.cover if getPostDesc
.next-post(class=hasPagePrev) .info-2
a(href=url_for(page.next.path)) .info-item-1!=getPostDesc
if theme.lazyload.enable else
img.next_cover.lazyload(data-src=`${pagination_cover}` onerror=`onerror=null;src='${theme.lodding_bg.post_page}'`) nav#pagination
else .pagination
img.next_cover(src=`${pagination_cover}` onerror=`onerror=null;src='${theme.lodding_bg.post_page}'`) if globalPageType === 'home'
.pagination-info - options.format = 'page/%d/#content-inner'
.label=_p('pagination.next') !=paginator(options)
.next_info=page.next.title

View File

@@ -0,0 +1,8 @@
- const { limit_day, message_prev, message_next, position} = theme.noticeOutdate
- const notice_data = { limitDay: limit_day, messagePrev: message_prev, messageNext: message_next, postUpdate: full_date(page.updated)}
if position === 'top'
#post-outdate-notice(data=notice_data hidden)
!=page.content
else
!=page.content
#post-outdate-notice(data=notice_data hidden)

View File

@@ -1,13 +1,23 @@
if (theme.post_copyright && theme.post_copyright.enable && page.copyright != false) if theme.post_copyright.enable && page.copyright !== false
.post-copyright - const author = page.copyright_author || config.author
.post-copyright__author - const authorHref = page.copyright_author_href || theme.post_copyright.author_href || config.url
span.post-copyright-meta= _p('post.copyright.author') + ": " - const url = page.copyright_url || page.permalink
span.post-copyright-info - const info = page.copyright_info || _p('post.copyright.copyright_content', theme.post_copyright.license_url, theme.post_copyright.license, config.url, config.title)
a(href=`mailto:${config.email}`) #[=config.author] .post-copyright
.post-copyright__type .post-copyright__author
span.post-copyright-meta= _p('post.copyright.link') + ": " span.post-copyright-meta
span.post-copyright-info i.fas.fa-circle-user.fa-fw
a(href=url_for(page.permalink))= theme.post_copyright.decode ? decodeURI(page.permalink) : page.permalink = _p('post.copyright.author') + ": "
.post-copyright__notice span.post-copyright-info
span.post-copyright-meta= _p('post.copyright.copyright_notice') + ": " a(href=authorHref)= author
span.post-copyright-info!= _p('post.copyright.copyright_content', theme.post_copyright.license_url, theme.post_copyright.license, config.url, config.title) .post-copyright__type
span.post-copyright-meta
i.fas.fa-square-arrow-up-right.fa-fw
= _p('post.copyright.link') + ": "
span.post-copyright-info
a(href=url_for(url))= theme.post_copyright.decode ? decodeURI(url) : url
.post-copyright__notice
span.post-copyright-meta
i.fas.fa-circle-exclamation.fa-fw
= _p('post.copyright.copyright_notice') + ": "
span.post-copyright-info!= info

View File

@@ -1,11 +1,12 @@
.post-reward .post-reward
a.reward-button.button--primary.button--animated .reward-button
i.fa.fa-qrcode i.fas.fa-qrcode
= ' ' + _p('donate') = theme.reward.text || _p('donate')
.reward-main .reward-main
ul.reward-all ul.reward-all
each item in theme.reward.QR_code each item in theme.reward.QR_code
li.reward-item - const clickTo = item.link || item.img
img.lazyload.post-qr-code__img(src=url_for((item.itemlist||item).img) alt=(item.itemlist||item).text) li.reward-item
.post-qr-code__desc=(item.itemlist||item).text a(href=url_for(clickTo) target='_blank')
img.post-qr-code-img(src=url_for(item.img) alt=item.text)
.post-qr-code-desc=item.text

View File

@@ -1,67 +0,0 @@
each article , index in page.posts.data
.recent-post-item
- var link = article.link || article.path
- var title = article.title || _p('no_title')
- var leftOrRight = index%2 == 0 ? 'left_radius' : 'right_radius'
- var post_cover = article.cover
- var no_cover = article.cover === false || !theme.cover.index_enable ? 'no-cover' : ''
if post_cover && theme.cover.index_enable
.post_cover(class=leftOrRight)
a(href=url_for(link) title=title)
if theme.lazyload.enable
img.post_bg.lazyload(data-src=`${post_cover}` onerror=`this.onerror=null;this.src='`+ url_for(theme.lodding_bg.post_page) + `'` alt=title)
else
img.post_bg(src=`${post_cover}` onerror=`this.onerror=null;this.src='`+ url_for(theme.lodding_bg.post_page) + `'` alt=title)
.recent-post-info(class=no_cover)
a.article-title(href=url_for(link) title=title)= title
.article-meta-wrap
if (article.top)
span.article-meta
i.fa.fa-thumb-tack.article-meta__icon.sticky
span.sticky= _p('sticky')
span.article-meta__separator |
if (theme.post_meta.page.date_type)
if (theme.post_meta.page.date_type === 'both')
time.post-meta__date
span.post-meta__date-created(title=_p('post.created') + ' ' + full_date(article.date))
i.fa.fa-calendar(aria-hidden="true")
=date(article.date, config.date_format)
span.article-meta__separator |
span.post-meta__date-updated(title=_p('post.updated') + ' ' + full_date(article.updated))
i.fa.fa-history(aria-hidden="true")
=date(article.updated, config.date_format)
else
- var data_type_updated = theme.post_meta.page.date_type === 'updated'
- var date_type = data_type_updated ? 'updated' : 'date'
- var date_icon = data_type_updated ? 'fa-history' :'fa-calendar'
- var date_title = data_type_updated ? _p('post.updated') : _p('post.created')
time.post-meta__date(title=date_title + ' ' + full_date(article[date_type]))
i.fa(class=date_icon aria-hidden="true")
=date(article[date_type], config.date_format)
if (theme.post_meta.page.categories && article.categories.data.length > 0)
span.article-meta
span.article-meta__separator |
each item, index in article.categories.data
i.fa.fa-inbox.article-meta__icon(aria-hidden="true")
a(href=url_for(item.path)).article-meta__categories #[=item.name]
if (index < article.categories.data.length - 1)
i.fa.fa-angle-right(aria-hidden="true")
if (theme.post_meta.page.tags && article.tags.data.length > 0)
span.article-meta.tags
span.article-meta__separator |
each item, index in article.tags.data
i.fa.fa-tag.article-meta__icon(aria-hidden="true")
a(href=url_for(item.path)).article-meta__tags #[=item.name]
if (index < article.tags.data.length - 1)
span.article-meta__link -
if theme.auto_excerpt && theme.auto_excerpt.enable
- const content = strip_html(article.content)
- let expert = content.substring(0, theme.auto_excerpt.length)
- content.length > theme.auto_excerpt.length ? expert += ' ...' : ''
.content!= expert
else
.content!= article.description
if theme.ad && theme.ad.index
if (index + 1) % 3 == 0
.recent-post-item.ad_height!=theme.ad.index

View File

@@ -1,23 +1,54 @@
section#rightside.rightside - const { readmode, translate, darkmode, aside, chat } = theme
#rightside-config-hide
if theme.readmode.enable && is_post() mixin rightsideItem(array)
i#readmode.fa.fa-book(title=_p('rightside.readmode_title')) each item in array
if is_post() case item
i#font_plus.fa.fa-plus(title=_p('rightside.font_plus_title')) when 'readmode'
i#font_minus.fa.fa-minus(title=_p('rightside.font_minus_title')) if globalPageType === 'post' && readmode
if theme.translate && theme.translate.enable button#readmode(type="button" title=_p('rightside.readmode_title'))
a#translateLink.translate_chn_to_cht(href="javascript:translatePage();" title=_p('rightside.translate_title') target="_self")= theme.translate.default i.fas.fa-book-open
if theme.darkmode.enable && theme.darkmode.button when 'translate'
- var lightModeIcon = theme.fontawesome_v5 && theme.fontawesome_v5.enable ? 'far fa-sun' : 'fa fa-sun-o' if translate.enable
- var darkModeIcon = theme.fontawesome_v5 && theme.fontawesome_v5.enable ? 'far fa-moon' : 'fa fa-moon-o' button#translateLink(type="button" title=_p('rightside.translate_title'))= translate.default
- var display_mode = theme.display_mode == 'dark' ? lightModeIcon : darkModeIcon when 'darkmode'
i#darkmode.darkmode(class=display_mode title=_p('rightside.night_mode_title')) if darkmode.enable && darkmode.button
#rightside-config-show button#darkmode(type="button" title=_p('rightside.night_mode_title'))
#rightside_config(title=_p("rightside.setting")) i.fas.fa-adjust
i.fa.fa-cog(aria-hidden="true") when 'hideAside'
if is_post() && page.comments !== false && isComment if aside.enable && aside.button && page.aside !== false
a#to_comment(href="#post-comment" title=_p("rightside.scroll_to_comment")) button#hide-aside-btn(type="button" title=_p('rightside.aside'))
i.scroll_to_comment.fa.fa-comments i.fas.fa-arrows-alt-h
if is_post() && page.toc !== false && theme.toc.enable && (toc(page.content) !== '' || page.encrypt == true ) when 'toc'
i#mobile-toc-button.fa.fa-list-ul.close(title=_p("rightside.toc") aria-hidden="true") if showToc
i.fa.fa-arrow-up#go-up(title=_p("rightside.back_to_top") aria-hidden="true") button#mobile-toc-button.close(type="button" title=_p("rightside.toc"))
i.fas.fa-list-ul
when 'chat'
if chat.rightside_button && chat.use
button#chat-btn(type="button" title=_p("rightside.chat") style="display:none")
i.fas.fa-message
when 'comment'
if commentsJsLoad
a#to_comment(href="#post-comment" title=_p("rightside.scroll_to_comment"))
i.fas.fa-comments
- const { enable, hide, show } = theme.rightside_item_order
- const hideArray = enable && hide ? hide.split(',') : ['readmode','translate','darkmode','hideAside']
- const showArray = enable && show ? show.split(',') : ['toc','chat','comment']
- const needCogBtn = (enable && hide) || (!enable && ((globalPageType === 'post' && (readmode || translate.enable || (darkmode.enable && darkmode.button))) || (translate.enable || (darkmode.enable && darkmode.button))))
#rightside
#rightside-config-hide
if hideArray.length
+rightsideItem(hideArray)
#rightside-config-show
if needCogBtn
button#rightside-config(type="button" title=_p("rightside.setting"))
i.fas.fa-cog(class=theme.rightside_config_animation ? 'fa-spin' : '')
if showArray.length
+rightsideItem(showArray)
button#go-up(type="button" title=_p("rightside.back_to_top"))
span.scroll-percent
i.fas.fa-arrow-up

View File

@@ -1,12 +0,0 @@
#algolia-search.search-dialog
#algolia-search-title.search-dialog__title Algolia
#algolia-input-panel
#algolia-search-input
hr
#algolia-search-results
#algolia-hits
#algolia-pagination
#algolia-stats
span.search-close-button
i.fa.fa-times
.search-mask

View File

@@ -1,5 +0,0 @@
if (theme.algolia_search.enable)
include ./algolia.pug
if (theme.local_search)
if (!theme.algolia_search.enable && theme.local_search.enable)
include ./local-search.pug

View File

@@ -1,23 +0,0 @@
#local-search.search-dialog
#local-search-title.search-dialog__title=_p("local_search.label")
#local-input-panel
#local-search-input
.local-search-box
input(placeholder=_p("local_search.input_placeholder") type="text").local-search-box--input
hr
#local-search-results
#local-hits
#local-stats
#hr.local-search-stats__hr
case config.language
when "zh-CN"
span=_p("local_search.by")
| #[a(href="https://github.com/wzpan/hexo-generator-search" style={'color': '#49B1F5'}) hexo-generator-search]
| #[span=_p("local_search.powered")]
when "en"
default
span=_p("local_search.powered_by")
| #[a(href="https://github.com/wzpan/hexo-generator-search" style={'color': '#49B1F5'}) hexo-generator-search]
span.search-close-button
i.fa.fa-times
.search-mask

View File

@@ -1,2 +0,0 @@
.addthis_inline_share_toolbox
script(src=`//s7.addthis.com/js/300/addthis_widget.js#pubid=${theme.addThis.pubid}` async)

View File

@@ -1,9 +0,0 @@
.addtoany
.a2a_kit.a2a_kit_size_32.a2a_default_style
each name in theme.addtoany.item
a(class="a2a_button_" + name)
a.a2a_dd(href="https://www.addtoany.com/share")
script(async src=url_for(theme.CDN.addtoany))

View File

@@ -1,7 +0,0 @@
.post_share
if theme.addThis.enable
include ./add-this.pug
else if theme.sharejs && theme.sharejs.enable
include ./share-js.pug
else if theme.addtoany.enable
include ./addtoany.pug

View File

@@ -1,4 +0,0 @@
if (theme.sharejs && theme.sharejs.enable)
.social-share(data-image= page.cover|| theme.avatar.img data-sites= theme.sharejs.sites)
link(rel="stylesheet" href=url_for(theme.CDN.sharejs_css))
script(src=url_for(theme.CDN.sharejs))

View File

@@ -1,25 +1,18 @@
- let showToc = is_post() && page.toc !== false && theme.toc.enable && (toc(page.content) !== '' || page.encrypt == true ) if theme.menu
- let autoOpenSidebar = theme.auto_open_sidebar.enable === true ? 'on' : '' #sidebar
#menu-mask
- #sidebar-menus
let tocNumber .avatar-img.text-center
if (page.toc_number !== undefined) tocNumber = page.toc_number img(src=url_for(theme.avatar.img) onerror=`this.onerror=null;this.src='${url_for(theme.error_img.flink)}'` alt="avatar")
else if (theme.toc.number !== undefined) tocNumber = theme.toc.number .site-data.text-center
else tocNumber = true a(href=`${url_for(config.archive_dir)}/`)
- .headline= _p('aside.articles')
.length-num= site.posts.length
if(showToc) a(href=`${url_for(config.tag_dir)}/`)
i.fa.fa-arrow-right#toggle-sidebar(aria-hidden="true" class=autoOpenSidebar) .headline= _p('aside.tags')
#sidebar .length-num= site.tags.length
.sidebar-toc a(href=`${url_for(config.category_dir)}/`)
div.sidebar-toc__title= _p('sidebar.catalog') .headline= _p('aside.categories')
div.sidebar-toc__progress .length-num= site.categories.length
span.progress-notice= _p('sidebar.have_read')
span.progress-num 0 != partial('includes/header/menu_item', {}, { cache: true })
span.progress-percentage %
div.sidebar-toc__progress-bar
if (page.encrypt == true)
div.sidebar-toc__content.toc-div-class(style="display:none")!=toc(page.origin, {list_number: tocNumber})
else
div.sidebar-toc__content!=toc(page.content, {list_number: tocNumber})

View File

@@ -0,0 +1,46 @@
script.
(() => {
const abcjsInit = () => {
const abcjsFn = () => {
setTimeout(() => {
const sheets = document.querySelectorAll(".abc-music-sheet")
for (let i = 0; i < sheets.length; i++) {
const ele = sheets[i]
if (ele.children.length > 0) continue
// Parse parameters from data-params attribute
let params = {}
const dp = ele.getAttribute("data-params")
if (dp) {
try {
params = JSON.parse(dp)
} catch (e) {
console.error("Failed to parse data-params:", e)
}
}
// Merge parsed parameters with the responsive option
// Ensures params content appears before responsive
const options = { ...params, responsive: "resize" }
// Render the music score using ABCJS.renderAbc
ABCJS.renderAbc(ele, ele.textContent, options)
}
}, 100)
}
if (typeof ABCJS === "object") {
abcjsFn()
} else {
btf.getScript("!{url_for(theme.asset.abcjs_basic_js)}").then(abcjsFn)
}
}
if (window.pjax) {
abcjsInit()
} else {
window.addEventListener("load", abcjsInit)
}
btf.addGlobalFn("encrypt", abcjsInit, "abcjs")
})()

View File

@@ -0,0 +1,3 @@
if theme.abcjs.enable
if theme.abcjs.per_page && (['post','page'].includes(globalPageType)) || page.abcjs
include ./abcjs.pug

23
layout/includes/third-party/aplayer.pug vendored Normal file
View File

@@ -0,0 +1,23 @@
link(rel='stylesheet' href=url_for(theme.asset.aplayer_css) media="print" onload="this.media='all'")
script(src=url_for(theme.asset.aplayer_js))
script(src=url_for(theme.asset.meting_js))
if theme.pjax.enable
script.
(() => {
const destroyAplayer = () => {
if (window.aplayers) {
for (let i = 0; i < window.aplayers.length; i++) {
if (!window.aplayers[i].options.fixed) {
window.aplayers[i].destroy()
}
}
}
}
const runMetingJS = () => {
typeof loadMeting === 'function' && document.getElementsByClassName('aplayer').length && loadMeting()
}
btf.addGlobalFn('pjaxSend', destroyAplayer, 'destroyAplayer')
btf.addGlobalFn('pjaxComplete', loadMeting, 'runMetingJS')
})()

View File

@@ -1 +0,0 @@
script#canvas_nest(color=theme.canvas_nest.color opacity=theme.canvas_nest.opacity zIndex=theme.canvas_nest.zIndex count=theme.canvas_nest.count mobile=`${theme.canvas_nest.mobile}` src=url_for(theme.CDN.canvas_nest))

View File

@@ -1 +0,0 @@
script(id="ribbon_piao" mobile=`${theme.canvas_ribbon_piao.mobile}` src=url_for(theme.CDN.canvas_ribbon_piao))

View File

@@ -1,3 +0,0 @@
script(defer id="ribbon" src=url_for(theme.CDN.canvas_ribbon) size=theme.canvas_ribbon.size
alpha=theme.canvas_ribbon.alpha zIndex=theme.canvas_ribbon.zIndex mobile=`${theme.canvas_ribbon.mobile}` data-click=`${theme.canvas_ribbon.click_to_change}`)

View File

@@ -0,0 +1,31 @@
- const { server, site } = theme.artalk
script.
(() => {
const getArtalkCount = async() => {
try {
const eleGroup = document.querySelectorAll('#recent-posts .artalk-count')
const keyArray = Array.from(eleGroup).map(i => i.getAttribute('data-page-key'))
const headerList = {
method: 'GET',
}
const searchParams = new URLSearchParams({
'site_name': '!{site}',
'page_keys': keyArray
})
const res = await fetch(`!{server}/api/v2/stats/page_comment?${searchParams}`, headerList)
const result = await res.json()
keyArray.forEach((key, index) => {
eleGroup[index].textContent = result.data[key] || 0
})
} catch (err) {
console.error(err)
}
}
window.pjax ? getArtalkCount() : window.addEventListener('load', getArtalkCount)
})()

View File

@@ -0,0 +1,25 @@
- const { shortname, apikey } = theme.disqus
script.
(() => {
const getCount = async () => {
try {
const eleGroup = document.querySelectorAll('#recent-posts .disqus-count')
const cleanedLinks = Array.from(eleGroup).map(i => `thread:link=${i.href.replace(/#post-comment$/, '')}`);
const res = await fetch(`https://disqus.com/api/3.0/threads/set.json?forum=!{shortname}&api_key=!{apikey}&${cleanedLinks.join('&')}`,{
method: 'GET'
})
const result = await res.json()
eleGroup.forEach(i => {
const cleanedLink = i.href.replace(/#post-comment$/, '')
const urlData = result.response.find(data => data.link === cleanedLink) || { posts: 0 }
i.textContent = urlData.posts
})
} catch (err) {
console.error(err)
}
}
window.pjax ? getCount() : window.addEventListener('load', getCount)
})()

View File

@@ -0,0 +1,18 @@
- const fbSDKVer = 'v20.0'
- const fbSDK = `https://connect.facebook.net/${theme.facebook_comments.lang}/sdk.js#xfbml=1&version=${fbSDKVer}`
script.
(()=>{
function loadFBComment () {
if (typeof FB === 'object') FB.XFBML.parse(document.getElementById('recent-posts'))
else {
let ele = document.createElement('script')
ele.setAttribute('src','!{fbSDK}')
ele.setAttribute('async', 'true')
ele.setAttribute('defer', 'true')
ele.setAttribute('crossorigin', 'anonymous')
document.body.appendChild(ele)
}
}
window.pjax ? loadFBComment() : window.addEventListener('load', loadFBComment)
})()

View File

@@ -0,0 +1,16 @@
case theme.comments.use[0]
when 'Twikoo'
include ./twikoo.pug
when 'Disqus'
when 'Disqusjs'
include ./disqus.pug
when 'Valine'
include ./valine.pug
when 'Waline'
include ./waline.pug
when 'Facebook Comments'
include ./fb.pug
when 'Remark42'
include ./remark42.pug
when 'Artalk'
include ./artalk.pug

View File

@@ -0,0 +1,18 @@
- const { host, siteId, option } = theme.remark42
script.
(()=>{
window.remark_config = Object.assign({
host: '!{host}',
site_id: '!{siteId}',
},!{JSON.stringify(option)})
function getCount () {
const s = document.createElement('script')
s.src = remark_config.host + '/web/counter.js'
s.defer = true
document.head.appendChild(s)
}
window.pjax ? getCount() : window.addEventListener('load', getCount)
})()

View File

@@ -0,0 +1,39 @@
script.
(() => {
const getCommentUrl = () => {
const eleGroup = document.querySelectorAll('#recent-posts .article-title')
let urlArray = []
eleGroup.forEach(i=>{
urlArray.push(i.getAttribute('href'))
})
return urlArray
}
const getCount = () => {
const runTwikoo = () => {
twikoo.getCommentsCount({
envId: '!{theme.twikoo.envId}',
region: '!{theme.twikoo.region}',
urls: getCommentUrl(),
includeReply: false
}).then(function (res) {
document.querySelectorAll('#recent-posts .twikoo-count').forEach((item,index) => {
if (res[index]) {
item.textContent = res[index].count
}
})
}).catch(function (err) {
console.log(err)
})
}
if (typeof twikoo === 'object') {
runTwikoo()
} else {
btf.getScript('!{url_for(theme.asset.twikoo)}').then(runTwikoo)
}
}
window.pjax ? getCount() : window.addEventListener('load', getCount)
})()

View File

@@ -0,0 +1,20 @@
script.
(() => {
function loadValine () {
function initValine () {
let initData = {
el: '#vcomment',
appId: '#{theme.valine.appId}',
appKey: '#{theme.valine.appKey}',
serverURLs: '#{theme.valine.serverURLs}'
}
const valine = new Valine(initData)
}
if (typeof Valine === 'function') initValine()
else btf.getScript('!{url_for(theme.asset.valine)}').then(initValine)
}
window.pjax ? loadValine() : window.addEventListener('load', loadValine)
})()

Some files were not shown because too many files have changed in this diff Show More