dev e3e1d9e6ceff cached
211 files
537.2 KB
162.9k tokens
26 symbols
1 requests
Download .txt
Showing preview only (587K chars total). Download the full file or copy to clipboard to get everything.
Repository: jerryc127/hexo-theme-butterfly
Branch: dev
Commit: e3e1d9e6ceff
Files: 211
Total size: 537.2 KB

Directory structure:
gitextract_p7rv7pkl/

├── .github/
│   ├── FUNDING.yml
│   ├── ISSUE_TEMPLATE/
│   │   ├── bug_report.yml
│   │   ├── config.yml
│   │   └── feature_request.yml
│   └── workflows/
│       ├── publish.yml
│       └── stale.yml
├── .gitignore
├── LICENSE
├── README.md
├── README_CN.md
├── _config.yml
├── languages/
│   ├── default.yml
│   ├── en.yml
│   ├── ja.yml
│   ├── ko.yml
│   ├── zh-CN.yml
│   ├── zh-HK.yml
│   └── zh-TW.yml
├── layout/
│   ├── archive.pug
│   ├── category.pug
│   ├── includes/
│   │   ├── additional-js.pug
│   │   ├── footer.pug
│   │   ├── head/
│   │   │   ├── Open_Graph.pug
│   │   │   ├── analytics.pug
│   │   │   ├── config.pug
│   │   │   ├── config_site.pug
│   │   │   ├── google_adsense.pug
│   │   │   ├── preconnect.pug
│   │   │   ├── pwa.pug
│   │   │   ├── site_verification.pug
│   │   │   └── structured_data.pug
│   │   ├── head.pug
│   │   ├── header/
│   │   │   ├── index.pug
│   │   │   ├── menu_item.pug
│   │   │   ├── nav.pug
│   │   │   ├── post-info.pug
│   │   │   └── social.pug
│   │   ├── layout.pug
│   │   ├── loading/
│   │   │   ├── fullpage-loading.pug
│   │   │   ├── index.pug
│   │   │   └── pace.pug
│   │   ├── mixins/
│   │   │   ├── article-sort.pug
│   │   │   └── indexPostUI.pug
│   │   ├── page/
│   │   │   ├── 404.pug
│   │   │   ├── categories.pug
│   │   │   ├── default-page.pug
│   │   │   ├── flink.pug
│   │   │   ├── shuoshuo.pug
│   │   │   └── tags.pug
│   │   ├── pagination.pug
│   │   ├── post/
│   │   │   ├── outdate-notice.pug
│   │   │   ├── post-copyright.pug
│   │   │   └── reward.pug
│   │   ├── rightside.pug
│   │   ├── sidebar.pug
│   │   ├── third-party/
│   │   │   ├── abcjs/
│   │   │   │   ├── abcjs.pug
│   │   │   │   └── index.pug
│   │   │   ├── aplayer.pug
│   │   │   ├── card-post-count/
│   │   │   │   ├── artalk.pug
│   │   │   │   ├── disqus.pug
│   │   │   │   ├── fb.pug
│   │   │   │   ├── index.pug
│   │   │   │   ├── remark42.pug
│   │   │   │   ├── twikoo.pug
│   │   │   │   ├── valine.pug
│   │   │   │   └── waline.pug
│   │   │   ├── chat/
│   │   │   │   ├── chatra.pug
│   │   │   │   ├── crisp.pug
│   │   │   │   ├── index.pug
│   │   │   │   └── tidio.pug
│   │   │   ├── comments/
│   │   │   │   ├── artalk.pug
│   │   │   │   ├── disqus.pug
│   │   │   │   ├── disqusjs.pug
│   │   │   │   ├── facebook_comments.pug
│   │   │   │   ├── giscus.pug
│   │   │   │   ├── gitalk.pug
│   │   │   │   ├── index.pug
│   │   │   │   ├── js.pug
│   │   │   │   ├── livere.pug
│   │   │   │   ├── remark42.pug
│   │   │   │   ├── twikoo.pug
│   │   │   │   ├── utterances.pug
│   │   │   │   ├── valine.pug
│   │   │   │   └── waline.pug
│   │   │   ├── effect.pug
│   │   │   ├── math/
│   │   │   │   ├── chartjs.pug
│   │   │   │   ├── index.pug
│   │   │   │   ├── katex.pug
│   │   │   │   ├── mathjax.pug
│   │   │   │   └── mermaid.pug
│   │   │   ├── newest-comments/
│   │   │   │   ├── artalk.pug
│   │   │   │   ├── common.pug
│   │   │   │   ├── disqus-comment.pug
│   │   │   │   ├── github-issues.pug
│   │   │   │   ├── index.pug
│   │   │   │   ├── remark42.pug
│   │   │   │   ├── twikoo-comment.pug
│   │   │   │   ├── valine.pug
│   │   │   │   └── waline.pug
│   │   │   ├── pjax.pug
│   │   │   ├── prismjs.pug
│   │   │   ├── search/
│   │   │   │   ├── algolia.pug
│   │   │   │   ├── docsearch.pug
│   │   │   │   ├── index.pug
│   │   │   │   └── local-search.pug
│   │   │   ├── share/
│   │   │   │   ├── addtoany.pug
│   │   │   │   ├── index.pug
│   │   │   │   └── share-js.pug
│   │   │   ├── subtitle.pug
│   │   │   └── umami_analytics.pug
│   │   └── widget/
│   │       ├── card_ad.pug
│   │       ├── card_announcement.pug
│   │       ├── card_archives.pug
│   │       ├── card_author.pug
│   │       ├── card_bottom_self.pug
│   │       ├── card_categories.pug
│   │       ├── card_newest_comment.pug
│   │       ├── card_post_series.pug
│   │       ├── card_post_toc.pug
│   │       ├── card_recent_post.pug
│   │       ├── card_tags.pug
│   │       ├── card_top_self.pug
│   │       ├── card_webinfo.pug
│   │       └── index.pug
│   ├── index.pug
│   ├── page.pug
│   ├── post.pug
│   └── tag.pug
├── package.json
├── plugins.yml
├── scripts/
│   ├── common/
│   │   ├── default_config.js
│   │   └── postDesc.js
│   ├── events/
│   │   ├── 404.js
│   │   ├── cdn.js
│   │   ├── init.js
│   │   ├── stylus.js
│   │   └── welcome.js
│   ├── filters/
│   │   ├── post_lazyload.js
│   │   └── random_cover.js
│   ├── helpers/
│   │   ├── aside_archives.js
│   │   ├── aside_categories.js
│   │   ├── getArchiveLength.js
│   │   ├── inject_head_js.js
│   │   ├── page.js
│   │   ├── related_post.js
│   │   └── series.js
│   └── tag/
│       ├── button.js
│       ├── chartjs.js
│       ├── flink.js
│       ├── gallery.js
│       ├── hide.js
│       ├── inlineImg.js
│       ├── label.js
│       ├── mermaid.js
│       ├── note.js
│       ├── score.js
│       ├── series.js
│       ├── tabs.js
│       └── timeline.js
└── source/
    ├── css/
    │   ├── _global/
    │   │   ├── function.styl
    │   │   └── index.styl
    │   ├── _highlight/
    │   │   ├── highlight/
    │   │   │   ├── diff.styl
    │   │   │   └── index.styl
    │   │   ├── highlight.styl
    │   │   ├── prismjs/
    │   │   │   ├── diff.styl
    │   │   │   ├── index.styl
    │   │   │   └── line-number.styl
    │   │   └── theme.styl
    │   ├── _layout/
    │   │   ├── aside.styl
    │   │   ├── chat.styl
    │   │   ├── comments.styl
    │   │   ├── footer.styl
    │   │   ├── head.styl
    │   │   ├── loading.styl
    │   │   ├── pagination.styl
    │   │   ├── post.styl
    │   │   ├── relatedposts.styl
    │   │   ├── reward.styl
    │   │   ├── rightside.styl
    │   │   ├── sidebar.styl
    │   │   └── third-party.styl
    │   ├── _mode/
    │   │   ├── darkmode.styl
    │   │   └── readmode.styl
    │   ├── _page/
    │   │   ├── 404.styl
    │   │   ├── archives.styl
    │   │   ├── categories.styl
    │   │   ├── common.styl
    │   │   ├── flink.styl
    │   │   ├── homepage.styl
    │   │   ├── shuoshuo.styl
    │   │   └── tags.styl
    │   ├── _search/
    │   │   ├── algolia.styl
    │   │   ├── index.styl
    │   │   └── local-search.styl
    │   ├── _tags/
    │   │   ├── button.styl
    │   │   ├── gallery.styl
    │   │   ├── hexo.styl
    │   │   ├── hide.styl
    │   │   ├── inlineImg.styl
    │   │   ├── label.styl
    │   │   ├── note.styl
    │   │   ├── series.styl
    │   │   ├── tabs.styl
    │   │   └── timeline.styl
    │   ├── index.styl
    │   └── var.styl
    └── js/
        ├── main.js
        ├── search/
        │   ├── algolia.js
        │   └── local-search.js
        ├── tw_cn.js
        └── utils.js

================================================
FILE CONTENTS
================================================

================================================
FILE: .github/FUNDING.yml
================================================
# 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']


================================================
FILE: .github/ISSUE_TEMPLATE/bug_report.yml
================================================
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

================================================
FILE: .github/ISSUE_TEMPLATE/config.yml
================================================
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'



================================================
FILE: .github/ISSUE_TEMPLATE/feature_request.yml
================================================
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

================================================
FILE: .github/workflows/publish.yml
================================================
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 }}

================================================
FILE: .github/workflows/stale.yml
================================================
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

================================================
FILE: .gitignore
================================================
.DS_Store
node_modules/


================================================
FILE: LICENSE
================================================

                                 Apache License
                           Version 2.0, January 2004
                        http://www.apache.org/licenses/

   TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION

   1. Definitions.

      "License" shall mean the terms and conditions for use, reproduction,
      and distribution as defined by Sections 1 through 9 of this document.

      "Licensor" shall mean the copyright owner or entity authorized by
      the copyright owner that is granting the License.

      "Legal Entity" shall mean the union of the acting entity and all
      other entities that control, are controlled by, or are under common
      control with that entity. For the purposes of this definition,
      "control" means (i) the power, direct or indirect, to cause the
      direction or management of such entity, whether by contract or
      otherwise, or (ii) ownership of fifty percent (50%) or more of the
      outstanding shares, or (iii) beneficial ownership of such entity.

      "You" (or "Your") shall mean an individual or Legal Entity
      exercising permissions granted by this License.

      "Source" form shall mean the preferred form for making modifications,
      including but not limited to software source code, documentation
      source, and configuration files.

      "Object" form shall mean any form resulting from mechanical
      transformation or translation of a Source form, including but
      not limited to compiled object code, generated documentation,
      and conversions to other media types.

      "Work" shall mean the work of authorship, whether in Source or
      Object form, made available under the License, as indicated by a
      copyright notice that is included in or attached to the work
      (an example is provided in the Appendix below).

      "Derivative Works" shall mean any work, whether in Source or Object
      form, that is based on (or derived from) the Work and for which the
      editorial revisions, annotations, elaborations, or other modifications
      represent, as a whole, an original work of authorship. For the purposes
      of this License, Derivative Works shall not include works that remain
      separable from, or merely link (or bind by name) to the interfaces of,
      the Work and Derivative Works thereof.

      "Contribution" shall mean any work of authorship, including
      the original version of the Work and any modifications or additions
      to that Work or Derivative Works thereof, that is intentionally
      submitted to Licensor for inclusion in the Work by the copyright owner
      or by an individual or Legal Entity authorized to submit on behalf of
      the copyright owner. For the purposes of this definition, "submitted"
      means any form of electronic, verbal, or written communication sent
      to the Licensor or its representatives, including but not limited to
      communication on electronic mailing lists, source code control systems,
      and issue tracking systems that are managed by, or on behalf of, the
      Licensor for the purpose of discussing and improving the Work, but
      excluding communication that is conspicuously marked or otherwise
      designated in writing by the copyright owner as "Not a Contribution."

      "Contributor" shall mean Licensor and any individual or Legal Entity
      on behalf of whom a Contribution has been received by Licensor and
      subsequently incorporated within the Work.

   2. Grant of Copyright License. Subject to the terms and conditions of
      this License, each Contributor hereby grants to You a perpetual,
      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
      copyright license to reproduce, prepare Derivative Works of,
      publicly display, publicly perform, sublicense, and distribute the
      Work and such Derivative Works in Source or Object form.

   3. Grant of Patent License. Subject to the terms and conditions of
      this License, each Contributor hereby grants to You a perpetual,
      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
      (except as stated in this section) patent license to make, have made,
      use, offer to sell, sell, import, and otherwise transfer the Work,
      where such license applies only to those patent claims licensable
      by such Contributor that are necessarily infringed by their
      Contribution(s) alone or by combination of their Contribution(s)
      with the Work to which such Contribution(s) was submitted. If You
      institute patent litigation against any entity (including a
      cross-claim or counterclaim in a lawsuit) alleging that the Work
      or a Contribution incorporated within the Work constitutes direct
      or contributory patent infringement, then any patent licenses
      granted to You under this License for that Work shall terminate
      as of the date such litigation is filed.

   4. Redistribution. You may reproduce and distribute copies of the
      Work or Derivative Works thereof in any medium, with or without
      modifications, and in Source or Object form, provided that You
      meet the following conditions:

      (a) You must give any other recipients of the Work or
          Derivative Works a copy of this License; and

      (b) You must cause any modified files to carry prominent notices
          stating that You changed the files; and

      (c) You must retain, in the Source form of any Derivative Works
          that You distribute, all copyright, patent, trademark, and
          attribution notices from the Source form of the Work,
          excluding those notices that do not pertain to any part of
          the Derivative Works; and

      (d) If the Work includes a "NOTICE" text file as part of its
          distribution, then any Derivative Works that You distribute must
          include a readable copy of the attribution notices contained
          within such NOTICE file, excluding those notices that do not
          pertain to any part of the Derivative Works, in at least one
          of the following places: within a NOTICE text file distributed
          as part of the Derivative Works; within the Source form or
          documentation, if provided along with the Derivative Works; or,
          within a display generated by the Derivative Works, if and
          wherever such third-party notices normally appear. The contents
          of the NOTICE file are for informational purposes only and
          do not modify the License. You may add Your own attribution
          notices within Derivative Works that You distribute, alongside
          or as an addendum to the NOTICE text from the Work, provided
          that such additional attribution notices cannot be construed
          as modifying the License.

      You may add Your own copyright statement to Your modifications and
      may provide additional or different license terms and conditions
      for use, reproduction, or distribution of Your modifications, or
      for any such Derivative Works as a whole, provided Your use,
      reproduction, and distribution of the Work otherwise complies with
      the conditions stated in this License.

   5. Submission of Contributions. Unless You explicitly state otherwise,
      any Contribution intentionally submitted for inclusion in the Work
      by You to the Licensor shall be under the terms and conditions of
      this License, without any additional terms or conditions.
      Notwithstanding the above, nothing herein shall supersede or modify
      the terms of any separate license agreement you may have executed
      with Licensor regarding such Contributions.

   6. Trademarks. This License does not grant permission to use the trade
      names, trademarks, service marks, or product names of the Licensor,
      except as required for reasonable and customary use in describing the
      origin of the Work and reproducing the content of the NOTICE file.

   7. Disclaimer of Warranty. Unless required by applicable law or
      agreed to in writing, Licensor provides the Work (and each
      Contributor provides its Contributions) on an "AS IS" BASIS,
      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
      implied, including, without limitation, any warranties or conditions
      of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
      PARTICULAR PURPOSE. You are solely responsible for determining the
      appropriateness of using or redistributing the Work and assume any
      risks associated with Your exercise of permissions under this License.

   8. Limitation of Liability. In no event and under no legal theory,
      whether in tort (including negligence), contract, or otherwise,
      unless required by applicable law (such as deliberate and grossly
      negligent acts) or agreed to in writing, shall any Contributor be
      liable to You for damages, including any direct, indirect, special,
      incidental, or consequential damages of any character arising as a
      result of this License or out of the use or inability to use the
      Work (including but not limited to damages for loss of goodwill,
      work stoppage, computer failure or malfunction, or any and all
      other commercial damages or losses), even if such Contributor
      has been advised of the possibility of such damages.

   9. Accepting Warranty or Additional Liability. While redistributing
      the Work or Derivative Works thereof, You may choose to offer,
      and charge a fee for, acceptance of support, warranty, indemnity,
      or other liability obligations and/or rights consistent with this
      License. However, in accepting such obligations, You may act only
      on Your own behalf and on Your sole responsibility, not on behalf
      of any other Contributor, and only if You agree to indemnify,
      defend, and hold each Contributor harmless for any liability
      incurred by, or claims asserted against, such Contributor by reason
      of your accepting any such warranty or additional liability.

   END OF TERMS AND CONDITIONS

   APPENDIX: How to apply the Apache License to your work.

      To apply the Apache License to your work, attach the following
      boilerplate notice, with the fields enclosed by brackets "[]"
      replaced with your own identifying information. (Don't include
      the brackets!)  The text should be enclosed in the appropriate
      comment syntax for the file format. We also recommend that a
      file or class name and description of purpose be included on the
      same "printed page" as the copyright notice for easier
      identification within third-party archives.

   Copyright [yyyy] [name of copyright owner]

   Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
   You may obtain a copy of the License at

       http://www.apache.org/licenses/LICENSE-2.0

   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.

================================================
FILE: README.md
================================================
<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

A modern, elegant and feature-rich theme for Hexo

![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)

📢 **Demo**: [Butterfly Official](https://butterfly.js.org/) | [CrazyWong's Blog](https://blog.crazywong.com/)

📖 **Documentation**: [English Docs](https://butterfly.js.org/en/posts/butterfly-docs-en-get-started/) | [中文文档](https://butterfly.js.org/posts/21cfbf15/)

![Butterfly Theme Preview](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/theme-butterfly-readme.png)

</div>

---

## 🚀 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
```

```bash
# Install development version (early access to new features)
git clone -b dev https://github.com/jerryc127/hexo-theme-butterfly.git themes/butterfly
```

#### Method 2: NPM Installation

> ⚠️ **Note**: NPM installation only supports Hexo 5.0.0 and above

```bash
npm install hexo-theme-butterfly
```

### ⚙️ Theme Configuration

1. **Enable Theme**: Modify your Hexo configuration file `_config.yml`:

```yaml
theme: butterfly
```

2. **Install Dependencies**: If you haven't installed pug and stylus renderers, please run:

```bash
npm install hexo-renderer-pug hexo-renderer-stylus --save
```

## ✨ Theme Features

### 🎨 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

### 📝 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>


================================================
FILE: README_CN.md
================================================
<div align="right">
  <a title="English" href="/README.md">English</a>
</div>

<div align="center">

<img src="./source/img/butterfly-icon.png" width="150" height="150" alt="Butterfly Logo" />

# hexo-theme-butterfly

一個適用於 Hexo 的現代化、美觀且功能豐富的主題

![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)

📢 **在線預覽**: [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/)

![Butterfly 主題預覽](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/theme-butterfly-readme.png)

</div>

---

## 🚀 快速開始

### 💾 安裝方式

#### 方式一:Git 安裝(推薦)

> 💡 **提示**: 如果您在中國大陸訪問 GitHub 速度較慢,可以使用 [Gitee 鏡像](https://gitee.com/immyw/hexo-theme-butterfly.git)

在您的 Hexo 博客根目錄下執行:

```bash
# 安裝穩定版本(推薦)
git clone -b master https://github.com/jerryc127/hexo-theme-butterfly.git themes/butterfly
```

```bash
# 安裝開發版本(搶先體驗新功能)
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: _config.yml
================================================
# --------------------------------------
# Hexo Butterfly Theme Configuration
# If you have any questions, please refer to the documentation
# Chinese: https://butterfly.js.org/
# English: https://butterfly.js.org/en/
# --------------------------------------

# --------------------------------------
# Navigation Settings
# --------------------------------------

nav:
  # Navigation bar logo image
  logo:
  display_title: true
  display_post_title: true
  # Whether to fix navigation bar
  fixed: false

menu:
  # Home: / || fas fa-home
  # List||fas fa-list:
  #   Music: /music/ || fas fa-music
  #   Movie: /movies/ || fas fa-video

# --------------------------------------
# Code Blocks Settings
# --------------------------------------

code_blocks:
  # Code block theme: darker / pale night / light / ocean / false
  theme: light
  macStyle: false
  # Code block height limit (unit: px)
  height_limit: false
  word_wrap: false

  # Toolbar
  copy: true
  language: true
  # true: shrink the code blocks | false: expand the code blocks | none: expand code blocks and hide the button
  shrink: false
  fullpage: false

# Social media links
# Formal:
#   icon: link || the description || color
social:
  # fab fa-github: https://github.com/xxxxx || Github || '#24292e'
  # fas fa-envelope: mailto:xxxxxx@gmail.com || Email || '#4a7dbe'

# --------------------------------------
# Image Settings
# --------------------------------------

favicon: /img/favicon.png

avatar:
  img: /img/butterfly-icon.png
  effect: false

# Disable all banner images
disable_top_img: false

# If the banner of page not setting, it will show the default_top_img
default_top_img:

# The banner image of index page
index_img:

# The banner image of archive page
archive_img:

# Note: tag page, not tags page
tag_img:

# The banner image of tag page, you can set the banner image for each tag
# Format:
#  - tag name: xxxxx
tag_per_img:

# Note: category page, not categories page
category_img:

# The banner image of category page, you can set the banner image for each category
# Format:
#  - category name: xxxxx
category_per_img:

# The background image of footer
footer_img: false

# Website Background
# Can set it to color, image URL or an array containing colors and/or image URLs
# If an array is provided, a random background will be selected from the array on each load
background:

cover:
  # Disable the cover or not
  index_enable: true
  aside_enable: true
  archives_enable: true
  # When cover is not set, the default cover is displayed
  default_cover:
    # - xxx.jpg

# Replace Broken Images
error_img:
  flink: /img/friend_404.gif
  post_page: /img/404.jpg

# A simple 404 page
error_404:
  enable: false
  subtitle: 'Page Not Found'
  background: /img/error-page.png

post_meta:
  # Home Page
  page:
    # Choose: created / updated / both
    date_type: created
    # Choose: date / relative
    date_format: date
    categories: true
    tags: false
    label: true
  post:
    # Choose: left / center
    position: left
    # Choose: created / updated / both
    date_type: both
    # Choose: date / relative
    date_format: date
    categories: true
    tags: true
    label: true

# --------------------------------------
# Index page settings
# --------------------------------------

# The top_img settings of home page
# default: top img - full screen, site info - middle
# The position of site info, eg: 300px/300em/300rem/10%
index_site_info_top:
# The height of top_img, eg: 300px/300em/300rem
index_top_img_height:

# The subtitle on homepage
subtitle:
  enable: false
  # Typewriter Effect
  effect: true
  # Customize typed.js
  # https://github.com/mattboldt/typed.js/#customization
  typed_option:
  # Source - Call the third-party service API (Chinese only)
  # It will show the source first, then show the content of sub
  # Choose: false/1/2/3
  # false - disable the function
  # 1 - hitokoto.cn
  # 2 - https://api.aa1.cn/doc/yiyan.html
  # 3 - jinrishici.com
  source: false
  # If you close the typewriter effect, the subtitle will only show the first line of sub
  sub:

# Article layout on the homepage
# 1: Cover on the left, info on the right
# 2: Cover on the right, info on the left
# 3: Cover and info alternate between left and right
# 4: Cover on top, info on the bottom
# 5: Info displayed on the cover
# 6: Masonry layout - Cover on top, info on the bottom
# 7: Masonry layout - Info displayed on the cover
index_layout: 3

# Display the article introduction on homepage
# 1: description
# 2: both (if the description exists, it will show description, or show the auto_excerpt)
# 3: auto_excerpt (default)
# false: do not show the article introduction
index_post_content:
  method: 3
  # If you set method to 2 or 3, the length need to config
  length: 500

# --------------------------------------
# Post Settings
# --------------------------------------

toc:
  post: true
  page: false
  number: true
  expand: false
  # Only for post
  style_simple: false
  scroll_percent: true

post_copyright:
  enable: true
  decode: false
  author_href:
  license: CC BY-NC-SA 4.0
  license_url: https://creativecommons.org/licenses/by-nc-sa/4.0/

# Sponsor/reward
reward:
  enable: false
  text:
  QR_code:
    # - img: /img/wechat.jpg
    #   link:
    #   text: wechat
    # - img: /img/alipay.jpg
    #   link:
    #   text: alipay

# Post edit
# Easily browse and edit blog source code online.
post_edit:
  enable: false
  # url: https://github.com/user-name/repo-name/edit/branch-name/subdirectory-name/
  # For example: https://github.com/jerryc127/butterfly.js.org/edit/main/source/
  url:

# Related Articles
related_post:
  enable: true
  # Number of posts displayed
  limit: 6
  # Choose: created / updated
  date_type: created

# Choose: 1 / 2 / false
# 1: The 'next post' will link to old post
# 2: The 'next post' will link to new post
# false: disable pagination
post_pagination: 1

# Displays outdated notice for a post
noticeOutdate:
  enable: false
  # Style: simple / flat
  style: flat
  # When will it be shown
  limit_day: 365
  # Position: top / bottom
  position: top
  message_prev: It has been
  message_next: days since the last update, the content of the article may be outdated.

# --------------------------------------
# Footer Settings
# --------------------------------------
footer:
  nav:
  owner:
    enable: true
    since: 2025
  # Copyright of theme and framework
  copyright:
    enable: true
    version: true
  custom_text:

# --------------------------------------
# Aside Settings
# --------------------------------------

aside:
  enable: true
  hide: false
  # Show the button to hide the aside in bottom right button
  button: true
  mobile: true
  # Position: left / right
  position: right
  display:
    archive: true
    tag: true
    category: true
  card_author:
    enable: true
    description:
    button:
      enable: true
      icon: fab fa-github
      text: Follow Me
      link: https://github.com/xxxxxx
  card_announcement:
    enable: true
    content: This is my Blog
  card_recent_post:
    enable: true
    # If set 0 will show all
    limit: 5
    # Sort: date / updated
    sort: date
    sort_order:
  card_newest_comments:
    enable: false
    sort_order:
    limit: 6
    # Unit: mins, save data to localStorage
    storage: 10
    avatar: true
  card_categories:
    enable: true
    # If set 0 will show all
    limit: 8
    # Choose: none / true / false
    expand: none
    sort_order:
  card_tags:
    enable: true
    # If set 0 will show all
    limit: 40
    color: false
    custom_colors:
    # Order of tags, random/name/length
    orderby: random
    # Sort of order. 1, asc for ascending; -1, desc for descending
    order: 1
    sort_order:
  card_archives:
    enable: true
    # Type: monthly / yearly
    type: monthly
    # Eg: YYYY年MM月
    format: MMMM YYYY
    # Sort of order. 1, asc for ascending; -1, desc for descending
    order: -1
    # If set 0 will show all
    limit: 8
    sort_order:
  card_post_series:
    enable: true
    # The title shows the series name
    series_title: false
    # Order by title or date
    orderBy: 'date'
    # Sort of order. 1, asc for ascending; -1, desc for descending
    order: -1
  card_webinfo:
    enable: true
    post_count: true
    last_push_date: true
    sort_order:
    # Time difference between publish date and now
    # Formal: Month/Day/Year Time or Year/Month/Day Time
    # Leave it empty if you don't enable this feature
    runtime_date:

# --------------------------------------
# Bottom right button
# --------------------------------------

# The distance between the bottom right button and the bottom (default unit: px)
rightside_bottom:

# Conversion between Traditional and Simplified Chinese
translate:
  enable: false
  # The text of a button
  default: 繁
  # the language of website (1 - Traditional Chinese/ 2 - Simplified Chinese)
  defaultEncoding: 2
  # Time delay
  translateDelay: 0
  # The text of the button when the language is Simplified Chinese
  msgToTraditionalChinese: '繁'
  # The text of the button when the language is Traditional Chinese
  msgToSimplifiedChinese: '簡'

# Read Mode
readmode: true

# Dark Mode
darkmode:
  enable: true
  # Toggle Button to switch dark/light mode
  button: true
  # Switch dark/light mode automatically
  # autoChangeMode: 1  Following System Settings, if the system doesn't support dark mode, it will switch dark mode between 6 pm to 6 am
  # autoChangeMode: 2  Switch dark mode between 6 pm to 6 am
  # autoChangeMode: false
  autoChangeMode: false
  # Set the light mode time. The value is between 0 and 24. If not set, the default value is 6 and 18
  start:
  end:

# Show scroll percent in scroll-to-top button
rightside_scroll_percent: false

# Don't modify the following settings unless you know how they work
# Choose: readmode,translate,darkmode,hideAside,toc,chat,comment
# Don't repeat the same value
rightside_item_order:
  enable: false
  # Default: readmode,translate,darkmode,hideAside
  hide:
  # Default: toc,chat,comment
  show:

# Animation for the bottom right config button
rightside_config_animation: true

# --------------------------------------
# Global Settings
# --------------------------------------

anchor:
  # When you scroll, the URL will update according to header id.
  auto_update: false
  # Click the headline to scroll and update the anchor
  click_to_scroll: false

photofigcaption: false

copy:
  enable: true
  # Add the copyright information after copied content
  copyright:
    enable: false
    limit_count: 150

# Need to install the hexo-wordcount plugin
wordcount:
  enable: false
  # Display the word count of the article in post meta
  post_wordcount: true
  # Display the time to read the article in post meta
  min2read: true
  # Display the total word count of the website in aside's webinfo
  total_wordcount: true

# Busuanzi count for PV / UV in site
busuanzi:
  site_uv: true
  site_pv: true
  page_pv: true

# --------------------------------------
# Math
# --------------------------------------

# About the per_page
# if you set it to true, it will load mathjax/katex script in each page
# if you set it to false, it will load mathjax/katex script according to your setting (add the 'mathjax: true' or 'katex: true' in page's front-matter)
math:
  # Choose: mathjax, katex
  # Leave it empty if you don't need math
  use:
  per_page: true
  hide_scrollbar: false

  mathjax:
    # Enable the contextual menu
    enableMenu: true
    # Choose: all / ams / none, This controls whether equations are numbered and how
    tags: none

  katex:
    # Enable the copy KaTeX formula
    copy_tex: false

# --------------------------------------
# Search
# --------------------------------------

search:
  # Choose: algolia_search / local_search / docsearch
  # leave it empty if you don't need search
  use:
  placeholder:

  # Algolia Search
  algolia_search:
    # Number of search results per page
    hitsPerPage: 6

  # Local Search
  local_search:
    # Preload the search data when the page loads.
    preload: false
    # Show top n results per article, show all results by setting to -1
    top_n_per_article: 1
    # Unescape html strings to the readable one.
    unescape: false
    # Enable pagination for search results
    pagination:
      enable: false
      # Number of search results per page
      hitsPerPage: 8
    CDN:

  # Docsearch
  # https://docsearch.algolia.com/
  docsearch:
    appId:
    apiKey:
    indexName:
    option:

# --------------------------------------
# Share System
# --------------------------------------

share:
  # Choose: sharejs / addtoany
  # Leave it empty if you don't need share
  use: sharejs

  # Share.js
  # https://github.com/overtrue/share.js
  sharejs:
    sites: facebook,x,wechat,weibo,qq

  # AddToAny
  # https://www.addtoany.com/
  addtoany:
    item: facebook,x,wechat,sina_weibo,facebook_messenger,email,copy_link

# --------------------------------------
# Comments System
# --------------------------------------

comments:
  # Up to two comments system, the first will be shown as default
  # Leave it empty if you don't need comments
  # Choose: Disqus/Disqusjs/Livere/Gitalk/Valine/Waline/Utterances/Facebook Comments/Twikoo/Giscus/Remark42/Artalk
  # Format of two comments system : Disqus,Waline
  use:
  # Display the comment name next to the button
  text: true
  # Lazyload: The comment system will be load when comment element enters the browser's viewport.
  # If you set it to true, the comment count will be invalid
  lazyload: false
  # Display comment count in post's top_img
  count: false
  # Display comment count in Home Page
  card_post_count: false

# Disqus
# https://disqus.com/
disqus:
  shortname:
  # For newest comments widget
  apikey:

# Alternative Disqus - Render comments with Disqus API
# https://github.com/SukkaW/DisqusJS
disqusjs:
  shortname:
  apikey:
  option:

# Livere
# https://www.livere.com/
livere:
  uid:

# Gitalk
# https://github.com/gitalk/gitalk
gitalk:
  client_id:
  client_secret:
  repo:
  owner:
  admin:
  option:

# Valine
# https://valine.js.org
valine:
  appId:
  appKey:
  avatar: monsterid
  # This configuration is suitable for domestic custom domain name users, overseas version will be automatically detected (no need to manually fill in)
  serverURLs:
  bg:
  # Use Valine visitor count as the page view count
  visitor: false
  option:

# Waline - A simple comment system with backend support fork from Valine
# https://waline.js.org/
waline:
  serverURL:
  bg:
  # Use Waline pageview count as the page view count
  pageview: false
  option:

# Utterances
# https://utteranc.es/
utterances:
  repo:
  # Issue Mapping: pathname/url/title/og:title
  issue_term: pathname
  # Theme: github-light/github-dark/github-dark-orange/icy-dark/dark-blue/photon-dark
  light_theme: github-light
  dark_theme: photon-dark
  js:
  option:

# Facebook Comments Plugin
# https://developers.facebook.com/docs/plugins/comments/
facebook_comments:
  app_id:
  # optional
  user_id:
  pageSize: 10
  # Choose: social / time / reverse_time
  order_by: social
  lang: en_US

# Twikoo
# https://github.com/imaegoo/twikoo
twikoo:
  envId:
  region:
  # Use Twikoo visitor count as the page view count
  visitor: false
  option:

# Giscus
# https://giscus.app/
giscus:
  repo:
  repo_id:
  category_id:
  light_theme: light
  dark_theme: dark
  js:
  option:

# Remark42
# https://remark42.com/docs/configuration/frontend/
remark42:
  host:
  siteId:
  option:

# Artalk
# https://artalk.js.org/guide/frontend/config.html
artalk:
  server:
  site:
  # Use Artalk visitor count as the page view count
  visitor: false
  option:

# --------------------------------------
# Chat Services
# --------------------------------------

chat:
  # Choose: chatra/tidio/crisp
  # Leave it empty if you don't need chat
  use:
  # Chat Button [recommend]
  # It will create a button in the bottom right corner of website, and hide the origin button
  rightside_button: false
  # The origin chat button is displayed when scrolling up, and the button is hidden when scrolling down
  button_hide_show: false

# https://chatra.io/
chatra:
  id:

# https://www.tidio.com/
tidio:
  public_key:

# https://crisp.chat/en/
crisp:
  website_id:

# --------------------------------------
# Analysis
# --------------------------------------

# https://tongji.baidu.com/web/welcome/login
baidu_analytics:

# https://analytics.google.com/analytics/web/
google_analytics:

# https://www.cloudflare.com/zh-tw/web-analytics/
cloudflare_analytics:

# https://clarity.microsoft.com/
microsoft_clarity:

# https://umami.is/
umami_analytics:
  enable: false
  # For self-hosted setups, configure the hostname of the Umami instance
  serverURL:
  script_name: script.js
  website_id:
  option:
  UV_PV:
    site_uv: false
    site_pv: false
    page_pv: false
    # Umami Cloud (API key) / self-hosted Umami (token)
    token:

# https://www.googletagmanager.com/
google_tag_manager:
  tag_id:
  # optional
  domain:

# --------------------------------------
# Advertisement
# --------------------------------------

# Google Adsense
google_adsense:
  enable: false
  auto_ads: true
  js: https://pagead2.googlesyndication.com/pagead/js/adsbygoogle.js
  client:
  enable_page_level_ads: true

# Insert ads manually
# Leave it empty if you don't need ads
ad:
  # Insert ads in the index (every three posts)
  index:
  # Insert ads in aside
  aside:
  # Insert ads in the post (before pagination)
  post:

# --------------------------------------
# Verification
# --------------------------------------

site_verification:
  # - name: google-site-verification
  #   content: xxxxxx
  # - name: baidu-site-verification
  #   content: xxxxxxx

# --------------------------------------
# Beautify / Effect
# --------------------------------------

# Theme color for customize
# Notice: color value must in double quotes like "#000" or may cause error!

# theme_color:
#   enable: true
#   main: "#49B1F5"
#   paginator: "#00c4b6"
#   button_hover: "#FF7242"
#   text_selection: "#00c4b6"
#   link_color: "#99a9bf"
#   meta_color: "#858585"
#   hr_color: "#A4D8FA"
#   code_foreground: "#F47466"
#   code_background: "rgba(27, 31, 35, .05)"
#   toc_color: "#00c4b6"
#   blockquote_padding_color: "#49b1f5"
#   blockquote_background_color: "#49b1f5"
#   scrollbar_color: "#49b1f5"
#   meta_theme_color_light: "ffffff"
#   meta_theme_color_dark: "#0d0d0d"

# The user interface setting of category and tag page
# Choose: index - same as Homepage UI / default - same as archives UI
# leave it empty or index
category_ui:
tag_ui:

# Rounded corners for UI elements
rounded_corners_ui: true

# Stretches the lines so that each line has equal width
text_align_justify: false

# Add a mask to the header and footer
mask:
  header: true
  footer: true

# Loading Animation
preloader:
  enable: false
  # source
  # 1. fullpage-loading
  # 2. pace (progress bar)
  source: 1
  # pace theme (see https://codebyzach.github.io/pace/)
  pace_css_url:

# Page Transition
enter_transitions: true

# Default display mode - light (default) / dark
display_mode: light

# Configuration for beautifying the content of the article
beautify:
  enable: false
  # Specify the field to beautify (site or post)
  field: post
  # Specify the icon to be used as a prefix for the title, such as '\f0c1'
  title_prefix_icon:
  # Specify the color of the title prefix icon, such as '#F47466'
  title_prefix_icon_color:

# Global font settings
# Don't modify the following settings unless you know how they work
font:
  global_font_size:
  code_font_size:
  font_family:
  code_font_family:

# Font settings for the site title and site subtitle
blog_title_font:
  font_link:
  font_family:

# The setting of divider icon
hr_icon:
  enable: true
  # The unicode value of Font Awesome icon, such as '\3423'
  icon:
  icon_top:

# Typewriter Effect
# https://github.com/disjukr/activate-power-mode
activate_power_mode:
  enable: false
  colorful: true
  shake: true
  mobile: false

# Background effects
# --------------------------------------

# canvas_ribbon
# See: https://github.com/hustcc/ribbon.js
canvas_ribbon:
  enable: false
  # The size of ribbon
  size: 150
  # The opacity of ribbon (0 ~ 1)
  alpha: 0.6
  zIndex: -1
  click_to_change: false
  mobile: false

# Fluttering Ribbon
canvas_fluttering_ribbon:
  enable: false
  mobile: false

# canvas_nest
# https://github.com/hustcc/canvas-nest.js
canvas_nest:
  enable: false
  # Color of lines, default: '0,0,0'; RGB values: (R,G,B).(note: use ',' to separate.)
  color: '0,0,255'
  # The opacity of line (0~1)
  opacity: 0.7
  # The z-index property of the background
  zIndex: -1
  # The number of lines
  count: 99
  mobile: false

# Mouse click effects: fireworks
fireworks:
  enable: false
  zIndex: 9999
  mobile: false

# Mouse click effects: Heart symbol
click_heart:
  enable: false
  mobile: false

# Mouse click effects: words
clickShowText:
  enable: false
  text:
    # - I
    # - LOVE
    # - YOU
  fontSize: 15px
  random: false
  mobile: false

# --------------------------------------
# Lightbox Settings
# --------------------------------------

# Choose: fancybox / medium_zoom
# https://github.com/francoischalifour/medium-zoom
# https://fancyapps.com/fancybox/
# Leave it empty if you don't need lightbox
lightbox:

# --------------------------------------
# Tag Plugins settings
# --------------------------------------

# Series
series:
  enable: false
  # Order by title or date
  orderBy: 'title'
  # Sort of order. 1, asc for ascending; -1, desc for descending
  order: 1
  number: true

# ABCJS - The ABC Music Notation Plugin
# https://github.com/paulrosen/abcjs
abcjs:
  enable: false
  per_page: true

# Mermaid
# https://github.com/mermaid-js/mermaid
mermaid:
  enable: false
  # Write Mermaid diagrams using code blocks
  code_write: false
  # built-in themes: default / forest / dark / neutral
  theme:
    light: default
    dark: dark
  # Enable "Open in New Tab" button to view diagram in a separate window
  open_in_new_tab: true
  # Enable zoom and pan interactions on diagrams
  zoom_pan: true

# chartjs
# see https://www.chartjs.org/docs/latest/
chartjs:
  enable: false
  # Do not modify unless you understand how they work.
  # The default settings are only used when the MD syntax is not specified.
  # General font color for the chart
  fontColor:
    light: 'rgba(0, 0, 0, 0.8)'
    dark: 'rgba(255, 255, 255, 0.8)'
  # General border color for the chart
  borderColor:
    light: 'rgba(0, 0, 0, 0.1)'
    dark: 'rgba(255, 255, 255, 0.2)'
  # Background color for scale labels on radar and polar area charts
  scale_ticks_backdropColor:
    light: 'transparent'
    dark: 'transparent'

# Note - Bootstrap Callout
note:
  # Note tag style values:
  #  - simple    bs-callout old alert style. Default.
  #  - modern    bs-callout new (v2-v3) alert style.
  #  - flat      flat callout style with background, like on Mozilla or StackOverflow.
  #  - disabled  disable all CSS styles import of note tag.
  style: flat
  icons: true
  border_radius: 3
  # Offset lighter of background in % for modern and flat styles (modern: -12 | 12; flat: -18 | 6).
  # Offset also applied to label tag variables. This option can work with disabled note tag.
  light_bg_offset: 0

# --------------------------------------
# Other Settings
# --------------------------------------

# https://github.com/MoOx/pjax
pjax:
  enable: false
  # Exclude the specified pages from pjax, such as '/music/'
  exclude:
    # - /xxxxxx/

# Inject the css and script (aplayer/meting)
aplayerInject:
  enable: false
  per_page: true

# Snackbar - Toast Notification
# https://github.com/polonel/SnackBar
# position: top-left / top-center / top-right / bottom-left / bottom-center / bottom-right
snackbar:
  enable: false
  position: bottom-left
  # The background color of Toast Notification in light mode and dark mode
  bg_light: '#49b1f5'
  bg_dark: '#1f1f1f'

# Instant.page
# https://instant.page/
instantpage: false

# Lazyload
# https://github.com/verlok/vanilla-lazyload
lazyload:
  enable: false
  # Use browser's native lazyload instead of vanilla-lazyload
  native: false
  # Specify the field to use lazyload (site or post)
  field: site
  placeholder:
  blur: false

# PWA
# See https://github.com/JLHwung/hexo-offline
# ---------------
pwa:
  enable: false
  manifest:
  apple_touch_icon:
  favicon_32_32:
  favicon_16_16:
  mask_icon:

# Open graph meta tags
# https://hexo.io/docs/helpers#open-graph
Open_Graph_meta:
  enable: true
  option:
    # twitter_card:
    # twitter_image:
    # twitter_id:
    # twitter_site:
    # google_plus:
    # fb_admins:
    # fb_app_id:

# Structured Data
# https://developers.google.com/search/docs/guides/intro-structured-data
structured_data:
  enable: false
  # Alternate name for the site, used in structured data
  # Format: ['name1', 'name2']
  alternate_name:

# Add the vendor prefixes to ensure compatibility
css_prefix: true

# Inject
# Insert the code to head (before '</head>' tag) and the bottom (before '</body>' tag)
inject:
  head:
    # - <link rel="stylesheet" href="/xxx.css">
  bottom:
    # - <script src="xxxx"></script>

# CDN Settings
# Don't modify the following settings unless you know how they work
CDN:
  # The CDN provider for internal and third-party scripts
  # Options for both: local/jsdelivr/unpkg/cdnjs/custom
  # Note: Dev version can only use 'local' for internal scripts
  # Note: When setting third-party scripts to 'local', you need to install hexo-butterfly-extjs
  internal_provider: local
  third_party_provider: jsdelivr

  # Add version number to url, true or false
  version: true

  # Custom format
  # For example: https://cdn.staticfile.org/${cdnjs_name}/${version}/${min_cdnjs_file}
  custom_format:

  option:
    # abcjs_basic_js:
    # activate_power_mode:
    # algolia_js:
    # algolia_search:
    # aplayer_css:
    # aplayer_js:
    # artalk_css:
    # artalk_js:
    # blueimp_md5:
    # busuanzi:
    # canvas_fluttering_ribbon:
    # canvas_nest:
    # canvas_ribbon:
    # chartjs:
    # click_heart:
    # clickShowText:
    # disqusjs:
    # disqusjs_css:
    # docsearch_css:
    # docsearch_js:
    # egjs_infinitegrid:
    # fancybox:
    # fancybox_css:
    # fireworks:
    # fontawesome:
    # gitalk:
    # gitalk_css:
    # giscus:
    # instantpage:
    # katex:
    # katex_copytex:
    # lazyload:
    # local_search:
    # main:
    # main_css:
    # mathjax:
    # medium_zoom:
    # mermaid:
    # meting_js:
    # prismjs_autoloader:
    # prismjs_js:
    # prismjs_lineNumber_js:
    # pjax:
    # sharejs:
    # sharejs_css:
    # snackbar:
    # snackbar_css:
    # translate:
    # twikoo:
    # typed:
    # utils:
    # valine:
    # waline_css:
    # waline_js:


================================================
FILE: languages/default.yml
================================================
footer:
  framework: Framework
  theme: Theme

copy:
  success: Copy Successful
  error: Copy Failed
  noSupport: Browser Not Supported

page:
  articles: All Articles
  tag: Tag
  category: Category
  archives: Archives

card_post_count: comments

no_title: Untitled

post:
  created: Created
  updated: Updated
  wordcount: Word Count
  min2read: Reading Time
  min2read_unit: mins
  page_pv: Post Views
  comments: Comments
  copyright:
    author: Author
    link: Link
    copyright_notice: Copyright Notice
    copyright_content: 'All articles on this blog are licensed under <a href="%s">%s</a> unless otherwise stated.'
  recommend: Related Articles
  edit: Edit
  back_to_home: Back to Home

search:
  title: Search
  load_data: Loading Database
  input_placeholder: Search for Posts
  algolia_search:
    hits_empty: 'No results found for: ${query}'
    hits_stats: '${hits} results found in ${time} ms'
  local_search:
    hits_empty: 'No results found for: ${query}'
    hits_stats: '${hits} articles found'

pagination:
  prev: Previous
  next: Next
  page_info: 'Page ${current} of ${total}' 

comment: Comments

aside:
  articles: Articles
  tags: Tags
  categories: Categories
  card_announcement: Announcement
  card_categories: Categories
  card_tags: Tags
  card_archives: Archives
  card_recent_post: Recent Posts
  card_webinfo:
    headline: Website Info
    article_name: Article Count
    runtime:
      name: Runtime
      unit: days
    last_push_date:
      name: Last Update
    site_wordcount: Total Word Count
    site_uv_name: Unique Visitors
    site_pv_name: Page Views
  more_button: View More
  card_newest_comments:
    headline: Latest Comments
    loading_text: Loading...
    error: Unable to retrieve comments, please check the configuration
    zero: No comments
    image: Image
    link: Link
    code: Code
  card_toc: Contents
  card_post_series: Post Series

date_suffix:
  just: Just now
  min: minutes ago
  hour: hours ago
  day: days ago
  month: months ago

donate: Sponsor
share: Share

rightside:
  readmode_title: Reading Mode
  translate_title: Toggle Between Traditional and Simplified Chinese
  night_mode_title: Toggle Between Light and Dark Mode
  back_to_top: Back to Top
  toc: Table of Contents
  scroll_to_comment: Scroll to Comments
  setting: Settings
  aside: Toggle Between Single-column and Double-column
  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


================================================
FILE: languages/en.yml
================================================
footer:
  framework: Framework
  theme: Theme

copy:
  success: Copy Successful
  error: Copy Failed
  noSupport: Browser Not Supported

page:
  articles: All Articles
  tag: Tag
  category: Category
  archives: Archives

card_post_count: comments

no_title: Untitled

post:
  created: Created
  updated: Updated
  wordcount: Word Count
  min2read: Reading Time
  min2read_unit: mins
  page_pv: Post Views
  comments: Comments
  copyright:
    author: Author
    link: Link
    copyright_notice: Copyright Notice
    copyright_content: 'All articles on this blog are licensed under <a href="%s">%s</a> unless otherwise stated.'
  recommend: Related Articles
  edit: Edit
  back_to_home: Back to Home

search:
  title: Search
  load_data: Loading Database
  input_placeholder: Search for Posts
  algolia_search:
    hits_empty: 'No results found for: ${query}'
    hits_stats: '${hits} results found in ${time} ms'
  local_search:
    hits_empty: 'No results found for: ${query}'
    hits_stats: '${hits} articles found'

pagination:
  prev: Previous
  next: Next
  page_info: 'Page ${current} of ${total}'

comment: Comments

aside:
  articles: Articles
  tags: Tags
  categories: Categories
  card_announcement: Announcement
  card_categories: Categories
  card_tags: Tags
  card_archives: Archives
  card_recent_post: Recent Posts
  card_webinfo:
    headline: Website Info
    article_name: Article Count
    runtime:
      name: Runtime
      unit: days
    last_push_date:
      name: Last Update
    site_wordcount: Total Word Count
    site_uv_name: Unique Visitors
    site_pv_name: Page Views
  more_button: View More
  card_newest_comments:
    headline: Latest Comments
    loading_text: Loading...
    error: Unable to retrieve comments, please check the configuration
    zero: No comments
    image: Image
    link: Link
    code: Code
  card_toc: Contents
  card_post_series: Post Series

date_suffix:
  just: Just now
  min: minutes ago
  hour: hours ago
  day: days ago
  month: months ago

donate: Sponsor
share: Share

rightside:
  readmode_title: Reading Mode
  translate_title: Toggle Between Traditional and Simplified Chinese
  night_mode_title: Toggle Between Light and Dark Mode
  back_to_top: Back to Top
  toc: Table of Contents
  scroll_to_comment: Scroll to Comments
  setting: Settings
  aside: Toggle Between Single-column and Double-column
  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


================================================
FILE: languages/ja.yml
================================================
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: ページが見つかりません


================================================
FILE: languages/ko.yml
================================================
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: 페이지를 찾을 수 없습니다.


================================================
FILE: languages/zh-CN.yml
================================================
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" target="_blank">%s</a> 许可协议。转载请注明来源 <a href="%s" target="_blank">%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: 页面未找到


================================================
FILE: languages/zh-HK.yml
================================================
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: 未找到頁面


================================================
FILE: languages/zh-TW.yml
================================================
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" target="_blank">%s</a> 授權協議。轉載請註明來源 <a href="%s" target="_blank">%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: 找不到頁面


================================================
FILE: layout/archive.pug
================================================
extends includes/layout.pug

block content
  include ./includes/mixins/article-sort.pug
  #archive
    .article-sort-title= `${_p('page.articles')} - ${getArchiveLength()}`
    +articleSort(page.posts)
    include includes/pagination.pug

================================================
FILE: layout/category.pug
================================================
extends includes/layout.pug

block content
  if theme.category_ui == 'index'
    include ./includes/mixins/indexPostUI.pug
    +indexPostUI
  else
    include ./includes/mixins/article-sort.pug
    #category
      .article-sort-title= _p('page.category') + ' - ' + page.category
      +articleSort(page.posts)
      include includes/pagination.pug

================================================
FILE: layout/includes/additional-js.pug
================================================
div
  script(src=url_for(theme.asset.utils))
  script(src=url_for(theme.asset.main))

  if theme.translate.enable
    script(src=url_for(theme.asset.translate))

  if theme.lightbox
    script(src=url_for(theme.asset[theme.lightbox]))

  if theme.instantpage
    script(src=url_for(theme.asset.instantpage), type='module')

  if theme.lazyload.enable && !theme.lazyload.native
    script(src=url_for(theme.asset.lazyload))

  if theme.snackbar.enable
    script(src=url_for(theme.asset.snackbar))

  .js-pjax
    if needLoadCountJs
      != partial("includes/third-party/card-post-count/index", {}, { cache: true })

    if loadSubJs
      include ./third-party/subtitle.pug

    include ./third-party/math/index.pug
    include ./third-party/abcjs/index.pug

    if commentsJsLoad
      include ./third-party/comments/js.pug

  != partial("includes/third-party/prismjs", {}, { cache: true })

  if theme.aside.enable && theme.aside.card_newest_comments.enable
    if theme.pjax.enable || (globalPageType !== 'post' && page.aside !== false)
      != partial("includes/third-party/newest-comments/index", {}, { cache: true })

  != fragment_cache('injectBottom', function(){return injectHtml(theme.inject.bottom)})

  != partial("includes/third-party/effect", {}, { cache: true })
  != partial("includes/third-party/chat/index", {}, { cache: true })

  if theme.aplayerInject && theme.aplayerInject.enable
    if theme.pjax.enable || theme.aplayerInject.per_page || page.aplayer
      include ./third-party/aplayer.pug

  if theme.pjax.enable
    != partial("includes/third-party/pjax", {}, { cache: true })

  if theme.umami_analytics.enable
    != partial("includes/third-party/umami_analytics", {}, { cache: true })

  if theme.busuanzi.site_uv || theme.busuanzi.site_pv || theme.busuanzi.page_pv
    script(async data-pjax src=theme.asset.busuanzi ? url_for(theme.asset.busuanzi) : '//busuanzi.ibruce.info/busuanzi/2.3/busuanzi.pure.mini.js')

  != partial('includes/third-party/search/index', {}, { cache: true })

  if theme.google_tag_manager && theme.google_tag_manager.tag_id
    noscript
      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")

================================================
FILE: layout/includes/footer.pug
================================================
- const { nav, owner, copyright, custom_text } = theme.footer

if nav
  .footer-flex
    for block in nav
      .footer-flex-items(style=`${ block.width ? 'flex-grow:' + block.width : '' }`)
        for blockItem in block.content
          .footer-flex-item
            .footer-flex-title= blockItem.title
            .footer-flex-content
              for subitem in blockItem.item
                if subitem.html
                  div!= subitem.html
                else if subitem.url
                  a(href=url_for(subitem.url), target='_blank' title=subitem.title)= subitem.title
                else if subitem.title
                  div!= subitem.title
.footer-other
  .footer-copyright
    if owner.enable
      - const currentYear = new Date().getFullYear()
      - const sinceYear = owner.since
      span.copyright
        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


================================================
FILE: layout/includes/head/Open_Graph.pug
================================================
if theme.Open_Graph_meta.enable
  -
    const coverVal = page.cover_type === 'img' ? page.cover : theme.avatar.img
    let ogOption = Object.assign({
      type: globalPageType === 'post' ? 'article' : 'website',
      image: coverVal ? full_url_for(coverVal) : '',
      fb_admins: theme.facebook_comments.user_id || '',
      fb_app_id: theme.facebook_comments.app_id || '',
    }, theme.Open_Graph_meta.option)
  -
  != open_graph(ogOption)
else
  - const description = page.description || page.content || page.title || config.description
  if description
    meta(name="description" content=truncate(description, 150))



================================================
FILE: layout/includes/head/analytics.pug
================================================
if theme.baidu_analytics
  script.
    var _hmt = _hmt || [];
    (function() {
      var hm = document.createElement("script");
      hm.src = "https://hm.baidu.com/hm.js?!{theme.baidu_analytics}";
      var s = document.getElementsByTagName("script")[0];
      s.parentNode.insertBefore(hm, s);
    })();
    btf.addGlobalFn('pjaxComplete', () => {
      _hmt.push(['_trackPageview',window.location.pathname])
    }, 'baidu_analytics')

if theme.google_analytics
  script(async src=`https://www.googletagmanager.com/gtag/js?id=${theme.google_analytics}`)
  script.
    window.dataLayer = window.dataLayer || []
    function gtag(){dataLayer.push(arguments)}
    gtag('js', new Date())
    gtag('config', '!{theme.google_analytics}')
    btf.addGlobalFn('pjaxComplete', () => {
      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')

================================================
FILE: layout/includes/head/config.pug
================================================
-
  let algolia = 'undefined'
  if (theme.search.use === 'algolia_search') {
    const { ALGOLIA_APP_ID, ALGOLIA_API_KEY, ALGOLIA_INDEX_NAME } = process.env
    const { appId, applicationID, apiKey, indexName } = config.algolia
    algolia = JSON.stringify({
      appId: ALGOLIA_APP_ID || appId || applicationID,
      apiKey: ALGOLIA_API_KEY || apiKey,
      indexName: ALGOLIA_INDEX_NAME || indexName,
      hitsPerPage: theme.search.algolia_search.hitsPerPage,
      // search languages
      languages: {
        hits_empty: _p("search.algolia_search.hits_empty"),
        hits_stats: _p("search.algolia_search.hits_stats"),
      }
    })
  }

  let localSearch = 'undefined'
  if (theme.search.use === 'local_search') {
    const { CDN, preload, top_n_per_article, pagination, unescape } = theme.search.local_search
    localSearch = JSON.stringify({
      path: CDN || config.root + config.search.path,
      preload,
      top_n_per_article,
      unescape,
      pagination: {
        enable: pagination.enable,
        hitsPerPage: pagination.hitsPerPage
      },
      languages: {
        // search languages
        hits_empty: _p("search.local_search.hits_empty"),
        hits_stats: _p("search.local_search.hits_stats"),
      }
    })
  }

  let translate = 'undefined'
  if (theme.translate && theme.translate.enable){
    translate = JSON.stringify({
      defaultEncoding: theme.translate.defaultEncoding,
      translateDelay: theme.translate.translateDelay,
      msgToTraditionalChinese: theme.translate.msgToTraditionalChinese,
      msgToSimplifiedChinese: theme.translate.msgToSimplifiedChinese
    })
  }

  let copyright = 'undefined'
  if (theme.copy.enable && theme.copy.copyright.enable){
    copyright = JSON.stringify({
      limitCount: theme.copy.copyright.limit_count,
      languages: {
        author: _p("copy_copyright.author") + ': ' + config.author,
        link: _p("copy_copyright.link") + ': ',
        source: _p("copy_copyright.source") + ': ' + config.title,
        info: _p("copy_copyright.info")
      }
    })
  }

  let Snackbar = 'undefined'
  if (theme.snackbar && theme.snackbar.enable) {
    Snackbar = JSON.stringify({
      chs_to_cht: _p("Snackbar.chs_to_cht"),
      cht_to_chs: _p("Snackbar.cht_to_chs"),
      day_to_night: _p("Snackbar.day_to_night"),
      night_to_day: _p("Snackbar.night_to_day"),
      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({
    plugin: highlightProvider,
    highlightCopy: copy,
    highlightLang: language,
    highlightHeightLimit: height_limit,
    highlightFullpage: fullpage,
    highlightMacStyle: macStyle
  })

script.
  const GLOBAL_CONFIG = {
    root: '!{config.root}',
    algolia: !{algolia},
    localSearch: !{localSearch},
    translate: !{translate},
    highlight: !{highlight},
    copy: {
      success: '!{_p("copy.success")}',
      error: '!{_p("copy.error")}',
      noSupport: '!{_p("copy.noSupport")}'
    },
    relativeDate: {
      homepage: !{theme.post_meta.page.date_format === 'relative'},
      post: !{theme.post_meta.post.date_format === 'relative'}
    },
    runtime: '!{theme.aside.card_webinfo.runtime_date ? _p("aside.card_webinfo.runtime.unit") : ""}',
    dateSuffix: {
      just: '!{_p("date_suffix.just")}',
      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}
  }


================================================
FILE: layout/includes/head/config_site.pug
================================================
-
  const titleVal = pageTitle.replace(/'/ig,"\\'")

  let isHighlightShrink
  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}'
  }


================================================
FILE: layout/includes/head/google_adsense.pug
================================================
if (theme.google_adsense && theme.google_adsense.enable)
  script(async src=theme.google_adsense.js)

  if theme.google_adsense.auto_ads
    script.
      (adsbygoogle = window.adsbygoogle || []).push({
        google_ad_client: '!{theme.google_adsense.client}',
        enable_page_level_ads: '!{theme.google_adsense.enable_page_level_ads}'
      });

================================================
FILE: layout/includes/head/preconnect.pug
================================================
-
  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")

================================================
FILE: layout/includes/head/pwa.pug
================================================
- const { manifest, theme_color, apple_touch_icon, favicon_32_32, favicon_16_16, mask_icon } = theme.pwa

link(rel="manifest" href=url_for(manifest))
if theme_color
  meta(name="msapplication-TileColor" content=theme_color)
if apple_touch_icon
  link(rel="apple-touch-icon" sizes="180x180" href=url_for(apple_touch_icon))
if favicon_32_32
  link(rel="icon" type="image/png" sizes="32x32" href=url_for(favicon_32_32))
if favicon_16_16
  link(rel="icon" type="image/png" sizes="16x16" href=url_for(favicon_16_16))
if mask_icon
  link(rel="mask-icon" href=url_for(mask_icon) color="#5bbad5")


================================================
FILE: layout/includes/head/site_verification.pug
================================================
if theme.site_verification
  each item in theme.site_verification
    meta(name=item.name content=item.content)

================================================
FILE: layout/includes/head/structured_data.pug
================================================
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}


================================================
FILE: layout/includes/head.pug
================================================
- var pageTitle
- globalPageType === 'archive' ? page.title = findArchivesTitle(page, theme.menu, date) : ''
case globalPageType
  when 'tag'
    - pageTitle = _p('page.tag') + ': ' + page.tag
  when 'category'
    - pageTitle = _p('page.category') + ': ' + page.category
  when '404'
    - pageTitle = _p('error404')
  default
    - pageTitle = page.title || config.title || ''


- var isSubtitle = config.subtitle ? ' - ' + config.subtitle : ''
- var tabTitle = globalPageType === 'home' || !pageTitle ? config.title + isSubtitle : pageTitle + ' | ' + config.title
- var pageAuthor = config.email ? config.author + ',' + config.email : 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 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(name="viewport" content="width=device-width, initial-scale=1.0,viewport-fit=cover")
title= tabTitle
meta(name="author" content=pageAuthor)
meta(name="copyright" content=pageCopyright)
meta(name ="format-detection" content="telephone=no")
meta(name="theme-color" content=themeColor)

//- Open_Graph
include ./head/Open_Graph.pug

//- Structured Data
include ./head/structured_data.pug

!=favicon_tag(theme.favicon || config.favicon)
link(rel="canonical" href=urlNoIndex(null,config.pretty_urls.trailing_index,config.pretty_urls.trailing_html))

//- 預解析
!=partial('includes/head/preconnect', {}, {cache: true})

//- 網站驗證
!=partial('includes/head/site_verification', {}, {cache: true})

//- PWA
if (theme.pwa && theme.pwa.enable)
  !=partial('includes/head/pwa', {}, {cache: true})

//- main css
link(rel='stylesheet', href=url_for(theme.asset.main_css))
link(rel='stylesheet', href=url_for(theme.asset.fontawesome))

if (theme.snackbar && theme.snackbar.enable)
  link(rel='stylesheet', href=url_for(theme.asset.snackbar_css) media="print" onload="this.media='all'")

if theme.lightbox === 'fancybox'
  link(rel='stylesheet' href=url_for(theme.asset.fancybox_css) media="print" onload="this.media='all'")

!=fragment_cache('injectHeadJs', function(){return inject_head_js()})

//- google_adsense
!=partial('includes/head/google_adsense', {}, {cache: true})

//- analytics
!=partial('includes/head/analytics', {}, {cache: true})

//- font
if theme.blog_title_font && theme.blog_title_font.font_link
  link(rel='stylesheet' href=url_for(theme.blog_title_font.font_link) media="print" onload="this.media='all'")

//- global config
!=partial('includes/head/config', {}, {cache: true})

include ./head/config_site.pug

!=fragment_cache('injectHead', function(){return injectHtml(theme.inject.head)})


================================================
FILE: layout/includes/header/index.pug
================================================
-
  const returnTopImg = img => img !== false ? img || theme.default_top_img : false
  const isFixedClass = theme.nav.fixed ? ' fixed' : ''
  var top_img = false
  let headerClassName = 'not-top-img'
  var bg_img = ''

if !theme.disable_top_img && page.top_img !== false
  case globalPageType
    when 'post'
      - top_img = page.top_img || page.cover || theme.default_top_img
    when 'page'
      - top_img = page.top_img || theme.default_top_img
    when 'tag'
      - top_img = theme.tag_per_img && theme.tag_per_img[page.tag] || returnTopImg(theme.tag_img)
    when 'category'
      - top_img = theme.category_per_img && theme.category_per_img[page.category] || returnTopImg(theme.category_img)
    when 'home'
      - top_img = returnTopImg(theme.index_img)
    when 'archive'
      - top_img = returnTopImg(theme.archive_img)
    default
      - top_img = page.top_img || theme.default_top_img

  if top_img !== false
    - bg_img = getBgPath(top_img)
    - headerClassName = globalPageType === 'home' ? 'full_page' : globalPageType === 'post' ? 'post-bg' : 'not-home-page'

header#page-header(class=`${headerClassName + isFixedClass}` style=bg_img)
  include ./nav.pug
  if top_img !== false
    if globalPageType === 'post'
      include ./post-info.pug
    else if globalPageType === 'home'
      #site-info
        h1#site-title=config.title
        if theme.subtitle.enable
          - var loadSubJs = true
          #site-subtitle
            span#subtitle
        if theme.social
          #site_social_icons
            !=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

================================================
FILE: layout/includes/header/menu_item.pug
================================================
if theme.menu
  .menus_items
    each value, label in theme.menu
      if typeof value !== 'object'
        .menus_item
          - const [link, icon] = value.split('||').map(part => trim(part))
          a.site-page(href=url_for(link))
            if icon
              i.fa-fw(class=icon)
            span= ' ' + label
      else
        .menus_item
          - const [groupLabel, groupIcon, groupClass] = label.split('||').map(part => trim(part))
          - const hideClass = groupClass === 'hide' ? 'hide' : ''
          span.site-page.group(class=hideClass)
            if groupIcon
              i.fa-fw(class=groupIcon)
            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

================================================
FILE: layout/includes/header/nav.pug
================================================
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

================================================
FILE: layout/includes/header/post-info.pug
================================================
- let comments = theme.comments
#post-info
  h1.post-title= page.title || _p('no_title')
    if theme.post_edit.enable
      a.post-edit-link(href=theme.post_edit.url + page.source title=_p('post.edit') target="_blank")
        i.fas.fa-pencil-alt

  #post-meta
    .meta-firstline
      if theme.post_meta.post.date_type
        span.post-meta-date
          if theme.post_meta.post.date_type === 'both'
            i.far.fa-calendar-alt.fa-fw.post-meta-icon
            span.post-meta-label= _p('post.created')
            time.post-meta-date-created(datetime=date_xml(page.date) title=_p('post.created') + ' ' + full_date(page.date))= date(page.date, config.date_format)
            span.post-meta-separator |
            i.fas.fa-history.fa-fw.post-meta-icon
            span.post-meta-label= _p('post.updated')
            time.post-meta-date-updated(datetime=date_xml(page.updated) title=_p('post.updated') + ' ' + full_date(page.updated))= date(page.updated, config.date_format)
          else
            - let data_type_update = theme.post_meta.post.date_type === 'updated'
            - let date_type = data_type_update ? 'updated' : 'date'
            - let date_icon = data_type_update ? 'fas fa-history' : 'far fa-calendar-alt'
            - let date_title = data_type_update ? _p('post.updated') : _p('post.created')
            i.fa-fw.post-meta-icon(class=date_icon)
            span.post-meta-label= date_title
            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.categories && page.categories.data.length > 0
        span.post-meta-categories
          if theme.post_meta.post.date_type
            span.post-meta-separator |
          each item, index in page.categories.data
            i.fas.fa-inbox.fa-fw.post-meta-icon
            a(href=url_for(item.path)).post-meta-categories #[=item.name]
            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)
      if postWordcount
        span.post-meta-separator |
        span.post-meta-wordcount
          if theme.wordcount.post_wordcount
            i.far.fa-file-word.fa-fw.post-meta-icon
            span.post-meta-label= _p('post.wordcount') + ':'
            span.word-count= wordcount(page.content)
            if theme.wordcount.min2read
              span.post-meta-separator |
          if theme.wordcount.min2read
            i.far.fa-clock.fa-fw.post-meta-icon
            span.post-meta-label= _p('post.min2read') + ':'
            span= min2read(page.content, {cn: 350, en: 160}) + _p('post.min2read_unit')

      //- for pv and count
      mixin pvBlock(parent_id, parent_class, parent_title)
        span.post-meta-separator |
        span(class=parent_class id=parent_id data-flag-title=parent_title)
          i.far.fa-eye.fa-fw.post-meta-icon
          span.post-meta-label= _p('post.page_pv') + ':'
          if block
            block

      mixin otherPV()
        if theme.umami_analytics.enable && theme.umami_analytics.UV_PV.page_pv
          +pvBlock('', '', '')
            span#umamiPV(data-path=url_for(page.path))
              i.fa-solid.fa-spinner.fa-spin
        else if theme.busuanzi.page_pv
          +pvBlock('', 'post-meta-pv-cv', '')
            span#busuanzi_value_page_pv
              i.fa-solid.fa-spinner.fa-spin

      - const commentUse = comments.use && comments.use[0]
      if page.comments !== false && commentUse && !comments.lazyload
        if commentUse === 'Valine' && theme.valine.visitor
          +pvBlock(url_for(page.path), 'leancloud_visitors', page.title)
            span.leancloud-visitors-count
              i.fa-solid.fa-spinner.fa-spin
        else if commentUse === 'Waline' && theme.waline.pageview
          +pvBlock('', '', '')
            span.waline-pageview-count(data-path=url_for(page.path))
              i.fa-solid.fa-spinner.fa-spin
        else if commentUse === 'Twikoo' && theme.twikoo.visitor
          +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

================================================
FILE: layout/includes/header/social.pug
================================================
each url, icon in theme.social
  -
    const [link, title, color] = url.split('||').map(i => trim(i))
    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)

================================================
FILE: layout/includes/layout.pug
================================================
- var globalPageType = getPageType(page, is_home)
- var htmlClassHideAside = theme.aside.enable && theme.aside.hide ? 'hide-aside' : ''
- page.aside = globalPageType === 'archive' ? theme.aside.display.archive: globalPageType === 'category' ? theme.aside.display.category : globalPageType === 'tag' ? theme.aside.display.tag : page.aside
- var hideAside = !theme.aside.enable || page.aside === false ? 'hide-aside' : ''
- var pageType = globalPageType === 'post' ? 'post' : 'page'
- pageType = page.type ? pageType + ' type-' + page.type : pageType

doctype html
html(lang=config.language data-theme=theme.display_mode class=htmlClassHideAside)
  head
    include ./head.pug
  body
    !=partial('includes/loading/index', {}, {cache: true})

    if theme.background
      if !Array.isArray(theme.background)
        #web_bg.bg-animation(style=getBgPath(theme.background))
      else
        #web_bg.bg-animation
        - const bgStyleArr = theme.background.map(getBgPath)
        script.
          (() => {
            const arr = !{JSON.stringify(bgStyleArr)}
            const webBgDiv = document.getElementById('web_bg')

            const setRandomBg = () => {
              webBgDiv.style = arr[Math.floor(Math.random() * arr.length)]
              requestAnimationFrame(() => webBgDiv.classList.add('bg-animation'))
            }

            document.addEventListener('pjax:send', () => {
              webBgDiv.style = ''
              webBgDiv.classList.remove('bg-animation')
            })

            document.addEventListener('pjax:complete', setRandomBg)
            document.addEventListener('DOMContentLoaded', setRandomBg)
          })()

    !=partial('includes/sidebar', {}, {cache: true})

    #body-wrap(class=pageType)
      include ./header/index.pug

      main#content-inner.layout(class=hideAside)
        if body
          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

================================================
FILE: layout/includes/loading/fullpage-loading.pug
================================================
#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')
    }
  })()

================================================
FILE: layout/includes/loading/index.pug
================================================
if theme.preloader.enable
  if theme.preloader.source === 1
    include ./fullpage-loading.pug
  else
    include ./pace.pug

================================================
FILE: layout/includes/loading/pace.pug
================================================
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))

================================================
FILE: layout/includes/mixins/article-sort.pug
================================================
mixin articleSort(posts)
  .article-sort
    - let year
    - posts.forEach(article => {
      - const tempYear = date(article.date, 'YYYY')
      - const noCoverClass = article.cover === false || !theme.cover.archives_enable ? 'no-article-cover' : ''
      - const title = article.title || _p('no_title')
      if tempYear !== year
        - year = tempYear
        .article-sort-item.year= year
      .article-sort-item(class=noCoverClass)
        if article.cover && theme.cover.archives_enable
          a.article-sort-item-img(href=url_for(article.path) title=title)
            if article.cover_type === 'img'
              img(src=url_for(article.cover) alt=title onerror=`this.onerror=null;this.src='${url_for(theme.error_img.post_page)}'`)
            else
              div(style=`background: ${article.cover}`)
        .article-sort-item-info
          .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
    - })

================================================
FILE: layout/includes/mixins/indexPostUI.pug
================================================
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

================================================
FILE: layout/includes/page/404.pug
================================================
- 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')


================================================
FILE: layout/includes/page/categories.pug
================================================
.category-lists!= list_categories()

================================================
FILE: layout/includes/page/default-page.pug
================================================
#article-container.container
  != page.content

================================================
FILE: layout/includes/page/flink.pug
================================================
#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


================================================
FILE: layout/includes/page/shuoshuo.pug
================================================
//- - 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)
      })()

================================================
FILE: layout/includes/page/tags.pug
================================================
.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})


================================================
FILE: layout/includes/pagination.pug
================================================
if page.total !== 1
  -
    var options = {
      prev_text: '<i class="fas fa-chevron-left fa-fw"></i>',
      next_text: '<i class="fas fa-chevron-right fa-fw"></i>',
      mid_size: 1,
      escape: false
    }

  if globalPageType === 'post'
    - let paginationOrder = theme.post_pagination === 2 ? { prev: page.prev, next: page.next } : { prev: page.next, next: page.prev }

    nav#pagination.pagination-post
      each direction, key in paginationOrder
        if direction
          - const getPostDesc = direction.postDesc || postDesc(direction)
          - let className = key === 'prev' ? (paginationOrder.next ? '' : 'full-width') : (paginationOrder.prev ? '' : 'full-width')
          - className = getPostDesc ? className : className + ' no-desc'

          a.pagination-related(class=className href=url_for(direction.path) title=direction.title)
            if direction.cover_type === 'img'
              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`)
            else
              .cover(style=`background: ${direction.cover || 'var(--default-bg-color)'}`)

            .info(class=key === 'prev' ? '' : 'text-right')
              .info-1
                .info-item-1=_p(`pagination.${key}`)
                .info-item-2!=direction.title
              if getPostDesc
                .info-2
                  .info-item-1!=getPostDesc
  else
    nav#pagination
      .pagination
        if globalPageType === 'home'
          - options.format = 'page/%d/#content-inner'
        !=paginator(options)

================================================
FILE: layout/includes/post/outdate-notice.pug
================================================
- 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)

================================================
FILE: layout/includes/post/post-copyright.pug
================================================
if theme.post_copyright.enable && page.copyright !== false
  - const author = page.copyright_author || config.author
  - const authorHref = page.copyright_author_href || theme.post_copyright.author_href || config.url
  - const url = page.copyright_url || page.permalink
  - const info = page.copyright_info || _p('post.copyright.copyright_content', theme.post_copyright.license_url, theme.post_copyright.license, config.url, config.title)
  .post-copyright
    .post-copyright__author
      span.post-copyright-meta
        i.fas.fa-circle-user.fa-fw
        = _p('post.copyright.author') + ": "
      span.post-copyright-info
        a(href=authorHref)= author
    .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

================================================
FILE: layout/includes/post/reward.pug
================================================
.post-reward
  .reward-button
    i.fas.fa-qrcode
    = theme.reward.text || _p('donate')
  .reward-main
    ul.reward-all
      each item in theme.reward.QR_code
        - const clickTo = item.link || item.img
        li.reward-item
          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

================================================
FILE: layout/includes/rightside.pug
================================================
- const { readmode, translate, darkmode, aside, chat } = theme

mixin rightsideItem(array)
  each item in array
    case item
      when 'readmode'
        if globalPageType === 'post' && readmode
          button#readmode(type="button" title=_p('rightside.readmode_title'))
            i.fas.fa-book-open
      when 'translate'
        if translate.enable
          button#translateLink(type="button" title=_p('rightside.translate_title'))= translate.default
      when 'darkmode'
        if darkmode.enable && darkmode.button
          button#darkmode(type="button" title=_p('rightside.night_mode_title'))
            i.fas.fa-adjust
      when 'hideAside'
        if aside.enable && aside.button && page.aside !== false
          button#hide-aside-btn(type="button" title=_p('rightside.aside'))
            i.fas.fa-arrows-alt-h
      when 'toc'
        if showToc
          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

================================================
FILE: layout/includes/sidebar.pug
================================================
if theme.menu
  #sidebar
    #menu-mask
    #sidebar-menus
      .avatar-img.text-center
        img(src=url_for(theme.avatar.img) onerror=`this.onerror=null;this.src='${url_for(theme.error_img.flink)}'` alt="avatar")
      .site-data.text-center
        a(href=`${url_for(config.archive_dir)}/`)
          .headline= _p('aside.articles')
          .length-num= site.posts.length
        a(href=`${url_for(config.tag_dir)}/`)
          .headline= _p('aside.tags')
          .length-num= site.tags.length
        a(href=`${url_for(config.category_dir)}/`)
          .headline= _p('aside.categories')
          .length-num= site.categories.length

      != partial('includes/header/menu_item', {}, { cache: true })

================================================
FILE: layout/includes/third-party/abcjs/abcjs.pug
================================================
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")
  })()


================================================
FILE: layout/includes/third-party/abcjs/index.pug
================================================
if theme.abcjs.enable
  if theme.abcjs.per_page && (['post','page'].includes(globalPageType)) || page.abcjs
    include ./abcjs.pug


================================================
FILE: layout/includes/third-party/aplayer.pug
================================================
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')
    })()


================================================
FILE: layout/includes/third-party/card-post-count/artalk.pug
================================================
- 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)
  })()

================================================
FILE: layout/includes/third-party/card-post-count/disqus.pug
================================================
- 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)
  })()


================================================
FILE: layout/includes/third-party/card-post-count/fb.pug
================================================
- 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)
  })()


================================================
FILE: layout/includes/third-party/card-post-count/index.pug
================================================
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

================================================
FILE: layout/includes/third-party/card-post-count/remark42.pug
================================================
- 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)
  })()

================================================
FILE: layout/includes/third-party/card-post-count/twikoo.pug
================================================
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)

  })()

================================================
FILE: layout/includes/third-party/card-post-count/valine.pug
================================================
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)
  })()


================================================
FILE: layout/includes/third-party/card-post-count/waline.pug
================================================
- const serverURL = theme.waline.serverURL.replace(/\/$/, '')
script.
  (() => {
    async function loadWaline () {
      try {
        const eleGroup = document.querySelectorAll('#recent-posts .waline-comment-count')
        const keyArray = Array.from(eleGroup).map(i => i.getAttribute('data-path'))

        const res = await fetch(`!{serverURL}/api/comment?type=count&url=${keyArray}`, { method: 'GET' })
        const result = await res.json()
        
        result.data.forEach((count, index) => {
          eleGroup[index].textContent = count
        })
      } catch (err) {
        console.error(err)
      }
    }

    window.pjax ? loadWaline() : window.addEventListener('load', loadWaline)
  })()


================================================
FILE: layout/includes/third-party/chat/chatra.pug
================================================
//- https://chatra.io/help/api/
script.
  (() => {
    window.ChatraID = '#{theme.chatra.id}'
    window.Chatra = window.Chatra || function() {
      (window.Chatra.q = window.Chatra.q || []).push(arguments)
    }

    btf.getScript('https://call.chatra.io/chatra.js').then(() => {
      const isChatBtn = !{theme.chat.rightside_button}
      const isChatHideShow = !{theme.chat.button_hide_show}

      if (isChatBtn) {
        const close = () => {
          Chatra('minimizeWidget')
          Chatra('hide')
        }

        const open = () => {
          Chatra('openChat', true)
          Chatra('show')
        }

        window.ChatraSetup = { startHidden: true }
      
        window.chatBtnFn = () => document.getElementById('chatra').classList.contains('chatra--expanded') ? close() : open()

        document.getElementById('chat-btn').style.display = 'block'
      } else if (isChatHideShow) {
        window.chatBtn = {
          hide: () => Chatra('hide'),
          show: () => Chatra('show')
        }
      }
    })
  })()




================================================
FILE: layout/includes/third-party/chat/crisp.pug
================================================
script.
  (() => {
    window.$crisp = ['safe', true]
    window.CRISP_WEBSITE_ID = "!{theme.crisp.website_id}"

    btf.getScript('https://client.crisp.chat/l.js').then(() => {
      const isChatBtn = !{theme.chat.rightside_button}
      const isChatHideShow = !{theme.chat.button_hide_show}

      if (isChatBtn) {
        const open = () => {
          $crisp.push(["do", "chat:show"])
          $crisp.push(["do", "chat:open"])
        }

        const close = () => $crisp.push(["do", "chat:hide"])

        close()

        $crisp.push(["on", "chat:closed", close])

        window.chatBtnFn = () => $crisp.is("chat:visible") ? close() : open()

        document.getElementById('chat-btn').style.display = 'block'
      } else if (isChatHideShow) {
        window.chatBtn = {
          hide: () => $crisp.push(["do", "chat:hide"]),
          show: () => $crisp.push(["do", "chat:show"])
        }
      }
    })
  })()

================================================
FILE: layout/includes/third-party/chat/index.pug
================================================
case theme.chat.use
  when 'chatra'
    include ./chatra.pug
  when 'tidio'
    include ./tidio.pug
  when 'crisp'
    include ./crisp.pug

================================================
FILE: layout/includes/third-party/chat/tidio.pug
================================================
script.
  (() => {
    btf.getScript('//code.tidio.co/!{theme.tidio.public_key}.js').then(() => {
      const isChatBtn = !{theme.chat.rightside_button}
      const isChatHideShow = !{theme.chat.button_hide_show}

      if (isChatBtn) {
        let isShow = false
        const close = () => {
          window.tidioChatApi.hide()
          isShow = false
        }
        
        const open = () => {
          window.tidioChatApi.open()
          window.tidioChatApi.show()
          isShow = true
        }

        const onTidioChatApiReady = () => {
          window.tidioChatApi.hide()
          window.tidioChatApi.on("close", close)
        }
        if (window.tidioChatApi) {
          window.tidioChatApi.on("ready", onTidioChatApiReady)
        } else {
          document.addEventListener("tidioChat-ready", onTidioChatApiReady)
        }

        window.chatBtnFn = () => {
          if (!window.tidioChatApi) return
          isShow ? close() : open()
        }

        document.getElementById('chat-btn').style.display = 'block'

      } else if (isChatHideShow) {
        window.chatBtn = {
          hide: () => window.tidioChatApi && window.tidioChatApi.hide(),
          show: () => window.tidioChatApi && window.tidioChatApi.show()
        }
      }
    })
  })()



================================================
FILE: layout/includes/third-party/comments/artalk.pug
================================================
- const { server, site, option } = theme.artalk
- const { use, lazyload } = theme.comments

script.
  (() => {
    let artalkItem = null
    const option = !{JSON.stringify(option)}
    const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'

    const destroyArtalk = () => {
      if (artalkItem) {
        artalkItem.destroy()
        artalkItem = null
      }
    }

    const artalkChangeMode = theme => artalkItem && artalkItem.setDarkMode(theme === 'dark')

    const initArtalk = (el = document, pageKey = location.pathname) => {
      artalkItem = Artalk.init({
        el: el.querySelector('#artalk-wrap'),
        server: '!{server}',
        site: '!{site}',
        darkMode: document.documentElement.getAttribute('data-theme') === 'dark',
        ...option,
        pageKey: isShuoshuo ? pageKey : (option && option.pageKey) || pageKey
      })

      if (GLOBAL_CONFIG.lightbox === 'null') return
      artalkItem.on('list-loaded', () => {
        artalkItem.ctx.get('list').getCommentNodes().forEach(comment => {
          const $content = comment.getRender().$content
          btf.loadLightbox($content.querySelectorAll('img:not([atk-emoticon])'))
        })
      })

      if (isShuoshuo) {
        window.shuoshuoComment.destroyArtalk = () => {
          destroyArtalk()
          if (el.children.length) {
            el.innerHTML = ''
            el.classList.add('no-comment')
          }
        }
      }

      btf.addGlobalFn('pjaxSendOnce', destroyArtalk, 'destroyArtalk')
      btf.addGlobalFn('themeChange', artalkChangeMode, 'artalk')
    }

    const loadArtalk = async (el, pageKey) => {
      if (typeof Artalk === 'object') initArtalk(el, pageKey)
      else {
        await btf.getCSS('!{url_for(theme.asset.artalk_css)}')
        await btf.getScript('!{url_for(theme.asset.artalk_js)}')
        initArtalk(el, pageKey)
      }
    }

    if (isShuoshuo) {
      '!{use[0]}' === 'Artalk'
        ? window.shuoshuoComment = { loadComment: loadArtalk }
        : window.loadOtherComment = loadArtalk
      return
    }

    if ('!{use[0]}' === 'Artalk' || !!{lazyload}) {
      if (!{lazyload}) btf.loadComment(document.getElementById('artalk-wrap'), loadArtalk)
      else setTimeout(loadArtalk, 100)
    } else {
      window.loadOtherComment = loadArtalk
    }
  })()

================================================
FILE: layout/includes/third-party/comments/disqus.pug
================================================
- const disqusPageTitle = page.title.replace(/'/ig,"\\'")
- const { shortname, apikey } = theme.disqus
- const { use, lazyload, count } = theme.comments

script.
  (() => {
    const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'

    const disqusReset = conf => {
      window.DISQUS && window.DISQUS.reset({
        reload: true,
        config: conf
      })
    }

    const loadDisqus = (el, path) => {
      if (isShuoshuo) {
        window.shuoshuoComment.destroyDisqus = () => {
          if (el.children.length) {
            el.innerHTML = ''
            el.classList.add('no-comment')
          }
        }
      }

      window.disqus_identifier = isShuoshuo ? path : '!{ url_for(page.path) }'
      window.disqus_url = isShuoshuo ? location.origin + path : '!{ page.permalink }'

      const disqus_config = function () {
        this.page.url = disqus_url
        this.page.identifier = disqus_identifier
        this.page.title = '!{ disqusPageTitle }'
      }

      if (window.DISQUS) disqusReset(disqus_config)
      else {
        const script = document.createElement('script')
        script.src = 'https://!{shortname}.disqus.com/embed.js'
        script.setAttribute('data-timestamp', +new Date())
        document.head.appendChild(script)
      }

      btf.addGlobalFn('themeChange', () => disqusReset(disqus_config), 'disqus')
    }

    const getCount = async() => {
      try {
        const eleGroup = document.querySelector('#post-meta .disqus-comment-count')
        if (!eleGroup) return
        const cleanedLinks = eleGroup.href.replace(/#post-comment$/, '')

        const res = await fetch(`https://disqus.com/api/3.0/threads/set.json?forum=!{shortname}&api_key=!{apikey}&thread:link=${cleanedLinks}`,{
          method: 'GET'
        })
        const result = await res.json()

        const count = result.response.length ? result.response[0].posts : 0
        eleGroup.textContent = count
      } catch (err) {
        console.error(err)
      }
    }

    if (isShuoshuo) {
      '!{use[0]}' === 'Disqus'
        ? window.shuoshuoComment = { loadComment: loadDisqus }
        : window.loadOtherComment = loadDisqus
      return
    }

    if ('!{use[0]}' === 'Disqus' || !!{lazyload}) {
      if (!{lazyload}) btf.loadComment(document.getElementById('disqus_thread'), loadDisqus)
      else {
        loadDisqus()
        !{ count ? `GLOBAL_CONFIG_SITE.pageType === 'post' && getCount()` : '' }
      }
    } else {
      window.loadOtherComment = loadDisqus
    }
  })()


================================================
FILE: layout/includes/third-party/comments/disqusjs.pug
================================================
- let disqusjsPageTitle = page.title && page.title.replace(/'/ig,"\\'")
- const { shortname:dqShortname, apikey:dqApikey, option:dqOption } = theme.disqusjs

script.
  (() => {
    const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'== 'shuoshuo'
    const dqOption = !{JSON.stringify(dqOption)}

    const destroyDisqusjs = () => {
      disqusjs.destroy()
      window.disqusjs = null
    }

    const themeChange = (el, path) => {
      destroyDisqusjs()
      initDisqusjs(el, path)
    }

    const initDisqusjs = (el = document, path) => {
      if (isShuoshuo) {
        window.shuoshuoComment.destroyDisqusjs = () => {
          destroyDisqusjs()
          if (el.children.length) {
            el.innerHTML = ''
            el.classList.add('no-comment')
          }
        }
      }

      disqusjs = new DisqusJS({
        shortname: '!{dqShortname}',
        title: '!{ disqusjsPageTitle }',
        apikey: '!{dqApikey}',
        ...dqOption,
        identifier: isShuoshuo ? path : (dqOption && dqOption.identifier) || '!{ url_for(page.path) }',
        url: isShuoshuo ? location.origin + path : (dqOption && dqOption.url) || '!{ page.permalink }'
      })

      disqusjs.render(el.querySelector('#disqusjs-wrap'))

      btf.addGlobalFn('themeChange', () => themeChange(el, path), 'disqusjs')
    }

    const loadDisqusjs = async(el, path) => {
      if (window.disqusJsLoad) initDisqusjs(el, path)
      else {
        await btf.getCSS('!{url_for(theme.asset.disqusjs_css)}')
        await btf.getScript('!{url_for(theme.asset.disqusjs)}')
        initDisqusjs(el, path)
        window.disqusJsLoad = true
      }
    }

    const getCount = async() => {
      try {
        const eleGroup = document.querySelector('#post-meta .disqusjs-comment-count')
        if (!eleGroup) return
        const cleanedLinks = eleGroup.href.replace(/#post-comment$/, '')

        const res = await fetch(`https://disqus.com/api/3.0/threads/set.json?forum=!{dqShortname}&api_key=!{dqApikey}&thread:link=${cleanedLinks}`,{
          method: 'GET'
        })
        const result = await res.json()
        const count = result.response.length ? result.response[0].posts : 0
        eleGroup.textContent = count
      } catch (err) {
        console.error(err)
      }
    }

    if (isShuoshuo) {
      '!{theme.comments.use[0]}' === 'Disqusjs'
        ? window.shuoshuoComment = { loadComment: loadDisqusjs }
        : window.loadOtherComment = loadDisqusjs
      return
    }

    if ('!{theme.comments.use[0]}' === 'Disqusjs' || !!{theme.comments.lazyload}) {
      if (!{theme.comments.lazyload}) btf.loadComment(document.getElementById('disqusjs-wrap'), loadDisqusjs)
      else {
        loadDisqusjs()
        !{ theme.comments.count ? `GLOBAL_CONFIG_SITE.pageType === 'post' && getCount()` : '' }
      }
    } else {
      window.loadOtherComment = loadDisqusjs
    }
  })()

================================================
FILE: layout/includes/third-party/comments/facebook_comments.pug
================================================
- const fbSDKVer = 'v20.0'
- const fbSDK = `https://connect.facebook.net/${theme.facebook_comments.lang}/sdk.js#xfbml=1&version=${fbSDKVer}`

script.
  (()=>{
    const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'== 'shuoshuo'

    const loadFBComment = (el = document, path) => {
      if (isShuoshuo) {
        window.shuoshuoComment.destroyFB = () => {
          if (el.children.length) {
            el.innerHTML = ''
            el.classList.add('no-comment')
          }
        }
      }

      document.getElementById('fb-root') ? '' : document.body.insertAdjacentHTML('afterend', '<div id="fb-root"></div>')

      const themeNow = document.documentElement.getAttribute('data-theme') === 'dark' ? 'dark' : 'light'
      const $fbComment = el.getElementsByClassName('fb-comments')[0]
      $fbComment.setAttribute('data-colorscheme',themeNow)
      $fbComment.setAttribute('data-href', isShuoshuo ? '!{urlNoIndex(page.permalink)}' + '#' + path : '!{urlNoIndex(page.permalink)}')

      if (typeof FB === 'object') {
        FB.XFBML.parse(document.getElementsByClassName('post-meta-commentcount')[0])
        FB.XFBML.parse(el.querySelector('#post-comment'))
      }
      else {
        let ele = document.createElement('script')
        ele.setAttribute('src','!{fbSDK}')
        ele.setAttribute('async', 'true')
        ele.setAttribute('defer', 'true')
        ele.setAttribute('crossorigin', 'anonymous')
        ele.setAttribute('id', 'facebook-jssdk')
        document.getElementById('fb-root').insertAdjacentElement('afterbegin',ele)
      }
    }

    const fbModeChange = theme => {
      const $fbComment = document.getElementsByClassName('fb-comments')[0]
      if ($fbComment && typeof FB === 'object') {
        $fbComment.setAttribute('data-colorscheme',theme)
        FB.XFBML.parse(document.getElementById('post-comment'))
      }
    }

    btf.addGlobalFn('themeChange', fbModeChange, 'facebook_comments')

    if (isShuoshuo) {
      '!{theme.comments.use[0]}' === 'Facebook Comments'
        ? window.shuoshuoComment = { loadComment: loadFBComment }
        : window.loadOtherComment = loadFBComment
      return
    }

    if ('!{theme.comments.use[0]}' === 'Facebook Comments' || !!{theme.comments.lazyload}) {
      if (!{theme.comments.lazyload}) btf.loadComment(document.querySelector('#post-comment .fb-comments'), loadFBComment)
      else loadFBComment()
    } else {
      window.loadOtherComment = loadFBComment
    }
  })()



================================================
FILE: layout/includes/third-party/comments/giscus.pug
================================================
- const { use, lazyload } = theme.comments
- const { repo, repo_id, category_id, light_theme, dark_theme, js, option } = theme.giscus
- const giscusUrl = js || 'https://giscus.app/client.js'
- const giscusOriginUrl = new URL(giscusUrl).origin

script.
  (() => {
    const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
    const option = !{JSON.stringify(option)}

    const getGiscusTheme = theme => theme === 'dark' ? '!{dark_theme}' : '!{light_theme}'

    const createScriptElement = config => {
      const ele = document.createElement('script')
      Object.entries(config).forEach(([key, value]) => {
        ele.setAttribute(key, value)
      })
      return ele
    }

    const loadGiscus = (el = document, key) => {
      const mappingConfig = isShuoshuo
        ? { 'data-mapping': 'specific', 'data-term': key }
        : { 'data-mapping': (option && option['data-mapping']) || 'pathname' }

      const giscusConfig = {
        src: '!{giscusUrl}',
        'data-repo': '!{repo}',
        'data-repo-id': '!{repo_id}',
        'data-category-id': '!{category_id}',
        'data-theme': getGiscusTheme(document.documentElement.getAttribute('data-theme')),
        'data-reactions-enabled': '1',
        crossorigin: 'anonymous',
        async: true,
        ...option,
        ...mappingConfig
      }

      const scriptElement = createScriptElement(giscusConfig)

      el.querySelector('#giscus-wrap').appendChild(scriptElement)

      if (isShuoshuo) {
        window.shuoshuoComment.destroyGiscus = () => {
          if (el.children.length) {
            el.innerHTML = ''
            el.classList.add('no-comment')
          }
        }
      }
    }

    const changeGiscusTheme = theme => {
      const iframe = document.querySelector('#giscus-wrap iframe')
      if (iframe) {
        const message = {
          giscus: {
            setConfig: {
              theme: getGiscusTheme(theme)
            }
          }
        }
        iframe.contentWindow.postMessage(message, '!{giscusOriginUrl}')
      }
    }

    btf.addGlobalFn('themeChange', changeGiscusTheme, 'giscus')

    if (isShuoshuo) {
      '!{use[0]}' === 'Giscus'
        ? window.shuoshuoComment = { loadComment: loadGiscus }
        : window.loadOtherComment = loadGiscus
      return
    }

    if ('!{use[0]}' === 'Giscus' || !!{lazyload}) {
      if (!{lazyload}) btf.loadComment(document.getElementById('giscus-wrap'), loadGiscus)
      else loadGiscus()
    } else {
      window.loadOtherComment = loadGiscus
    }
  })()


================================================
FILE: layout/includes/third-party/comments/gitalk.pug
================================================
- const { client_id, client_secret, repo, owner, admin, option } = theme.gitalk

script.
  (() => {
    const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
    const option = !{JSON.stringify(option)}

    const commentCount = n => {
      const isCommentCount = document.querySelector('#post-meta .gitalk-comment-count')
      if (isCommentCount) {
        isCommentCount.textContent= n
      }
    }

    const initGitalk = (el, path) => {
      if (isShuoshuo) {
        window.shuoshuoComment.destroyGitalk = () => {
          if (el.children.length) {
            el.innerHTML = ''
            el.classList.add('no-comment')
          }
        }
      }

      const gitalk = new Gitalk({
        clientID: '!{client_id}',
        clientSecret: '!{client_secret}',
        repo: '!{repo}',
        owner: '!{owner}',
        admin: ['!{admin}'],
        updateCountCallback: commentCount,
        ...option,
        id: isShuoshuo ? path : (option && option.id) || '!{md5(page.path)}'
      })

      gitalk.render('gitalk-container')
    }

    const loadGitalk = async(el, path) => {
      if (typeof Gitalk === 'function') initGitalk(el, path)
      else {
        await btf.getCSS('!{url_for(theme.asset.gitalk_css)}')
        await btf.getScript('!{url_for(theme.asset.gitalk)}')
        initGitalk(el, path)
      }
    }

    if (isShuoshuo) {
      '!{theme.comments.use[0]}' === 'Gitalk'
        ? window.shuoshuoComment = { loadComment: loadGitalk }
        : window.loadOtherComment = loadGitalk
      return
    }

    if ('!{theme.comments.use[0]}' === 'Gitalk' || !!{theme.comments.lazyload}) {
      if (!{theme.comments.lazyload}) btf.loadComment(document.getElementById('gitalk-container'), loadGitalk)
      else loadGitalk()
    } else {
      window.loadOtherComment = loadGitalk
    }
  })()





================================================
FILE: layout/includes/third-party/comments/index.pug
================================================
- let defaultComment = theme.comments.use[0]
hr.custom-hr
#post-comment
  .comment-head
    .comment-headline
      i.fas.fa-comments.fa-fw
      span= ' ' + _p('comment')

    if theme.comments.use.length > 1
      .comment-switch
        span.first-comment=defaultComment
        span#switch-btn
        span.second-comment=theme.comments.use[1]


  .comment-wrap
    each name in theme.comments.use
      div
        case name
          when 'Disqus'
            #disqus_thread
          when 'Valine'
            #vcomment.vcomment
          when 'Disqusjs'
            #disqusjs-wrap
          when 'Livere'
            #lv-container(data-id="city" data-uid=theme.livere.uid)
          when 'Gitalk'
            #gitalk-container
          when 'Utterances'
            #utterances-wrap
          when 'Twikoo'
            #twikoo-wrap
          when 'Waline'
            #waline-wrap
          when 'Giscus'
            #giscus-wrap
          when 'Facebook Comments'
            .fb-comments(data-colorscheme = theme.display_mode === 'dark' ? 'dark' : 'light'
                        data-numposts= theme.facebook_comments.pageSize || 10
                        data-order-by= theme.facebook_comments.order_by || 'social'
                        data-width="100%")
          when 'Remark42'
            #remark42
          when 'Artalk'
            #artalk-wrap


================================================
FILE: layout/includes/third-party/comments/js.pug
================================================
each name in theme.comments.use
  case name
    when 'Valine'
      !=partial('includes/third-party/comments/valine', {}, {cache: true})
    when 'Disqus'
      include ./disqus.pug
    when 'Disqusjs'
      include ./disqusjs.pug
    when 'Livere'
      !=partial('includes/third-party/comments/livere', {}, {cache: true})
    when 'Gitalk'
      include ./gitalk.pug
    when 'Utterances'
      !=partial('includes/third-party/comments/utterances', {}, {cache: true})
    when 'Twikoo'
      !=partial('includes/third-party/comments/twikoo', {}, {cache: true})
    when 'Waline'
      !=partial('includes/third-party/comments/waline', {}, {cache: true})
    when 'Giscus'
      !=partial('includes/third-party/comments/giscus', {}, {cache: true})
    when 'Facebook Comments'
      include ./facebook_comments.pug
    when 'Remark42'
      !=partial('includes/third-party/comments/remark42', {}, {cache: true})
    when 'Artalk'
      !=partial('includes/third-party/comments/artalk', {}, {cache: true})

================================================
FILE: layout/includes/third-party/comments/livere.pug
================================================
- const { use, lazyload } = theme.comments

script.
  (() => {
    const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'

    const loadLivere = (el, path) => {
      window.livereOptions = {
        refer: path || location.pathname
      }

      if (isShuoshuo) {
        window.shuoshuoComment.destroyLivere = () => {
          if (el.children.length) {
            el.innerHTML = ''
            el.classList.add('no-comment')
          }
        }
      }

      if (typeof LivereTower === 'object') window.LivereTower.init()
      else {
        (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');
      }
    }

    if (isShuoshuo) {
      '!{use[0]}' === 'Livere'
        ? window.shuoshuoComment = { loadComment: loadLivere }
        : window.loadOtherComment = loadLivere
      return
    }

    if ('!{use[0]}' === 'Livere' || !!{lazyload}) {
      if (!{lazyload}) btf.loadComment(document.getElementById('lv-container'), loadLivere)
      else loadLivere()
    } else {
      window.loadOtherComment = loadLivere
    }
  })()

================================================
FILE: layout/includes/third-party/comments/remark42.pug
================================================
- const { host, siteId, option } = theme.remark42

script.
  (() => {
    const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
    const options = !{JSON.stringify(option)}

    const loadScript = src => {
      const script = document.createElement('script')
      script.src = src
      script.defer = true
      document.head.appendChild(script)
    }

    const addRemark42 = () => loadScript('!{host}/web/embed.js')

    const getCount = () => document.querySelector('.remark42__counter') && loadScript('!{host}/web/count.js')

    const destroyRemark42 = () => window.remark42Instance && window.remark42Instance.destroy()

    const initRemark42 = remark_config => {
      if (window.REMARK42) {
        destroyRemark42()
        window.remark42Instance = window.REMARK42.createInstance({
          ...remark_config
        })
      }
    }

    const loadRemark42 = (el, path) => {
      if (isShuoshuo) {
        window.shuoshuoComment.destroyRemark42 = () => {
          destroyRemark42()
          if (el.children.length) {
            el.innerHTML = ''
            el.classList.add('no-comment')
          }
        }
      }

      window.remark_config = {
        host: '!{host}',
        site_id: '!{siteId}',
        theme: document.documentElement.getAttribute('data-theme') === 'dark' ? 'dark' : 'light',
        ...options,
        url: isShuoshuo ? window.location.origin + path : (options && options.url) || window.location.origin + window.location.pathname
      }

      if (window.REMARK42) {
        initRemark42(remark_config)
        getCount()
      } else {
        addRemark42()
        window.addEventListener('REMARK42::ready', () => {
          initRemark42(remark_config)
          getCount()
        })
      }
    }

    const remarkChangeMode = theme => window.REMARK42 && window.REMARK42.changeTheme(theme)

    btf.addGlobalFn('themeChange', remarkChangeMode, 'remark42')

    if (isShuoshuo) {
      '!{theme.comments.use[0]}' === 'Remark42'
        ? window.shuoshuoComment = { loadComment: loadRemark42 }
        : window.loadOtherComment = loadRemark42
      return
    }

    if ('!{theme.comments.use[0]}' === 'Remark42' || !!{theme.comments.lazyload}) {
      if (!{theme.comments.lazyload}) btf.loadComment(document.getElementById('remark42'), loadRemark42)
      else loadRemark42()
    } else {
      window.loadOtherComment = loadRemark42
    }
  })()

================================================
FILE: layout/includes/third-party/comments/twikoo.pug
================================================
- const { envId, region, option } = theme.twikoo
- const { use, lazyload, count } = theme.comments

script.
  (() => {
    const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
    const option = !{JSON.stringify(option)}

    const getCount = () => {
      const countELement = document.getElementById('twikoo-count')
      if(!countELement) return
      twikoo.getCommentsCount({
        envId: '!{envId}',
        region: '!{region}',
        urls: [window.location.pathname],
        includeReply: false
      }).then(res => {
        countELement.textContent = res[0].count
      }).catch(err => {
        console.error(err)
      })
    }

    const init = (el = document, path = location.pathname) => {
      twikoo.init({
        el: el.querySelector('#twikoo-wrap'),
        envId: '!{envId}',
        region: '!{region}',
        onCommentLoaded: () => {
          btf.loadLightbox(document.querySelectorAll('#twikoo .tk-content img:not(.tk-owo-emotion)'))
        },
        ...option,
        path: isShuoshuo ? path : (option && option.path) || path
      })

      !{count ? `GLOBAL_CONFIG_SITE.pageType === 'post' && getCount()` : ''}

      isShuoshuo && (window.shuoshuoComment.destroyTwikoo = () => {
        if (el.children.length) {
          el.innerHTML = ''
          el.classList.add('no-comment')
        }
      })
    }

    const loadTwikoo = (el, path) => {
      if (typeof twikoo === 'object') setTimeout(() => init(el, path), 0)
      else btf.getScript('!{url_for(theme.asset.twikoo)}').then(() => init(el, path))
    }

    if (isShuoshuo) {
      '!{use[0]}' === 'Twikoo'
        ? window.shuoshuoComment = { loadComment: loadTwikoo }
        : window.loadOtherComment = loadTwikoo
      return
    }

    if ('!{use[0]}' === 'Twikoo' || !!{lazyload}) {
      if (!{lazyload}) btf.loadComment(document.getElementById('twikoo-wrap'), loadTwikoo)
      else loadTwikoo()
    } else {
      window.loadOtherComment = loadTwikoo
    }
  })()

================================================
FILE: layout/includes/third-party/comments/utterances.pug
================================================
- const { use, lazyload } = theme.comments
- const { repo, issue_term, light_theme, dark_theme, js, option } = theme.utterances
- const utterancesUrl = js || 'https://utteranc.es/client.js'
- const utterancesOriginUrl = new URL(utterancesUrl).origin

script.
  (() => {
    const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
    const option = !{JSON.stringify(option)}
    const getUtterancesTheme = theme => theme === 'dark' ? '#{dark_theme}' : '#{light_theme}'

    const loadUtterances = (el = document, key) => {
      if (isShuoshuo) {
        window.shuoshuoComment.destroyUtterances = () => {
          if (el.children.length) {
            el.innerHTML = ''
            el.classList.add('no-comment')
          }
        }
      }

      const config = {
        src: '!{utterancesUrl}',
        repo: '!{repo}',
        theme: getUtterancesTheme(document.documentElement.getAttribute('data-theme')),
        crossorigin: 'anonymous',
        async: true,
        ...option,
        'issue-term': isShuoshuo ? key : (option && option['issue-term']) || '!{issue_term}'
      }

      const ele = document.createElement('script')
      Object.entries(config).forEach(([key, value]) => ele.setAttribute(key, value))
      el.querySelector('#utterances-wrap').appendChild(ele)
    }

    const changeUtterancesTheme = theme => {
      const iframe = document.querySelector('#utterances-wrap iframe')
      if (iframe) {
        const message = {
          type: 'set-theme',
          theme: getUtterancesTheme(theme)
        };
        iframe.contentWindow.postMessage(message, '!{utterancesOriginUrl}')
      }
    }

    btf.addGlobalFn('themeChange', changeUtterancesTheme, 'utterances')

    if (isShuoshuo) {
      '!{use[0]}' === 'Utterances'
        ? window.shuoshuoComment = { loadComment: loadUtterances }
        : window.loadOtherComment = loadUtterances
      return
    }
    
    if ('!{use[0]}' === 'Utterances' || !!{lazyload}) {
      if (!{lazyload}) btf.loadComment(document.getElementById('utterances-wrap'), loadUtterances)
      else loadUtterances()
    } else {
      window.loadOtherComment = loadUtterances
    }
  })()

================================================
FILE: layout/includes/third-party/comments/valine.pug
================================================
- const { use, lazyload } = theme.comments
- const { appId, appKey, avatar, serverURLs, visitor, option } = theme.valine

- let emojiMaps = '""'
if site.data.valine
  - emojiMaps = JSON.stringify(site.data.valine)

script.
  (() => {
    const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
    const option = !{JSON.stringify(option)}

    const initValine = (el, path) => {
      if (isShuoshuo) {
        window.shuoshuoComment.destroyValine = () => {
          if (el.children.length) {
            el.innerHTML = ''
            el.classList.add('no-comment')
          }
        }
      }

      const valineConfig = {
        el: '#vcomment',
        appId: '#{appId}',
        appKey: '#{appKey}',
        avatar: '#{avatar}',
        serverURLs: '#{serverURLs}',
        emojiMaps: !{emojiMaps},
        visitor: #{visitor},
        ...option,
        path: isShuoshuo ? path : (option && option.path) || window.location.pathname
      }

      new Valine(valineConfig)
    }

    const loadValine = async (el, path) => {
      if (typeof Valine === 'function') {
        initValine(el, path)
      } else {
        await btf.getScript('!{url_for(theme.asset.valine)}')
        initValine(el, path)
      }
    }

    if (isShuoshuo) {
      '!{use[0]}' === 'Valine'
        ? window.shuoshuoComment = { loadComment: loadValine }
        : window.loadOtherComment = loadValine
      return
    }

    if ('!{use[0]}' === 'Valine' || !!{lazyload}) {
      if (!{lazyload}) btf.loadComment(document.getElementById('vcomment'),loadValine)
      else setTimeout(loadValine, 0)
    } else {
      window.loadOtherComment = loadValine
    }
  })()

================================================
FILE: layout/includes/third-party/comments/waline.pug
================================================
- const { serverURL, option, pageview } = theme.waline
- const { lazyload, count, use } = theme.comments

script.
  (() => {
    let initFn = window.walineFn || null
    const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
    const option = !{JSON.stringify(option)}

    const destroyWaline = ele => ele.destroy()

    const initWaline = (Fn, el = document, path = window.location.pathname) => {
      const waline = Fn({
        el: el.querySelector('#waline-wrap'),
        serverURL: '!{serverURL}',
        pageview: !{lazyload ? false : pageview},
        dark: 'html[data-theme="dark"]',
        comment: !{lazyload ? false : count},
        ...option,
        path: isShuoshuo ? path : (option && option.path) || path
      })

      if (isShuoshuo) {
        window.shuoshuoComment.destroyWaline = () => {
          destroyWaline(waline)
          if (el.children.length) {
            el.innerHTML = ''
            el.classList.add('no-comment')
          }
        }
      }
    }

    const loadWaline = (el, path) => {
      if (initFn) initWaline(initFn, el, path)
      else {
        btf.getCSS('!{url_for(theme.asset.waline_css)}')
          .then(() => import('!{url_for(theme.asset.waline_js)}'))
          .then(({ init }) => {
            initFn = init || Waline.init
            initWaline(initFn, el, path)
            window.walineFn = initFn
          })
      }
    }

    if (isShuoshuo) {
      '!{use[0]}' === 'Waline'
        ? window.shuoshuoComment = { loadComment: loadWaline } 
        : window.loadOtherComment = loadWaline
      return
    }

    if ('!{use[0]}' === 'Waline' || !!{lazyload}) {
      if (!{lazyload}) btf.loadComment(document.getElementById('waline-wrap'),loadWaline)
      else setTimeout(loadWaline, 0)
    } else {
      window.loadOtherComment = loadWaline
    }
  })()



================================================
FILE: layout/includes/third-party/effect.pug
================================================
if theme.fireworks && theme.fireworks.enable
  canvas.fireworks(mobile=`${theme.fireworks.mobile}`)
  script(src=url_for(theme.asset.fireworks))

if (theme.canvas_ribbon && theme.canvas_ribbon.enable)
  script(defer id="ribbon" src=url_for(theme.asset.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}`)

if (theme.canvas_fluttering_ribbon && theme.canvas_fluttering_ribbon.enable)
  script(defer id="fluttering_ribbon" mobile=`${theme.canvas_fluttering_ribbon.mobile}` src=url_for(theme.asset.canvas_fluttering_ribbon))

if (theme.canvas_nest && theme.canvas_nest.enable)
  script#canvas_nest(defer 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.asset.canvas_nest))

if theme.activate_power_mode.enable
  script(src=url_for(theme.asset.activate_power_mode))
  script.
    POWERMODE.colorful = !{theme.activate_power_mode.colorful};
    POWERMODE.shake = !{theme.activate_power_mode.shake};
    POWERMODE.mobile = !{theme.activate_power_mode.mobile};
    document.body.addEventListener('input', POWERMODE);

//- 鼠標特效
if theme.click_heart && theme.click_heart.enable
  script#click-heart(src=url_for(theme.asset.click_heart) async mobile=`${theme.click_heart.mobile}`)

if theme.clickShowText && theme.clickShowText.enable
  script#click-show-text(
    src= url_for(theme.asset.clickShowText)
    data-mobile= `${theme.clickShowText.mobile}`
    data-text= theme.clickShowText.text.join(",")
    data-fontsize= theme.clickShowText.fontSize
    data-random= `${theme.clickShowText.random}`
    async
  )

================================================
FILE: layout/includes/third-party/math/chartjs.pug
================================================
- const { fontColor, borderColor, scale_ticks_backdropColor } = theme.chartjs

script.
  (() => {
    const applyThemeDefaultsConfig = theme => {
      if (theme === 'dark-mode') {
        Chart.defaults.color = "!{fontColor.dark}"
        Chart.defaults.borderColor = "!{borderColor.dark}"
        Chart.defaults.scale.ticks.backdropColor = "!{scale_ticks_backdropColor.dark}"
      } else {
        Chart.defaults.color = "!{fontColor.light}"
        Chart.defaults.borderColor = "!{borderColor.light}"
        Chart.defaults.scale.ticks.backdropColor = "!{scale_ticks_backdropColor.light}"
      }
    }

    // Recursively traverse the config object and automatically apply theme-specific color schemes
    const applyThemeConfig = (obj, theme) => {
      if (typeof obj !== 'object' || obj === null) return

      Object.keys(obj).forEach(key => {
        const value = obj[key]
        // If the property is an object and has theme-specific options, apply them
        if (typeof value === 'object' && value !== null) {
          if (value[theme]) {
            obj[key] = value[theme] // Apply the value for the current theme
          } else {
            // Recursively process child objects
            applyThemeConfig(value, theme)
          }
        }
      })
    }

    const runChartJS = ele => {
      window.loadChartJS = true

      Array.from(ele).forEach((item, index) => {
        const chartSrc = item.firstElementChild
        const chartID = item.getAttribute('data-chartjs-id') || ('chartjs-' + index) // Use custom ID or default ID
        const width = item.getAttribute('data-width')
        const existingCanvas = document.getElementById(chartID)

        // If a canvas already exists, remove it to avoid rendering duplicates
        if (existingCanvas) {
            existingCanvas.parentNode.remove()
        }

        const chartDefinition = chartSrc.textContent
        const canvas = document.createElement('canvas')
        canvas.id = chartID

        const div = document.createElement('div')
        div.className = 'chartjs-wrap'

        if (width) {
          div.style.width = width
        }

        div.appendChild(canvas)
        chartSrc.insertAdjacentElement('afterend', div)

        const ctx = document.getElementById(chartID).getContext('2d')

        const config = JSON.parse(chartDefinition)

        const theme = document.documentElement.getAttribute('data-theme') === 'dark' ? 'dark-mode' : 'light-mode'

        // Set default styles (initial setup)
        applyThemeDefaultsConfig(theme)

        // Automatically traverse the config and apply dual-mode color schemes
        applyThemeConfig(config, theme)

        new Chart(ctx, config)
      })
    }

    const loadChartJS = () => {
      const chartJSEle = document.querySelectorAll('#article-container .chartjs-container')
      if (chartJSEle.length === 0) return

      window.loadChartJS ? runChartJS(chartJSEle) : btf.getScript('!{url_for(t
Download .txt
gitextract_p7rv7pkl/

├── .github/
│   ├── FUNDING.yml
│   ├── ISSUE_TEMPLATE/
│   │   ├── bug_report.yml
│   │   ├── config.yml
│   │   └── feature_request.yml
│   └── workflows/
│       ├── publish.yml
│       └── stale.yml
├── .gitignore
├── LICENSE
├── README.md
├── README_CN.md
├── _config.yml
├── languages/
│   ├── default.yml
│   ├── en.yml
│   ├── ja.yml
│   ├── ko.yml
│   ├── zh-CN.yml
│   ├── zh-HK.yml
│   └── zh-TW.yml
├── layout/
│   ├── archive.pug
│   ├── category.pug
│   ├── includes/
│   │   ├── additional-js.pug
│   │   ├── footer.pug
│   │   ├── head/
│   │   │   ├── Open_Graph.pug
│   │   │   ├── analytics.pug
│   │   │   ├── config.pug
│   │   │   ├── config_site.pug
│   │   │   ├── google_adsense.pug
│   │   │   ├── preconnect.pug
│   │   │   ├── pwa.pug
│   │   │   ├── site_verification.pug
│   │   │   └── structured_data.pug
│   │   ├── head.pug
│   │   ├── header/
│   │   │   ├── index.pug
│   │   │   ├── menu_item.pug
│   │   │   ├── nav.pug
│   │   │   ├── post-info.pug
│   │   │   └── social.pug
│   │   ├── layout.pug
│   │   ├── loading/
│   │   │   ├── fullpage-loading.pug
│   │   │   ├── index.pug
│   │   │   └── pace.pug
│   │   ├── mixins/
│   │   │   ├── article-sort.pug
│   │   │   └── indexPostUI.pug
│   │   ├── page/
│   │   │   ├── 404.pug
│   │   │   ├── categories.pug
│   │   │   ├── default-page.pug
│   │   │   ├── flink.pug
│   │   │   ├── shuoshuo.pug
│   │   │   └── tags.pug
│   │   ├── pagination.pug
│   │   ├── post/
│   │   │   ├── outdate-notice.pug
│   │   │   ├── post-copyright.pug
│   │   │   └── reward.pug
│   │   ├── rightside.pug
│   │   ├── sidebar.pug
│   │   ├── third-party/
│   │   │   ├── abcjs/
│   │   │   │   ├── abcjs.pug
│   │   │   │   └── index.pug
│   │   │   ├── aplayer.pug
│   │   │   ├── card-post-count/
│   │   │   │   ├── artalk.pug
│   │   │   │   ├── disqus.pug
│   │   │   │   ├── fb.pug
│   │   │   │   ├── index.pug
│   │   │   │   ├── remark42.pug
│   │   │   │   ├── twikoo.pug
│   │   │   │   ├── valine.pug
│   │   │   │   └── waline.pug
│   │   │   ├── chat/
│   │   │   │   ├── chatra.pug
│   │   │   │   ├── crisp.pug
│   │   │   │   ├── index.pug
│   │   │   │   └── tidio.pug
│   │   │   ├── comments/
│   │   │   │   ├── artalk.pug
│   │   │   │   ├── disqus.pug
│   │   │   │   ├── disqusjs.pug
│   │   │   │   ├── facebook_comments.pug
│   │   │   │   ├── giscus.pug
│   │   │   │   ├── gitalk.pug
│   │   │   │   ├── index.pug
│   │   │   │   ├── js.pug
│   │   │   │   ├── livere.pug
│   │   │   │   ├── remark42.pug
│   │   │   │   ├── twikoo.pug
│   │   │   │   ├── utterances.pug
│   │   │   │   ├── valine.pug
│   │   │   │   └── waline.pug
│   │   │   ├── effect.pug
│   │   │   ├── math/
│   │   │   │   ├── chartjs.pug
│   │   │   │   ├── index.pug
│   │   │   │   ├── katex.pug
│   │   │   │   ├── mathjax.pug
│   │   │   │   └── mermaid.pug
│   │   │   ├── newest-comments/
│   │   │   │   ├── artalk.pug
│   │   │   │   ├── common.pug
│   │   │   │   ├── disqus-comment.pug
│   │   │   │   ├── github-issues.pug
│   │   │   │   ├── index.pug
│   │   │   │   ├── remark42.pug
│   │   │   │   ├── twikoo-comment.pug
│   │   │   │   ├── valine.pug
│   │   │   │   └── waline.pug
│   │   │   ├── pjax.pug
│   │   │   ├── prismjs.pug
│   │   │   ├── search/
│   │   │   │   ├── algolia.pug
│   │   │   │   ├── docsearch.pug
│   │   │   │   ├── index.pug
│   │   │   │   └── local-search.pug
│   │   │   ├── share/
│   │   │   │   ├── addtoany.pug
│   │   │   │   ├── index.pug
│   │   │   │   └── share-js.pug
│   │   │   ├── subtitle.pug
│   │   │   └── umami_analytics.pug
│   │   └── widget/
│   │       ├── card_ad.pug
│   │       ├── card_announcement.pug
│   │       ├── card_archives.pug
│   │       ├── card_author.pug
│   │       ├── card_bottom_self.pug
│   │       ├── card_categories.pug
│   │       ├── card_newest_comment.pug
│   │       ├── card_post_series.pug
│   │       ├── card_post_toc.pug
│   │       ├── card_recent_post.pug
│   │       ├── card_tags.pug
│   │       ├── card_top_self.pug
│   │       ├── card_webinfo.pug
│   │       └── index.pug
│   ├── index.pug
│   ├── page.pug
│   ├── post.pug
│   └── tag.pug
├── package.json
├── plugins.yml
├── scripts/
│   ├── common/
│   │   ├── default_config.js
│   │   └── postDesc.js
│   ├── events/
│   │   ├── 404.js
│   │   ├── cdn.js
│   │   ├── init.js
│   │   ├── stylus.js
│   │   └── welcome.js
│   ├── filters/
│   │   ├── post_lazyload.js
│   │   └── random_cover.js
│   ├── helpers/
│   │   ├── aside_archives.js
│   │   ├── aside_categories.js
│   │   ├── getArchiveLength.js
│   │   ├── inject_head_js.js
│   │   ├── page.js
│   │   ├── related_post.js
│   │   └── series.js
│   └── tag/
│       ├── button.js
│       ├── chartjs.js
│       ├── flink.js
│       ├── gallery.js
│       ├── hide.js
│       ├── inlineImg.js
│       ├── label.js
│       ├── mermaid.js
│       ├── note.js
│       ├── score.js
│       ├── series.js
│       ├── tabs.js
│       └── timeline.js
└── source/
    ├── css/
    │   ├── _global/
    │   │   ├── function.styl
    │   │   └── index.styl
    │   ├── _highlight/
    │   │   ├── highlight/
    │   │   │   ├── diff.styl
    │   │   │   └── index.styl
    │   │   ├── highlight.styl
    │   │   ├── prismjs/
    │   │   │   ├── diff.styl
    │   │   │   ├── index.styl
    │   │   │   └── line-number.styl
    │   │   └── theme.styl
    │   ├── _layout/
    │   │   ├── aside.styl
    │   │   ├── chat.styl
    │   │   ├── comments.styl
    │   │   ├── footer.styl
    │   │   ├── head.styl
    │   │   ├── loading.styl
    │   │   ├── pagination.styl
    │   │   ├── post.styl
    │   │   ├── relatedposts.styl
    │   │   ├── reward.styl
    │   │   ├── rightside.styl
    │   │   ├── sidebar.styl
    │   │   └── third-party.styl
    │   ├── _mode/
    │   │   ├── darkmode.styl
    │   │   └── readmode.styl
    │   ├── _page/
    │   │   ├── 404.styl
    │   │   ├── archives.styl
    │   │   ├── categories.styl
    │   │   ├── common.styl
    │   │   ├── flink.styl
    │   │   ├── homepage.styl
    │   │   ├── shuoshuo.styl
    │   │   └── tags.styl
    │   ├── _search/
    │   │   ├── algolia.styl
    │   │   ├── index.styl
    │   │   └── local-search.styl
    │   ├── _tags/
    │   │   ├── button.styl
    │   │   ├── gallery.styl
    │   │   ├── hexo.styl
    │   │   ├── hide.styl
    │   │   ├── inlineImg.styl
    │   │   ├── label.styl
    │   │   ├── note.styl
    │   │   ├── series.styl
    │   │   ├── tabs.styl
    │   │   └── timeline.styl
    │   ├── index.styl
    │   └── var.styl
    └── js/
        ├── main.js
        ├── search/
        │   ├── algolia.js
        │   └── local-search.js
        ├── tw_cn.js
        └── utils.js
Download .txt
SYMBOL INDEX (26 symbols across 5 files)

FILE: scripts/events/init.js
  function checkHexoEnvironment (line 10) | function checkHexoEnvironment (hexo) {
  function loadDefaultConfig (line 37) | function loadDefaultConfig () {
  function processCommentConfig (line 50) | function processCommentConfig (themeConfig) {

FILE: scripts/tag/gallery.js
  constant DEFAULT_LIMIT (line 14) | const DEFAULT_LIMIT = 10
  constant DEFAULT_FIRST_LIMIT (line 15) | const DEFAULT_FIRST_LIMIT = 10
  constant IMAGE_REGEX (line 16) | const IMAGE_REGEX = /!\[(.*?)\]\(([^\s]*)\s*(?:["'](.*?)["']?)?\s*\)/g

FILE: scripts/tag/series.js
  function series (line 30) | function series (args) {

FILE: source/js/search/algolia.js
  method searchInput (line 246) | get searchInput () { return document.querySelector('#algolia-search-inpu...
  method hits (line 247) | get hits () { return document.getElementById('algolia-hits') }
  method hitsEmpty (line 248) | get hitsEmpty () { return document.getElementById('algolia-hits-empty') }
  method hitsList (line 249) | get hitsList () { return document.querySelector('#algolia-hits .ais-Hits...
  method hitsWrapper (line 250) | get hitsWrapper () { return document.querySelector('#algolia-hits .ais-H...
  method pagination (line 251) | get pagination () { return document.getElementById('algolia-pagination') }
  method paginationList (line 252) | get paginationList () { return document.querySelector('#algolia-paginati...
  method stats (line 253) | get stats () { return document.querySelector('#algolia-info .ais-Stats-t...

FILE: source/js/search/local-search.js
  class LocalSearch (line 7) | class LocalSearch {
    method constructor (line 8) | constructor ({
    method getIndexByWord (line 20) | getIndexByWord (words, text, caseSensitive = false) {
    method mergeIntoSlice (line 57) | mergeIntoSlice (start, end, index) {
    method highlightKeyword (line 92) | highlightKeyword (val, slice) {
    method getResultItems (line 104) | getResultItems (keywords) {
    method fetchData (line 173) | fetchData () {
    method highlightText (line 200) | highlightText (node, slice, className) {
    method highlightSearchWords (line 219) | highlightSearchWords (body) {
  method pagination (line 266) | get pagination () { return document.getElementById('local-search-paginat...
  method paginationList (line 267) | get paginationList () { return document.querySelector('#local-search-pag...
Condensed preview — 211 files, each showing path, character count, and a content snippet. Download the .json file or copy for the full structured content (601K chars).
[
  {
    "path": ".github/FUNDING.yml",
    "chars": 849,
    "preview": "# These are supported funding model platforms\n\ngithub: # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [u"
  },
  {
    "path": ".github/ISSUE_TEMPLATE/bug_report.yml",
    "chars": 2173,
    "preview": "name: Bug report\r\ndescription: Create a report to help us improve\r\ntitle: '[Bug]: '\r\n\r\nbody:\r\n  - type: markdown\r\n    at"
  },
  {
    "path": ".github/ISSUE_TEMPLATE/config.yml",
    "chars": 517,
    "preview": "blank_issues_enabled: false\r\ncontact_links:\r\n  - name: Questions about Butterfly\r\n    url: https://github.com/jerryc127/"
  },
  {
    "path": ".github/ISSUE_TEMPLATE/feature_request.yml",
    "chars": 371,
    "preview": "name: Feature request\r\ndescription: Suggest an idea for this project\r\ntitle: '[Feature]: '\r\n\r\nbody:\r\n  - type: textarea\r"
  },
  {
    "path": ".github/workflows/publish.yml",
    "chars": 432,
    "preview": "name: npm publish\r\n\r\non:\r\n  release:\r\n    types: [created]\r\njobs:\r\n  build:\r\n    runs-on: ubuntu-latest\r\n    steps:\r\n   "
  },
  {
    "path": ".github/workflows/stale.yml",
    "chars": 752,
    "preview": "name: 'Close stale issues and PRs'\r\non:\r\n  schedule:\r\n    - cron: '30 1 * * *'\r\n\r\njobs:\r\n  stale:\r\n    runs-on: ubuntu-l"
  },
  {
    "path": ".gitignore",
    "chars": 24,
    "preview": ".DS_Store\nnode_modules/\n"
  },
  {
    "path": "LICENSE",
    "chars": 11558,
    "preview": "\r\n                                 Apache License\r\n                           Version 2.0, January 2004\r\n               "
  },
  {
    "path": "README.md",
    "chars": 7840,
    "preview": "<div align=\"right\">\n  <a title=\"中文\" href=\"/README_CN.md\">中文</a>\n</div>\n\n<div align=\"center\">\n\n<img src=\"./source/img/but"
  },
  {
    "path": "README_CN.md",
    "chars": 5422,
    "preview": "<div align=\"right\">\r\n  <a title=\"English\" href=\"/README.md\">English</a>\r\n</div>\r\n\r\n<div align=\"center\">\r\n\r\n<img src=\"./s"
  },
  {
    "path": "_config.yml",
    "chars": 28051,
    "preview": "# --------------------------------------\r\n# Hexo Butterfly Theme Configuration\r\n# If you have any questions, please refe"
  },
  {
    "path": "languages/default.yml",
    "chars": 3014,
    "preview": "footer:\r\n  framework: Framework\r\n  theme: Theme\r\n\r\ncopy:\r\n  success: Copy Successful\r\n  error: Copy Failed\r\n  noSupport:"
  },
  {
    "path": "languages/en.yml",
    "chars": 3013,
    "preview": "footer:\r\n  framework: Framework\r\n  theme: Theme\r\n\r\ncopy:\r\n  success: Copy Successful\r\n  error: Copy Failed\r\n  noSupport:"
  },
  {
    "path": "languages/ja.yml",
    "chars": 2369,
    "preview": "footer:\r\n  framework: フレームワーク\r\n  theme: テーマ\r\n\r\ncopy:\r\n  success: コピー成功\r\n  error: コピー失敗\r\n  noSupport: ブラウザが対応していません\r\n\r\npa"
  },
  {
    "path": "languages/ko.yml",
    "chars": 2330,
    "preview": "footer:\r\n  framework: 프레임워크\r\n  theme: 테마\r\n\r\ncopy:\r\n  success: 복사 성공\r\n  error: 복사 실패\r\n  noSupport: 브라우저가 지원되지 않음\r\n\r\npage:"
  },
  {
    "path": "languages/zh-CN.yml",
    "chars": 2172,
    "preview": "footer:\r\n  framework: 框架\r\n  theme: 主题\r\n\r\ncopy:\r\n  success: 复制成功\r\n  error: 复制失败\r\n  noSupport: 浏览器不支持\r\n\r\npage:\r\n  articles"
  },
  {
    "path": "languages/zh-HK.yml",
    "chars": 2117,
    "preview": "footer:\r\n  framework: 框架\r\n  theme: 主題\r\n\r\ncopy:\r\n  success: 複製成功\r\n  error: 複製失敗\r\n  noSupport: 瀏覽器不支援\r\n\r\npage:\r\n  articles"
  },
  {
    "path": "languages/zh-TW.yml",
    "chars": 2160,
    "preview": "footer:\r\n  framework: 框架\r\n  theme: 主題\r\n\r\ncopy:\r\n  success: 複製成功\r\n  error: 複製失敗\r\n  noSupport: 瀏覽器不支援\r\n\r\npage:\r\n  articles"
  },
  {
    "path": "layout/archive.pug",
    "chars": 244,
    "preview": "extends includes/layout.pug\r\n\r\nblock content\r\n  include ./includes/mixins/article-sort.pug\r\n  #archive\r\n    .article-sor"
  },
  {
    "path": "layout/category.pug",
    "chars": 358,
    "preview": "extends includes/layout.pug\r\n\r\nblock content\r\n  if theme.category_ui == 'index'\r\n    include ./includes/mixins/indexPost"
  },
  {
    "path": "layout/includes/additional-js.pug",
    "chars": 2388,
    "preview": "div\r\n  script(src=url_for(theme.asset.utils))\r\n  script(src=url_for(theme.asset.main))\r\n\r\n  if theme.translate.enable\r\n "
  },
  {
    "path": "layout/includes/footer.pug",
    "chars": 1614,
    "preview": "- const { nav, owner, copyright, custom_text } = theme.footer\r\n\r\nif nav\r\n  .footer-flex\r\n    for block in nav\r\n      .fo"
  },
  {
    "path": "layout/includes/head/Open_Graph.pug",
    "chars": 640,
    "preview": "if theme.Open_Graph_meta.enable\r\n  -\r\n    const coverVal = page.cover_type === 'img' ? page.cover : theme.avatar.img\r\n  "
  },
  {
    "path": "layout/includes/head/analytics.pug",
    "chars": 2233,
    "preview": "if theme.baidu_analytics\r\n  script.\r\n    var _hmt = _hmt || [];\r\n    (function() {\r\n      var hm = document.createElemen"
  },
  {
    "path": "layout/includes/head/config.pug",
    "chars": 4485,
    "preview": "-\r\n  let algolia = 'undefined'\r\n  if (theme.search.use === 'algolia_search') {\r\n    const { ALGOLIA_APP_ID, ALGOLIA_API_"
  },
  {
    "path": "layout/includes/head/config_site.pug",
    "chars": 947,
    "preview": "-\r\n  const titleVal = pageTitle.replace(/'/ig,\"\\\\'\")\r\n\r\n  let isHighlightShrink\r\n  if (theme.code_blocks.shrink == 'none"
  },
  {
    "path": "layout/includes/head/google_adsense.pug",
    "chars": 359,
    "preview": "if (theme.google_adsense && theme.google_adsense.enable)\r\n  script(async src=theme.google_adsense.js)\r\n\r\n  if theme.goog"
  },
  {
    "path": "layout/includes/head/preconnect.pug",
    "chars": 1396,
    "preview": "-\r\n  const { internal_provider, third_party_provider, custom_format } = theme.CDN\r\n  const providers = {\r\n    'jsdelivr'"
  },
  {
    "path": "layout/includes/head/pwa.pug",
    "chars": 602,
    "preview": "- const { manifest, theme_color, apple_touch_icon, favicon_32_32, favicon_16_16, mask_icon } = theme.pwa\r\n\r\nlink(rel=\"ma"
  },
  {
    "path": "layout/includes/head/site_verification.pug",
    "chars": 113,
    "preview": "if theme.site_verification\r\n  each item in theme.site_verification\r\n    meta(name=item.name content=item.content)"
  },
  {
    "path": "layout/includes/head/structured_data.pug",
    "chars": 2167,
    "preview": "if theme.structured_data\r\n  if page.layout === 'post'\r\n    -\r\n      // https://developers.google.com/search/docs/appeara"
  },
  {
    "path": "layout/includes/head.pug",
    "chars": 2995,
    "preview": "- var pageTitle\r\n- globalPageType === 'archive' ? page.title = findArchivesTitle(page, theme.menu, date) : ''\r\ncase glob"
  },
  {
    "path": "layout/includes/header/index.pug",
    "chars": 1937,
    "preview": "-\r\n  const returnTopImg = img => img !== false ? img || theme.default_top_img : false\r\n  const isFixedClass = theme.nav."
  },
  {
    "path": "layout/includes/header/menu_item.pug",
    "chars": 1060,
    "preview": "if theme.menu\r\n  .menus_items\r\n    each value, label in theme.menu\r\n      if typeof value !== 'object'\r\n        .menus_i"
  },
  {
    "path": "layout/includes/header/nav.pug",
    "chars": 841,
    "preview": "nav#nav\r\n  span#blog-info\r\n    a.nav-site-title(href=url_for('/'))\r\n      if theme.nav.logo\r\n        img.site-icon(src=u"
  },
  {
    "path": "layout/includes/header/post-info.pug",
    "chars": 6894,
    "preview": "- let comments = theme.comments\r\n#post-info\r\n  h1.post-title= page.title || _p('no_title')\r\n    if theme.post_edit.enabl"
  },
  {
    "path": "layout/includes/header/social.pug",
    "chars": 343,
    "preview": "each url, icon in theme.social\r\n  -\r\n    const [link, title, color] = url.split('||').map(i => trim(i))\r\n    const href "
  },
  {
    "path": "layout/includes/layout.pug",
    "chars": 2338,
    "preview": "- var globalPageType = getPageType(page, is_home)\r\n- var htmlClassHideAside = theme.aside.enable && theme.aside.hide ? '"
  },
  {
    "path": "layout/includes/loading/fullpage-loading.pug",
    "chars": 1245,
    "preview": "#loading-box\r\n  .loading-left-bg\r\n  .loading-right-bg\r\n  .spinner-box\r\n    .configure-border-1\r\n      .configure-core\r\n "
  },
  {
    "path": "layout/includes/loading/index.pug",
    "chars": 128,
    "preview": "if theme.preloader.enable\r\n  if theme.preloader.source === 1\r\n    include ./fullpage-loading.pug\r\n  else\r\n    include ./"
  },
  {
    "path": "layout/includes/loading/pace.pug",
    "chars": 298,
    "preview": "script.\r\n  window.paceOptions = {\r\n    restartOnPushState: false\r\n  }\r\n\r\n  btf.addGlobalFn('pjaxSend', () => {\r\n    Pace"
  },
  {
    "path": "layout/includes/mixins/article-sort.pug",
    "chars": 1205,
    "preview": "mixin articleSort(posts)\r\n  .article-sort\r\n    - let year\r\n    - posts.forEach(article => {\r\n      - const tempYear = da"
  },
  {
    "path": "layout/includes/mixins/indexPostUI.pug",
    "chars": 6307,
    "preview": "mixin indexPostUI()\r\n  - const indexLayout = theme.index_layout\r\n  - const masonryLayoutClass = [6, 7].includes(indexLay"
  },
  {
    "path": "layout/includes/page/404.pug",
    "chars": 268,
    "preview": "- var top_img_404 = theme.error_404.background || theme.default_top_img\r\n\r\n.error-content\r\n  .error-img\r\n    img(src=url"
  },
  {
    "path": "layout/includes/page/categories.pug",
    "chars": 35,
    "preview": ".category-lists!= list_categories()"
  },
  {
    "path": "layout/includes/page/default-page.pug",
    "chars": 47,
    "preview": "#article-container.container\r\n  != page.content"
  },
  {
    "path": "layout/includes/page/flink.pug",
    "chars": 3509,
    "preview": "#article-container.container\r\n  .flink\r\n    - let { content, random, flink_url } = page\r\n    - let pageContent = content"
  },
  {
    "path": "layout/includes/page/shuoshuo.pug",
    "chars": 12958,
    "preview": "//- - author:\r\n//-   avatar:\r\n//-   date:\r\n//-   content:\r\n//-   tags:\r\n//-     - tag1\r\n//-     - tag2\r\n\r\n- page.toc = f"
  },
  {
    "path": "layout/includes/page/tags.pug",
    "chars": 217,
    "preview": ".tag-cloud-list.text-center\r\n  !=cloudTags({source: site.tags, orderby: page.orderby || 'random', order: page.order || 1"
  },
  {
    "path": "layout/includes/pagination.pug",
    "chars": 1705,
    "preview": "if page.total !== 1\r\n  -\r\n    var options = {\r\n      prev_text: '<i class=\"fas fa-chevron-left fa-fw\"></i>',\r\n      next"
  },
  {
    "path": "layout/includes/post/outdate-notice.pug",
    "chars": 380,
    "preview": "- const { limit_day, message_prev, message_next, position} = theme.noticeOutdate\r\n- const notice_data = { limitDay: limi"
  },
  {
    "path": "layout/includes/post/post-copyright.pug",
    "chars": 1134,
    "preview": "if theme.post_copyright.enable && page.copyright !== false\r\n  - const author = page.copyright_author || config.author\r\n "
  },
  {
    "path": "layout/includes/post/reward.pug",
    "chars": 404,
    "preview": ".post-reward\r\n  .reward-button\r\n    i.fas.fa-qrcode\r\n    = theme.reward.text || _p('donate')\r\n  .reward-main\r\n    ul.rew"
  },
  {
    "path": "layout/includes/rightside.pug",
    "chars": 2302,
    "preview": "- const { readmode, translate, darkmode, aside, chat } = theme\r\n\r\nmixin rightsideItem(array)\r\n  each item in array\r\n    "
  },
  {
    "path": "layout/includes/sidebar.pug",
    "chars": 729,
    "preview": "if theme.menu\r\n  #sidebar\r\n    #menu-mask\r\n    #sidebar-menus\r\n      .avatar-img.text-center\r\n        img(src=url_for(th"
  },
  {
    "path": "layout/includes/third-party/abcjs/abcjs.pug",
    "chars": 1392,
    "preview": "script.\r\n  (() => {\r\n    const abcjsInit = () => {\r\n      const abcjsFn = () => {\r\n        setTimeout(() => {\r\n         "
  },
  {
    "path": "layout/includes/third-party/abcjs/index.pug",
    "chars": 135,
    "preview": "if theme.abcjs.enable\r\n  if theme.abcjs.per_page && (['post','page'].includes(globalPageType)) || page.abcjs\r\n    includ"
  },
  {
    "path": "layout/includes/third-party/aplayer.pug",
    "chars": 820,
    "preview": "link(rel='stylesheet' href=url_for(theme.asset.aplayer_css) media=\"print\" onload=\"this.media='all'\")\r\nscript(src=url_for"
  },
  {
    "path": "layout/includes/third-party/card-post-count/artalk.pug",
    "chars": 918,
    "preview": "- const { server, site } = theme.artalk\r\n\r\nscript.\r\n  (() => {\r\n    const getArtalkCount = async() => {\r\n      try {\r\n  "
  },
  {
    "path": "layout/includes/third-party/card-post-count/disqus.pug",
    "chars": 945,
    "preview": "- const { shortname, apikey } = theme.disqus\r\nscript.\r\n  (() => {\r\n    const getCount = async () => {\r\n      try {\r\n    "
  },
  {
    "path": "layout/includes/third-party/card-post-count/fb.pug",
    "chars": 686,
    "preview": "- const fbSDKVer = 'v20.0'\r\n- const fbSDK = `https://connect.facebook.net/${theme.facebook_comments.lang}/sdk.js#xfbml=1"
  },
  {
    "path": "layout/includes/third-party/card-post-count/index.pug",
    "chars": 357,
    "preview": "case theme.comments.use[0]\r\n  when 'Twikoo'\r\n    include ./twikoo.pug\r\n  when 'Disqus'\r\n  when 'Disqusjs'\r\n    include ."
  },
  {
    "path": "layout/includes/third-party/card-post-count/remark42.pug",
    "chars": 484,
    "preview": "- const { host, siteId, option } = theme.remark42\r\n\r\nscript.\r\n  (()=>{\r\n    window.remark_config = Object.assign({\r\n    "
  },
  {
    "path": "layout/includes/third-party/card-post-count/twikoo.pug",
    "chars": 1118,
    "preview": "script.\r\n  (() => {\r\n    const getCommentUrl = () => {\r\n      const eleGroup = document.querySelectorAll('#recent-posts "
  },
  {
    "path": "layout/includes/third-party/card-post-count/valine.pug",
    "chars": 581,
    "preview": "script.\r\n  (() => {\r\n    function loadValine () {\r\n      function initValine () {\r\n        let initData = {\r\n          e"
  },
  {
    "path": "layout/includes/third-party/card-post-count/waline.pug",
    "chars": 732,
    "preview": "- const serverURL = theme.waline.serverURL.replace(/\\/$/, '')\r\nscript.\r\n  (() => {\r\n    async function loadWaline () {\r\n"
  },
  {
    "path": "layout/includes/third-party/chat/chatra.pug",
    "chars": 1083,
    "preview": "//- https://chatra.io/help/api/\r\nscript.\r\n  (() => {\r\n    window.ChatraID = '#{theme.chatra.id}'\r\n    window.Chatra = wi"
  },
  {
    "path": "layout/includes/third-party/chat/crisp.pug",
    "chars": 955,
    "preview": "script.\r\n  (() => {\r\n    window.$crisp = ['safe', true]\r\n    window.CRISP_WEBSITE_ID = \"!{theme.crisp.website_id}\"\r\n\r\n  "
  },
  {
    "path": "layout/includes/third-party/chat/index.pug",
    "chars": 144,
    "preview": "case theme.chat.use\r\n  when 'chatra'\r\n    include ./chatra.pug\r\n  when 'tidio'\r\n    include ./tidio.pug\r\n  when 'crisp'\r"
  },
  {
    "path": "layout/includes/third-party/chat/tidio.pug",
    "chars": 1334,
    "preview": "script.\r\n  (() => {\r\n    btf.getScript('//code.tidio.co/!{theme.tidio.public_key}.js').then(() => {\r\n      const isChatB"
  },
  {
    "path": "layout/includes/third-party/comments/artalk.pug",
    "chars": 2383,
    "preview": "- const { server, site, option } = theme.artalk\r\n- const { use, lazyload } = theme.comments\r\n\r\nscript.\r\n  (() => {\r\n    "
  },
  {
    "path": "layout/includes/third-party/comments/disqus.pug",
    "chars": 2601,
    "preview": "- const disqusPageTitle = page.title.replace(/'/ig,\"\\\\'\")\r\n- const { shortname, apikey } = theme.disqus\r\n- const { use, "
  },
  {
    "path": "layout/includes/third-party/comments/disqusjs.pug",
    "chars": 2981,
    "preview": "- let disqusjsPageTitle = page.title && page.title.replace(/'/ig,\"\\\\'\")\r\n- const { shortname:dqShortname, apikey:dqApike"
  },
  {
    "path": "layout/includes/third-party/comments/facebook_comments.pug",
    "chars": 2541,
    "preview": "- const fbSDKVer = 'v20.0'\r\n- const fbSDK = `https://connect.facebook.net/${theme.facebook_comments.lang}/sdk.js#xfbml=1"
  },
  {
    "path": "layout/includes/third-party/comments/giscus.pug",
    "chars": 2613,
    "preview": "- const { use, lazyload } = theme.comments\r\n- const { repo, repo_id, category_id, light_theme, dark_theme, js, option } "
  },
  {
    "path": "layout/includes/third-party/comments/gitalk.pug",
    "chars": 1896,
    "preview": "- const { client_id, client_secret, repo, owner, admin, option } = theme.gitalk\r\n\r\nscript.\r\n  (() => {\r\n    const isShuo"
  },
  {
    "path": "layout/includes/third-party/comments/index.pug",
    "chars": 1415,
    "preview": "- let defaultComment = theme.comments.use[0]\r\nhr.custom-hr\r\n#post-comment\r\n  .comment-head\r\n    .comment-headline\r\n     "
  },
  {
    "path": "layout/includes/third-party/comments/js.pug",
    "chars": 1030,
    "preview": "each name in theme.comments.use\r\n  case name\r\n    when 'Valine'\r\n      !=partial('includes/third-party/comments/valine',"
  },
  {
    "path": "layout/includes/third-party/comments/livere.pug",
    "chars": 1382,
    "preview": "- const { use, lazyload } = theme.comments\r\n\r\nscript.\r\n  (() => {\r\n    const isShuoshuo = GLOBAL_CONFIG_SITE.pageType =="
  },
  {
    "path": "layout/includes/third-party/comments/remark42.pug",
    "chars": 2486,
    "preview": "- const { host, siteId, option } = theme.remark42\r\n\r\nscript.\r\n  (() => {\r\n    const isShuoshuo = GLOBAL_CONFIG_SITE.page"
  },
  {
    "path": "layout/includes/third-party/comments/twikoo.pug",
    "chars": 2043,
    "preview": "- const { envId, region, option } = theme.twikoo\r\n- const { use, lazyload, count } = theme.comments\r\n\r\nscript.\r\n  (() =>"
  },
  {
    "path": "layout/includes/third-party/comments/utterances.pug",
    "chars": 2225,
    "preview": "- const { use, lazyload } = theme.comments\r\n- const { repo, issue_term, light_theme, dark_theme, js, option } = theme.ut"
  },
  {
    "path": "layout/includes/third-party/comments/valine.pug",
    "chars": 1717,
    "preview": "- const { use, lazyload } = theme.comments\r\n- const { appId, appKey, avatar, serverURLs, visitor, option } = theme.valin"
  },
  {
    "path": "layout/includes/third-party/comments/waline.pug",
    "chars": 1899,
    "preview": "- const { serverURL, option, pageview } = theme.waline\r\n- const { lazyload, count, use } = theme.comments\r\n\r\nscript.\r\n  "
  },
  {
    "path": "layout/includes/third-party/effect.pug",
    "chars": 1822,
    "preview": "if theme.fireworks && theme.fireworks.enable\r\n  canvas.fireworks(mobile=`${theme.fireworks.mobile}`)\r\n  script(src=url_f"
  },
  {
    "path": "layout/includes/third-party/math/chartjs.pug",
    "chars": 3375,
    "preview": "- const { fontColor, borderColor, scale_ticks_backdropColor } = theme.chartjs\r\n\r\nscript.\r\n  (() => {\r\n    const applyThe"
  },
  {
    "path": "layout/includes/third-party/math/index.pug",
    "chars": 397,
    "preview": "case theme.math.use\r\n  when 'mathjax'\r\n    if (theme.math.per_page && (['post','page'].includes(globalPageType))) || pag"
  },
  {
    "path": "layout/includes/third-party/math/katex.pug",
    "chars": 453,
    "preview": "script.\r\n  (async () => {\r\n    const showKatex = () => {\r\n      document.querySelectorAll('#article-container .katex').f"
  },
  {
    "path": "layout/includes/third-party/math/mathjax.pug",
    "chars": 2821,
    "preview": "//- Mathjax 4/5\r\n- const { tags, enableMenu } = theme.math.mathjax\r\nscript.\r\n  (() => {\r\n    const loadMathjax = () => {"
  },
  {
    "path": "layout/includes/third-party/math/mermaid.pug",
    "chars": 12654,
    "preview": "script.\r\n  (() => {\r\n    const parseViewBox = viewBox => {\r\n      if (!viewBox) return null\r\n      const parts = viewBox"
  },
  {
    "path": "layout/includes/third-party/newest-comments/artalk.pug",
    "chars": 2821,
    "preview": "- const { server, site, option } = theme.artalk\r\n- const avatarCdn = (option !== null && option.gravatar && option.grava"
  },
  {
    "path": "layout/includes/third-party/newest-comments/common.pug",
    "chars": 2633,
    "preview": "script.\r\n  window.newestComments = {\r\n    changeContent: content => {\r\n      if (content === '') return content\r\n\r\n     "
  },
  {
    "path": "layout/includes/third-party/newest-comments/disqus-comment.pug",
    "chars": 1183,
    "preview": "!= partial(\"includes/third-party/newest-comments/common.pug\", {}, { cache: true })\r\n\r\nscript.\r\n  window.addEventListener"
  },
  {
    "path": "layout/includes/third-party/newest-comments/github-issues.pug",
    "chars": 2078,
    "preview": "!= partial(\"includes/third-party/newest-comments/common.pug\", {}, { cache: true })\r\n\r\nscript.\r\n  window.addEventListener"
  },
  {
    "path": "layout/includes/third-party/newest-comments/index.pug",
    "chars": 1008,
    "preview": "- let { use } = theme.comments\r\n\r\nif use\r\n  -\r\n    let forum,apiKey,userRepo\r\n    let { limit:newestCommentsLimit } = th"
  },
  {
    "path": "layout/includes/third-party/newest-comments/remark42.pug",
    "chars": 1108,
    "preview": "- const { host, siteId } = theme.remark42\r\n!= partial(\"includes/third-party/newest-comments/common.pug\", {}, { cache: tr"
  },
  {
    "path": "layout/includes/third-party/newest-comments/twikoo-comment.pug",
    "chars": 1392,
    "preview": "!= partial(\"includes/third-party/newest-comments/common.pug\", {}, { cache: true })\r\n\r\nscript.\r\n  window.addEventListener"
  },
  {
    "path": "layout/includes/third-party/newest-comments/valine.pug",
    "chars": 1865,
    "preview": "- let default_avatar = theme.valine.avatar\r\n\r\nscript(src=url_for(theme.asset.blueimp_md5))\r\n!= partial(\"includes/third-p"
  },
  {
    "path": "layout/includes/third-party/newest-comments/waline.pug",
    "chars": 1154,
    "preview": "- const serverURL = theme.waline.serverURL.replace(/\\/$/, '')\r\n\r\n!= partial(\"includes/third-party/newest-comments/common"
  },
  {
    "path": "layout/includes/third-party/pjax.pug",
    "chars": 2672,
    "preview": "- var pjaxExclude = 'a:not([target=\"_blank\"])'\r\nif theme.pjax.exclude\r\n  each val in theme.pjax.exclude\r\n    - pjaxExclu"
  },
  {
    "path": "layout/includes/third-party/prismjs.pug",
    "chars": 788,
    "preview": "- const { prismjs_js, prismjs_autoloader, prismjs_lineNumber_js } = theme.asset\r\n- const { prismjs, syntax_highlighter }"
  },
  {
    "path": "layout/includes/third-party/search/algolia.pug",
    "chars": 7280,
    "preview": "#algolia-search\r\n  .search-dialog\r\n    nav.search-nav\r\n      span.search-dialog-title= _p('search.title')\r\n      i.fas.f"
  },
  {
    "path": "layout/includes/third-party/search/docsearch.pug",
    "chars": 923,
    "preview": "- const { placeholder, docsearch: { appId, apiKey, indexName, option } } = theme.search\r\n\r\n.docsearch-wrap\r\n  #docsearch"
  },
  {
    "path": "layout/includes/third-party/search/index.pug",
    "chars": 177,
    "preview": "case theme.search.use\r\n  when 'algolia_search'\r\n    include ./algolia.pug\r\n  when 'local_search'\r\n    include ./local-se"
  },
  {
    "path": "layout/includes/third-party/search/local-search.pug",
    "chars": 683,
    "preview": "#local-search\r\n  .search-dialog\r\n    nav.search-nav\r\n      span.search-dialog-title= _p('search.title')\r\n      i.fas.fa-"
  },
  {
    "path": "layout/includes/third-party/share/addtoany.pug",
    "chars": 299,
    "preview": ".addtoany\n  .a2a_kit.a2a_kit_size_32.a2a_default_style\n    - let addtoanyItem = theme.share.addtoany.item.split(',')\n   "
  },
  {
    "path": "layout/includes/third-party/share/index.pug",
    "chars": 222,
    "preview": "- const { use } = theme.share\r\n\r\nif use\r\n  .post-share\r\n    case use\r\n      when 'addtoany'\r\n        !=partial('includes"
  },
  {
    "path": "layout/includes/third-party/share/share-js.pug",
    "chars": 309,
    "preview": "- const coverVal = page.cover_type === 'img' ? page.cover : theme.avatar.img\r\n.social-share(data-image=url_for(coverVal)"
  },
  {
    "path": "layout/includes/third-party/subtitle.pug",
    "chars": 3607,
    "preview": "- const { effect, source, sub, typed_option } = theme.subtitle\r\n- let subContent = sub || new Array()\r\n\r\nscript.\r\n  wind"
  },
  {
    "path": "layout/includes/third-party/umami_analytics.pug",
    "chars": 3948,
    "preview": "- let { serverURL, script_name, website_id, option, UV_PV } = theme.umami_analytics\r\n- const isServerURL = !!serverURL\r\n"
  },
  {
    "path": "layout/includes/widget/card_ad.pug",
    "chars": 79,
    "preview": "if theme.ad && theme.ad.aside\r\n  .card-widget.ads-wrap\r\n    != theme.ad.aside\r\n"
  },
  {
    "path": "layout/includes/widget/card_announcement.pug",
    "chars": 237,
    "preview": "if theme.aside.card_announcement.enable\r\n  .card-widget.card-announcement\r\n    .item-headline\r\n      i.fas.fa-bullhorn.f"
  },
  {
    "path": "layout/includes/widget/card_archives.pug",
    "chars": 436,
    "preview": "if theme.aside.card_archives.enable\r\n  .card-widget.card-archives\r\n    - let type = theme.aside.card_archives.type || 'm"
  },
  {
    "path": "layout/includes/widget/card_author.pug",
    "chars": 1094,
    "preview": "if theme.aside.card_author.enable\r\n  .card-widget.card-info.text-center\r\n    .avatar-img\r\n      img(src=url_for(theme.av"
  },
  {
    "path": "layout/includes/widget/card_bottom_self.pug",
    "chars": 310,
    "preview": "if site.data.widget && site.data.widget.bottom\r\n  each item in site.data.widget.bottom\r\n    .card-widget(class=item.clas"
  },
  {
    "path": "layout/includes/widget/card_categories.pug",
    "chars": 270,
    "preview": "if theme.aside.card_categories.enable\r\n  if site.categories.length\r\n    .card-widget.card-categories\r\n      !=aside_cate"
  },
  {
    "path": "layout/includes/widget/card_newest_comment.pug",
    "chars": 359,
    "preview": "if theme.aside.card_newest_comments.enable && theme.comments.use && !['Livere','Facebook Comments','Giscus'].includes(th"
  },
  {
    "path": "layout/includes/widget/card_post_series.pug",
    "chars": 1131,
    "preview": "if theme.aside.card_post_series.enable\r\n  - const array = fragment_cache('seriesArr', groupPosts)\r\n  .card-widget.card-p"
  },
  {
    "path": "layout/includes/widget/card_post_toc.pug",
    "chars": 603,
    "preview": "- let tocNumber = typeof page.toc_number === 'boolean' ? page.toc_number : theme.toc.number\r\n- let tocExpand = typeof pa"
  },
  {
    "path": "layout/includes/widget/card_recent_post.pug",
    "chars": 1609,
    "preview": "if theme.aside.card_recent_post.enable\r\n  .card-widget.card-recent-post\r\n    .item-headline\r\n      i.fas.fa-history\r\n   "
  },
  {
    "path": "layout/includes/widget/card_tags.pug",
    "chars": 727,
    "preview": "if theme.aside.card_tags.enable\r\n  if site.tags.length\r\n    .card-widget.card-tags\r\n      .item-headline\r\n        i.fas."
  },
  {
    "path": "layout/includes/widget/card_top_self.pug",
    "chars": 253,
    "preview": "if site.data.widget && site.data.widget.top\r\n  each item in site.data.widget.top\r\n    .card-widget(class=item.class_name"
  },
  {
    "path": "layout/includes/widget/card_webinfo.pug",
    "chars": 2038,
    "preview": "if theme.aside.card_webinfo.enable\r\n  .card-widget.card-webinfo\r\n    .item-headline\r\n      i.fas.fa-chart-line\r\n      sp"
  },
  {
    "path": "layout/includes/widget/index.pug",
    "chars": 1673,
    "preview": "#aside-content.aside-content\r\n  //- post\r\n  if globalPageType === 'post'\r\n    - const tocStyle = page.toc_style_simple\r\n"
  },
  {
    "path": "layout/index.pug",
    "chars": 105,
    "preview": "extends includes/layout.pug\r\n\r\nblock content\r\n  include ./includes/mixins/indexPostUI.pug\r\n  +indexPostUI"
  },
  {
    "path": "layout/page.pug",
    "chars": 926,
    "preview": "extends includes/layout.pug\r\n\r\nblock content\r\n  - const noCardLayout = ['shuoshuo', '404'].includes(page.type) ? 'nc' : "
  },
  {
    "path": "layout/post.pug",
    "chars": 1187,
    "preview": "extends includes/layout.pug\r\n\r\nblock content\r\n  #post\r\n    if top_img === false\r\n      include includes/header/post-info"
  },
  {
    "path": "layout/tag.pug",
    "chars": 338,
    "preview": "extends includes/layout.pug\r\n\r\nblock content\r\n  if theme.tag_ui == 'index'\r\n    include ./includes/mixins/indexPostUI.pu"
  },
  {
    "path": "package.json",
    "chars": 881,
    "preview": "{\r\n  \"name\": \"hexo-theme-butterfly\",\r\n  \"version\": \"5.5.4\",\r\n  \"description\": \"A Simple and Card UI Design theme for Hex"
  },
  {
    "path": "plugins.yml",
    "chars": 4489,
    "preview": "abcjs_basic_js:\r\n  name: abcjs\r\n  file: dist/abcjs-basic-min.js\r\n  version: 6.6.0\r\nactivate_power_mode:\r\n  name: butterf"
  },
  {
    "path": "scripts/common/default_config.js",
    "chars": 10828,
    "preview": "// Butterfly 主題默認配置\n// Default configuration for Butterfly theme\n\nmodule.exports = {\n  nav: {\n    logo: null,\n    displa"
  },
  {
    "path": "scripts/common/postDesc.js",
    "chars": 1045,
    "preview": "'use strict'\r\n\r\nconst { stripHTML, truncate } = require('hexo-util')\r\n\r\n// Truncates the given content to a specified le"
  },
  {
    "path": "scripts/events/404.js",
    "chars": 345,
    "preview": "/**\r\n * Butterfly\r\n * 404 error page\r\n */\r\n\r\n'use strict'\r\n\r\nhexo.extend.generator.register('404', function (locals) {\r\n"
  },
  {
    "path": "scripts/events/cdn.js",
    "chars": 2869,
    "preview": "/**\r\n * Butterfly\r\n * Merge CDN\r\n */\r\n\r\n'use strict'\r\n\r\nconst { version } = require('../../package.json')\r\nconst path = "
  },
  {
    "path": "scripts/events/init.js",
    "chars": 2304,
    "preview": "const { deepMerge } = require('hexo-util')\nconst path = require('path')\n\n// Cache default config to avoid repeated file "
  },
  {
    "path": "scripts/events/stylus.js",
    "chars": 855,
    "preview": "/**\r\n * Stylus renderer\r\n */\r\n\r\n'use strict'\r\n\r\nhexo.extend.filter.register('stylus:renderer', style => {\r\n  const { syn"
  },
  {
    "path": "scripts/events/welcome.js",
    "chars": 679,
    "preview": "hexo.on('ready', () => {\r\n  const { version } = require('../../package.json')\r\n  hexo.log.info(`\r\n  ===================="
  },
  {
    "path": "scripts/filters/post_lazyload.js",
    "chars": 1645,
    "preview": "/**\r\n * Butterfly\r\n * Lazyload filter\r\n * Replace src with data-lazy-src for lazy loading\r\n */\r\n\r\n'use strict'\r\n\r\nconst "
  },
  {
    "path": "scripts/filters/random_cover.js",
    "chars": 2445,
    "preview": "/**\r\n * Random cover for posts\r\n */\r\n\r\n'use strict'\r\n\r\nhexo.extend.generator.register('post', locals => {\r\n  const imgTe"
  },
  {
    "path": "scripts/helpers/aside_archives.js",
    "chars": 3273,
    "preview": "'use strict'\r\n\r\nhexo.extend.helper.register('aside_archives', function (options = {}) {\r\n  const { config, page, site, u"
  },
  {
    "path": "scripts/helpers/aside_categories.js",
    "chars": 3441,
    "preview": "'use strict'\r\n\r\nhexo.extend.helper.register('aside_categories', function (categories, options = {}) {\r\n  if (!categories"
  },
  {
    "path": "scripts/helpers/getArchiveLength.js",
    "chars": 1358,
    "preview": "hexo.extend.helper.register('getArchiveLength', function () {\r\n  const archiveGenerator = hexo.config.archive_generator\r"
  },
  {
    "path": "scripts/helpers/inject_head_js.js",
    "chars": 5534,
    "preview": "'use strict'\r\n\r\nhexo.extend.helper.register('inject_head_js', function () {\r\n  const { darkmode, aside, pjax } = this.th"
  },
  {
    "path": "scripts/helpers/page.js",
    "chars": 6671,
    "preview": "'use strict'\r\n\r\nconst { truncateContent, postDesc } = require('../common/postDesc')\r\nconst { prettyUrls } = require('hex"
  },
  {
    "path": "scripts/helpers/related_post.js",
    "chars": 3077,
    "preview": "/* eslint-disable camelcase */\r\n/**\r\n * Butterfly\r\n * Related Posts\r\n * According the tag\r\n */\r\n\r\n'use strict'\r\n\r\nconst "
  },
  {
    "path": "scripts/helpers/series.js",
    "chars": 601,
    "preview": "'use strict'\r\n\r\nhexo.extend.helper.register('groupPosts', function () {\r\n  const getGroupArray = array => {\r\n    return "
  },
  {
    "path": "scripts/tag/button.js",
    "chars": 649,
    "preview": "/**\r\n * Button\r\n * {% btn url text icon option %}\r\n * option: color outline center block larger\r\n * color : default/blue"
  },
  {
    "path": "scripts/tag/chartjs.js",
    "chars": 1596,
    "preview": "/**\r\n * Butterfly\r\n * chartjs\r\n * https://www.chartjs.org/\r\n * {% chartjs [width, abreast, chartId] %}\r\n * <!-- chart --"
  },
  {
    "path": "scripts/tag/flink.js",
    "chars": 1222,
    "preview": "/**\r\n * flink\r\n */\r\n\r\n'use strict'\r\n\r\nconst urlFor = require('hexo-util').url_for.bind(hexo)\r\n\r\nconst flinkFn = (args, c"
  },
  {
    "path": "scripts/tag/gallery.js",
    "chars": 2269,
    "preview": "/**\r\n * Butterfly\r\n * galleryGroup and gallery\r\n * {% galleryGroup [name] [descr] [url] [img] %}\r\n *\r\n * {% gallery [but"
  },
  {
    "path": "scripts/tag/hide.js",
    "chars": 1956,
    "preview": "/**\r\n * Butterfly\r\n * @example\r\n * hideInline\r\n * {% hideInline content,display,bg,color %}\r\n * content不能包含當引號,可用 &apos;"
  },
  {
    "path": "scripts/tag/inlineImg.js",
    "chars": 541,
    "preview": "/**\r\n * inlineImg\r\n * @param {Array} args - Image name and height\r\n * @param {string} args[0] - Image name\r\n * @param {n"
  },
  {
    "path": "scripts/tag/label.js",
    "chars": 283,
    "preview": "/**\r\n * Butterfly\r\n * label\r\n * {% label text color %}\r\n */\r\n\r\n'use strict'\r\n\r\nconst addLabel = args => {\r\n  const [text"
  },
  {
    "path": "scripts/tag/mermaid.js",
    "chars": 433,
    "preview": "/**\r\n * Butterfly\r\n * mermaid\r\n * https://github.com/mermaid-js/mermaid\r\n */\r\n\r\n'use strict'\r\n\r\nconst { escapeHTML } = r"
  },
  {
    "path": "scripts/tag/note.js",
    "chars": 801,
    "preview": "/**\r\n * note.js\r\n * transplant from hexo-theme-next\r\n * Modify by Jerry\r\n */\r\n\r\n'use strict'\r\n\r\nconst postNote = (args, "
  },
  {
    "path": "scripts/tag/score.js",
    "chars": 1398,
    "preview": "/**\r\n * Music Score\r\n * {% score %}\r\n */\r\n\r\n'use strict'\r\n\r\nconst score = (args, content) => {\r\n  // Escape HTML tags an"
  },
  {
    "path": "scripts/tag/series.js",
    "chars": 1687,
    "preview": "/**\r\n * series plugin\r\n * Syntax:\r\n *  {% series [series name] %}\r\n * Usage:\r\n * {% series %}\r\n * {% series series1 %}\r\n"
  },
  {
    "path": "scripts/tag/tabs.js",
    "chars": 1864,
    "preview": "/**\r\n * Tabs\r\n * transplant from hexo-theme-next\r\n * modify by Jerry\r\n */\r\n\r\n'use strict'\r\n\r\nconst postTabs = (args, con"
  },
  {
    "path": "scripts/tag/timeline.js",
    "chars": 1534,
    "preview": "/**\r\n * Timeline tag for Hexo\r\n * Syntax:\r\n * {% timeline [headline],[color] %}\r\n * <!-- timeline [title] -->\r\n * [conte"
  },
  {
    "path": "source/css/_global/function.styl",
    "chars": 6473,
    "preview": ".limit-one-line\r\n  overflow: hidden\r\n  text-overflow: ellipsis\r\n  white-space: nowrap\r\n\r\n.limit-more-line\r\n  display: -w"
  },
  {
    "path": "source/css/_global/index.styl",
    "chars": 7965,
    "preview": ":root\r\n  --global-font-size: $font-size\r\n  --global-bg: $body-bg\r\n  --font-color: $font-black\r\n  --hr-border: lighten($t"
  },
  {
    "path": "source/css/_highlight/highlight/diff.styl",
    "chars": 1331,
    "preview": "figure.highlight\r\n  table\r\n    @extend $scrollbar-style\r\n\r\n  pre .deletion\r\n    color: $highlight-deletion\r\n\r\n  pre .add"
  },
  {
    "path": "source/css/_highlight/highlight/index.styl",
    "chars": 821,
    "preview": "if $highlight_theme != false\r\n  @require 'diff'\r\n\r\n.container\r\n  figure.highlight\r\n    .line\r\n      if wordWrap\r\n       "
  },
  {
    "path": "source/css/_highlight/highlight.styl",
    "chars": 6634,
    "preview": "$highlight_theme = hexo-config('code_blocks.theme')\r\n$highlight_macstyle = hexo-config('code_blocks.macStyle')\r\nwordWrap"
  },
  {
    "path": "source/css/_highlight/prismjs/diff.styl",
    "chars": 5353,
    "preview": "if $highlight_theme == 'light'\r\n  // prism-base16-ateliersulphurpool.light\r\n  pre[class*='language-']\r\n    .token.functi"
  },
  {
    "path": "source/css/_highlight/prismjs/index.styl",
    "chars": 348,
    "preview": "if $prismjs_line_number\r\n  @require 'line-number'\r\n\r\nif $highlight_theme != false\r\n  @require 'diff'\r\n\r\n.container\r\n  pr"
  },
  {
    "path": "source/css/_highlight/prismjs/line-number.styl",
    "chars": 1079,
    "preview": ".container\r\n  pre[class*='language-']\r\n    &.line-numbers\r\n      position: relative\r\n      padding-left: 3.8em\r\n      co"
  },
  {
    "path": "source/css/_highlight/theme.styl",
    "chars": 3794,
    "preview": "if $highlight_theme == 'darker'\r\n  $highlight-background = #212121\r\n  $highlight-selection = #61616150\r\n  $highlight-for"
  },
  {
    "path": "source/css/_layout/aside.styl",
    "chars": 8867,
    "preview": "#aside-content\r\n  width: 26%\r\n\r\n  +minWidth900()\r\n    if hexo-config('aside.position') == 'right'\r\n      padding-left: 1"
  },
  {
    "path": "source/css/_layout/chat.styl",
    "chars": 287,
    "preview": "// chat\r\nif hexo-config('chat.rightside_button') == true\r\n  if hexo-config('chat.use') == 'chatra'\r\n    #chatra:not(.cha"
  },
  {
    "path": "source/css/_layout/comments.styl",
    "chars": 1722,
    "preview": "#post-comment\r\n  .comment-head\r\n    margin-bottom: 20px\r\n\r\n    &:after\r\n      display: block\r\n      clear: both\r\n      c"
  },
  {
    "path": "source/css/_layout/footer.styl",
    "chars": 1717,
    "preview": "#footer\r\n  position: relative\r\n  background-color: $light-blue\r\n  background-attachment: scroll\r\n  background-position: "
  },
  {
    "path": "source/css/_layout/head.styl",
    "chars": 9198,
    "preview": "#page-header\r\n  position: relative\r\n  width: 100%\r\n  background-color: $light-blue\r\n  background-position: center center"
  },
  {
    "path": "source/css/_layout/loading.styl",
    "chars": 2009,
    "preview": "if hexo-config('preloader.enable') && hexo-config('preloader.source') == 1\r\n  .loading-bg\r\n    position: fixed\r\n    z-in"
  },
  {
    "path": "source/css/_layout/pagination.styl",
    "chars": 1920,
    "preview": "#pagination\r\n  .pagination\r\n    margin-top: 20px\r\n    text-align: center\r\n\r\n  .page-number\r\n    &.current\r\n      backgro"
  },
  {
    "path": "source/css/_layout/post.styl",
    "chars": 4912,
    "preview": "beautify()\r\n  headStyle(fontsize)\r\n    padding-left: unit(fontsize + 8, 'px')\r\n\r\n    &:before\r\n      font-size: unit(fon"
  },
  {
    "path": "source/css/_layout/relatedposts.styl",
    "chars": 571,
    "preview": ".relatedPosts\r\n  margin-top: 40px\r\n\r\n  & > .headline\r\n    margin-bottom: 5px\r\n    font-weight: 700\r\n    font-size: 1.43e"
  },
  {
    "path": "source/css/_layout/reward.styl",
    "chars": 1644,
    "preview": ".post-reward\r\n  position: relative\r\n  margin-top: 80px\r\n  width: 100%\r\n  text-align: center\r\n  pointer-events: none\r\n\r\n "
  },
  {
    "path": "source/css/_layout/rightside.styl",
    "chars": 2151,
    "preview": "#rightside\r\n  position: fixed\r\n  right: -48px\r\n  bottom: $rightside-bottom\r\n  z-index: 100\r\n  opacity: 0\r\n  transition: "
  },
  {
    "path": "source/css/_layout/sidebar.styl",
    "chars": 2313,
    "preview": "#sidebar\r\n  #menu-mask\r\n    position: fixed\r\n    z-index: 102\r\n    display: none\r\n    width: 100%\r\n    height: 100%\r\n   "
  },
  {
    "path": "source/css/_layout/third-party.styl",
    "chars": 4397,
    "preview": "#vcomment\r\n  font-size: 1.1em\r\n\r\n  .vbtn\r\n    border: none\r\n    background: var(--btn-bg)\r\n    color: var(--btn-color)\r\n"
  },
  {
    "path": "source/css/_mode/darkmode.styl",
    "chars": 6990,
    "preview": "if hexo-config('darkmode.enable') || hexo-config('display_mode') == 'dark'\r\n  [data-theme='dark']\r\n    --global-bg: dark"
  },
  {
    "path": "source/css/_mode/readmode.styl",
    "chars": 4201,
    "preview": "if hexo-config('readmode')\r\n  .read-mode\r\n    --font-color: #4c4948\r\n    --readmode-light-color: #fff\r\n    --white: #4c4"
  },
  {
    "path": "source/css/_page/404.styl",
    "chars": 1347,
    "preview": "if hexo-config('error_404.enable')\r\n  .type-404\r\n    .error-content\r\n      @extend .cardHover\r\n      overflow: hidden\r\n "
  },
  {
    "path": "source/css/_page/archives.styl",
    "chars": 2298,
    "preview": ".article-sort\r\n  margin-left: 10px\r\n  padding-left: 20px\r\n  border-left: 2px solid lighten($light-blue, 20)\r\n\r\n  &-title"
  },
  {
    "path": "source/css/_page/categories.styl",
    "chars": 583,
    "preview": ".category-lists\r\n  .category-title\r\n    font-size: 2.57em\r\n\r\n    +maxWidth768()\r\n      font-size: 2em\r\n\r\n  .category-lis"
  },
  {
    "path": "source/css/_page/common.styl",
    "chars": 1035,
    "preview": "#body-wrap\r\n  display: flex\r\n  flex-direction: column\r\n  min-height: 100vh\r\n\r\n.layout\r\n  display: flex\r\n  flex: 1 auto\r\n"
  },
  {
    "path": "source/css/_page/flink.styl",
    "chars": 2007,
    "preview": ".container\r\n  .flink\r\n    margin-bottom: 20px\r\n\r\n    .flink-list\r\n      overflow: auto\r\n      padding: 10px 10px 0\r\n    "
  },
  {
    "path": "source/css/_page/homepage.styl",
    "chars": 4302,
    "preview": "$indexLayout = hexo-config('index_layout') || 1\r\n$indexEnable = hexo-config('cover.index_enable')\r\n\r\n#recent-posts\r\n  .r"
  },
  {
    "path": "source/css/_page/shuoshuo.styl",
    "chars": 3548,
    "preview": "#article-container\r\n  .shuoshuo-item\r\n    @extend .cardHover\r\n    margin-bottom: 20px\r\n    padding: 35px 30px 30px\r\n\r\n  "
  },
  {
    "path": "source/css/_page/tags.styl",
    "chars": 2118,
    "preview": ".tag-cloud\r\n  &-list\r\n    animation: tagsFadeIn .6s cubic-bezier(.4, 0, .2, 1)\r\n\r\n    &:hover a:not(:hover)\r\n      opaci"
  },
  {
    "path": "source/css/_search/algolia.styl",
    "chars": 266,
    "preview": "#algolia-search\r\n  .search-dialog\r\n    .ais-Hits-list\r\n      +maxWidth768()\r\n        min-height: calc(var(--search-heigh"
  },
  {
    "path": "source/css/_search/index.styl",
    "chars": 5116,
    "preview": ".search-dialog\r\n  position: fixed\r\n  top: 10%\r\n  left: 50%\r\n  z-index: 1001\r\n  display: none\r\n  margin-left: -300px\r\n  p"
  },
  {
    "path": "source/css/_search/local-search.styl",
    "chars": 464,
    "preview": "#local-search\r\n  .search-dialog\r\n    .search-result-list\r\n      +maxWidth768()\r\n        if hexo-config('search.local_sea"
  },
  {
    "path": "source/css/_tags/button.styl",
    "chars": 1621,
    "preview": ".container\r\n  .btn-center\r\n    margin: 0 0 20px\r\n    text-align: center\r\n\r\n  .btn-beautify\r\n    display: inline-block\r\n "
  },
  {
    "path": "source/css/_tags/gallery.styl",
    "chars": 4658,
    "preview": ".container\r\n  figure.gallery-group\r\n    position: relative\r\n    float: left\r\n    overflow: hidden\r\n    margin: 6px 4px\r\n"
  },
  {
    "path": "source/css/_tags/hexo.styl",
    "chars": 485,
    "preview": "// pullquote\r\nblockquote\r\n  &.pullquote\r\n    position: relative\r\n    max-width: 45%\r\n    font-size: 110%\r\n\r\n    &.left\r\n"
  },
  {
    "path": "source/css/_tags/hide.styl",
    "chars": 1184,
    "preview": "// tag-hide\r\n.hide-inline,\r\n.hide-block\r\n  & > .hide-button\r\n    display: inline-block\r\n    padding: 5px 18px\r\n    backg"
  },
  {
    "path": "source/css/_tags/inlineImg.styl",
    "chars": 117,
    "preview": ".container\r\n  .inline-img\r\n    display: inline\r\n    margin: 0 3px\r\n    height: 1.1em\r\n    vertical-align: text-bottom"
  },
  {
    "path": "source/css/_tags/label.styl",
    "chars": 269,
    "preview": ".hl-label\r\n  padding: 2px 4px\r\n  color: var(--btn-color, $btn-color)\r\n  addBorderRadius(3)\r\n\r\n  &.default\r\n    backgroun"
  }
]

// ... and 11 more files (download for full content)

About this extraction

This page contains the full source code of the jerryc127/hexo-theme-butterfly GitHub repository, extracted and formatted as plain text for AI agents and large language models (LLMs). The extraction includes 211 files (537.2 KB), approximately 162.9k tokens, and a symbol index with 26 extracted functions, classes, methods, constants, and types. Use this with OpenClaw, Claude, ChatGPT, Cursor, Windsurf, or any other AI tool that accepts text input. You can copy the full output to your clipboard or download it as a .txt file.

Extracted by GitExtract — free GitHub repo to text converter for AI. Built by Nikandr Surkov.

Copied to clipboard!