Compare commits

...

120 Commits
5.1.0 ... dev

Author SHA1 Message Date
Jerry
13315f6710 improvement: fancybox 手機頁面不顯示 toolbar 2026-04-08 15:53:06 +08:00
Jerry
e3e1d9e6ce feat: 新增 Mermaid 圖表功能的選項,支持在新標籤頁中打開和縮放平移互動
fix: 更新插件版本以修正相容性問題
refactor: 優化代碼結構,提升可讀性
2026-01-26 22:48:18 +08:00
Jerry
954598b45c Merge branch 'dev' of https://github.com/jerryc127/hexo-theme-butterfly into dev 2026-01-16 01:16:49 +08:00
Jerry
b6afb6edad fix: 修正 ABCJS 渲染使用 textContent 以提高音樂譜顯示準確性
style: 移除多餘的換行符號以保持代碼整潔
2026-01-16 01:16:47 +08:00
Jerry Wong
b06e704281 Merge pull request #1777 from bugwz/mermaid_svg_action
Add pan and zoom support for mermaid diagrams
2026-01-15 23:30:45 +08:00
Jerry
e4597deded update 2026-01-15 23:28:14 +08:00
Jerry Wong
b034637bc1 Merge pull request #1778 from Void4m0n/feature/cloudtags-custom-colors
feat(cloudTags): add custom colors support for cloudTags on card_tags widget & tags page Pr_Satus(Finished)
2026-01-15 23:08:29 +08:00
Void4m0n
bdc97cc22b fix(tagcloud): independent tags page and card_tags, drop colormode, use inline css, keep tag-color classes 2026-01-12 13:18:09 +01:00
Void4m0n
5878e3b7ee refactor(tagcloud): switch custom_colors from inline styles to CSS classes 2026-01-12 11:42:43 +01:00
Void4m0n
d15620198c feat(card_tags): add colormode/custom_colors to cloudTags call 2026-01-12 11:42:43 +01:00
Void4m0n
3b4b143c27 feat(cloudTags): add custom colors support for cloudTags on tags page 2026-01-12 11:42:43 +01:00
Jerry
2bab33d312 Merge branch 'dev' of https://github.com/jerryc127/hexo-theme-butterfly into dev 2026-01-10 13:19:26 +08:00
Jerry
b0951197a9 feat: 優化代碼高亮系統並修復樣式問題 (v5.5.4-b1)
- 簡化 config.pug 中的高亮設置檢測邏輯
- 重構滾動條樣式,提取為可重用的 $scrollbar-style
- 優化 JavaScript 代碼高亮邏輯,增加對默認高亮的支持
- 修復打賞彈窗的邊框圓角應用位置
- 改善代碼高亮工具的檢測和處理流程
2026-01-10 13:19:23 +08:00
Jerry Wong
572321a00f Merge pull request #1782 from akFace/dev
优化页面标题过长显示的问题
2026-01-06 17:21:24 +08:00
墨迹
e7f796e110 Update head.styl 2026-01-04 00:36:44 +08:00
Jerry Wong
02c8e8e3b5 Merge pull request #1784 from jerryc127/revert-1776-css/text-autospace
Revert "chore: use `text-autospace` by default"
2026-01-04 00:03:07 +08:00
Jerry Wong
edbd4d10c3 Revert "chore: use text-autospace by default" 2026-01-04 00:02:54 +08:00
墨迹
2cdf93c58c Restore page header styles and layout
Restored the page header styles with background and layout properties.
2026-01-02 15:16:27 +08:00
bugwz
2c5e8c7cbb Optimize mermaid view 2025-12-21 19:19:42 +08:00
Jerry Wong
80da110698 Merge pull request #1776 from DeepChirp/css/text-autospace
chore: use `text-autospace` by default
2025-12-21 12:37:37 +08:00
bugwz
0cc950a111 Add view button 2025-12-20 22:24:10 +08:00
bugwz
dd2d051b3d Add pan and zoom support for mermaid diagrams 2025-12-20 20:57:32 +08:00
DeepChirp
3f3102d10b chore: use text-autospace by default 2025-12-16 23:34:24 +08:00
Jerry
c771efa9bc Merge branch 'dev' of https://github.com/jerryc127/hexo-theme-butterfly into dev 2025-12-10 19:23:00 +08:00
Jerry Wong
1ebad6b1b7 Merge pull request #1769 from Void4m0n/add-post-pagination-cover-dev
Allow to set specific post pagination cover
2025-12-10 19:19:34 +08:00
Jerry
10c967957b 恢复修改 2025-12-10 19:18:12 +08:00
Jerry
f1397da086 chore: release v5.5.3
- Bump version from 5.5.3-b2 to 5.5.3 in package.json
- Update third-party dependencies:
  * algolia_search: 5.43.0 -> 5.46.0
  * docsearch: 4.3.1 -> 4.3.2
  * fancybox: 6.1.4 -> 6.1.7
  * katex: 0.16.25 -> 0.16.27
  * mermaid: 11.12.1 -> 11.12.2
  * waline: 3.7.1 -> 3.8.0

perf: optimize JavaScript performance
- Add defer attribute to script tags in pjax and prismjs
- Improve DOM content loading timing in pjax
- Optimize utilities with better throttle implementation
- Cache header positions for TOC performance
- Optimize related posts generation with Maps
- Improve archive helpers performance

fix: improve error handling and UI fixes
- Replace process.exit with proper error throwing
- Fix tooltip positioning with boundary checks
- Add btn-effects to readmode exit button
- Fix element height calculation for hidden elements
- Improve image filters in dark mode (brightness .88, contrast .95)

style: code improvements and consistency
- Refactor random cover generation with generator pattern
- Optimize data processing in helpers
- Clean up unused functions and improve code structure
- Fix Chinese translation: 页 -> 頁
2025-12-10 19:16:03 +08:00
Void4m0n
bb7b206369 support asset_folder for pagination_cover yaml 2025-11-25 21:37:48 +01:00
Void4m0n
586805ceb2 Search post pagination_cover yaml for prev/next card img 2025-11-25 21:37:21 +01:00
Jerry Wong
7dc7942230 Merge pull request #1764 from zhdbk3/dev
feat: 随机背景
2025-11-25 14:54:04 +08:00
Jerry Wong
8ae57ac9bf Update layout.pug 2025-11-25 14:52:42 +08:00
着火的冰块nya
57a7db7a52 fix: 随机背景适配 Pjax 2025-11-25 12:29:48 +08:00
Jerry
4225d23cb6 feat: 升級到 v5.5.3-b2 並優化 hide 標籤組件
- 升級版本號到 5.5.3-b2
- 優化 hide 標籤的 toggle 按鈕樣式,添加旋轉圖標動畫
- 增強 Umami Analytics,在 URL 參數中添加 path 欄位
- 為 toggle-button 添加暗色模式支援
- 修復代碼格式問題
2025-11-21 16:33:27 +08:00
着火的冰块nya
c882e84cd2 feat: 随机背景(修正上次的错误) 2025-11-18 10:27:29 +08:00
着火的冰块nya
29a6fa455a Merge branch 'jerryc127:dev' into dev 2025-11-17 18:21:26 +08:00
Jerry
7985bdda9e - fix: 修正 Umami Analytics API 參數與資料處理邏輯
- style: 最佳化程式碼高亮工具列顯示與間距設定
- improve: 改善複製工具提示的定位邏輯
- bump: 版本號更新至 5.5.3
2025-11-17 15:42:00 +08:00
Jerry Wong
bdf77e328d Merge pull request #1765 from zhdbk3/fix-1762
fix: 修复 busuanzi data-pjax src
2025-11-17 14:46:39 +08:00
Jerry Wong
65a077a00d Refactor additional-js.pug structure and scripts 2025-11-17 14:45:38 +08:00
Jerry Wong
17fc32b59d Merge pull request #1759 from DeepChirp/fix/umami-v3
fix(umami): adapt umami v3
2025-11-17 14:39:56 +08:00
着火的冰块nya
83574c75ad fix: 修复 busuanzi data-pjax src 2025-11-17 11:21:22 +08:00
着火的冰块nya
eb263085cf feat: 随机背景 2025-11-16 10:34:08 +08:00
DeepChirp
15b7a018c6 fix(umami): adapt umami v3 2025-11-08 11:54:34 +08:00
Jerry
439014bbb6 chore: 升級版本至 5.5.2 並更新相依套件
- perf(highlight): 改善複製提示 UI,新增浮動動畫效果
- fix(shuoshuo): 為說說內容中的圖片新增延遲載入支援
- fix(mermaid): 支援在標籤助手中使用自訂配置選項
- fix: 使用 url_for() 處理主題資源路徑 (busuanzi, artalk)
- refactor(aside_archives): 重新命名 url_for 變數以保持一致性
- refactor(inject_head_js): 使用解構賦值命名方式
- chore(deps): 更新 hexo-util 至 4.0.0
- chore(plugins): 更新多個 CDN 套件版本
- style(highlight): 優化程式碼區塊工具列佈局與溢位處理
- fix(zh-CN): 修正分頁文字使用正確的簡體字「页」
2025-11-07 22:19:16 +08:00
Jerry
2d4765202d Merge branch 'dev' of https://github.com/jerryc127/hexo-theme-butterfly into dev 2025-10-02 14:32:19 +08:00
Jerry
5e766ac40a chore: bump version to 5.5.1 and update dependencies
- Upgrade version from 5.5.0 to 5.5.1
- Update third-party dependencies: algolia, disqusjs, docsearch, fancybox, fontawesome, mermaid
- Replace Twitter with X in share configurations
- Enable CDN version numbers by default
- Fix shuoshuo page JSON security with safeJSON helper
- Improve image lazy loading regex to handle minified HTML
- Fix search result HTML structure and styling
- Add margin-top to search result numbering for better alignment
2025-10-02 14:32:08 +08:00
Jerry Wong
4c3a782610 Merge pull request #1736 from DeepChirp/fix-zh-cn
fix(languages): 将大陆地区用字改为简体(`頁`→`页`)
2025-09-09 18:31:48 +08:00
DeepChirp
8645a4355d fix(languages): 将大陆地区用字改为简体( 2025-09-09 16:36:25 +08:00
Jerry
4226c95818 Merge branch 'dev' of https://github.com/jerryc127/hexo-theme-butterfly into dev 2025-09-09 15:40:14 +08:00
Jerry
67c95cee0c 🔍 搜索功能改進:
- 新增本地搜索分頁配置 (enablePagination, hitsPerPage)
- 重構 Algolia 搜索邏輯,支援多索引和更好的錯誤處理
- 優化搜索 UI 樣式,包括分頁按鈕和響應式設計
- 改進搜索結果顯示,新增編號和更好的高亮處理

📦 依賴項更新:
- 更新 plugins.yml 中的多個插件版本 (abcjs, algolia, aplayer 等)
- 更新 package.json 版本號為 5.5.0

🎨 UI/UX 優化:
- 改進側邊欄和目錄的動畫效果
- 優化樣式佈局,調整寬度百分比
- 新增說說頁面的分頁導航組件
- 改進右側邊欄按鈕樣式

🐛 錯誤處理和代碼優化:
- 修復 Umami Analytics 的錯誤處理和數據驗證
- 改進懶加載圖片的正則表達式,避免匹配腳本標籤
- 移除未使用的變數和改進代碼結構
- 新增說說內容的 Markdown 渲染支援

🔧 其他改進:
- 更新翻譯功能,移除箭頭函數語法以提升相容性
2025-09-09 15:40:08 +08:00
Jerry Wong
0a72f43fae Merge pull request #1732 from DeepChirp/umami-script-name
feat(umami): 支持自定义脚本名称
2025-09-08 20:04:49 +08:00
Jerry Wong
006414da98 Merge pull request #1734 from DeepChirp/fix-pjax-failure
fix(Pjax): 增加错误处理以防止Pjax失效
2025-09-08 20:04:20 +08:00
DeepChirp
c61f55a773 fix(Pjax): 增加错误处理以防止Pjax失效 2025-09-05 23:43:14 +08:00
DeepChirp
6feac51de7 feat(umami): 支持自定义脚本名称 2025-09-03 12:56:07 +08:00
Jerry Wong
fcd760011a Update FUNDING.yml 2025-08-27 11:35:54 +08:00
Jerry
ec1a226774 Fix merge conflicts 2025-08-19 14:55:15 +08:00
Jerry
5ee24defc3 feat: 預設關閉 structured_data
feat: 升級一些項目依賴
feat: 重寫 README.md 和 README_CN.md,改進文檔結構和內容
feat: tags 標籤插件夜間模式調整
feat: 加按鈕懸停效果和動畫
fix: 修復右下角箭頭圖標位置沒有居中的 bug
feat: 增加右下角箭頭和滾動百分比的切換效果
improvement: 優化 tags 頁標籤雲顯示效果
improvement: 整合部分js到 init.js
improvement: 統一 CSS 變數使用,改進主題一致性
2025-08-19 14:49:39 +08:00
Jerry Wong
58818a0630 Merge pull request #1718 from Windsland52/dev
fix(gitalk): 修复 MD5 函数导致的评论聚合问题
2025-08-14 13:51:09 +08:00
Jerry Wong
a551b9277f Merge pull request #1720 from DeepChirp/structured_data
feat(structured_data): add `alternateNames` for backup
2025-08-14 13:50:33 +08:00
DeepChirp
1b5bc97431 feat(structured_data): add alternateNames for backup 2025-07-31 19:37:25 +08:00
Windsland52
0da72787fa fix(gitalk): 修复 MD5 函数导致的评论聚合问题
- 修改 url_for 调用添加 {relative: false} 参数
- 解决空字符串 MD5 导致多页面评论混合的问题
2025-07-30 08:27:46 +08:00
Jerry
0e9b8f5b69 Merge branch 'dev' of https://github.com/jerryc127/hexo-theme-butterfly into dev 2025-07-27 15:26:37 +08:00
Jerry
89b3626c84 fix: 修復 aside 最後一個 沒有 margin-bottom 的 bug
feat: 適配 fortawesome v7
improvement: 優化 loading 加載
2025-07-27 15:26:28 +08:00
Jerry Wong
c5417d4532 Merge pull request #1699 from DeepChirp/copyright-space
在`©`后添加空格
2025-07-23 15:21:23 +08:00
DeepChirp
dd8a18e8f4 chore: clean up unnecessary comments 2025-07-15 13:41:26 +08:00
Jerry
a61e216452 fix: 修正 abcjs 報錯的 bug 2025-07-09 13:24:42 +08:00
DeepChirp
b89f165be3 chore: add a space after © 2025-07-07 13:14:38 +08:00
Jerry
d2eacd2d8a feat: 增加導航欄顯示文章標題的選項,並更新懶加載設置 2025-07-04 23:26:29 +08:00
Jerry
5bfc1da03b feat: 更新版本號並升級多個插件的依賴版本,改善功能和相容性 2025-07-04 23:12:47 +08:00
Jerry
21c238e5c1 Merge branch 'dev' of https://github.com/jerryc127/hexo-theme-butterfly into dev 2025-06-25 16:27:06 +08:00
Jerry
60fa703fd3 feat: 增加導覽欄顯示文章標題的選項 feat: 文章頁增加'返回首頁'文字 improvement: 當 per_page 為 0 時,頁面不顯示導覽列 2025-06-25 16:25:46 +08:00
Jerry Wong
7353de6c2e Merge pull request #1695 from DeepChirp/structured_data
feat(structured_data): 为网站首页添加结构化数据
2025-06-25 15:17:08 +08:00
DeepChirp
a9fe9f5332 feat(structured_data): 为网站首页添加结构化数据 2025-06-12 11:43:32 +08:00
Jerry
73de62a6e1 Merge branch 'dev' of https://github.com/jerryc127/hexo-theme-butterfly into dev 2025-05-09 17:57:40 +08:00
Jerry
cf059bd533 fix: 更新 package.json 和 plugins.yml 中的版本號
fix: 修正 truncateContent 函數, 加密文章不显示自动擷取內容
feat: 增加首頁樣式以支持單詞換行
fix: 修正 truncateContent 函數以正確處理自動擷取內容
fix: 修复 card_archives 计数 bug
fix: 修正分頁順序邏輯
2025-05-09 17:52:01 +08:00
Jerry Wong
8cb726ddaa Merge pull request #1683 from SamirLiu127/dev
feat: Add google tag manager
2025-05-09 17:44:32 +08:00
SamirLiu
baee803689 feat: move noscript to additional-js 2025-05-09 13:28:01 +08:00
SamirLiu
023d82820d feat: update merge_config.js 2025-05-09 13:23:17 +08:00
SamirLiu
b7f1610859 update: DataLayer parameter 2025-05-08 14:03:08 +08:00
SamirLiu
5bac44a284 feat: add dataLayer push event pjaxComplete 2025-05-08 13:51:09 +08:00
SamirLiu
9c5294b40c fix: update analytics.pug for google tag manager under pjax 2025-05-07 19:21:39 +08:00
SamirLiu
a7315b6bfc feat: add google tag manager config 2025-05-07 19:17:39 +08:00
SamirLiu
cb3778c686 feat: add google tag manager noscript 2025-05-07 19:17:39 +08:00
Jerry Wong
2ce969023c Merge pull request #1686 from guozhenyi/dev
chore: 补充缺少的依赖库
2025-05-07 18:28:03 +08:00
gzy
7c697c15e8 chore: 补充缺少的依赖库 2025-04-30 16:56:46 +08:00
Jerry
ca030589fb feat: 更新頁腳配置,增加導航欄和版權信息,改進樣式和結構
fix: 修改 getBgPath 函數以使用 this.url_for 獲取圖片的正確路徑
feat: 增加右側配置按鈕的動畫效果
2025-03-23 23:20:12 +08:00
Jerry Wong
a3f6b625ed Update package.json 2025-03-04 16:14:26 +08:00
Jerry
628d1bbe52 Merge branch 'dev' of https://github.com/jerryc127/hexo-theme-butterfly into dev 2025-03-02 15:27:34 +08:00
Jerry
ede4f8bfea update 2025-03-02 15:27:16 +08:00
Jerry Wong
feb3346cf6 Merge pull request #1650 from TephrocactusMYC/dev
feat(score tag): support RenderAbc options
2025-03-02 15:13:15 +08:00
Yuchen Mu
576fa5c80e fix: resolve issues from previous commit about feat(score tag) 2025-03-02 10:23:40 +08:00
Yuchen Mu
6019cc8b7c Merge branch 'jerryc127:dev' into dev 2025-03-02 10:12:00 +08:00
Jerry Wong
7f9409553a Merge pull request #1649 from icemyst/patch-1
Update pjax.pug
2025-03-02 01:06:20 +08:00
Yuchen Mu
2de7d34b2b feat(score tag): support RenderAbc options
Modified the score tag to pass custom options to the abcjs RenderAbc interface.
This enhancement enables proper rendering of guitar tablature with custom
settings such as instrument tuning, labels, and formatting options.
2025-02-24 19:03:46 +08:00
冰梦
4b8a610c08 Update pjax.pug
自定义404.html跳转更平滑,使用 window.location.href = e.request.responseURL 加载,自定义404.html加载速度缓慢,可能会出现loading二次加载
2025-02-20 14:10:13 +08:00
Jerry
13720bd94d .DS_Store banished! 2025-02-16 21:26:08 +08:00
Jerry
3028b08526 update 2025-02-16 20:53:10 +08:00
Jerry
f605e6dc89 update 2025-02-16 20:38:17 +08:00
Jerry Wong
92913a6193 Merge pull request #1635 from cxyfer/dev 2025-01-17 12:16:06 +08:00
cxyfer
1e20234d74 fix: update addtoany item reference for correct sharing functionality 2025-01-17 02:19:21 +08:00
myw
1f3ea55890 fix: 文章頁分頁不顯示的 bug
improvement: 優化部分插件導致文章頁分頁樣式錯亂的 bug
2025-01-13 00:21:47 +08:00
myw
3d4bf30948 fix: 修復隨機封面死循環的問題 2025-01-12 15:32:55 +08:00
myw
172a8ee846 更新依赖 2025-01-11 01:26:06 +08:00
myw
5331f4ef9e Merge branch 'dev' of https://github.com/jerryc127/hexo-theme-butterfly into dev 2024-12-10 20:41:14 +08:00
myw
0d0001c808 feat: 更新 plugins.yml 中的依賴版本至最新
feat: 優化 aside_archives ,改進性能和可讀性
feat: 改善 inlineImg 和 timeline 標籤的文檔,優化時間線邏輯
feat: 更新 gallery 標籤以支持額外參數,優化圖片顯示邏輯
improvement: 優化隨機封面過濾器邏輯, 避免連續重複
feat: 最新評論限制顯示 1-10 條之間
fix: artalk 的最新評論顯示待定或者封禁的評論的 bug
2024-12-10 20:35:58 +08:00
Jerry Wong
13db106172 Merge pull request #1619 from Henry-ZHR/dev
fix: 代码字体先 fallback 到 monospace 再到中文和 sans-serif
2024-12-08 00:53:59 +08:00
Henry-ZHR
f42048792e fix: 代码字体先 fallback 到 monospace 再到中文和 sans-serif 2024-12-07 20:43:16 +08:00
myw
247c1b664d feat: 更新 lazyload 配置,支持原生 lazyload 功能
feat: 代碼優化
feat: 優化 pageType 邏輯
fix: 修復解密文章後, chartjs 沒有加載的 bug
2024-11-30 13:38:39 +08:00
myw
f7483d59b5 Merge branch 'dev' of https://github.com/jerryc127/hexo-theme-butterfly into dev 2024-11-15 14:56:56 +08:00
myw
b7771e5938 feat: 移除 pangu.js
feat: js 加载完成才显示聊天按钮
2024-11-15 14:53:40 +08:00
Jerry Wong
74c555fb37 Merge pull request #1602 from DeepChirp/structured_data
feat: 添加结构化数据支持
2024-11-11 18:46:57 +08:00
DeepChirp
648ca6eb4f chore: 简化结构化数据配置,移除不必要的嵌套 2024-11-11 18:42:46 +08:00
DeepChirp
e5a52d5621 fix: 移除结构化数据中不合规范的微数据 2024-11-11 16:29:28 +08:00
Jerry Wong
1045f66a51 Merge pull request #1603 from DeepChirp/image
chore: add `avif` to the list of supported image formats
2024-11-11 15:13:02 +08:00
DeepChirp
5338a2be99 chore: add avif to the list of supported image formats 2024-11-07 18:09:01 +08:00
DeepChirp
3ea138178d feat: 添加结构化数据支持 2024-11-07 17:55:33 +08:00
myw
91c8c5cd4b feat: 過期通知優化,可單獨文章關閉
fix: 修復説説評論 css 受主題影響的 bug
2024-11-05 18:01:36 +08:00
myw
fb4ab20169 update 2024-11-02 19:38:15 +08:00
myw
7b6a386a14 update 2024-11-02 18:59:33 +08:00
myw
f91ce41a66 update 2024-11-02 18:58:20 +08:00
myw
d7bfcf36c9 fix: 修复 说说时间 timezone bug 2024-10-22 23:16:43 +08:00
214 changed files with 18505 additions and 15610 deletions

2
.github/FUNDING.yml vendored
View File

@@ -10,4 +10,4 @@ liberapay: # Replace with a single Liberapay username
issuehunt: # Replace with a single IssueHunt username issuehunt: # Replace with a single IssueHunt username
otechie: # Replace with a single Otechie username otechie: # Replace with a single Otechie username
lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry
custom: ['https://buy.stripe.com/3cs6rP6YA91sbbG5kk','https://jsd.012700.xyz/gh/jerryc127/CDN/Photo/wechat.jpg'] # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2'] custom: ['https://buy.stripe.com/3cs6rP6YA91sbbG5kk'] # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2']

View File

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

View File

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

View File

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

View File

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

View File

@@ -1,19 +1,19 @@
name: 'Close stale issues and PRs' name: 'Close stale issues and PRs'
on: on:
schedule: schedule:
- cron: '30 1 * * *' - cron: '30 1 * * *'
jobs: jobs:
stale: stale:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/stale@v5 - uses: actions/stale@v5
with: with:
days-before-issue-stale: 30 days-before-issue-stale: 30
days-before-pr-stale: -1 days-before-pr-stale: -1
days-before-close: 7 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.' 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.' close-pr-message: 'This issue has not seen any activity since it was marked stale. Closing.'
stale-issue-label: 'Stale' stale-issue-label: 'Stale'
exempt-issue-labels: 'pinned,bug,enhancement,documentation,Plan' exempt-issue-labels: 'pinned,bug,enhancement,documentation,Plan'
operations-per-run: 1000 operations-per-run: 1000

2
.gitignore vendored Normal file
View File

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

402
LICENSE
View File

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

226
README.md
View File

@@ -1,117 +1,193 @@
<div align="right"> <div align="right">
<a title="Chinese" href="/README_CN.md">中文</a> <a title="中文" href="/README_CN.md">中文</a>
</div> </div>
<div align="center"> <div align="center">
<img src="./source/img/butterfly-icon.png" width="150" height="150" /> <img src="./source/img/butterfly-icon.png" width="150" height="150" alt="Butterfly Logo" />
# hexo-theme-butterfly # 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) ![master version](https://img.shields.io/github/package-json/v/jerryc127/hexo-theme-butterfly/master?color=%231ab1ad&label=master)
![master version](https://img.shields.io/github/package-json/v/jerryc127/hexo-theme-butterfly/dev?label=dev) ![dev version](https://img.shields.io/github/package-json/v/jerryc127/hexo-theme-butterfly/dev?label=dev)
![https://img.shields.io/npm/v/hexo-theme-butterfly?color=%09%23bf00ff](https://img.shields.io/npm/v/hexo-theme-butterfly?color=%09%23bf00ff) ![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+-0e83c) ![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) ![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](https://butterfly.js.org/) / [CrazyWong](https://blog.crazywong.com/) 📢 **Demo**: [Butterfly Official](https://butterfly.js.org/) | [CrazyWong's Blog](https://blog.crazywong.com/)
📖 Docs: [English](https://butterfly.js.org/en/posts/butterfly-docs-en-get-started/) / [Chinese](https://butterfly.js.org/posts/21cfbf15/) 📖 **Documentation**: [English Docs](https://butterfly.js.org/en/posts/butterfly-docs-en-get-started/) | [中文文档](https://butterfly.js.org/posts/21cfbf15/)
![](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/theme-butterfly-readme.png) ![Butterfly Theme Preview](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/theme-butterfly-readme.png)
</div> </div>
--- ---
## 💻 Installation ## 🚀 Quick Start
### GIT ### 💾 Installation
> If you are in Mainland China, you can download in [Gitee](https://gitee.com/immyw/hexo-theme-butterfly.git) #### Method 1: Git Installation (Recommended)
Stable branch [recommend]: > 💡 **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 git clone -b master https://github.com/jerryc127/hexo-theme-butterfly.git themes/butterfly
``` ```
Dev branch: ```bash
# Install development version (early access to new features)
```
git clone -b dev https://github.com/jerryc127/hexo-theme-butterfly.git themes/butterfly git clone -b dev https://github.com/jerryc127/hexo-theme-butterfly.git themes/butterfly
``` ```
### NPM #### Method 2: NPM Installation
> It supports Hexo 5.0.0 or later > ⚠️ **Note**: NPM installation only supports Hexo 5.0.0 and above
In Hexo site root directory ```bash
npm install hexo-theme-butterfly
```powershell
npm i hexo-theme-butterfly
``` ```
## ⚙ Configuration ### Theme Configuration
Set theme in the hexo work folder's root config file `_config.yml`: 1. **Enable Theme**: Modify your Hexo configuration file `_config.yml`:
> theme: butterfly ```yaml
theme: butterfly
```
If you don't have pug & stylus renderer, try this: 2. **Install Dependencies**: If you haven't installed pug and stylus renderers, please run:
> npm install hexo-renderer-pug hexo-renderer-stylus ```bash
npm install hexo-renderer-pug hexo-renderer-stylus --save
```
## 🎉 Features ## ✨ Theme Features
- [x] Card UI Design ### 🎨 Design Style
- [x] Rounded Design/Squared Design - [x] **Card-based Design** - Modern card-style layout
- [X] Support sub-menu - [x] **Rounded/Square Design** - Customizable border styles
- [x] Two-column layout - [x] **Responsive Design** - Perfect adaptation to all screen sizes
- [x] Responsive Web Design - [x] **Two-column Layout** - Optimized reading experience
- [x] Dark Mode - [x] **Dark Mode** - Eye-friendly night mode
- [x] Pjax
- [x] Read Mode
- [x] Conversion between Traditional and Simplified Chinese
- [X] TOC catalog is available for both computers and mobile phones
- [X] Built-in Syntax Highlighting Themes (darker/pale night/light/ocean), also support customization
- [X] Code Blocks (Display code language/close or expand Code Blocks/Copy Button/word wrap)
- [X] Disable copy/Add a Copyright Notice to the Copied Text
- [X] Search (Algolia Search/Local Search)
- [x] Mathjax and Katex
- [x] Built-in 404 page
- [x] WordCount
- [x] Related articles
- [x] Displays outdated notice for a post
- [x] Share (Sharejs/Addtoany)
- [X] Comment (Disqus/Disqusjs/Livere/Gitalk/Valine/Waline/Utterances/Facebook Comments/Twikoo/Giscus/Remark42/artalk)
- [x] Multiple Comment System Support
- [x] Online Chats (Chatra/Tidio/Crisp)
- [x] Web analytics
- [x] Google AdSense
- [x] Webmaster Verification
- [x] Change website colour scheme
- [x] Typewriter Effect: activate_power_mode
- [x] Background effects (Canvas ribbon/canvas_ribbon_piao/canvas_nest)
- [x] Mouse click effects (Fireworks/Heart/Text)
- [x] Preloader/Loading Animation/pace.js
- [x] Busuanzi visitor counter
- [x] Medium Zoom/Fancybox
- [x] Mermaid
- [x] Chart.js
- [x] Justified Gallery
- [x] Lazyload images
- [x] Instantpage/Pangu/Snackbar notification toast/PWA......
## Contributors ### 📝 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
<a href="https://github.com/jerryc127/hexo-theme-butterfly/graphs/contributors"> ### 🔍 Search & Navigation
<img src="https://contrib.rocks/image?repo=jerryc127/hexo-theme-butterfly" /> - [x] **Multiple Search Options** - Algolia Search / Local Search / Docsearch
</a> - [x] **Built-in 404** - Beautiful 404 error page
- [x] **Pjax Support** - Smooth page transition experience
## 📷 Screenshots ### 🎨 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
![](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-1.jpg) ### 💬 Social Interaction
![](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-2.jpg) - [x] **Multiple Comment Systems** - Disqus/Gitalk/Valine/Waline/Twikoo/Giscus/Artalk etc.
![](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-3.jpg) - [x] **Dual Comments Support** - Enable two comment systems simultaneously
![](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-4.jpg) - [x] **Share Features** - Sharejs/Addtoany sharing components
- [x] **Live Chat** - Chatra/Tidio/Crisp instant messaging
### 📊 Analytics & Statistics
- [x] **Visit Statistics** - Busuanzi counter
- [x] **Site Analytics** - Google Analytics/Baidu Analytics/Cloudflare Analytics/Microsoft Clarity/Umami
- [x] **Webmaster Verification** - Major search engine verification
- [x] **Ad Support** - Google AdSense/custom ad slots
### 🎪 Visual Effects
- [x] **Typing Effects** - activate_power_mode animations
- [x] **Background Effects** - Static ribbons/dynamic ribbons/floating ribbons/Canvas Nest
- [x] **Mouse Effects** - Fireworks/hearts/text click effects
- [x] **Loading Animations** - Preloader and pace.js progress bars
- [x] **Image Effects** - Medium Zoom/Fancybox image lightbox
- [x] **Lazy Loading** - Image lazy loading optimization
### 🛠️ Advanced Features
- [x] **PWA Support** - Progressive Web App
- [x] **Copy Protection** - Disable text copying/copyright info append
- [x] **Theme Customization** - Custom site color schemes
- [x] **Chart Support** - Mermaid flowcharts/Chart.js data charts
- [x] **Music Notation** - ABCJS music notation support
- [x] **Music Player** - APlayer/Meting music playback
- [x] **Article Series** - Series article organization
- [x] **Instantpage** - Page preloading acceleration
- [x] **Snackbar** - Elegant notification messages
## 🤝 Contributors
Thanks to all the developers who have contributed to the Butterfly theme!
[![Contributors](https://contrib.rocks/image?repo=jerryc127/hexo-theme-butterfly)](https://github.com/jerryc127/hexo-theme-butterfly/graphs/contributors)
## 📸 Screenshots
<div align="center">
![Theme Demo](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-1.jpg)
![Theme Demo](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-2.jpg)
![Theme Demo](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-3.jpg)
![Theme Demo](https://cdn.jsdelivr.net/gh/jerryc127/CDN@m2/img/butterfly-readme-screenshots-4.jpg)
</div>
## ⭐ Star History
[![Star History Chart](https://api.star-history.com/svg?repos=jerryc127/hexo-theme-butterfly&type=Date)](https://star-history.com/#jerryc127/hexo-theme-butterfly&Date)
## 🤝 Building a Better Theme Together
We believe **the power of open source comes from everyone's participation**! Whether you're a developer, designer, or user, you can contribute to the development of the Butterfly theme.
### 💬 Get Help & Support
- 🐛 **Found a bug?** → [GitHub Issues](https://github.com/jerryc127/hexo-theme-butterfly/issues) - Let's solve it together!
- 💡 **Have ideas?** → [GitHub Discussions](https://github.com/jerryc127/hexo-theme-butterfly/discussions) - Share your creative ideas!
- 📚 **Learning to use?** → [Official Documentation](https://butterfly.js.org/) - Detailed usage guide
- 💬 **Real-time discussion?** → [Telegram Group](https://t.me/bu2fly) - Chat with community members
### 🎯 Contributing
Want to make Butterfly better? We welcome any form of contribution:
- **🔧 Code Contributions** - Fix bugs, add new features, optimize performance
- **📝 Documentation** - Improve docs, translate content, write tutorials
- **🎨 Design Suggestions** - UI/UX improvements, theme colors, icon design
- **🧪 Testing & Feedback** - Test new features, report issues, provide user experience
- **💰 Financial Support** - [Sponsor the Project](https://buy.stripe.com/3cs6rP6YA91sbbG5kk) - Support long-term development
## 📄 License
This project is licensed under the [Apache 2.0](LICENSE) License.
## 🙏 Acknowledgments
This theme is developed based on [hexo-theme-melody](https://github.com/Molunerfinn/hexo-theme-melody). Thanks to the original author for their excellent work that provided inspiration and foundation!
Thanks to all friends who have contributed to the development of the Butterfly theme. Your support has made this theme continuously improve and progress.
---
<div align="center">
**✨ If this theme helps you, please give us a ⭐ Star! ✨**
</div>

View File

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

File diff suppressed because it is too large Load Diff

View File

@@ -1,121 +1,123 @@
footer: footer:
framework: Framework framework: Framework
theme: Theme theme: Theme
copy: copy:
success: Copy Successful success: Copy Successful
error: Copy Failed error: Copy Failed
noSupport: Browser Not Supported noSupport: Browser Not Supported
page: page:
articles: All Articles articles: All Articles
tag: Tag tag: Tag
category: Category category: Category
archives: Archives archives: Archives
card_post_count: comments card_post_count: comments
no_title: Untitled no_title: Untitled
post: post:
created: Created created: Created
updated: Updated updated: Updated
wordcount: Word Count wordcount: Word Count
min2read: Reading Time min2read: Reading Time
min2read_unit: mins min2read_unit: mins
page_pv: Post Views page_pv: Post Views
comments: Comments comments: Comments
copyright: copyright:
author: Author author: Author
link: Link link: Link
copyright_notice: Copyright Notice copyright_notice: Copyright Notice
copyright_content: 'All articles on this blog are licensed under <a href="%s">%s</a> unless otherwise stated.' copyright_content: 'All articles on this blog are licensed under <a href="%s">%s</a> unless otherwise stated.'
recommend: Related Articles recommend: Related Articles
edit: Edit edit: Edit
back_to_home: Back to Home
search:
title: Search search:
load_data: Loading Database title: Search
input_placeholder: Search for Posts load_data: Loading Database
algolia_search: input_placeholder: Search for Posts
hits_empty: 'No results found for: ${query}' algolia_search:
hits_stats: '${hits} results found in ${time} ms' hits_empty: 'No results found for: ${query}'
local_search: hits_stats: '${hits} results found in ${time} ms'
hits_empty: 'No results found for: ${query}' local_search:
hits_stats: '${hits} articles found' hits_empty: 'No results found for: ${query}'
hits_stats: '${hits} articles found'
pagination:
prev: Previous pagination:
next: Next prev: Previous
next: Next
comment: Comments page_info: 'Page ${current} of ${total}'
aside: comment: Comments
articles: Articles
tags: Tags aside:
categories: Categories articles: Articles
card_announcement: Announcement tags: Tags
card_categories: Categories categories: Categories
card_tags: Tags card_announcement: Announcement
card_archives: Archives card_categories: Categories
card_recent_post: Recent Posts card_tags: Tags
card_webinfo: card_archives: Archives
headline: Website Info card_recent_post: Recent Posts
article_name: Article Count card_webinfo:
runtime: headline: Website Info
name: Runtime article_name: Article Count
unit: days runtime:
last_push_date: name: Runtime
name: Last Update unit: days
site_wordcount: Total Word Count last_push_date:
site_uv_name: Unique Visitors name: Last Update
site_pv_name: Page Views site_wordcount: Total Word Count
more_button: View More site_uv_name: Unique Visitors
card_newest_comments: site_pv_name: Page Views
headline: Latest Comments more_button: View More
loading_text: Loading... card_newest_comments:
error: Unable to retrieve comments, please check the configuration headline: Latest Comments
zero: No comments loading_text: Loading...
image: Image error: Unable to retrieve comments, please check the configuration
link: Link zero: No comments
code: Code image: Image
card_toc: Contents link: Link
card_post_series: Post Series code: Code
card_toc: Contents
date_suffix: card_post_series: Post Series
just: Just now
min: minutes ago date_suffix:
hour: hours ago just: Just now
day: days ago min: minutes ago
month: months ago hour: hours ago
day: days ago
donate: Sponsor month: months ago
share: Share
donate: Sponsor
rightside: share: Share
readmode_title: Reading Mode
translate_title: Toggle Between Traditional and Simplified Chinese rightside:
night_mode_title: Toggle Between Light and Dark Mode readmode_title: Reading Mode
back_to_top: Back to Top translate_title: Toggle Between Traditional and Simplified Chinese
toc: Table of Contents night_mode_title: Toggle Between Light and Dark Mode
scroll_to_comment: Scroll to Comments back_to_top: Back to Top
setting: Settings toc: Table of Contents
aside: Toggle Between Single-column and Double-column scroll_to_comment: Scroll to Comments
chat: Chat setting: Settings
aside: Toggle Between Single-column and Double-column
copy_copyright: chat: Chat
author: Author
link: Link copy_copyright:
source: Source author: Author
info: Copyright belongs to the author. For commercial use, please contact the author for authorization. For non-commercial use, please indicate the source. link: Link
source: Source
Snackbar: info: Copyright belongs to the author. For commercial use, please contact the author for authorization. For non-commercial use, please indicate the source.
chs_to_cht: You have switched to Traditional Chinese
cht_to_chs: You have switched to Simplified Chinese Snackbar:
day_to_night: You have switched to Dark Mode chs_to_cht: You have switched to Traditional Chinese
night_to_day: You have switched to Light Mode cht_to_chs: You have switched to Simplified Chinese
day_to_night: You have switched to Dark Mode
loading: Loading... night_to_day: You have switched to Light Mode
load_more: Load More
loading: Loading...
error404: Page Not Found load_more: Load More
error404: Page Not Found

View File

@@ -1,121 +1,123 @@
footer: footer:
framework: Framework framework: Framework
theme: Theme theme: Theme
copy: copy:
success: Copy Successful success: Copy Successful
error: Copy Failed error: Copy Failed
noSupport: Browser Not Supported noSupport: Browser Not Supported
page: page:
articles: All Articles articles: All Articles
tag: Tag tag: Tag
category: Category category: Category
archives: Archives archives: Archives
card_post_count: comments card_post_count: comments
no_title: Untitled no_title: Untitled
post: post:
created: Created created: Created
updated: Updated updated: Updated
wordcount: Word Count wordcount: Word Count
min2read: Reading Time min2read: Reading Time
min2read_unit: mins min2read_unit: mins
page_pv: Post Views page_pv: Post Views
comments: Comments comments: Comments
copyright: copyright:
author: Author author: Author
link: Link link: Link
copyright_notice: Copyright Notice copyright_notice: Copyright Notice
copyright_content: 'All articles on this blog are licensed under <a href="%s">%s</a> unless otherwise stated.' copyright_content: 'All articles on this blog are licensed under <a href="%s">%s</a> unless otherwise stated.'
recommend: Related Articles recommend: Related Articles
edit: Edit edit: Edit
back_to_home: Back to Home
search:
title: Search search:
load_data: Loading Database title: Search
input_placeholder: Search for Posts load_data: Loading Database
algolia_search: input_placeholder: Search for Posts
hits_empty: 'No results found for: ${query}' algolia_search:
hits_stats: '${hits} results found in ${time} ms' hits_empty: 'No results found for: ${query}'
local_search: hits_stats: '${hits} results found in ${time} ms'
hits_empty: 'No results found for: ${query}' local_search:
hits_stats: '${hits} articles found' hits_empty: 'No results found for: ${query}'
hits_stats: '${hits} articles found'
pagination:
prev: Previous pagination:
next: Next prev: Previous
next: Next
comment: Comments page_info: 'Page ${current} of ${total}'
aside: comment: Comments
articles: Articles
tags: Tags aside:
categories: Categories articles: Articles
card_announcement: Announcement tags: Tags
card_categories: Categories categories: Categories
card_tags: Tags card_announcement: Announcement
card_archives: Archives card_categories: Categories
card_recent_post: Recent Posts card_tags: Tags
card_webinfo: card_archives: Archives
headline: Website Info card_recent_post: Recent Posts
article_name: Article Count card_webinfo:
runtime: headline: Website Info
name: Runtime article_name: Article Count
unit: days runtime:
last_push_date: name: Runtime
name: Last Update unit: days
site_wordcount: Total Word Count last_push_date:
site_uv_name: Unique Visitors name: Last Update
site_pv_name: Page Views site_wordcount: Total Word Count
more_button: View More site_uv_name: Unique Visitors
card_newest_comments: site_pv_name: Page Views
headline: Latest Comments more_button: View More
loading_text: Loading... card_newest_comments:
error: Unable to retrieve comments, please check the configuration headline: Latest Comments
zero: No comments loading_text: Loading...
image: Image error: Unable to retrieve comments, please check the configuration
link: Link zero: No comments
code: Code image: Image
card_toc: Contents link: Link
card_post_series: Post Series code: Code
card_toc: Contents
date_suffix: card_post_series: Post Series
just: Just now
min: minutes ago date_suffix:
hour: hours ago just: Just now
day: days ago min: minutes ago
month: months ago hour: hours ago
day: days ago
donate: Sponsor month: months ago
share: Share
donate: Sponsor
rightside: share: Share
readmode_title: Reading Mode
translate_title: Toggle Between Traditional and Simplified Chinese rightside:
night_mode_title: Toggle Between Light and Dark Mode readmode_title: Reading Mode
back_to_top: Back to Top translate_title: Toggle Between Traditional and Simplified Chinese
toc: Table of Contents night_mode_title: Toggle Between Light and Dark Mode
scroll_to_comment: Scroll to Comments back_to_top: Back to Top
setting: Settings toc: Table of Contents
aside: Toggle Between Single-column and Double-column scroll_to_comment: Scroll to Comments
chat: Chat setting: Settings
aside: Toggle Between Single-column and Double-column
copy_copyright: chat: Chat
author: Author
link: Link copy_copyright:
source: Source author: Author
info: Copyright belongs to the author. For commercial use, please contact the author for authorization. For non-commercial use, please indicate the source. link: Link
source: Source
Snackbar: info: Copyright belongs to the author. For commercial use, please contact the author for authorization. For non-commercial use, please indicate the source.
chs_to_cht: You have switched to Traditional Chinese
cht_to_chs: You have switched to Simplified Chinese Snackbar:
day_to_night: You have switched to Dark Mode chs_to_cht: You have switched to Traditional Chinese
night_to_day: You have switched to Light Mode cht_to_chs: You have switched to Simplified Chinese
day_to_night: You have switched to Dark Mode
loading: Loading... night_to_day: You have switched to Light Mode
load_more: Load More
loading: Loading...
error404: Page Not Found load_more: Load More
error404: Page Not Found

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

@@ -1,60 +1,61 @@
div div
script(src=url_for(theme.asset.utils)) script(src=url_for(theme.asset.utils))
script(src=url_for(theme.asset.main)) script(src=url_for(theme.asset.main))
if theme.translate.enable if theme.translate.enable
script(src=url_for(theme.asset.translate)) script(src=url_for(theme.asset.translate))
if theme.lightbox if theme.lightbox
script(src=url_for(theme.asset[theme.lightbox])) script(src=url_for(theme.asset[theme.lightbox]))
if theme.instantpage if theme.instantpage
script(src=url_for(theme.asset.instantpage), type='module') script(src=url_for(theme.asset.instantpage), type='module')
if theme.lazyload.enable if theme.lazyload.enable && !theme.lazyload.native
script(src=url_for(theme.asset.lazyload)) script(src=url_for(theme.asset.lazyload))
if theme.snackbar.enable if theme.snackbar.enable
script(src=url_for(theme.asset.snackbar)) script(src=url_for(theme.asset.snackbar))
if theme.pangu.enable .js-pjax
!= partial("includes/third-party/pangu.pug", {}, { cache: true }) if needLoadCountJs
!= partial("includes/third-party/card-post-count/index", {}, { cache: true })
.js-pjax
if needLoadCountJs if loadSubJs
!= partial("includes/third-party/card-post-count/index", {}, { cache: true }) include ./third-party/subtitle.pug
if loadSubJs include ./third-party/math/index.pug
include ./third-party/subtitle.pug include ./third-party/abcjs/index.pug
include ./third-party/math/index.pug if commentsJsLoad
include ./third-party/abcjs/index.pug include ./third-party/comments/js.pug
if commentsJsLoad != partial("includes/third-party/prismjs", {}, { cache: true })
include ./third-party/comments/js.pug
if theme.aside.enable && theme.aside.card_newest_comments.enable
!= partial("includes/third-party/prismjs", {}, { cache: true }) if theme.pjax.enable || (globalPageType !== 'post' && page.aside !== false)
!= partial("includes/third-party/newest-comments/index", {}, { cache: true })
if theme.aside.enable && theme.aside.card_newest_comments.enable
if theme.pjax.enable || (!is_post() && page.aside !== false) != fragment_cache('injectBottom', function(){return injectHtml(theme.inject.bottom)})
!= partial("includes/third-party/newest-comments/index", {}, { cache: true })
!= partial("includes/third-party/effect", {}, { cache: true })
!= fragment_cache('injectBottom', function(){return injectHtml(theme.inject.bottom)}) != partial("includes/third-party/chat/index", {}, { cache: true })
!= partial("includes/third-party/effect", {}, { cache: true }) if theme.aplayerInject && theme.aplayerInject.enable
!= partial("includes/third-party/chat/index", {}, { cache: true }) if theme.pjax.enable || theme.aplayerInject.per_page || page.aplayer
include ./third-party/aplayer.pug
if theme.aplayerInject && theme.aplayerInject.enable
if theme.pjax.enable || theme.aplayerInject.per_page || page.aplayer if theme.pjax.enable
include ./third-party/aplayer.pug != partial("includes/third-party/pjax", {}, { cache: true })
if theme.pjax.enable if theme.umami_analytics.enable
!= partial("includes/third-party/pjax", {}, { cache: true }) != partial("includes/third-party/umami_analytics", {}, { cache: true })
if theme.umami_analytics.enable if theme.busuanzi.site_uv || theme.busuanzi.site_pv || theme.busuanzi.page_pv
!= partial("includes/third-party/umami_analytics", {}, { cache: true }) script(async data-pjax src=theme.asset.busuanzi ? url_for(theme.asset.busuanzi) : '//busuanzi.ibruce.info/busuanzi/2.3/busuanzi.pure.mini.js')
if theme.busuanzi.site_uv || theme.busuanzi.site_pv || theme.busuanzi.page_pv != partial('includes/third-party/search/index', {}, { cache: true })
script(async data-pjax src= theme.asset.busuanzi || '//busuanzi.ibruce.info/busuanzi/2.3/busuanzi.pure.mini.js')
if theme.google_tag_manager && theme.google_tag_manager.tag_id
!= partial('includes/third-party/search/index', {}, { cache: true }) 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")

View File

@@ -1,18 +1,39 @@
#footer-wrap - const { nav, owner, copyright, custom_text } = theme.footer
if theme.footer.owner.enable
- const currentYear = new Date().getFullYear() if nav
- const sinceYear = theme.footer.owner.since .footer-flex
.copyright for block in nav
if sinceYear && sinceYear != currentYear .footer-flex-items(style=`${ block.width ? 'flex-grow:' + block.width : '' }`)
!= `&copy;${sinceYear} - ${currentYear} By ${config.author}` for blockItem in block.content
else .footer-flex-item
!= `&copy;${currentYear} By ${config.author}` .footer-flex-title= blockItem.title
if theme.footer.copyright .footer-flex-content
.framework-info for subitem in blockItem.item
span= _p('footer.framework') + ' ' if subitem.html
a(href='https://hexo.io')= 'Hexo' div!= subitem.html
span.footer-separator | else if subitem.url
span= _p('footer.theme') + ' ' a(href=url_for(subitem.url), target='_blank' title=subitem.title)= subitem.title
a(href='https://github.com/jerryc127/hexo-theme-butterfly')= 'Butterfly' else if subitem.title
if theme.footer.custom_text div!= subitem.title
.footer_custom_text!= theme.footer.custom_text .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

View File

@@ -1,68 +1,77 @@
- var pageTitle - var pageTitle
- is_archive() ? page.title = findArchivesTitle(page, theme.menu, date) : '' - globalPageType === 'archive' ? page.title = findArchivesTitle(page, theme.menu, date) : ''
- if (is_tag()) pageTitle = _p('page.tag') + ': ' + page.tag case globalPageType
- else if (is_category()) pageTitle = _p('page.category') + ': ' + page.category when 'tag'
- else if (is_current('/404.html', [strict])) pageTitle = _p('error404') - pageTitle = _p('page.tag') + ': ' + page.tag
- else pageTitle = page.title || config.title || '' when 'category'
- pageTitle = _p('page.category') + ': ' + page.category
- var isSubtitle = config.subtitle ? ' - ' + config.subtitle : '' when '404'
- var tabTitle = is_home() || !pageTitle ? config.title + isSubtitle : pageTitle + ' | ' + config.title - pageTitle = _p('error404')
- var pageAuthor = config.email ? config.author + ',' + config.email : config.author default
- var pageCopyright = config.copyright || config.author - pageTitle = page.title || config.title || ''
- 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 - var isSubtitle = config.subtitle ? ' - ' + config.subtitle : ''
- var tabTitle = globalPageType === 'home' || !pageTitle ? config.title + isSubtitle : pageTitle + ' | ' + config.title
meta(charset='UTF-8') - var pageAuthor = config.email ? config.author + ',' + config.email : config.author
meta(http-equiv="X-UA-Compatible" content="IE=edge") - var pageCopyright = config.copyright || config.author
meta(name="viewport" content="width=device-width, initial-scale=1.0,viewport-fit=cover") - var themeColorLight = theme.theme_color && theme.theme_color.enable && theme.theme_color.meta_theme_color_light || '#ffffff'
title= tabTitle - var themeColorDark = theme.theme_color && theme.theme_color.enable && theme.theme_color.meta_theme_color_dark || '#0d0d0d'
meta(name="author" content=pageAuthor) - var themeColor = theme.display_mode === 'dark' ? themeColorDark : themeColorLight
meta(name="copyright" content=pageCopyright)
meta(name ="format-detection" content="telephone=no") meta(charset='UTF-8')
meta(name="theme-color" content=themeColor) meta(http-equiv="X-UA-Compatible" content="IE=edge")
meta(name="viewport" content="width=device-width, initial-scale=1.0,viewport-fit=cover")
//- Open_Graph title= tabTitle
include ./head/Open_Graph.pug meta(name="author" content=pageAuthor)
meta(name="copyright" content=pageCopyright)
!=favicon_tag(theme.favicon || config.favicon) meta(name ="format-detection" content="telephone=no")
link(rel="canonical" href=urlNoIndex(null,config.pretty_urls.trailing_index,config.pretty_urls.trailing_html)) meta(name="theme-color" content=themeColor)
//- 預解析 //- Open_Graph
!=partial('includes/head/preconnect', {}, {cache: true}) include ./head/Open_Graph.pug
//- 網站驗證 //- Structured Data
!=partial('includes/head/site_verification', {}, {cache: true}) include ./head/structured_data.pug
//- PWA !=favicon_tag(theme.favicon || config.favicon)
if (theme.pwa && theme.pwa.enable) link(rel="canonical" href=urlNoIndex(null,config.pretty_urls.trailing_index,config.pretty_urls.trailing_html))
!=partial('includes/head/pwa', {}, {cache: true})
//- 預解析
//- main css !=partial('includes/head/preconnect', {}, {cache: true})
link(rel='stylesheet', href=url_for(theme.asset.main_css))
link(rel='stylesheet', href=url_for(theme.asset.fontawesome)) //- 網站驗證
!=partial('includes/head/site_verification', {}, {cache: true})
if (theme.snackbar && theme.snackbar.enable)
link(rel='stylesheet', href=url_for(theme.asset.snackbar_css) media="print" onload="this.media='all'") //- PWA
if (theme.pwa && theme.pwa.enable)
if theme.lightbox === 'fancybox' !=partial('includes/head/pwa', {}, {cache: true})
link(rel='stylesheet' href=url_for(theme.asset.fancybox_css) media="print" onload="this.media='all'")
//- main css
!=fragment_cache('injectHeadJs', function(){return inject_head_js()}) link(rel='stylesheet', href=url_for(theme.asset.main_css))
link(rel='stylesheet', href=url_for(theme.asset.fontawesome))
//- google_adsense
!=partial('includes/head/google_adsense', {}, {cache: true}) if (theme.snackbar && theme.snackbar.enable)
link(rel='stylesheet', href=url_for(theme.asset.snackbar_css) media="print" onload="this.media='all'")
//- analytics
!=partial('includes/head/analytics', {}, {cache: true}) if theme.lightbox === 'fancybox'
link(rel='stylesheet' href=url_for(theme.asset.fancybox_css) media="print" onload="this.media='all'")
//- font
if theme.blog_title_font && theme.blog_title_font.font_link !=fragment_cache('injectHeadJs', function(){return inject_head_js()})
link(rel='stylesheet' href=url_for(theme.blog_title_font.font_link) media="print" onload="this.media='all'")
//- google_adsense
//- global config !=partial('includes/head/google_adsense', {}, {cache: true})
!=partial('includes/head/config', {}, {cache: true})
//- analytics
include ./head/config_site.pug !=partial('includes/head/analytics', {}, {cache: true})
!=fragment_cache('injectHead', function(){return injectHtml(theme.inject.head)}) //- 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)})

View File

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

View File

@@ -1,34 +1,45 @@
if theme.baidu_analytics if theme.baidu_analytics
script. script.
var _hmt = _hmt || []; var _hmt = _hmt || [];
(function() { (function() {
var hm = document.createElement("script"); var hm = document.createElement("script");
hm.src = "https://hm.baidu.com/hm.js?!{theme.baidu_analytics}"; hm.src = "https://hm.baidu.com/hm.js?!{theme.baidu_analytics}";
var s = document.getElementsByTagName("script")[0]; var s = document.getElementsByTagName("script")[0];
s.parentNode.insertBefore(hm, s); s.parentNode.insertBefore(hm, s);
})(); })();
btf.addGlobalFn('pjaxComplete', () => { btf.addGlobalFn('pjaxComplete', () => {
_hmt.push(['_trackPageview',window.location.pathname]) _hmt.push(['_trackPageview',window.location.pathname])
}, 'baidu_analytics') }, 'baidu_analytics')
if theme.google_analytics if theme.google_analytics
script(async src=`https://www.googletagmanager.com/gtag/js?id=${theme.google_analytics}`) script(async src=`https://www.googletagmanager.com/gtag/js?id=${theme.google_analytics}`)
script. script.
window.dataLayer = window.dataLayer || [] window.dataLayer = window.dataLayer || []
function gtag(){dataLayer.push(arguments)} function gtag(){dataLayer.push(arguments)}
gtag('js', new Date()) gtag('js', new Date())
gtag('config', '!{theme.google_analytics}') gtag('config', '!{theme.google_analytics}')
btf.addGlobalFn('pjaxComplete', () => { btf.addGlobalFn('pjaxComplete', () => {
gtag('config', '!{theme.google_analytics}', {'page_path': window.location.pathname}) gtag('config', '!{theme.google_analytics}', {'page_path': window.location.pathname})
}, 'google_analytics') }, 'google_analytics')
if theme.cloudflare_analytics if theme.cloudflare_analytics
script(defer data-pjax src='https://static.cloudflareinsights.com/beacon.min.js' data-cf-beacon=`{"token": "${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 if theme.microsoft_clarity
script. script.
(function(c,l,a,r,i,t,y){ (function(c,l,a,r,i,t,y){
c[a]=c[a]||function(){(c[a].q=c[a].q||[]).push(arguments)}; 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; 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); y=l.getElementsByTagName(r)[0];y.parentNode.insertBefore(t,y);
})(window, document, "clarity", "script", "!{theme.microsoft_clarity}"); })(window, document, "clarity", "script", "!{theme.microsoft_clarity}");
if (theme.google_tag_manager && theme.google_tag_manager.tag_id)
script.
(function(w,d,s,l,i){w[l]=w[l]||[];w[l].push({'gtm.start':
new Date().getTime(),event:'gtm.js'});var f=d.getElementsByTagName(s)[0],
j=d.createElement(s),dl=l!='dataLayer'?'&l='+l:'';j.async=true;j.src=
"!{theme.google_tag_manager.domain ? theme.google_tag_manager.domain : 'https://www.googletagmanager.com'}/gtm.js?id="+i+dl;f.parentNode.insertBefore(j,f);
})(window,document,'script','dataLayer','!{theme.google_tag_manager.tag_id}');
btf.addGlobalFn('pjaxComplete', () => {
dataLayer.push({'event': 'pjaxComplete', 'page_title': document.title, 'page_location': location.href, 'page_path': window.location.pathname})
}, 'google_tag_manager')

View File

@@ -1,137 +1,125 @@
- -
let algolia = 'undefined' let algolia = 'undefined'
if (theme.search.use === 'algolia_search') { if (theme.search.use === 'algolia_search') {
const { ALGOLIA_APP_ID, ALGOLIA_API_KEY, ALGOLIA_INDEX_NAME } = process.env const { ALGOLIA_APP_ID, ALGOLIA_API_KEY, ALGOLIA_INDEX_NAME } = process.env
const { appId, applicationID, apiKey, indexName } = config.algolia const { appId, applicationID, apiKey, indexName } = config.algolia
algolia = JSON.stringify({ algolia = JSON.stringify({
appId: ALGOLIA_APP_ID || appId || applicationID, appId: ALGOLIA_APP_ID || appId || applicationID,
apiKey: ALGOLIA_API_KEY || apiKey, apiKey: ALGOLIA_API_KEY || apiKey,
indexName: ALGOLIA_INDEX_NAME || indexName, indexName: ALGOLIA_INDEX_NAME || indexName,
hitsPerPage: theme.search.algolia_search.hitsPerPage, hitsPerPage: theme.search.algolia_search.hitsPerPage,
// search languages // search languages
languages: { languages: {
input_placeholder: theme.search.placeholder || _p("search.input_placeholder"), hits_empty: _p("search.algolia_search.hits_empty"),
hits_empty: _p("search.algolia_search.hits_empty"), hits_stats: _p("search.algolia_search.hits_stats"),
hits_stats: _p("search.algolia_search.hits_stats"), }
} })
}) }
}
let localSearch = 'undefined'
let localSearch = 'undefined' if (theme.search.use === 'local_search') {
if (theme.search.use === 'local_search') { const { CDN, preload, top_n_per_article, pagination, unescape } = theme.search.local_search
const { CDN, preload, top_n_per_article, unescape } = theme.search.local_search localSearch = JSON.stringify({
localSearch = JSON.stringify({ path: CDN || config.root + config.search.path,
path: CDN || config.root + config.search.path, preload,
preload, top_n_per_article,
top_n_per_article, unescape,
unescape, pagination: {
languages: { enable: pagination.enable,
// search languages hitsPerPage: pagination.hitsPerPage
hits_empty: _p("search.local_search.hits_empty"), },
hits_stats: _p("search.local_search.hits_stats"), 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, let translate = 'undefined'
translateDelay: theme.translate.translateDelay, if (theme.translate && theme.translate.enable){
msgToTraditionalChinese: theme.translate.msgToTraditionalChinese, translate = JSON.stringify({
msgToSimplifiedChinese: theme.translate.msgToSimplifiedChinese 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, let copyright = 'undefined'
languages: { if (theme.copy.enable && theme.copy.copyright.enable){
author: _p("copy_copyright.author") + ': ' + config.author, copyright = JSON.stringify({
link: _p("copy_copyright.link") + ': ', limitCount: theme.copy.copyright.limit_count,
source: _p("copy_copyright.source") + ': ' + config.title, languages: {
info: _p("copy_copyright.info") 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"), let Snackbar = 'undefined'
cht_to_chs: _p("Snackbar.cht_to_chs"), if (theme.snackbar && theme.snackbar.enable) {
day_to_night: _p("Snackbar.day_to_night"), Snackbar = JSON.stringify({
night_to_day: _p("Snackbar.night_to_day"), chs_to_cht: _p("Snackbar.chs_to_cht"),
bgLight: theme.snackbar.bg_light, cht_to_chs: _p("Snackbar.cht_to_chs"),
bgDark: theme.snackbar.bg_dark, day_to_night: _p("Snackbar.day_to_night"),
position: theme.snackbar.position, night_to_day: _p("Snackbar.night_to_day"),
}) bgLight: theme.snackbar.bg_light,
} bgDark: theme.snackbar.bg_dark,
position: theme.snackbar.position,
let noticeOutdate = 'undefined' })
if (theme.noticeOutdate && theme.noticeOutdate.enable) { }
noticeOutdate = JSON.stringify({
limitDay: theme.noticeOutdate.limit_day, let highlightProvider = config.syntax_highlighter || (config.highlight.enable ? 'highlight.js' : config.prismjs.enable ? 'prismjs' : null)
position: theme.noticeOutdate.position, const { copy, language, height_limit, fullpage, macStyle, shrink } = theme.code_blocks
messagePrev: theme.noticeOutdate.message_prev, let highlight = JSON.stringify({
messageNext: theme.noticeOutdate.message_next, plugin: highlightProvider,
}) highlightCopy: copy,
} highlightLang: language,
highlightHeightLimit: height_limit,
let highlight = 'undefined' highlightFullpage: fullpage,
let syntaxHighlighter = config.syntax_highlighter highlightMacStyle: macStyle
let highlightEnable = syntaxHighlighter ? ['highlight.js', 'prismjs'].includes(syntaxHighlighter) : (config.highlight.enable || config.prismjs.enable) })
if (highlightEnable) {
const { copy, language, height_limit, fullpage, macStyle } = theme.code_blocks script.
highlight = JSON.stringify({ const GLOBAL_CONFIG = {
plugin: syntaxHighlighter ? syntaxHighlighter : config.highlight.enable ? 'highlight.js' : 'prismjs', root: '!{config.root}',
highlightCopy: copy, algolia: !{algolia},
highlightLang: language, localSearch: !{localSearch},
highlightHeightLimit: height_limit, translate: !{translate},
highlightFullpage: fullpage, highlight: !{highlight},
highlightMacStyle: macStyle copy: {
}) success: '!{_p("copy.success")}',
} error: '!{_p("copy.error")}',
noSupport: '!{_p("copy.noSupport")}'
script. },
const GLOBAL_CONFIG = { relativeDate: {
root: '!{config.root}', homepage: !{theme.post_meta.page.date_format === 'relative'},
algolia: !{algolia}, post: !{theme.post_meta.post.date_format === 'relative'}
localSearch: !{localSearch}, },
translate: !{translate}, runtime: '!{theme.aside.card_webinfo.runtime_date ? _p("aside.card_webinfo.runtime.unit") : ""}',
noticeOutdate: !{noticeOutdate}, dateSuffix: {
highlight: !{highlight}, just: '!{_p("date_suffix.just")}',
copy: { min: '!{_p("date_suffix.min")}',
success: '!{_p("copy.success")}', hour: '!{_p("date_suffix.hour")}',
error: '!{_p("copy.error")}', day: '!{_p("date_suffix.day")}',
noSupport: '!{_p("copy.noSupport")}' month: '!{_p("date_suffix.month")}'
}, },
relativeDate: { copyright: !{copyright},
homepage: !{theme.post_meta.page.date_format === 'relative'}, lightbox: '!{ theme.lightbox || 'null' }',
post: !{theme.post_meta.post.date_format === 'relative'} Snackbar: !{Snackbar},
}, infinitegrid: {
runtime: '!{theme.aside.card_webinfo.runtime_date ? _p("aside.card_webinfo.runtime.unit") : ""}', js: '!{url_for(theme.asset.egjs_infinitegrid)}',
dateSuffix: { buttonText: '!{_p("load_more")}'
just: '!{_p("date_suffix.just")}', },
min: '!{_p("date_suffix.min")}', isPhotoFigcaption: !{theme.photofigcaption},
hour: '!{_p("date_suffix.hour")}', islazyloadPlugin: !{theme.lazyload.enable && !theme.lazyload.native},
day: '!{_p("date_suffix.day")}', isAnchor: !{theme.anchor.auto_update || false},
month: '!{_p("date_suffix.month")}' percent: {
}, toc: !{theme.toc.scroll_percent},
copyright: !{copyright}, rightside: !{theme.rightside_scroll_percent},
lightbox: '!{ theme.lightbox || 'null' }', },
Snackbar: !{Snackbar}, autoDarkmode: !{theme.darkmode.enable && theme.darkmode.autoChangeMode === 1}
infinitegrid: { }
js: '!{url_for(theme.asset.egjs_infinitegrid)}',
buttonText: '!{_p("load_more")}'
},
isPhotoFigcaption: !{theme.photofigcaption},
islazyload: !{theme.lazyload.enable},
isAnchor: !{theme.anchor.auto_update || false},
percent: {
toc: !{theme.toc.scroll_percent},
rightside: !{theme.rightside_scroll_percent},
},
autoDarkmode: !{theme.darkmode.enable && theme.darkmode.autoChangeMode === 1}
}

View File

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

View File

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

View File

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

View File

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

View File

@@ -1,3 +1,3 @@
if theme.site_verification if theme.site_verification
each item in theme.site_verification each item in theme.site_verification
meta(name=item.name content=item.content) meta(name=item.name content=item.content)

View File

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

View File

@@ -1,53 +1,52 @@
- -
const returnTopImg = img => img !== false ? img || theme.default_top_img : false const returnTopImg = img => img !== false ? img || theme.default_top_img : false
const isFixedClass = theme.nav.fixed ? ' fixed' : '' const isFixedClass = theme.nav.fixed ? ' fixed' : ''
var top_img = false var top_img = false
let headerClassName = 'not-top-img' let headerClassName = 'not-top-img'
var bg_img = '' var bg_img = ''
if !theme.disable_top_img && page.top_img !== false if !theme.disable_top_img && page.top_img !== false
if is_post() case globalPageType
- top_img = page.top_img || page.cover || theme.default_top_img when 'post'
else if is_page() - top_img = page.top_img || page.cover || theme.default_top_img
- top_img = page.top_img || theme.default_top_img when 'page'
else if is_tag() - top_img = page.top_img || theme.default_top_img
- top_img = theme.tag_per_img && theme.tag_per_img[page.tag] when 'tag'
- top_img = top_img || returnTopImg(theme.tag_img) - top_img = theme.tag_per_img && theme.tag_per_img[page.tag] || returnTopImg(theme.tag_img)
else if is_category() when 'category'
- top_img = theme.category_per_img && theme.category_per_img[page.category] - top_img = theme.category_per_img && theme.category_per_img[page.category] || returnTopImg(theme.category_img)
- top_img = top_img || returnTopImg(theme.category_img) when 'home'
else if is_home() - top_img = returnTopImg(theme.index_img)
- top_img = returnTopImg(theme.index_img) when 'archive'
else if is_archive() - top_img = returnTopImg(theme.archive_img)
- top_img = returnTopImg(theme.archive_img) default
else - top_img = page.top_img || theme.default_top_img
- top_img = page.top_img || theme.default_top_img
if top_img !== false
if top_img !== false - bg_img = getBgPath(top_img)
- bg_img = getBgPath(top_img) - headerClassName = globalPageType === 'home' ? 'full_page' : globalPageType === 'post' ? 'post-bg' : 'not-home-page'
- headerClassName = is_home() ? 'full_page' : is_post() ? 'post-bg' : 'not-home-page'
header#page-header(class=`${headerClassName + isFixedClass}` style=bg_img)
header#page-header(class=`${headerClassName + isFixedClass}` style=bg_img) include ./nav.pug
include ./nav.pug if top_img !== false
if top_img !== false if globalPageType === 'post'
if is_post() include ./post-info.pug
include ./post-info.pug else if globalPageType === 'home'
else if is_home() #site-info
#site-info h1#site-title=config.title
h1#site-title=config.title if theme.subtitle.enable
if theme.subtitle.enable - var loadSubJs = true
- var loadSubJs = true #site-subtitle
#site-subtitle span#subtitle
span#subtitle if theme.social
if theme.social #site_social_icons
#site_social_icons !=partial('includes/header/social', {}, {cache: true})
!=partial('includes/header/social', {}, {cache: true}) #scroll-down
#scroll-down i.fas.fa-angle-down.scroll-down-effects
i.fas.fa-angle-down.scroll-down-effects else
else #page-site-info
#page-site-info h1#site-title=page.title || page.tag || page.category
h1#site-title=page.title || page.tag || page.category else
else //- improve seo
//- improvement seo if globalPageType !== 'post'
if !is_post()
h1.title-seo=page.title || page.tag || page.category || config.title h1.title-seo=page.title || page.tag || page.category || config.title

View File

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

View File

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

View File

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

View File

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

View File

@@ -1,36 +1,59 @@
- var htmlClassHideAside = theme.aside.enable && theme.aside.hide ? 'hide-aside' : '' - var globalPageType = getPageType(page, is_home)
- page.aside = is_archive() ? theme.aside.display.archive: is_category() ? theme.aside.display.category : is_tag() ? theme.aside.display.tag : page.aside - var htmlClassHideAside = theme.aside.enable && theme.aside.hide ? 'hide-aside' : ''
- var hideAside = !theme.aside.enable || page.aside === false ? '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 pageType = is_post() ? 'post' : 'page' - var hideAside = !theme.aside.enable || page.aside === false ? 'hide-aside' : ''
- pageType = page.type ? pageType + ' type-' + page.type : pageType - 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) doctype html
head html(lang=config.language data-theme=theme.display_mode class=htmlClassHideAside)
include ./head.pug head
body include ./head.pug
!=partial('includes/loading/index', {}, {cache: true}) body
!=partial('includes/loading/index', {}, {cache: true})
if theme.background
#web_bg(style=getBgPath(theme.background)) if theme.background
if !Array.isArray(theme.background)
!=partial('includes/sidebar', {}, {cache: true}) #web_bg.bg-animation(style=getBgPath(theme.background))
else
#body-wrap(class=pageType) #web_bg.bg-animation
include ./header/index.pug - const bgStyleArr = theme.background.map(getBgPath)
script.
main#content-inner.layout(class=hideAside) (() => {
if body const arr = !{JSON.stringify(bgStyleArr)}
div!= body const webBgDiv = document.getElementById('web_bg')
else
block content const setRandomBg = () => {
if theme.aside.enable && page.aside !== false webBgDiv.style = arr[Math.floor(Math.random() * arr.length)]
include widget/index.pug requestAnimationFrame(() => webBgDiv.classList.add('bg-animation'))
}
- const footerBg = theme.footer_img
- const footer_bg = footerBg ? footerBg === true ? bg_img : getBgPath(footerBg) : '' document.addEventListener('pjax:send', () => {
footer#footer(style=footer_bg) webBgDiv.style = ''
!=partial('includes/footer', {}, {cache: true}) webBgDiv.classList.remove('bg-animation')
})
include ./rightside.pug
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 include ./additional-js.pug

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

@@ -1,37 +1,38 @@
- if page.total !== 1
var options = { -
prev_text: '<i class="fas fa-chevron-left fa-fw"></i>', var options = {
next_text: '<i class="fas fa-chevron-right fa-fw"></i>', prev_text: '<i class="fas fa-chevron-left fa-fw"></i>',
mid_size: 1, next_text: '<i class="fas fa-chevron-right fa-fw"></i>',
escape: false mid_size: 1,
} escape: false
}
if is_post()
- let paginationOrder = theme.post_pagination === 1 ? { prev: page.prev, next: page.next } : { prev: page.next, next: page.prev } 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 nav#pagination.pagination-post
if direction each direction, key in paginationOrder
- const getPostDesc = direction.postDesc || postDesc(direction) if direction
- let className = key === 'prev' ? (paginationOrder.next ? '' : 'full-width') : (paginationOrder.prev ? '' : 'full-width') - const getPostDesc = direction.postDesc || postDesc(direction)
- className = getPostDesc ? className : className + ' no-desc' - 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' a.pagination-related(class=className href=url_for(direction.path) title=direction.title)
img.cover(src=url_for(direction.cover) onerror=`onerror=null;src='${url_for(theme.error_img.post_page)}'` alt=`cover of ${key === 'prev' ? 'previous' : 'next'} post`) if direction.cover_type === 'img'
else img.cover(src=url_for(direction.pagination_cover || direction.cover) onerror=`onerror=null;src='${url_for(theme.error_img.post_page)}'` alt=`cover of ${key === 'prev' ? 'previous' : 'next'} post`)
.cover(style=`background: ${direction.cover || 'var(--default-bg-color)'}`) else
.cover(style=`background: ${direction.cover || 'var(--default-bg-color)'}`)
.info(class=key === 'prev' ? '' : 'text-right')
.info-1 .info(class=key === 'prev' ? '' : 'text-right')
.info-item-1=_p(`pagination.${key}`) .info-1
.info-item-2!=direction.title .info-item-1=_p(`pagination.${key}`)
if getPostDesc .info-item-2!=direction.title
.info-2 if getPostDesc
.info-item-1!=getPostDesc .info-2
else .info-item-1!=getPostDesc
nav#pagination else
.pagination nav#pagination
if is_home() .pagination
- options.format = 'page/%d/#content-inner' if globalPageType === 'home'
!=paginator(options) - options.format = 'page/%d/#content-inner'
!=paginator(options)

View File

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

View File

@@ -1,23 +1,23 @@
if theme.post_copyright.enable && page.copyright !== false if theme.post_copyright.enable && page.copyright !== false
- const author = page.copyright_author || config.author - const author = page.copyright_author || config.author
- const authorHref = page.copyright_author_href || theme.post_copyright.author_href || config.url - const authorHref = page.copyright_author_href || theme.post_copyright.author_href || config.url
- const url = page.copyright_url || page.permalink - 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) - 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
.post-copyright__author .post-copyright__author
span.post-copyright-meta span.post-copyright-meta
i.fas.fa-circle-user.fa-fw i.fas.fa-circle-user.fa-fw
= _p('post.copyright.author') + ": " = _p('post.copyright.author') + ": "
span.post-copyright-info span.post-copyright-info
a(href=authorHref)= author a(href=authorHref)= author
.post-copyright__type .post-copyright__type
span.post-copyright-meta span.post-copyright-meta
i.fas.fa-square-arrow-up-right.fa-fw i.fas.fa-square-arrow-up-right.fa-fw
= _p('post.copyright.link') + ": " = _p('post.copyright.link') + ": "
span.post-copyright-info span.post-copyright-info
a(href=url_for(url))= theme.post_copyright.decode ? decodeURI(url) : url a(href=url_for(url))= theme.post_copyright.decode ? decodeURI(url) : url
.post-copyright__notice .post-copyright__notice
span.post-copyright-meta span.post-copyright-meta
i.fas.fa-circle-exclamation.fa-fw i.fas.fa-circle-exclamation.fa-fw
= _p('post.copyright.copyright_notice') + ": " = _p('post.copyright.copyright_notice') + ": "
span.post-copyright-info!= info span.post-copyright-info!= info

View File

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

View File

@@ -1,61 +1,54 @@
- const { readmode, translate, darkmode, aside, chat } = theme - const { readmode, translate, darkmode, aside, chat } = theme
mixin rightsideItem(array)
each item in array mixin rightsideItem(array)
case item each item in array
when 'readmode' case item
if is_post() && readmode when 'readmode'
button#readmode(type="button" title=_p('rightside.readmode_title')) if globalPageType === 'post' && readmode
i.fas.fa-book-open button#readmode(type="button" title=_p('rightside.readmode_title'))
when 'translate' i.fas.fa-book-open
if translate.enable when 'translate'
button#translateLink(type="button" title=_p('rightside.translate_title'))= translate.default if translate.enable
when 'darkmode' button#translateLink(type="button" title=_p('rightside.translate_title'))= translate.default
if darkmode.enable && darkmode.button when 'darkmode'
button#darkmode(type="button" title=_p('rightside.night_mode_title')) if darkmode.enable && darkmode.button
i.fas.fa-adjust button#darkmode(type="button" title=_p('rightside.night_mode_title'))
when 'hideAside' i.fas.fa-adjust
if aside.enable && aside.button && page.aside !== false when 'hideAside'
button#hide-aside-btn(type="button" title=_p('rightside.aside')) if aside.enable && aside.button && page.aside !== false
i.fas.fa-arrows-alt-h button#hide-aside-btn(type="button" title=_p('rightside.aside'))
when 'toc' i.fas.fa-arrows-alt-h
if showToc when 'toc'
button#mobile-toc-button.close(type="button" title=_p("rightside.toc")) if showToc
i.fas.fa-list-ul button#mobile-toc-button.close(type="button" title=_p("rightside.toc"))
when 'chat' i.fas.fa-list-ul
if chat.rightside_button && chat.use when 'chat'
button#chat-btn(type="button" title=_p("rightside.chat")) if chat.rightside_button && chat.use
i.fas.fa-message button#chat-btn(type="button" title=_p("rightside.chat") style="display:none")
when 'comment' i.fas.fa-message
if commentsJsLoad when 'comment'
a#to_comment(href="#post-comment" title=_p("rightside.scroll_to_comment")) if commentsJsLoad
i.fas.fa-comments a#to_comment(href="#post-comment" title=_p("rightside.scroll_to_comment"))
i.fas.fa-comments
#rightside
- const { enable, hide, show } = theme.rightside_item_order - const { enable, hide, show } = theme.rightside_item_order
- const hideArray = enable ? hide && hide.split(',') : ['readmode','translate','darkmode','hideAside'] - const hideArray = enable && hide ? hide.split(',') : ['readmode','translate','darkmode','hideAside']
- const showArray = enable ? show && show.split(',') : ['toc','chat','comment'] - 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-config-hide #rightside
if hideArray #rightside-config-hide
+rightsideItem(hideArray) if hideArray.length
#rightside-config-show +rightsideItem(hideArray)
if enable
if hide #rightside-config-show
button#rightside-config(type="button" title=_p("rightside.setting")) if needCogBtn
i.fas.fa-cog.fa-spin button#rightside-config(type="button" title=_p("rightside.setting"))
else i.fas.fa-cog(class=theme.rightside_config_animation ? 'fa-spin' : '')
if is_post()
if (readmode || translate.enable || (darkmode.enable && darkmode.button)) if showArray.length
button#rightside-config(type="button" title=_p("rightside.setting")) +rightsideItem(showArray)
i.fas.fa-cog.fa-spin
else if translate.enable || (darkmode.enable && darkmode.button) button#go-up(type="button" title=_p("rightside.back_to_top"))
button#rightside-config(type="button" title=_p("rightside.setting")) span.scroll-percent
i.fas.fa-cog.fa-spin
if showArray
+rightsideItem(showArray)
button#go-up(type="button" title=_p("rightside.back_to_top"))
span.scroll-percent
i.fas.fa-arrow-up i.fas.fa-arrow-up

View File

@@ -1,18 +1,18 @@
if theme.menu if theme.menu
#sidebar #sidebar
#menu-mask #menu-mask
#sidebar-menus #sidebar-menus
.avatar-img.text-center .avatar-img.text-center
img(src=url_for(theme.avatar.img) onerror=`onerror=null;src='${theme.error_img.flink}'` alt="avatar") 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 .site-data.text-center
a(href=url_for(config.archive_dir) + '/') a(href=`${url_for(config.archive_dir)}/`)
.headline= _p('aside.articles') .headline= _p('aside.articles')
.length-num= site.posts.length .length-num= site.posts.length
a(href=url_for(config.tag_dir) + '/' ) a(href=`${url_for(config.tag_dir)}/`)
.headline= _p('aside.tags') .headline= _p('aside.tags')
.length-num= site.tags.length .length-num= site.tags.length
a(href=url_for(config.category_dir) + '/') a(href=`${url_for(config.category_dir)}/`)
.headline= _p('aside.categories') .headline= _p('aside.categories')
.length-num= site.categories.length .length-num= site.categories.length
!=partial('includes/header/menu_item', {}, {cache: true}) != partial('includes/header/menu_item', {}, { cache: true })

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

@@ -1,21 +1,21 @@
- const serverURL = theme.waline.serverURL.replace(/\/$/, '') - const serverURL = theme.waline.serverURL.replace(/\/$/, '')
script. script.
(() => { (() => {
async function loadWaline () { async function loadWaline () {
try { try {
const eleGroup = document.querySelectorAll('#recent-posts .waline-comment-count') const eleGroup = document.querySelectorAll('#recent-posts .waline-comment-count')
const keyArray = Array.from(eleGroup).map(i => i.getAttribute('data-path')) 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 res = await fetch(`!{serverURL}/api/comment?type=count&url=${keyArray}`, { method: 'GET' })
const result = await res.json() const result = await res.json()
result.data.forEach((count, index) => { result.data.forEach((count, index) => {
eleGroup[index].textContent = count eleGroup[index].textContent = count
}) })
} catch (err) { } catch (err) {
console.error(err) console.error(err)
} }
} }
window.pjax ? loadWaline() : window.addEventListener('load', loadWaline) window.pjax ? loadWaline() : window.addEventListener('load', loadWaline)
})() })()

View File

@@ -1,42 +1,38 @@
//- https://chatra.io/help/api/ //- https://chatra.io/help/api/
script. script.
(() => { (() => {
const isChatBtn = !{theme.chat.rightside_button} window.ChatraID = '#{theme.chatra.id}'
const isChatHideShow = !{theme.chat.button_hide_show} window.Chatra = window.Chatra || function() {
(window.Chatra.q = window.Chatra.q || []).push(arguments)
if (isChatBtn) { }
const close = () => {
Chatra('minimizeWidget') btf.getScript('https://call.chatra.io/chatra.js').then(() => {
Chatra('hide') const isChatBtn = !{theme.chat.rightside_button}
} const isChatHideShow = !{theme.chat.button_hide_show}
const open = () => { if (isChatBtn) {
Chatra('openChat', true) const close = () => {
Chatra('show') Chatra('minimizeWidget')
} Chatra('hide')
}
window.ChatraSetup = { startHidden: true }
const open = () => {
window.chatBtnFn = () => { Chatra('openChat', true)
document.getElementById('chatra').classList.contains('chatra--expanded') ? close() : open() Chatra('show')
} }
} else if (isChatHideShow) {
window.chatBtn = { window.ChatraSetup = { startHidden: true }
hide: () => Chatra('hide'),
show: () => Chatra('show') window.chatBtnFn = () => document.getElementById('chatra').classList.contains('chatra--expanded') ? close() : open()
}
} document.getElementById('chat-btn').style.display = 'block'
} else if (isChatHideShow) {
(function(d, w, c) { window.chatBtn = {
w.ChatraID = '#{theme.chatra.id}' hide: () => Chatra('hide'),
var s = d.createElement('script') show: () => Chatra('show')
w[c] = w[c] || function() { }
(w[c].q = w[c].q || []).push(arguments) }
} })
s.async = true })()
s.src = 'https://call.chatra.io/chatra.js'
if (d.head) d.head.appendChild(s)
})(document, window, 'Chatra')
})()

View File

@@ -1,37 +1,32 @@
script. script.
(() => { (() => {
window.$crisp = []; window.$crisp = ['safe', true]
window.CRISP_WEBSITE_ID = "!{theme.crisp.website_id}"; window.CRISP_WEBSITE_ID = "!{theme.crisp.website_id}"
(function () {
d = document; btf.getScript('https://client.crisp.chat/l.js').then(() => {
s = d.createElement("script"); const isChatBtn = !{theme.chat.rightside_button}
s.src = "https://client.crisp.chat/l.js"; const isChatHideShow = !{theme.chat.button_hide_show}
s.async = 1;
d.getElementsByTagName("head")[0].appendChild(s); if (isChatBtn) {
})(); const open = () => {
$crisp.push(["safe", true]) $crisp.push(["do", "chat:show"])
$crisp.push(["do", "chat:open"])
const isChatBtn = !{theme.chat.rightside_button} }
const isChatHideShow = !{theme.chat.button_hide_show}
const close = () => $crisp.push(["do", "chat:hide"])
if (isChatBtn) {
const open = () => { close()
$crisp.push(["do", "chat:show"])
$crisp.push(["do", "chat:open"]) $crisp.push(["on", "chat:closed", close])
}
window.chatBtnFn = () => $crisp.is("chat:visible") ? close() : open()
const close = () => $crisp.push(["do", "chat:hide"])
document.getElementById('chat-btn').style.display = 'block'
close() } else if (isChatHideShow) {
window.chatBtn = {
$crisp.push(["on", "chat:closed", close]) hide: () => $crisp.push(["do", "chat:hide"]),
show: () => $crisp.push(["do", "chat:show"])
window.chatBtnFn = () => $crisp.is("chat:visible") ? close() : open() }
}
} else if (isChatHideShow) { })
window.chatBtn = {
hide: () => $crisp.push(["do", "chat:hide"]),
show: () => $crisp.push(["do", "chat:show"])
}
}
})() })()

View File

@@ -1,7 +1,7 @@
case theme.chat.use case theme.chat.use
when 'chatra' when 'chatra'
include ./chatra.pug include ./chatra.pug
when 'tidio' when 'tidio'
include ./tidio.pug include ./tidio.pug
when 'crisp' when 'crisp'
include ./crisp.pug include ./crisp.pug

View File

@@ -1,41 +1,45 @@
script(src=`//code.tidio.co/${theme.tidio.public_key}.js` async) script.
script. (() => {
(() => { btf.getScript('//code.tidio.co/!{theme.tidio.public_key}.js').then(() => {
const isChatBtn = !{theme.chat.rightside_button} const isChatBtn = !{theme.chat.rightside_button}
const isChatHideShow = !{theme.chat.button_hide_show} const isChatHideShow = !{theme.chat.button_hide_show}
if (isChatBtn) { if (isChatBtn) {
let isShow = false let isShow = false
const close = () => { const close = () => {
window.tidioChatApi.hide() window.tidioChatApi.hide()
isShow = false isShow = false
} }
const open = () => { const open = () => {
window.tidioChatApi.open() window.tidioChatApi.open()
window.tidioChatApi.show() window.tidioChatApi.show()
isShow = true isShow = true
} }
const onTidioChatApiReady = () => { const onTidioChatApiReady = () => {
window.tidioChatApi.hide() window.tidioChatApi.hide()
window.tidioChatApi.on("close", close) window.tidioChatApi.on("close", close)
} }
if (window.tidioChatApi) { if (window.tidioChatApi) {
window.tidioChatApi.on("ready", onTidioChatApiReady) window.tidioChatApi.on("ready", onTidioChatApiReady)
} else { } else {
document.addEventListener("tidioChat-ready", onTidioChatApiReady) document.addEventListener("tidioChat-ready", onTidioChatApiReady)
} }
window.chatBtnFn = () => { window.chatBtnFn = () => {
if (!window.tidioChatApi) return if (!window.tidioChatApi) return
isShow ? close() : open() isShow ? close() : open()
} }
} else if (isChatHideShow) {
window.chatBtn = { document.getElementById('chat-btn').style.display = 'block'
hide: () => window.tidioChatApi && window.tidioChatApi.hide(),
show: () => window.tidioChatApi && window.tidioChatApi.show() } else if (isChatHideShow) {
} window.chatBtn = {
} hide: () => window.tidioChatApi && window.tidioChatApi.hide(),
})() show: () => window.tidioChatApi && window.tidioChatApi.show()
}
}
})
})()

View File

@@ -1,55 +1,73 @@
- const { server, site, option } = theme.artalk - const { server, site, option } = theme.artalk
- const { use, lazyload } = theme.comments - const { use, lazyload } = theme.comments
script. script.
(() => { (() => {
let artalkItem = null let artalkItem = null
const initArtalk = () => { const option = !{JSON.stringify(option)}
artalkItem = Artalk.init(Object.assign({ const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
el: '#artalk-wrap',
server: '!{server}', const destroyArtalk = () => {
site: '!{site}', if (artalkItem) {
pageKey: location.pathname, artalkItem.destroy()
darkMode: document.documentElement.getAttribute('data-theme') === 'dark', artalkItem = null
},!{JSON.stringify(option)})) }
}
if (GLOBAL_CONFIG.lightbox === 'null') return
artalkItem.on('list-loaded', () => { const artalkChangeMode = theme => artalkItem && artalkItem.setDarkMode(theme === 'dark')
artalkItem.ctx.get('list').getCommentNodes().forEach(comment => {
const $content = comment.getRender().$content const initArtalk = (el = document, pageKey = location.pathname) => {
btf.loadLightbox($content.querySelectorAll('img:not([atk-emoticon])')) artalkItem = Artalk.init({
}) el: el.querySelector('#artalk-wrap'),
}) server: '!{server}',
site: '!{site}',
const destroyArtalk = () => { darkMode: document.documentElement.getAttribute('data-theme') === 'dark',
artalkItem.destroy() ...option,
} pageKey: isShuoshuo ? pageKey : (option && option.pageKey) || pageKey
})
btf.addGlobalFn('pjaxSendOnce', destroyArtalk, 'destroyArtalk')
} if (GLOBAL_CONFIG.lightbox === 'null') return
artalkItem.on('list-loaded', () => {
const loadArtalk = async () => { artalkItem.ctx.get('list').getCommentNodes().forEach(comment => {
if (typeof Artalk === 'object') initArtalk() const $content = comment.getRender().$content
else { btf.loadLightbox($content.querySelectorAll('img:not([atk-emoticon])'))
await btf.getCSS('!{theme.asset.artalk_css}') })
await btf.getScript('!{theme.asset.artalk_js}') })
initArtalk()
} if (isShuoshuo) {
} window.shuoshuoComment.destroyArtalk = () => {
destroyArtalk()
const artalkChangeMode = theme => { if (el.children.length) {
const artalkWrap = document.getElementById('artalk-wrap') el.innerHTML = ''
if (!(artalkWrap && artalkWrap.children.length)) return el.classList.add('no-comment')
const isDark = theme === 'dark' }
artalkItem.setDarkMode(isDark) }
} }
btf.addGlobalFn('themeChange', artalkChangeMode, 'artalk') btf.addGlobalFn('pjaxSendOnce', destroyArtalk, 'destroyArtalk')
btf.addGlobalFn('themeChange', artalkChangeMode, 'artalk')
if ('!{use[0]}' === 'Artalk' || !!{lazyload}) { }
if (!{lazyload}) btf.loadComment(document.getElementById('artalk-wrap'), loadArtalk)
else setTimeout(loadArtalk, 100) const loadArtalk = async (el, pageKey) => {
} else { if (typeof Artalk === 'object') initArtalk(el, pageKey)
window.loadOtherComment = loadArtalk 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
}
})() })()

View File

@@ -1,59 +1,80 @@
- const disqusPageTitle = page.title.replace(/'/ig,"\\'") - const disqusPageTitle = page.title.replace(/'/ig,"\\'")
- const { shortname, apikey } = theme.disqus - const { shortname, apikey } = theme.disqus
- const { use, lazyload, count } = theme.comments - const { use, lazyload, count } = theme.comments
script. script.
(() => { (() => {
const disqus_config = function () { const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
this.page.url = '!{ page.permalink }'
this.page.identifier = '!{ url_for(page.path) }' const disqusReset = conf => {
this.page.title = '!{ disqusPageTitle }' window.DISQUS && window.DISQUS.reset({
} reload: true,
config: conf
const disqusReset = () => { })
window.DISQUS && window.DISQUS.reset({ }
reload: true,
config: disqus_config const loadDisqus = (el, path) => {
}) if (isShuoshuo) {
} window.shuoshuoComment.destroyDisqus = () => {
if (el.children.length) {
btf.addGlobalFn('themeChange', disqusReset, 'disqus') el.innerHTML = ''
el.classList.add('no-comment')
const loadDisqus = () =>{ }
if (window.DISQUS) disqusReset() }
else { }
const script = document.createElement('script')
script.src = 'https://!{shortname}.disqus.com/embed.js' window.disqus_identifier = isShuoshuo ? path : '!{ url_for(page.path) }'
script.setAttribute('data-timestamp', +new Date()) window.disqus_url = isShuoshuo ? location.origin + path : '!{ page.permalink }'
document.head.appendChild(script)
} const disqus_config = function () {
} this.page.url = disqus_url
this.page.identifier = disqus_identifier
const getCount = async() => { this.page.title = '!{ disqusPageTitle }'
try { }
const eleGroup = document.querySelector('#post-meta .disqus-comment-count')
if (!eleGroup) return if (window.DISQUS) disqusReset(disqus_config)
const cleanedLinks = eleGroup.href.replace(/#post-comment$/, '') else {
const script = document.createElement('script')
const res = await fetch(`https://disqus.com/api/3.0/threads/set.json?forum=!{shortname}&api_key=!{apikey}&thread:link=${cleanedLinks}`,{ script.src = 'https://!{shortname}.disqus.com/embed.js'
method: 'GET' script.setAttribute('data-timestamp', +new Date())
}) document.head.appendChild(script)
const result = await res.json() }
const count = result.response.length ? result.response[0].posts : 0 btf.addGlobalFn('themeChange', () => disqusReset(disqus_config), 'disqus')
eleGroup.textContent = count }
} catch (err) {
console.error(err) const getCount = async() => {
} try {
} const eleGroup = document.querySelector('#post-meta .disqus-comment-count')
if (!eleGroup) return
if ('!{use[0]}' === 'Disqus' || !!{lazyload}) { const cleanedLinks = eleGroup.href.replace(/#post-comment$/, '')
if (!{lazyload}) btf.loadComment(document.getElementById('disqus_thread'), loadDisqus)
else { const res = await fetch(`https://disqus.com/api/3.0/threads/set.json?forum=!{shortname}&api_key=!{apikey}&thread:link=${cleanedLinks}`,{
loadDisqus() method: 'GET'
!{ count ? 'GLOBAL_CONFIG_SITE.isPost && getCount()' : '' } })
} const result = await res.json()
} else {
window.loadOtherComment = loadDisqus 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
}
})()

View File

@@ -1,64 +1,87 @@
- let disqusjsPageTitle = page.title.replace(/'/ig,"\\'") - let disqusjsPageTitle = page.title && page.title.replace(/'/ig,"\\'")
- const { shortname:dqShortname, apikey:dqApikey, option:dqOption } = theme.disqusjs - const { shortname:dqShortname, apikey:dqApikey, option:dqOption } = theme.disqusjs
script. script.
(() => { (() => {
const initDisqusjs = () => { const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'== 'shuoshuo'
window.disqusjs = null const dqOption = !{JSON.stringify(dqOption)}
disqusjs = new DisqusJS(Object.assign({
shortname: '!{dqShortname}', const destroyDisqusjs = () => {
identifier: '!{ url_for(page.path) }', disqusjs.destroy()
url: '!{ page.permalink }', window.disqusjs = null
title: '!{ disqusjsPageTitle }', }
apikey: '!{dqApikey}',
},!{JSON.stringify(dqOption)})) const themeChange = (el, path) => {
destroyDisqusjs()
disqusjs.render(document.getElementById('disqusjs-wrap')) initDisqusjs(el, path)
} }
const themeChange = () => { const initDisqusjs = (el = document, path) => {
const ele = document.getElementById('disqus_thread') if (isShuoshuo) {
if(!ele) return window.shuoshuoComment.destroyDisqusjs = () => {
disqusjs.destroy() destroyDisqusjs()
initDisqusjs() if (el.children.length) {
} el.innerHTML = ''
el.classList.add('no-comment')
btf.addGlobalFn('themeChange', themeChange, 'disqusjs') }
}
const loadDisqusjs = async() => { }
if (window.disqusJsLoad) initDisqusjs()
else { disqusjs = new DisqusJS({
await btf.getCSS('!{url_for(theme.asset.disqusjs_css)}') shortname: '!{dqShortname}',
await btf.getScript('!{url_for(theme.asset.disqusjs)}') title: '!{ disqusjsPageTitle }',
initDisqusjs() apikey: '!{dqApikey}',
window.disqusJsLoad = true ...dqOption,
} identifier: isShuoshuo ? path : (dqOption && dqOption.identifier) || '!{ url_for(page.path) }',
} url: isShuoshuo ? location.origin + path : (dqOption && dqOption.url) || '!{ page.permalink }'
})
const getCount = async() => {
try { disqusjs.render(el.querySelector('#disqusjs-wrap'))
const eleGroup = document.querySelector('#post-meta .disqusjs-comment-count')
if (!eleGroup) return btf.addGlobalFn('themeChange', () => themeChange(el, path), 'disqusjs')
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}`,{ const loadDisqusjs = async(el, path) => {
method: 'GET' if (window.disqusJsLoad) initDisqusjs(el, path)
}) else {
const result = await res.json() await btf.getCSS('!{url_for(theme.asset.disqusjs_css)}')
const count = result.response.length ? result.response[0].posts : 0 await btf.getScript('!{url_for(theme.asset.disqusjs)}')
eleGroup.textContent = count initDisqusjs(el, path)
} catch (err) { window.disqusJsLoad = true
console.error(err) }
} }
}
const getCount = async() => {
if ('!{theme.comments.use[0]}' === 'Disqusjs' || !!{theme.comments.lazyload}) { try {
if (!{theme.comments.lazyload}) btf.loadComment(document.getElementById('disqusjs-wrap'), loadDisqusjs) const eleGroup = document.querySelector('#post-meta .disqusjs-comment-count')
else { if (!eleGroup) return
loadDisqusjs() const cleanedLinks = eleGroup.href.replace(/#post-comment$/, '')
!{ theme.comments.count ? 'GLOBAL_CONFIG_SITE.isPost && getCount()' : '' }
} const res = await fetch(`https://disqus.com/api/3.0/threads/set.json?forum=!{dqShortname}&api_key=!{dqApikey}&thread:link=${cleanedLinks}`,{
} else { method: 'GET'
window.loadOtherComment = loadDisqusjs })
} 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
}
})() })()

View File

@@ -1,46 +1,64 @@
- const fbSDKVer = 'v20.0' - const fbSDKVer = 'v20.0'
- const fbSDK = `https://connect.facebook.net/${theme.facebook_comments.lang}/sdk.js#xfbml=1&version=${fbSDKVer}` - const fbSDK = `https://connect.facebook.net/${theme.facebook_comments.lang}/sdk.js#xfbml=1&version=${fbSDKVer}`
script. script.
(()=>{ (()=>{
const loadFBComment = () => { const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'== 'shuoshuo'
document.getElementById('fb-root') ? '' : document.body.insertAdjacentHTML('afterend', '<div id="fb-root"></div>')
const loadFBComment = (el = document, path) => {
const themeNow = document.documentElement.getAttribute('data-theme') === 'dark' ? 'dark' : 'light' if (isShuoshuo) {
const $fbComment = document.getElementsByClassName('fb-comments')[0] window.shuoshuoComment.destroyFB = () => {
$fbComment.setAttribute('data-colorscheme',themeNow) if (el.children.length) {
$fbComment.setAttribute('data-href', '!{urlNoIndex(page.permalink)}') el.innerHTML = ''
el.classList.add('no-comment')
if (typeof FB === 'object') { }
FB.XFBML.parse(document.getElementsByClassName('post-meta-commentcount')[0]) }
FB.XFBML.parse(document.getElementById('post-comment')) }
}
else { document.getElementById('fb-root') ? '' : document.body.insertAdjacentHTML('afterend', '<div id="fb-root"></div>')
let ele = document.createElement('script')
ele.setAttribute('src','!{fbSDK}') const themeNow = document.documentElement.getAttribute('data-theme') === 'dark' ? 'dark' : 'light'
ele.setAttribute('async', 'true') const $fbComment = el.getElementsByClassName('fb-comments')[0]
ele.setAttribute('defer', 'true') $fbComment.setAttribute('data-colorscheme',themeNow)
ele.setAttribute('crossorigin', 'anonymous') $fbComment.setAttribute('data-href', isShuoshuo ? '!{urlNoIndex(page.permalink)}' + '#' + path : '!{urlNoIndex(page.permalink)}')
ele.setAttribute('id', 'facebook-jssdk')
document.getElementById('fb-root').insertAdjacentElement('afterbegin',ele) if (typeof FB === 'object') {
} FB.XFBML.parse(document.getElementsByClassName('post-meta-commentcount')[0])
} FB.XFBML.parse(el.querySelector('#post-comment'))
}
const fbModeChange = theme => { else {
const $fbComment = document.getElementsByClassName('fb-comments')[0] let ele = document.createElement('script')
if ($fbComment && typeof FB === 'object') { ele.setAttribute('src','!{fbSDK}')
$fbComment.setAttribute('data-colorscheme',theme) ele.setAttribute('async', 'true')
FB.XFBML.parse(document.getElementById('post-comment')) ele.setAttribute('defer', 'true')
} ele.setAttribute('crossorigin', 'anonymous')
} ele.setAttribute('id', 'facebook-jssdk')
document.getElementById('fb-root').insertAdjacentElement('afterbegin',ele)
btf.addGlobalFn('themeChange', fbModeChange, 'facebook_comments') }
}
if ('!{theme.comments.use[0]}' === 'Facebook Comments' || !!{theme.comments.lazyload}) {
if (!{theme.comments.lazyload}) btf.loadComment(document.querySelector('#post-comment .fb-comments'), loadFBComment) const fbModeChange = theme => {
else loadFBComment() const $fbComment = document.getElementsByClassName('fb-comments')[0]
} else { if ($fbComment && typeof FB === 'object') {
window.loadOtherComment = loadFBComment $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
}
})()

View File

@@ -1,52 +1,82 @@
- const { use, lazyload } = theme.comments - const { use, lazyload } = theme.comments
- const { repo, repo_id, category_id, light_theme, dark_theme, js, option } = theme.giscus - const { repo, repo_id, category_id, light_theme, dark_theme, js, option } = theme.giscus
- const giscusUrl = js || 'https://giscus.app/client.js' - const giscusUrl = js || 'https://giscus.app/client.js'
- const giscusOriginUrl = new URL(giscusUrl).origin - const giscusOriginUrl = new URL(giscusUrl).origin
script. script.
(()=>{ (() => {
const getGiscusTheme = theme => theme === 'dark' ? '!{dark_theme}' : '!{light_theme}' const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
const option = !{JSON.stringify(option)}
const loadGiscus = () => {
const config = Object.assign({ const getGiscusTheme = theme => theme === 'dark' ? '!{dark_theme}' : '!{light_theme}'
src: '!{giscusUrl}',
'data-repo': '!{repo}', const createScriptElement = config => {
'data-repo-id': '!{repo_id}', const ele = document.createElement('script')
'data-category-id': '!{category_id}', Object.entries(config).forEach(([key, value]) => {
'data-mapping': 'pathname', ele.setAttribute(key, value)
'data-theme': getGiscusTheme(document.documentElement.getAttribute('data-theme')), })
'data-reactions-enabled': '1', return ele
crossorigin: 'anonymous', }
async: true
},!{JSON.stringify(option)}) const loadGiscus = (el = document, key) => {
const mappingConfig = isShuoshuo
const ele = document.createElement('script') ? { 'data-mapping': 'specific', 'data-term': key }
for (let key in config) { : { 'data-mapping': (option && option['data-mapping']) || 'pathname' }
ele.setAttribute(key, config[key])
} const giscusConfig = {
document.getElementById('giscus-wrap').appendChild(ele) src: '!{giscusUrl}',
} 'data-repo': '!{repo}',
'data-repo-id': '!{repo_id}',
const changeGiscusTheme = theme => { 'data-category-id': '!{category_id}',
const iframe = document.querySelector('#giscus-wrap iframe') 'data-theme': getGiscusTheme(document.documentElement.getAttribute('data-theme')),
if (iframe) { 'data-reactions-enabled': '1',
const message = { crossorigin: 'anonymous',
giscus: { async: true,
setConfig: { ...option,
theme: getGiscusTheme(theme) ...mappingConfig
} }
}
} const scriptElement = createScriptElement(giscusConfig)
iframe.contentWindow.postMessage(message, '!{giscusOriginUrl}')
} el.querySelector('#giscus-wrap').appendChild(scriptElement)
}
if (isShuoshuo) {
btf.addGlobalFn('themeChange', changeGiscusTheme, 'giscus') window.shuoshuoComment.destroyGiscus = () => {
if (el.children.length) {
if ('!{use[0]}' === 'Giscus' || !!{lazyload}) { el.innerHTML = ''
if (!{lazyload}) btf.loadComment(document.getElementById('giscus-wrap'), loadGiscus) el.classList.add('no-comment')
else loadGiscus() }
} else { }
window.loadOtherComment= loadGiscus }
} }
})()
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
}
})()

View File

@@ -1,44 +1,64 @@
- const { client_id, client_secret, repo, owner, admin, option } = theme.gitalk - const { client_id, client_secret, repo, owner, admin, option } = theme.gitalk
script. script.
(() => { (() => {
const initGitalk = () => { const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
const gitalk = new Gitalk(Object.assign({ const option = !{JSON.stringify(option)}
clientID: '!{client_id}',
clientSecret: '!{client_secret}', const commentCount = n => {
repo: '!{repo}', const isCommentCount = document.querySelector('#post-meta .gitalk-comment-count')
owner: '!{owner}', if (isCommentCount) {
admin: ['!{admin}'], isCommentCount.textContent= n
id: '!{md5(page.path)}', }
updateCountCallback: commentCount }
},!{JSON.stringify(option)}))
const initGitalk = (el, path) => {
gitalk.render('gitalk-container') if (isShuoshuo) {
} window.shuoshuoComment.destroyGitalk = () => {
if (el.children.length) {
const loadGitalk = async() => { el.innerHTML = ''
if (typeof Gitalk === 'function') initGitalk() el.classList.add('no-comment')
else { }
await btf.getCSS('!{url_for(theme.asset.gitalk_css)}') }
await btf.getScript('!{url_for(theme.asset.gitalk)}') }
initGitalk()
} const gitalk = new Gitalk({
} clientID: '!{client_id}',
clientSecret: '!{client_secret}',
const commentCount = n => { repo: '!{repo}',
const isCommentCount = document.querySelector('#post-meta .gitalk-comment-count') owner: '!{owner}',
if (isCommentCount) { admin: ['!{admin}'],
isCommentCount.textContent= n updateCountCallback: commentCount,
} ...option,
} id: isShuoshuo ? path : (option && option.id) || '!{md5(page.path)}'
})
if ('!{theme.comments.use[0]}' === 'Gitalk' || !!{theme.comments.lazyload}) {
if (!{theme.comments.lazyload}) btf.loadComment(document.getElementById('gitalk-container'), loadGitalk) gitalk.render('gitalk-container')
else loadGitalk() }
} else {
window.loadOtherComment = loadGitalk 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
}
})()

View File

@@ -1,46 +1,46 @@
- let defaultComment = theme.comments.use[0] - let defaultComment = theme.comments.use[0]
hr.custom-hr hr.custom-hr
#post-comment #post-comment
.comment-head .comment-head
.comment-headline .comment-headline
i.fas.fa-comments.fa-fw i.fas.fa-comments.fa-fw
span= ' ' + _p('comment') span= ' ' + _p('comment')
if theme.comments.use.length > 1 if theme.comments.use.length > 1
.comment-switch .comment-switch
span.first-comment=defaultComment span.first-comment=defaultComment
span#switch-btn span#switch-btn
span.second-comment=theme.comments.use[1] span.second-comment=theme.comments.use[1]
.comment-wrap .comment-wrap
each name in theme.comments.use each name in theme.comments.use
div div
case name case name
when 'Disqus' when 'Disqus'
#disqus_thread #disqus_thread
when 'Valine' when 'Valine'
#vcomment.vcomment #vcomment.vcomment
when 'Disqusjs' when 'Disqusjs'
#disqusjs-wrap #disqusjs-wrap
when 'Livere' when 'Livere'
#lv-container(data-id="city" data-uid=theme.livere.uid) #lv-container(data-id="city" data-uid=theme.livere.uid)
when 'Gitalk' when 'Gitalk'
#gitalk-container #gitalk-container
when 'Utterances' when 'Utterances'
#utterances-wrap #utterances-wrap
when 'Twikoo' when 'Twikoo'
#twikoo-wrap #twikoo-wrap
when 'Waline' when 'Waline'
#waline-wrap #waline-wrap
when 'Giscus' when 'Giscus'
#giscus-wrap #giscus-wrap
when 'Facebook Comments' when 'Facebook Comments'
.fb-comments(data-colorscheme = theme.display_mode === 'dark' ? 'dark' : 'light' .fb-comments(data-colorscheme = theme.display_mode === 'dark' ? 'dark' : 'light'
data-numposts= theme.facebook_comments.pageSize || 10 data-numposts= theme.facebook_comments.pageSize || 10
data-order-by= theme.facebook_comments.order_by || 'social' data-order-by= theme.facebook_comments.order_by || 'social'
data-width="100%") data-width="100%")
when 'Remark42' when 'Remark42'
#remark42 #remark42
when 'Artalk' when 'Artalk'
#artalk-wrap #artalk-wrap

View File

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

View File

@@ -1,25 +1,47 @@
- const { use, lazyload } = theme.comments - const { use, lazyload } = theme.comments
script. script.
(()=>{ (() => {
const loadLivere = () => { const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
if (typeof LivereTower === 'object') window.LivereTower.init()
else { const loadLivere = (el, path) => {
(function(d, s) { window.livereOptions = {
var j, e = d.getElementsByTagName(s)[0]; refer: path || location.pathname
if (typeof LivereTower === 'function') { return; } }
j = d.createElement(s);
j.src = 'https://cdn-city.livere.com/js/embed.dist.js'; if (isShuoshuo) {
j.async = true; window.shuoshuoComment.destroyLivere = () => {
e.parentNode.insertBefore(j, e); if (el.children.length) {
})(document, 'script'); el.innerHTML = ''
} el.classList.add('no-comment')
} }
}
if ('!{use[0]}' === 'Livere' || !!{lazyload}) { }
if (!{lazyload}) btf.loadComment(document.getElementById('lv-container'), loadLivere)
else loadLivere() if (typeof LivereTower === 'object') window.LivereTower.init()
} else { else {
window.loadOtherComment = loadLivere (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
}
})() })()

View File

@@ -1,68 +1,78 @@
- const { host, siteId, option } = theme.remark42 - const { host, siteId, option } = theme.remark42
script.
var remark_config = Object.assign({ script.
host: '!{host}', (() => {
site_id: '!{siteId}', const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
components: ['embed'], const options = !{JSON.stringify(option)}
theme: document.documentElement.getAttribute('data-theme') === 'dark' ? 'dark' : 'light'
},!{JSON.stringify(option)}) const loadScript = src => {
const script = document.createElement('script')
function addRemark42(){ script.src = src
for (let i = 0; i < remark_config.components.length; i++) { script.defer = true
const s = document.createElement('script') document.head.appendChild(script)
s.src = remark_config.host + '/web/' + remark_config.components[i] + '.js' }
s.defer = true
document.head.appendChild(s) const addRemark42 = () => loadScript('!{host}/web/embed.js')
}
} const getCount = () => document.querySelector('.remark42__counter') && loadScript('!{host}/web/count.js')
function initRemark42() { const destroyRemark42 = () => window.remark42Instance && window.remark42Instance.destroy()
if (window.REMARK42) {
if (this.remark42Instance) { const initRemark42 = remark_config => {
this.remark42Instance.destroy() if (window.REMARK42) {
} destroyRemark42()
window.remark42Instance = window.REMARK42.createInstance({
this.remark42Instance = window.REMARK42.createInstance({ ...remark_config
...remark_config })
}) }
} }
}
const loadRemark42 = (el, path) => {
function getCount () { if (isShuoshuo) {
const ele = document.querySelector('.remark42__counter') window.shuoshuoComment.destroyRemark42 = () => {
if (ele) { destroyRemark42()
const s = document.createElement('script') if (el.children.length) {
s.src = remark_config.host + '/web/counter.js' el.innerHTML = ''
s.defer = true el.classList.add('no-comment')
document.head.appendChild(s) }
} }
} }
function loadRemark42 () { window.remark_config = {
if (window.REMARK42) { host: '!{host}',
this.initRemark42() site_id: '!{siteId}',
getCount() theme: document.documentElement.getAttribute('data-theme') === 'dark' ? 'dark' : 'light',
} else { ...options,
addRemark42() url: isShuoshuo ? window.location.origin + path : (options && options.url) || window.location.origin + window.location.pathname
window.addEventListener('REMARK42::ready', () => { }
this.initRemark42()
getCount() if (window.REMARK42) {
}) initRemark42(remark_config)
} getCount()
} } else {
addRemark42()
function remarkChangeMode (theme) { window.addEventListener('REMARK42::ready', () => {
if (!window.REMARK42) return initRemark42(remark_config)
window.REMARK42.changeTheme(theme) getCount()
} })
}
btf.addGlobalFn('themeChange', remarkChangeMode, 'remark42') }
if ('!{theme.comments.use[0]}' === 'Remark42' || !!{theme.comments.lazyload}) { const remarkChangeMode = theme => window.REMARK42 && window.REMARK42.changeTheme(theme)
if (!{theme.comments.lazyload}) btf.loadComment(document.getElementById('remark42'), loadRemark42)
else loadRemark42() btf.addGlobalFn('themeChange', remarkChangeMode, 'remark42')
} else {
function loadOtherComment () { if (isShuoshuo) {
loadRemark42() '!{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
}
})()

View File

@@ -1,45 +1,64 @@
- const { envId, region, option } = theme.twikoo - const { envId, region, option } = theme.twikoo
- const { use, lazyload, count } = theme.comments - const { use, lazyload, count } = theme.comments
script. script.
(() => { (() => {
const getCount = () => { const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
const countELement = document.getElementById('twikoo-count') const option = !{JSON.stringify(option)}
if(!countELement) return
twikoo.getCommentsCount({ const getCount = () => {
envId: '!{envId}', const countELement = document.getElementById('twikoo-count')
region: '!{region}', if(!countELement) return
urls: [window.location.pathname], twikoo.getCommentsCount({
includeReply: false envId: '!{envId}',
}).then(res => { region: '!{region}',
countELement.textContent = res[0].count urls: [window.location.pathname],
}).catch(err => { includeReply: false
console.error(err) }).then(res => {
}) countELement.textContent = res[0].count
} }).catch(err => {
console.error(err)
const init = () => { })
twikoo.init(Object.assign({ }
el: '#twikoo-wrap',
envId: '!{envId}', const init = (el = document, path = location.pathname) => {
region: '!{region}', twikoo.init({
onCommentLoaded: () => { el: el.querySelector('#twikoo-wrap'),
btf.loadLightbox(document.querySelectorAll('#twikoo .tk-content img:not(.tk-owo-emotion)')) envId: '!{envId}',
} region: '!{region}',
}, !{JSON.stringify(option)})) onCommentLoaded: () => {
btf.loadLightbox(document.querySelectorAll('#twikoo .tk-content img:not(.tk-owo-emotion)'))
!{count ? 'GLOBAL_CONFIG_SITE.isPost && getCount()' : ''} },
} ...option,
path: isShuoshuo ? path : (option && option.path) || path
const loadTwikoo = () => { })
if (typeof twikoo === 'object') setTimeout(init,0)
else btf.getScript('!{url_for(theme.asset.twikoo)}').then(init) !{count ? `GLOBAL_CONFIG_SITE.pageType === 'post' && getCount()` : ''}
}
isShuoshuo && (window.shuoshuoComment.destroyTwikoo = () => {
if ('!{use[0]}' === 'Twikoo' || !!{lazyload}) { if (el.children.length) {
if (!{lazyload}) btf.loadComment(document.getElementById('twikoo-wrap'), loadTwikoo) el.innerHTML = ''
else loadTwikoo() el.classList.add('no-comment')
} else { }
window.loadOtherComment = loadTwikoo })
} }
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
}
})() })()

View File

@@ -1,47 +1,63 @@
- const { use, lazyload } = theme.comments - const { use, lazyload } = theme.comments
- const { repo, issue_term, light_theme, dark_theme, js, option } = theme.utterances - const { repo, issue_term, light_theme, dark_theme, js, option } = theme.utterances
- const utterancesUrl = js || 'https://utteranc.es/client.js' - const utterancesUrl = js || 'https://utteranc.es/client.js'
- const utterancesOriginUrl = new URL(utterancesUrl).origin - const utterancesOriginUrl = new URL(utterancesUrl).origin
script. script.
(() => { (() => {
const getUtterancesTheme = theme => theme === 'dark' ? '#{dark_theme}' : '#{light_theme}' const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
const option = !{JSON.stringify(option)}
const loadUtterances = () => { const getUtterancesTheme = theme => theme === 'dark' ? '#{dark_theme}' : '#{light_theme}'
const config = Object.assign({
id: 'utterances_comment', const loadUtterances = (el = document, key) => {
src: '!{utterancesUrl}', if (isShuoshuo) {
repo: '!{repo}', window.shuoshuoComment.destroyUtterances = () => {
'issue-term': '!{issue_term}', if (el.children.length) {
theme: getUtterancesTheme(document.documentElement.getAttribute('data-theme')), el.innerHTML = ''
crossorigin: 'anonymous', el.classList.add('no-comment')
async: true }
},!{JSON.stringify(option)}) }
}
const ele = document.createElement('script')
for (let key in config) { const config = {
ele.setAttribute(key, config[key]) src: '!{utterancesUrl}',
} repo: '!{repo}',
document.getElementById('utterances-wrap').appendChild(ele) theme: getUtterancesTheme(document.documentElement.getAttribute('data-theme')),
} crossorigin: 'anonymous',
async: true,
const changeUtterancesTheme = theme => { ...option,
const iframe = document.querySelector('#utterances-wrap iframe') 'issue-term': isShuoshuo ? key : (option && option['issue-term']) || '!{issue_term}'
if (iframe) { }
const message = {
type: 'set-theme', const ele = document.createElement('script')
theme: getUtterancesTheme(theme) Object.entries(config).forEach(([key, value]) => ele.setAttribute(key, value))
}; el.querySelector('#utterances-wrap').appendChild(ele)
iframe.contentWindow.postMessage(message, '!{utterancesOriginUrl}') }
}
} const changeUtterancesTheme = theme => {
const iframe = document.querySelector('#utterances-wrap iframe')
btf.addGlobalFn('themeChange', changeUtterancesTheme, 'utterances') if (iframe) {
const message = {
if ('!{use[0]}' === 'Utterances' || !!{lazyload}) { type: 'set-theme',
if (!{lazyload}) btf.loadComment(document.getElementById('utterances-wrap'), loadUtterances) theme: getUtterancesTheme(theme)
else loadUtterances() };
} else { iframe.contentWindow.postMessage(message, '!{utterancesOriginUrl}')
window.loadOtherComment = loadUtterances }
} }
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
}
})() })()

View File

@@ -1,38 +1,60 @@
- const { use, lazyload } = theme.comments - const { use, lazyload } = theme.comments
- const { appId, appKey, avatar, serverURLs, visitor, option } = theme.valine - const { appId, appKey, avatar, serverURLs, visitor, option } = theme.valine
- let emojiMaps = '""' - let emojiMaps = '""'
if site.data.valine if site.data.valine
- emojiMaps = JSON.stringify(site.data.valine) - emojiMaps = JSON.stringify(site.data.valine)
script. script.
(() => { (() => {
const initValine = () => { const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
const valine = new Valine(Object.assign({ const option = !{JSON.stringify(option)}
el: '#vcomment',
appId: '#{appId}', const initValine = (el, path) => {
appKey: '#{appKey}', if (isShuoshuo) {
avatar: '#{avatar}', window.shuoshuoComment.destroyValine = () => {
serverURLs: '#{serverURLs}', if (el.children.length) {
emojiMaps: !{emojiMaps}, el.innerHTML = ''
path: window.location.pathname, el.classList.add('no-comment')
visitor: #{visitor} }
}, !{JSON.stringify(option)})) }
} }
const loadValine = async () => { const valineConfig = {
if (typeof Valine === 'function') initValine() el: '#vcomment',
else { appId: '#{appId}',
await btf.getScript('!{url_for(theme.asset.valine)}') appKey: '#{appKey}',
initValine() avatar: '#{avatar}',
} serverURLs: '#{serverURLs}',
} emojiMaps: !{emojiMaps},
visitor: #{visitor},
if ('!{use[0]}' === 'Valine' || !!{lazyload}) { ...option,
if (!{lazyload}) btf.loadComment(document.getElementById('vcomment'),loadValine) path: isShuoshuo ? path : (option && option.path) || window.location.pathname
else setTimeout(loadValine, 0) }
} else {
window.loadOtherComment = loadValine 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
}
})()

View File

@@ -1,45 +1,61 @@
- const { serverURL, option, pageview } = theme.waline - const { serverURL, option, pageview } = theme.waline
- const { lazyload, count, use } = theme.comments - const { lazyload, count, use } = theme.comments
script. script.
(() => { (() => {
let initFn = window.walineFn || null let initFn = window.walineFn || null
const isShuoshuo = GLOBAL_CONFIG_SITE.pageType === 'shuoshuo'
const initWaline = (Fn) => { const option = !{JSON.stringify(option)}
const waline = Fn(Object.assign({
el: '#waline-wrap', const destroyWaline = ele => ele.destroy()
serverURL: '!{serverURL}',
pageview: !{lazyload ? false : pageview}, const initWaline = (Fn, el = document, path = window.location.pathname) => {
dark: 'html[data-theme="dark"]', const waline = Fn({
path: window.location.pathname, el: el.querySelector('#waline-wrap'),
comment: !{lazyload ? false : count}, serverURL: '!{serverURL}',
}, !{JSON.stringify(option)})) pageview: !{lazyload ? false : pageview},
dark: 'html[data-theme="dark"]',
const destroyWaline = () => { comment: !{lazyload ? false : count},
waline.destroy() ...option,
} path: isShuoshuo ? path : (option && option.path) || path
})
btf.addGlobalFn('pjaxSendOnce', destroyWaline, 'destroyWaline')
} if (isShuoshuo) {
window.shuoshuoComment.destroyWaline = () => {
const loadWaline = () => { destroyWaline(waline)
if (initFn) initWaline(initFn) if (el.children.length) {
else { el.innerHTML = ''
btf.getCSS('!{url_for(theme.asset.waline_css)}') el.classList.add('no-comment')
.then(() => import('!{url_for(theme.asset.waline_js)}')) }
.then(({ init }) => { }
initFn = init || Waline.init }
initWaline(initFn) }
window.walineFn = initFn
}) const loadWaline = (el, path) => {
} if (initFn) initWaline(initFn, el, path)
} else {
btf.getCSS('!{url_for(theme.asset.waline_css)}')
if ('!{use[0]}' === 'Waline' || !!{lazyload}) { .then(() => import('!{url_for(theme.asset.waline_js)}'))
if (!{lazyload}) btf.loadComment(document.getElementById('waline-wrap'),loadWaline) .then(({ init }) => {
else setTimeout(loadWaline, 0) initFn = init || Waline.init
} else { initWaline(initFn, el, path)
window.loadOtherComment = loadWaline 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
}
})()

View File

@@ -1,35 +1,35 @@
if theme.fireworks && theme.fireworks.enable if theme.fireworks && theme.fireworks.enable
canvas.fireworks(mobile=`${theme.fireworks.mobile}`) canvas.fireworks(mobile=`${theme.fireworks.mobile}`)
script(src=url_for(theme.asset.fireworks)) script(src=url_for(theme.asset.fireworks))
if (theme.canvas_ribbon && theme.canvas_ribbon.enable) if (theme.canvas_ribbon && theme.canvas_ribbon.enable)
script(defer id="ribbon" src=url_for(theme.asset.canvas_ribbon) size=theme.canvas_ribbon.size 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}`) 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) 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)) 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) 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)) 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 if theme.activate_power_mode.enable
script(src=url_for(theme.asset.activate_power_mode)) script(src=url_for(theme.asset.activate_power_mode))
script. script.
POWERMODE.colorful = !{theme.activate_power_mode.colorful}; POWERMODE.colorful = !{theme.activate_power_mode.colorful};
POWERMODE.shake = !{theme.activate_power_mode.shake}; POWERMODE.shake = !{theme.activate_power_mode.shake};
POWERMODE.mobile = !{theme.activate_power_mode.mobile}; POWERMODE.mobile = !{theme.activate_power_mode.mobile};
document.body.addEventListener('input', POWERMODE); document.body.addEventListener('input', POWERMODE);
//- 鼠標特效 //- 鼠標特效
if theme.click_heart && theme.click_heart.enable if theme.click_heart && theme.click_heart.enable
script#click-heart(src=url_for(theme.asset.click_heart) async mobile=`${theme.click_heart.mobile}`) script#click-heart(src=url_for(theme.asset.click_heart) async mobile=`${theme.click_heart.mobile}`)
if theme.clickShowText && theme.clickShowText.enable if theme.clickShowText && theme.clickShowText.enable
script#click-show-text( script#click-show-text(
src= url_for(theme.asset.clickShowText) src= url_for(theme.asset.clickShowText)
data-mobile= `${theme.clickShowText.mobile}` data-mobile= `${theme.clickShowText.mobile}`
data-text= theme.clickShowText.text.join(",") data-text= theme.clickShowText.text.join(",")
data-fontsize= theme.clickShowText.fontSize data-fontsize= theme.clickShowText.fontSize
data-random= `${theme.clickShowText.random}` data-random= `${theme.clickShowText.random}`
async async
) )

View File

@@ -1,91 +1,91 @@
- const { fontColor, borderColor, scale_ticks_backdropColor } = theme.chartjs - const { fontColor, borderColor, scale_ticks_backdropColor } = theme.chartjs
script. script.
(() => { (() => {
const $chartjs = document.querySelectorAll('#article-container .chartjs-container') const applyThemeDefaultsConfig = theme => {
if ($chartjs.length === 0) return if (theme === 'dark-mode') {
Chart.defaults.color = "!{fontColor.dark}"
const applyThemeDefaultsConfig = theme => { Chart.defaults.borderColor = "!{borderColor.dark}"
if (theme === 'dark-mode') { Chart.defaults.scale.ticks.backdropColor = "!{scale_ticks_backdropColor.dark}"
Chart.defaults.color = "!{fontColor.dark}" } else {
Chart.defaults.borderColor = "!{borderColor.dark}" Chart.defaults.color = "!{fontColor.light}"
Chart.defaults.scale.ticks.backdropColor = "!{scale_ticks_backdropColor.dark}" Chart.defaults.borderColor = "!{borderColor.light}"
} else { Chart.defaults.scale.ticks.backdropColor = "!{scale_ticks_backdropColor.light}"
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
// Recursively traverse the config object and automatically apply theme-specific color schemes
const applyThemeConfig = (obj, theme) => { Object.keys(obj).forEach(key => {
if (typeof obj !== 'object' || obj === null) return const value = obj[key]
// If the property is an object and has theme-specific options, apply them
Object.keys(obj).forEach(key => { if (typeof value === 'object' && value !== null) {
const value = obj[key] if (value[theme]) {
// If the property is an object and has theme-specific options, apply them obj[key] = value[theme] // Apply the value for the current theme
if (typeof value === 'object' && value !== null) { } else {
if (value[theme]) { // Recursively process child objects
obj[key] = value[theme] // Apply the value for the current theme applyThemeConfig(value, theme)
} else { }
// Recursively process child objects }
applyThemeConfig(value, theme) })
} }
}
}) const runChartJS = ele => {
} window.loadChartJS = true
const runChartJS = () => { Array.from(ele).forEach((item, index) => {
window.loadChartJS = true const chartSrc = item.firstElementChild
const chartID = item.getAttribute('data-chartjs-id') || ('chartjs-' + index) // Use custom ID or default ID
Array.from($chartjs).forEach((item, index) => { const width = item.getAttribute('data-width')
const chartSrc = item.firstElementChild const existingCanvas = document.getElementById(chartID)
const chartID = item.getAttribute('data-chartjs-id') || ('chartjs-' + index) // Use custom ID or default ID
const width = item.getAttribute('data-width') // If a canvas already exists, remove it to avoid rendering duplicates
const existingCanvas = document.getElementById(chartID) if (existingCanvas) {
existingCanvas.parentNode.remove()
// 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 chartDefinition = chartSrc.textContent
const canvas = document.createElement('canvas') const div = document.createElement('div')
canvas.id = chartID div.className = 'chartjs-wrap'
const div = document.createElement('div') if (width) {
div.className = 'chartjs-wrap' div.style.width = width
}
if (width) {
div.style.width = width div.appendChild(canvas)
} chartSrc.insertAdjacentElement('afterend', div)
div.appendChild(canvas) const ctx = document.getElementById(chartID).getContext('2d')
chartSrc.insertAdjacentElement('afterend', div)
const config = JSON.parse(chartDefinition)
const ctx = document.getElementById(chartID).getContext('2d')
const theme = document.documentElement.getAttribute('data-theme') === 'dark' ? 'dark-mode' : 'light-mode'
const config = JSON.parse(chartDefinition)
// Set default styles (initial setup)
const theme = document.documentElement.getAttribute('data-theme') === 'dark' ? 'dark-mode' : 'light-mode' applyThemeDefaultsConfig(theme)
// Set default styles (initial setup) // Automatically traverse the config and apply dual-mode color schemes
applyThemeDefaultsConfig(theme) applyThemeConfig(config, theme)
// Automatically traverse the config and apply dual-mode color schemes new Chart(ctx, config)
applyThemeConfig(config, theme) })
}
new Chart(ctx, config)
}) const loadChartJS = () => {
} const chartJSEle = document.querySelectorAll('#article-container .chartjs-container')
if (chartJSEle.length === 0) return
const loadChartJS = () => {
window.loadChartJS ? runChartJS() : btf.getScript('!{url_for(theme.asset.chartjs)}').then(runChartJS) window.loadChartJS ? runChartJS(chartJSEle) : btf.getScript('!{url_for(theme.asset.chartjs)}').then(() => runChartJS(chartJSEle))
} }
// Listen for theme change events // Listen for theme change events
btf.addGlobalFn('themeChange', runChartJS, 'chartjs') btf.addGlobalFn('themeChange', loadChartJS, 'chartjs')
btf.addGlobalFn('encrypt', runChartJS, 'chartjs') btf.addGlobalFn('encrypt', loadChartJS, 'chartjs')
window.pjax ? loadChartJS() : document.addEventListener('DOMContentLoaded', loadChartJS) window.pjax ? loadChartJS() : document.addEventListener('DOMContentLoaded', loadChartJS)
})() })()

View File

@@ -1,14 +1,14 @@
case theme.math.use case theme.math.use
when 'mathjax' when 'mathjax'
if (theme.math.per_page && (is_post() || is_page())) || page.mathjax if (theme.math.per_page && (['post','page'].includes(globalPageType))) || page.mathjax
include ./mathjax.pug include ./mathjax.pug
when 'katex' when 'katex'
if (theme.math.per_page && (is_post() || is_page())) || page.katex if (theme.math.per_page && (['post','page'].includes(globalPageType))) || page.katex
include ./katex.pug include ./katex.pug
if theme.mermaid.enable if theme.mermaid.enable
include ./mermaid.pug include ./mermaid.pug
if theme.chartjs.enable if theme.chartjs.enable
include ./chartjs.pug include ./chartjs.pug

View File

@@ -1,16 +1,16 @@
script. script.
(async () => { (async () => {
const showKatex = () => { const showKatex = () => {
document.querySelectorAll('#article-container .katex').forEach(el => el.classList.add('katex-show')) document.querySelectorAll('#article-container .katex').forEach(el => el.classList.add('katex-show'))
} }
if (!window.katex_js_css) { if (!window.katex_js_css) {
window.katex_js_css = true window.katex_js_css = true
await btf.getCSS('!{url_for(theme.asset.katex)}') await btf.getCSS('!{url_for(theme.asset.katex)}')
if (!{theme.math.katex.copy_tex}) { if (!{theme.math.katex.copy_tex}) {
await btf.getScript('!{url_for(theme.asset.katex_copytex)}') await btf.getScript('!{url_for(theme.asset.katex_copytex)}')
} }
} }
showKatex() showKatex()
})() })()

View File

@@ -1,47 +1,78 @@
//- Mathjax 3 //- Mathjax 4/5
- const { tags, enableMenu } = theme.math.mathjax - const { tags, enableMenu } = theme.math.mathjax
script. script.
(() => { (() => {
const loadMathjax = () => { const loadMathjax = () => {
if (!window.MathJax) { if (!window.MathJax) {
window.MathJax = { window.MathJax = {
tex: { loader: {
inlineMath: [['$', '$'], ['\\(', '\\)']], load: [
tags: '!{tags}', // Four font extension packages (optional)
}, //- '[tex]/bbm',
chtml: { //- '[tex]/bboldx',
scale: 1.1 //- '[tex]/dsfont',
}, '[tex]/mhchem'
options: { ],
enableMenu: !{enableMenu}, paths: {
renderActions: { 'mathjax-newcm': '[mathjax]/../@mathjax/mathjax-newcm-font',
findScript: [10, doc => {
for (const node of document.querySelectorAll('script[type^="math/tex"]')) { //- // Four font extension packages (optional)
const display = !!node.type.match(/; *mode=display/) //- 'mathjax-bbm-extension': '[mathjax]/../@mathjax/mathjax-bbm-font-extension',
const math = new doc.options.MathItem(node.textContent, doc.inputJax[0], display) //- 'mathjax-bboldx-extension': '[mathjax]/../@mathjax/mathjax-bboldx-font-extension',
const text = document.createTextNode('') //- 'mathjax-dsfont-extension': '[mathjax]/../@mathjax/mathjax-dsfont-font-extension',
node.parentNode.replaceChild(text, node) 'mathjax-mhchem-extension': '[mathjax]/../@mathjax/mathjax-mhchem-font-extension'
math.start = {node: text, delim: '', n: 0} }
math.end = {node: text, delim: '', n: 0} },
doc.math.push(math) output: {
} font: 'mathjax-newcm',
}, ''] },
} tex: {
} inlineMath: [['$', '$'], ['\\(', '\\)']],
} tags: '!{tags}',
packages: {
const script = document.createElement('script') '[+]': [
script.src = '!{url_for(theme.asset.mathjax)}' 'mhchem'
script.id = 'MathJax-script' ]
script.async = true }
document.head.appendChild(script) },
} else { chtml: {
MathJax.startup.document.state(0) scale: 1.1
MathJax.texReset() },
MathJax.typesetPromise() options: {
} enableMenu: !{enableMenu},
} menuOptions: {
settings: {
btf.addGlobalFn('encrypt', loadMathjax, 'mathjax') enrich: false // Turn off Braille and voice narration text automatic generation
window.pjax ? loadMathjax() : window.addEventListener('load', loadMathjax) }
},
renderActions: {
findScript: [10, doc => {
for (const node of document.querySelectorAll('script[type^="math/tex"]')) {
const display = !!node.type.match(/; *mode=display/)
const math = new doc.options.MathItem(node.textContent, doc.inputJax[0], display)
const text = document.createTextNode('')
node.parentNode.replaceChild(text, node)
math.start = {node: text, delim: '', n: 0}
math.end = {node: text, delim: '', n: 0}
doc.math.push(math)
}
}, '']
}
}
}
const script = document.createElement('script')
script.src = '!{url_for(theme.asset.mathjax)}'
script.id = 'MathJax-script'
script.async = true
document.head.appendChild(script)
} else {
MathJax.startup.document.state(0)
MathJax.texReset()
MathJax.typesetPromise()
}
}
btf.addGlobalFn('encrypt', loadMathjax, 'mathjax')
window.pjax ? loadMathjax() : window.addEventListener('load', loadMathjax)
})() })()

View File

@@ -1,51 +1,323 @@
script. script.
(() => { (() => {
const runMermaid = ele => { const parseViewBox = viewBox => {
window.loadMermaid = true if (!viewBox) return null
const theme = document.documentElement.getAttribute('data-theme') === 'dark' ? '!{theme.mermaid.theme.dark}' : '!{theme.mermaid.theme.light}' const parts = viewBox.trim().split(/[\s,]+/).map(n => Number(n))
if (parts.length !== 4 || parts.some(n => Number.isNaN(n))) return null
ele.forEach((item, index) => { return parts
const mermaidSrc = item.firstElementChild }
const mermaidThemeConfig = `%%{init:{ 'theme':'${theme}'}}%%\n`
const mermaidID = `mermaid-${index}` const getSvgViewBox = svg => {
const mermaidDefinition = mermaidThemeConfig + mermaidSrc.textContent const attr = parseViewBox(svg.getAttribute('viewBox'))
if (attr) return attr
const renderFn = mermaid.render(mermaidID, mermaidDefinition)
const renderMermaid = svg => { // Fallback: use bbox to build a viewBox
mermaidSrc.insertAdjacentHTML('afterend', svg) try {
} const bbox = svg.getBBox()
if (bbox && bbox.width && bbox.height) return [bbox.x, bbox.y, bbox.width, bbox.height]
// mermaid v9 and v10 compatibility } catch (e) {
typeof renderFn === 'string' ? renderMermaid(renderFn) : renderFn.then(({ svg }) => renderMermaid(svg)) // getBBox may fail on some edge cases; ignore
}) }
}
const w = Number(svg.getAttribute('width')) || 0
const codeToMermaid = () => { const h = Number(svg.getAttribute('height')) || 0
const codeMermaidEle = document.querySelectorAll('pre > code.mermaid') if (w > 0 && h > 0) return [0, 0, w, h]
if (codeMermaidEle.length === 0) return return [0, 0, 100, 100]
}
codeMermaidEle.forEach(ele => {
const preEle = document.createElement('pre') const setSvgViewBox = (svg, vb) => {
preEle.className = 'mermaid-src' svg.setAttribute('viewBox', `${vb[0]} ${vb[1]} ${vb[2]} ${vb[3]}`)
preEle.hidden = true }
preEle.textContent = ele.textContent
const newEle = document.createElement('div') const clamp = (v, min, max) => Math.max(min, Math.min(max, v))
newEle.className = 'mermaid-wrap'
newEle.appendChild(preEle) const openSvgInNewTab = ({ source, initViewBox }) => {
ele.parentNode.replaceWith(newEle) const getClonedSvg = () => {
}) if (typeof source === 'string') {
} const template = document.createElement('template')
template.innerHTML = source.trim()
const loadMermaid = () => { const svg = template.content.querySelector('svg')
if (!{theme.mermaid.code_write}) codeToMermaid() return svg ? svg.cloneNode(true) : null
const $mermaid = document.querySelectorAll('#article-container .mermaid-wrap') }
if ($mermaid.length === 0) return if (source && typeof source.cloneNode === 'function') {
return source.cloneNode(true)
const runMermaidFn = () => runMermaid($mermaid) }
btf.addGlobalFn('themeChange', runMermaidFn, 'mermaid') return null
window.loadMermaid ? runMermaidFn() : btf.getScript('!{url_for(theme.asset.mermaid)}').then(runMermaidFn) }
}
const clone = getClonedSvg()
btf.addGlobalFn('encrypt', loadMermaid, 'mermaid') if (!clone) return
window.pjax ? loadMermaid() : document.addEventListener('DOMContentLoaded', loadMermaid) if (initViewBox && initViewBox.length === 4) {
})() clone.setAttribute('viewBox', initViewBox.join(' '))
}
if (!clone.getAttribute('xmlns')) clone.setAttribute('xmlns', 'http://www.w3.org/2000/svg')
if (!clone.getAttribute('xmlns:xlink') && clone.outerHTML.includes('xlink:')) {
clone.setAttribute('xmlns:xlink', 'http://www.w3.org/1999/xlink')
}
// inject background to match current theme
const isDark = document.documentElement.getAttribute('data-theme') === 'dark'
const bg = getComputedStyle(document.body).backgroundColor || (isDark ? '#1e1e1e' : '#ffffff')
if (!clone.style.background) clone.style.background = bg
const serializer = new XMLSerializer()
const svgSource = serializer.serializeToString(clone)
const htmlSource = `<!doctype html><html><head><meta charset="utf-8" />
<style>
html, body { width: 100%; height: 100%; margin: 0; display: flex; align-items: center; justify-content: center; background: ${bg}; }
svg { max-width: 100%; max-height: 100%; height: auto; width: auto; }
</style>
</head><body>${svgSource}</body></html>`
const blob = new Blob([htmlSource], { type: 'text/html;charset=utf-8' })
const url = URL.createObjectURL(blob)
window.open(url, '_blank', 'noopener')
setTimeout(() => URL.revokeObjectURL(url), 30000)
}
const attachMermaidViewerButton = wrap => {
let btn = wrap.querySelector('.mermaid-open-btn')
if (!btn) {
btn = document.createElement('button')
btn.type = 'button'
btn.className = 'mermaid-open-btn'
wrap.appendChild(btn)
}
btn.innerHTML = '<i class="fa fa-search fa-fw" aria-hidden="true"></i>'
if (!btn.__mermaidViewerBound) {
btn.addEventListener('click', e => {
e.preventDefault()
e.stopPropagation()
const svg = wrap.__mermaidOriginalSvg || wrap.querySelector('svg')
if (!svg) return
const initViewBox = wrap.__mermaidInitViewBox
if (typeof svg === 'string') {
openSvgInNewTab({ source: svg, initViewBox })
return
}
openSvgInNewTab({ source: svg, initViewBox })
})
btn.__mermaidViewerBound = true
}
}
// Zoom around a point (px, py) in the SVG viewport (in viewBox coordinates)
const zoomAtPoint = (vb, factor, px, py) => {
const w = vb[2] * factor
const h = vb[3] * factor
const nx = px - (px - vb[0]) * factor
const ny = py - (py - vb[1]) * factor
return [nx, ny, w, h]
}
const initMermaidGestures = wrap => {
const svg = wrap.querySelector('svg')
if (!svg) return
// Ensure viewBox exists so gestures always work
const initVb = getSvgViewBox(svg)
wrap.__mermaidInitViewBox = initVb
wrap.__mermaidCurViewBox = initVb.slice()
setSvgViewBox(svg, initVb)
// Avoid binding multiple times on themeChange/pjax
if (wrap.__mermaidGestureBound) return
wrap.__mermaidGestureBound = true
// Helper: map client (viewport) coordinate -> viewBox coordinate
const clientToViewBox = (clientX, clientY) => {
const rect = svg.getBoundingClientRect()
const vb = wrap.__mermaidCurViewBox || getSvgViewBox(svg)
const x = vb[0] + (clientX - rect.left) * (vb[2] / rect.width)
const y = vb[1] + (clientY - rect.top) * (vb[3] / rect.height)
return { x, y, rect, vb }
}
const state = {
pointers: new Map(),
startVb: null,
startDist: 0,
startCenter: null
}
const clampVb = vb => {
const init = wrap.__mermaidInitViewBox || vb
const minW = init[2] * 0.1
const maxW = init[2] * 10
const minH = init[3] * 0.1
const maxH = init[3] * 10
vb[2] = clamp(vb[2], minW, maxW)
vb[3] = clamp(vb[3], minH, maxH)
return vb
}
const setCurVb = vb => {
vb = clampVb(vb)
wrap.__mermaidCurViewBox = vb
setSvgViewBox(svg, vb)
}
const onPointerDown = e => {
// Allow only primary button for mouse
if (e.pointerType === 'mouse' && e.button !== 0) return
svg.setPointerCapture(e.pointerId)
state.pointers.set(e.pointerId, { x: e.clientX, y: e.clientY })
if (state.pointers.size === 1) {
state.startVb = (wrap.__mermaidCurViewBox || getSvgViewBox(svg)).slice()
} else if (state.pointers.size === 2) {
const pts = [...state.pointers.values()]
const dx = pts[0].x - pts[1].x
const dy = pts[0].y - pts[1].y
state.startDist = Math.hypot(dx, dy)
state.startVb = (wrap.__mermaidCurViewBox || getSvgViewBox(svg)).slice()
state.startCenter = { x: (pts[0].x + pts[1].x) / 2, y: (pts[0].y + pts[1].y) / 2 }
}
}
const onPointerMove = e => {
if (!state.pointers.has(e.pointerId)) return
state.pointers.set(e.pointerId, { x: e.clientX, y: e.clientY })
// Pan with 1 pointer
if (state.pointers.size === 1 && state.startVb) {
const p = [...state.pointers.values()][0]
const prev = { x: e.clientX - e.movementX, y: e.clientY - e.movementY }
// movementX/Y unreliable on touch, compute from stored last position
const last = wrap.__mermaidLastSinglePointer || p
const dxClient = p.x - last.x
const dyClient = p.y - last.y
wrap.__mermaidLastSinglePointer = p
const { rect } = clientToViewBox(p.x, p.y)
const vb = (wrap.__mermaidCurViewBox || getSvgViewBox(svg)).slice()
const dx = dxClient * (vb[2] / rect.width)
const dy = dyClient * (vb[3] / rect.height)
setCurVb([vb[0] - dx, vb[1] - dy, vb[2], vb[3]])
return
}
// Pinch zoom with 2 pointers
if (state.pointers.size === 2 && state.startVb && state.startDist > 0) {
const pts = [...state.pointers.values()]
const dx = pts[0].x - pts[1].x
const dy = pts[0].y - pts[1].y
const dist = Math.hypot(dx, dy)
if (!dist) return
const factor = state.startDist / dist // dist bigger => zoom in (viewBox smaller)
const cx = (pts[0].x + pts[1].x) / 2
const cy = (pts[0].y + pts[1].y) / 2
const centerClient = { x: cx, y: cy }
const pxy = clientToViewBox(centerClient.x, centerClient.y)
const cpx = pxy.x
const cpy = pxy.y
const vb = zoomAtPoint(state.startVb, factor, cpx, cpy)
setCurVb(vb)
}
}
const onPointerUpOrCancel = e => {
state.pointers.delete(e.pointerId)
if (state.pointers.size === 0) {
state.startVb = null
state.startDist = 0
state.startCenter = null
wrap.__mermaidLastSinglePointer = null
} else if (state.pointers.size === 1) {
// reset single pointer baseline to avoid jump
wrap.__mermaidLastSinglePointer = [...state.pointers.values()][0]
}
}
// Wheel zoom (mouse/trackpad)
const onWheel = e => {
// ctrlKey on mac trackpad pinch; we treat both as zoom
e.preventDefault()
const delta = e.deltaY
const zoomFactor = delta > 0 ? 1.1 : 0.9
const { x, y } = clientToViewBox(e.clientX, e.clientY)
const vb = (wrap.__mermaidCurViewBox || getSvgViewBox(svg)).slice()
setCurVb(zoomAtPoint(vb, zoomFactor, x, y))
}
const onDblClick = () => {
const init = wrap.__mermaidInitViewBox
if (!init) return
wrap.__mermaidCurViewBox = init.slice()
setSvgViewBox(svg, init)
}
svg.addEventListener('pointerdown', onPointerDown)
svg.addEventListener('pointermove', onPointerMove)
svg.addEventListener('pointerup', onPointerUpOrCancel)
svg.addEventListener('pointercancel', onPointerUpOrCancel)
svg.addEventListener('wheel', onWheel, { passive: false })
svg.addEventListener('dblclick', onDblClick)
}
const runMermaid = ele => {
window.loadMermaid = true
const theme = document.documentElement.getAttribute('data-theme') === 'dark' ? '!{theme.mermaid.theme.dark}' : '!{theme.mermaid.theme.light}'
ele.forEach((item, index) => {
const mermaidSrc = item.firstElementChild
// Clear old render (themeChange/pjax will rerun)
const oldSvg = item.querySelector('svg')
if (oldSvg) oldSvg.remove()
item.__mermaidGestureBound = false
const config = mermaidSrc.dataset.config ? JSON.parse(mermaidSrc.dataset.config) : {}
if (!config.theme) {
config.theme = theme
}
const mermaidThemeConfig = `%%{init: ${JSON.stringify(config)}}%%\n`
const mermaidID = `mermaid-${index}`
const mermaidDefinition = mermaidThemeConfig + mermaidSrc.textContent
const renderFn = mermaid.render(mermaidID, mermaidDefinition)
const renderMermaid = svg => {
mermaidSrc.insertAdjacentHTML('afterend', svg)
if (!{theme.mermaid.zoom_pan}) initMermaidGestures(item)
item.__mermaidOriginalSvg = svg
if (!{theme.mermaid.open_in_new_tab}) attachMermaidViewerButton(item)
}
// mermaid v9 and v10 compatibility
typeof renderFn === 'string' ? renderMermaid(renderFn) : renderFn.then(({ svg }) => renderMermaid(svg))
})
}
const codeToMermaid = () => {
const codeMermaidEle = document.querySelectorAll('pre > code.mermaid')
if (codeMermaidEle.length === 0) return
codeMermaidEle.forEach(ele => {
const preEle = document.createElement('pre')
preEle.className = 'mermaid-src'
preEle.hidden = true
preEle.textContent = ele.textContent
const newEle = document.createElement('div')
newEle.className = 'mermaid-wrap'
newEle.appendChild(preEle)
ele.parentNode.replaceWith(newEle)
})
}
const loadMermaid = () => {
if (!{theme.mermaid.code_write}) codeToMermaid()
const $mermaid = document.querySelectorAll('#article-container .mermaid-wrap')
if ($mermaid.length === 0) return
const runMermaidFn = () => runMermaid($mermaid)
btf.addGlobalFn('themeChange', runMermaidFn, 'mermaid')
window.loadMermaid ? runMermaidFn() : btf.getScript('!{url_for(theme.asset.mermaid)}').then(runMermaidFn)
}
btf.addGlobalFn('encrypt', loadMermaid, 'mermaid')
window.pjax ? loadMermaid() : document.addEventListener('DOMContentLoaded', loadMermaid)
})()

View File

@@ -1,64 +1,67 @@
- const { server, site, option } = theme.artalk - const { server, site, option } = theme.artalk
- const avatarCdn = (option !== null && option.gravatar && option.gravatar.mirror) || '' - const avatarCdn = (option !== null && option.gravatar && option.gravatar.mirror) || ''
- const avatarDefault = (option !== null && option.gravatar && (option.gravatar.params || option.gravatar.default)) || '' - const avatarDefault = (option !== null && option.gravatar && (option.gravatar.params || option.gravatar.default)) || ''
!= partial("includes/third-party/newest-comments/common.pug", {}, { cache: true }) != partial("includes/third-party/newest-comments/common.pug", {}, { cache: true })
script. script.
window.addEventListener('load', () => { window.addEventListener('load', () => {
const keyName = 'artalk-newest-comments' const keyName = 'artalk-newest-comments'
const { changeContent, generateHtml, run } = window.newestComments const { changeContent, generateHtml, run } = window.newestComments
const getAvatarValue = async () => { const getAvatarValue = async () => {
const predefinedAvatarCdn = '!{avatarCdn}' const predefinedAvatarCdn = '!{avatarCdn}'
const predefinedAvatarDefault = '!{avatarDefault}' const predefinedAvatarDefault = '!{avatarDefault}'
const avatarDefaultFormat = e => e.startsWith('d=') ? e : `d=${e}` const avatarDefaultFormat = e => e.startsWith('d=') ? e : `d=${e}`
if (predefinedAvatarCdn && predefinedAvatarDefault) { if (predefinedAvatarCdn && predefinedAvatarDefault) {
return { avatarCdn: predefinedAvatarCdn, avatarDefault: avatarDefaultFormat(predefinedAvatarDefault) } return { avatarCdn: predefinedAvatarCdn, avatarDefault: avatarDefaultFormat(predefinedAvatarDefault) }
} }
try { try {
const res = await fetch('!{server}/api/v2/conf') const res = await fetch('!{server}/api/v2/conf')
const result = await res.json() const result = await res.json()
const { mirror, params, default: defaults } = result.frontend_conf.gravatar const { mirror, params, default: defaults } = result.frontend_conf.gravatar
const avatarCdn = predefinedAvatarCdn || mirror const avatarCdn = predefinedAvatarCdn || mirror
let avatarDefault = avatarDefaultFormat(predefinedAvatarDefault || params || defaults) let avatarDefault = avatarDefaultFormat(predefinedAvatarDefault || params || defaults)
return { avatarCdn, avatarDefault} return { avatarCdn, avatarDefault}
} catch (e) { } catch (e) {
console.error(e) console.error(e)
return { avatarCdn: predefinedAvatarCdn, avatarDefault: avatarDefaultFormat(predefinedAvatarDefault) } return { avatarCdn: predefinedAvatarCdn, avatarDefault: avatarDefaultFormat(predefinedAvatarDefault) }
} }
} }
const searchParams = new URLSearchParams({ const searchParams = new URLSearchParams({
'site_name': '!{site}', 'site_name': '!{site}',
'limit': '!{theme.aside.card_newest_comments.limit}', 'limit': '!{newestCommentsLimit * 2}', // Fetch more comments to filter pending comments
}) })
const getComment = async (ele) => { const getComment = async (ele) => {
try { try {
const res = await fetch(`!{server}/api/v2/stats/latest_comments?${searchParams}`) const res = await fetch(`!{server}/api/v2/stats/latest_comments?${searchParams}`)
const result = await res.json() const result = await res.json()
const { avatarCdn, avatarDefault } = await getAvatarValue() const { avatarCdn, avatarDefault } = await getAvatarValue()
const artalk = result.data.map(e => { const artalk = result.data
const avatar = avatarCdn && e.email_encrypted ? `${avatarCdn}${e.email_encrypted}?${avatarDefault}` : '' .filter(e => !e.is_pending) // Filter pending comments
return { .slice(0, !{newestCommentsLimit}) // Limit the number of comments
'avatar': avatar, .map(e => {
'content': changeContent(e.content_marked), const avatar = avatarCdn && e.email_encrypted ? `${avatarCdn}${e.email_encrypted}?${avatarDefault}` : ''
'nick': e.nick, return {
'url': e.page_url, 'avatar': avatar,
'date': e.date, 'content': changeContent(e.content_marked),
} 'nick': e.nick,
}) 'url': e.page_url,
btf.saveToLocal.set(keyName, JSON.stringify(artalk), !{theme.aside.card_newest_comments.storage}/(60*24)) 'date': e.date,
generateHtml(artalk, ele) }
} catch (e) { })
console.log(e) btf.saveToLocal.set(keyName, JSON.stringify(artalk), !{theme.aside.card_newest_comments.storage}/(60*24))
ele.textContent= "!{_p('aside.card_newest_comments.error')}" generateHtml(artalk, ele)
} } catch (e) {
} console.log(e)
ele.textContent= "!{_p('aside.card_newest_comments.error')}"
run(keyName, getComment) }
}
run(keyName, getComment)
}) })

View File

@@ -1,60 +1,61 @@
script. script.
window.newestComments = { window.newestComments = {
changeContent: content => { changeContent: content => {
if (content === '') return content if (content === '') return content
content = content.replace(/<img.*?src="(.*?)"?[^\>]+>/ig, '[!{_p("aside.card_newest_comments.image")}]') // replace image link content = content.replace(/<img.*?src="(.*?)"?[^\>]+>/ig, '[!{_p("aside.card_newest_comments.image")}]') // replace image link
content = content.replace(/<a[^>]+?href=["']?([^"']+)["']?[^>]*>([^<]+)<\/a>/gi, '[!{_p("aside.card_newest_comments.link")}]') // replace url content = content.replace(/<a[^>]+?href=["']?([^"']+)["']?[^>]*>([^<]+)<\/a>/gi, '[!{_p("aside.card_newest_comments.link")}]') // replace url
content = content.replace(/<pre><code>.*?<\/pre>/gi, '[!{_p("aside.card_newest_comments.code")}]') // replace code content = content.replace(/<pre><code>.*?<\/pre>/gi, '[!{_p("aside.card_newest_comments.code")}]') // replace code
content = content.replace(/<code>.*?<\/code>/gi, '[!{_p("aside.card_newest_comments.code")}]') // replace code content = content.replace(/<code>.*?<\/code>/gi, '[!{_p("aside.card_newest_comments.code")}]') // replace code
content = content.replace(/<[^>]+>/g, "") // remove html tag content = content.replace(/<[^>]+>/g, "") // remove html tag
if (content.length > 150) { if (content.length > 150) {
content = content.substring(0, 150) + '...' content = content.substring(0, 150) + '...'
} }
return content return content
}, },
generateHtml: (array, ele) => { generateHtml: (array, ele) => {
let result = '' let result = ''
if (array.length) { if (array.length) {
for (let i = 0; i < array.length; i++) { for (let i = 0; i < array.length; i++) {
result += '<div class="aside-list-item">' result += '<div class="aside-list-item">'
if (!{theme.aside.card_newest_comments.avatar} && array[i].avatar) { if (!{theme.aside.card_newest_comments.avatar} && array[i].avatar) {
const imgAttr = '!{theme.lazyload.enable ? "data-lazy-src" : "src"}' const imgAttr = '!{theme.lazyload.enable && !theme.lazyload.native ? "data-lazy-src" : "src"}'
result += `<a href="${array[i].url}" class="thumbnail"><img ${imgAttr}="${array[i].avatar}" alt="${array[i].nick}"></a>` const lazyloadNative = '!{theme.lazyload.enable && theme.lazyload.native ? "loading=\"lazy\"" : ""}'
} result += `<a href="${array[i].url}" class="thumbnail"><img ${imgAttr}="${array[i].avatar}" alt="${array[i].nick}" ${lazyloadNative}></a>`
}
result += `<div class="content">
<a class="comment" href="${array[i].url}" title="${array[i].content}">${array[i].content}</a> result += `<div class="content">
<div class="name"><span>${array[i].nick} / </span><time datetime="${array[i].date}">${btf.diffDate(array[i].date, true)}</time></div> <a class="comment" href="${array[i].url}" title="${array[i].content}">${array[i].content}</a>
</div></div>` <div class="name"><span>${array[i].nick} / </span><time datetime="${array[i].date}">${btf.diffDate(array[i].date, true)}</time></div>
} </div></div>`
} else { }
result += '!{_p("aside.card_newest_comments.zero")}' } else {
} result += '!{_p("aside.card_newest_comments.zero")}'
}
ele.innerHTML = result
window.lazyLoadInstance && window.lazyLoadInstance.update() ele.innerHTML = result
window.pjax && window.pjax.refresh(ele) window.lazyLoadInstance && window.lazyLoadInstance.update()
}, window.pjax && window.pjax.refresh(ele)
},
newestCommentInit: (name, getComment) => {
const $dom = document.querySelector('#card-newest-comments .aside-list') newestCommentInit: (name, getComment) => {
if ($dom) { const $dom = document.querySelector('#card-newest-comments .aside-list')
const data = btf.saveToLocal.get(name) if ($dom) {
if (data) { const data = btf.saveToLocal.get(name)
newestComments.generateHtml(JSON.parse(data), $dom) if (data) {
} else { newestComments.generateHtml(JSON.parse(data), $dom)
getComment($dom) } else {
} getComment($dom)
} }
}, }
},
run: (name, getComment) => {
newestComments.newestCommentInit(name, getComment) run: (name, getComment) => {
btf.addGlobalFn('pjaxComplete', () => newestComments.newestCommentInit(name, getComment), name) newestComments.newestCommentInit(name, getComment)
} btf.addGlobalFn('pjaxComplete', () => newestComments.newestCommentInit(name, getComment), name)
}
} }

View File

@@ -1,34 +1,34 @@
!= partial("includes/third-party/newest-comments/common.pug", {}, { cache: true }) != partial("includes/third-party/newest-comments/common.pug", {}, { cache: true })
script. script.
window.addEventListener('load', () => { window.addEventListener('load', () => {
const keyName = 'disqus-newest-comments' const keyName = 'disqus-newest-comments'
const { changeContent, generateHtml, run } = window.newestComments const { changeContent, generateHtml, run } = window.newestComments
const getComment = ele => { const getComment = ele => {
fetch('https://disqus.com/api/3.0/forums/listPosts.json?forum=!{forum}&related=thread&limit=!{theme.aside.card_newest_comments.limit}&api_key=!{apiKey}') fetch('https://disqus.com/api/3.0/forums/listPosts.json?forum=!{forum}&related=thread&limit=!{newestCommentsLimit}&api_key=!{apiKey}')
.then(response => response.json()) .then(response => response.json())
.then(data => { .then(data => {
const disqusArray = data.response.map(item => { const disqusArray = data.response.map(item => {
return { return {
'avatar': item.author.avatar.cache, 'avatar': item.author.avatar.cache,
'content': changeContent(item.message), 'content': changeContent(item.message),
'nick': item.author.name, 'nick': item.author.name,
'url': item.url, 'url': item.url,
'date': item.createdAt 'date': item.createdAt
} }
}) })
btf.saveToLocal.set(keyName, JSON.stringify(disqusArray), !{theme.aside.card_newest_comments.storage}/(60*24)) btf.saveToLocal.set(keyName, JSON.stringify(disqusArray), !{theme.aside.card_newest_comments.storage}/(60*24))
generateHtml(disqusArray, ele) generateHtml(disqusArray, ele)
}).catch(e => { }).catch(e => {
console.error(e) console.error(e)
ele.textContent= "!{_p('aside.card_newest_comments.error')}" ele.textContent= "!{_p('aside.card_newest_comments.error')}"
}) })
} }
run(keyName, getComment) run(keyName, getComment)
}) })

View File

@@ -1,62 +1,62 @@
!= partial("includes/third-party/newest-comments/common.pug", {}, { cache: true }) != partial("includes/third-party/newest-comments/common.pug", {}, { cache: true })
script. script.
window.addEventListener('load', () => { window.addEventListener('load', () => {
const keyName = 'github-newest-comments' const keyName = 'github-newest-comments'
const { changeContent, generateHtml, run } = window.newestComments const { changeContent, generateHtml, run } = window.newestComments
const findTrueUrl = (array, ele) => { const findTrueUrl = (array, ele) => {
Promise.all(array.map(item => Promise.all(array.map(item =>
fetch(item.url).then(resp => resp.json()).then(data => { fetch(item.url).then(resp => resp.json()).then(data => {
let urlArray = data.body ? data.body.match(/(https?:\/\/)?([\da-z\.-]+)\.([a-z\.]{2,6})([\/\w \.-]*)*\/?/ig) : [] let urlArray = data.body ? data.body.match(/(https?:\/\/)?([\da-z\.-]+)\.([a-z\.]{2,6})([\/\w \.-]*)*\/?/ig) : []
if (!Array.isArray(urlArray) || urlArray.length === 0) { if (!Array.isArray(urlArray) || urlArray.length === 0) {
urlArray = [`${data.html_url}`] urlArray = [`${data.html_url}`]
} }
if (data.user.login === 'utterances-bot') { if (data.user.login === 'utterances-bot') {
return urlArray.pop() return urlArray.pop()
} else { } else {
return urlArray.shift() return urlArray.shift()
} }
}) })
)).then(res => { )).then(res => {
array = array.map((i,index)=> { array = array.map((i,index)=> {
return { return {
...i, ...i,
url: res[index] url: res[index]
} }
}) })
btf.saveToLocal.set(keyName, JSON.stringify(array), !{theme.aside.card_newest_comments.storage}/(60*24)) btf.saveToLocal.set(keyName, JSON.stringify(array), !{theme.aside.card_newest_comments.storage}/(60*24))
generateHtml(array, ele) generateHtml(array, ele)
}); });
} }
const getComment = ele => { const getComment = ele => {
fetch('https://api.github.com/repos/!{userRepo}/issues/comments?sort=updated&direction=desc&per_page=!{theme.aside.card_newest_comments.limit}&page=1',{ fetch('https://api.github.com/repos/!{userRepo}/issues/comments?sort=updated&direction=desc&per_page=!{newestCommentsLimit}&page=1',{
"headers": { "headers": {
Accept: 'application/vnd.github.v3.html+json' Accept: 'application/vnd.github.v3.html+json'
} }
}) })
.then(response => response.json()) .then(response => response.json())
.then(data => { .then(data => {
const githubArray = data.map(item => { const githubArray = data.map(item => {
return { return {
'avatar': item.user.avatar_url, 'avatar': item.user.avatar_url,
'content': changeContent(item.body_html || item.body), 'content': changeContent(item.body_html || item.body),
'nick': item.user.login, 'nick': item.user.login,
'url': item.issue_url, 'url': item.issue_url,
'date': item.updated_at 'date': item.updated_at
} }
}) })
findTrueUrl(githubArray, ele) findTrueUrl(githubArray, ele)
}).catch(e => { }).catch(e => {
console.error(e) console.error(e)
ele.textContent= "!{_p('aside.card_newest_comments.error')}" ele.textContent= "!{_p('aside.card_newest_comments.error')}"
}) })
} }
run(keyName, getComment) run(keyName, getComment)
}) })

View File

@@ -1,30 +1,34 @@
- let { use } = theme.comments - let { use } = theme.comments
if use if use
- let forum,apiKey,userRepo -
case use[0] let forum,apiKey,userRepo
when 'Valine' let { limit:newestCommentsLimit } = theme.aside.card_newest_comments
include ./valine.pug if (newestCommentsLimit > 10 || newestCommentsLimit < 1) newestCommentsLimit = 6
when 'Waline'
include ./waline.pug case use[0]
when 'Twikoo' when 'Valine'
include ./twikoo-comment.pug include ./valine.pug
when 'Disqus' when 'Waline'
- forum = theme.disqus.shortname include ./waline.pug
- apiKey = theme.disqus.apikey when 'Twikoo'
include ./disqus-comment.pug include ./twikoo-comment.pug
when 'Disqusjs' when 'Disqus'
- forum = theme.disqusjs.shortname - forum = theme.disqus.shortname
- apiKey = theme.disqusjs.apikey - apiKey = theme.disqus.apikey
include ./disqus-comment.pug include ./disqus-comment.pug
when 'Gitalk' when 'Disqusjs'
- let { repo,owner } = theme.gitalk - forum = theme.disqusjs.shortname
- userRepo = owner + '/' + repo - apiKey = theme.disqusjs.apikey
include ./github-issues.pug include ./disqus-comment.pug
when 'Utterances' when 'Gitalk'
- userRepo = theme.utterances.repo - let { repo,owner } = theme.gitalk
include ./github-issues.pug - userRepo = owner + '/' + repo
when 'Remark42' include ./github-issues.pug
include ./remark42.pug when 'Utterances'
when 'Artalk' - userRepo = theme.utterances.repo
include ./github-issues.pug
when 'Remark42'
include ./remark42.pug
when 'Artalk'
include ./artalk.pug include ./artalk.pug

View File

@@ -1,30 +1,31 @@
- const { host, siteId } = theme.remark42 - const { host, siteId } = theme.remark42
!= partial("includes/third-party/newest-comments/common.pug", {}, { cache: true })
script.
window.addEventListener('load', () => { script.
const keyName = 'remark42-newest-comments' window.addEventListener('load', () => {
const { changeContent, generateHtml, run } = window.newestComments const keyName = 'remark42-newest-comments'
const { changeContent, generateHtml, run } = window.newestComments
const getComment = ele => {
fetch('!{host}/api/v1/last/!{theme.aside.card_newest_comments.limit}?site=!{siteId}') const getComment = ele => {
.then(response => response.json()) fetch('!{host}/api/v1/last/!{newestCommentsLimit}?site=!{siteId}')
.then(data => { .then(response => response.json())
const remark42 = data.map(e => { .then(data => {
return { const remark42 = data.map(e => {
'avatar': e.user.picture, return {
'content': changeContent(e.text), 'avatar': e.user.picture,
'nick': e.user.name, 'content': changeContent(e.text),
'url': e.locator.url, 'nick': e.user.name,
'date': e.time, 'url': e.locator.url,
} 'date': e.time,
}) }
btf.saveToLocal.set(keyName, JSON.stringify(remark42), !{theme.aside.card_newest_comments.storage}/(60*24)) })
generateHtml(remark42, ele) btf.saveToLocal.set(keyName, JSON.stringify(remark42), !{theme.aside.card_newest_comments.storage}/(60*24))
}).catch(e => { generateHtml(remark42, ele)
console.error(e) }).catch(e => {
ele.textContent= "!{_p('aside.card_newest_comments.error')}" console.error(e)
}) ele.textContent= "!{_p('aside.card_newest_comments.error')}"
} })
}
run(keyName, getComment)
}) run(keyName, getComment)
})

View File

@@ -1,45 +1,45 @@
!= partial("includes/third-party/newest-comments/common.pug", {}, { cache: true }) != partial("includes/third-party/newest-comments/common.pug", {}, { cache: true })
script. script.
window.addEventListener('load', () => { window.addEventListener('load', () => {
const keyName = 'twikoo-newest-comments' const keyName = 'twikoo-newest-comments'
const { changeContent, generateHtml, run } = window.newestComments const { changeContent, generateHtml, run } = window.newestComments
const getComment = ele => { const getComment = ele => {
const runTwikoo = () => { const runTwikoo = () => {
twikoo.getRecentComments({ twikoo.getRecentComments({
envId: '!{theme.twikoo.envId}', envId: '!{theme.twikoo.envId}',
region: '!{theme.twikoo.region}', region: '!{theme.twikoo.region}',
pageSize: !{theme.aside.card_newest_comments.limit}, pageSize: !{newestCommentsLimit},
includeReply: true includeReply: true
}).then(res => { }).then(res => {
const twikooArray = res.map(e => { const twikooArray = res.map(e => {
return { return {
'content': changeContent(e.comment), 'content': changeContent(e.comment),
'avatar': e.avatar, 'avatar': e.avatar,
'nick': e.nick, 'nick': e.nick,
'url': e.url + '#' + e.id, 'url': e.url + '#' + e.id,
'date': new Date(e.created).toISOString() 'date': new Date(e.created).toISOString()
} }
}) })
btf.saveToLocal.set(keyName, JSON.stringify(twikooArray), !{theme.aside.card_newest_comments.storage}/(60*24)) btf.saveToLocal.set(keyName, JSON.stringify(twikooArray), !{theme.aside.card_newest_comments.storage}/(60*24))
generateHtml(twikooArray, ele) generateHtml(twikooArray, ele)
}).catch(err => { }).catch(err => {
console.error(err) console.error(err)
ele.textContent= "!{_p('aside.card_newest_comments.error')}" ele.textContent= "!{_p('aside.card_newest_comments.error')}"
}) })
} }
if (typeof twikoo === 'object') { if (typeof twikoo === 'object') {
runTwikoo() runTwikoo()
} else { } else {
btf.getScript('!{url_for(theme.asset.twikoo)}').then(runTwikoo) btf.getScript('!{url_for(theme.asset.twikoo)}').then(runTwikoo)
} }
} }
run(keyName, getComment) run(keyName, getComment)
}) })

View File

@@ -1,51 +1,51 @@
- let default_avatar = theme.valine.avatar - let default_avatar = theme.valine.avatar
script(src=url_for(theme.asset.blueimp_md5)) script(src=url_for(theme.asset.blueimp_md5))
!= partial("includes/third-party/newest-comments/common.pug", {}, { cache: true }) != partial("includes/third-party/newest-comments/common.pug", {}, { cache: true })
script. script.
window.addEventListener('load', () => { window.addEventListener('load', () => {
const keyName = 'valine-newest-comments' const keyName = 'valine-newest-comments'
const { changeContent, generateHtml, run } = window.newestComments const { changeContent, generateHtml, run } = window.newestComments
const getIcon = (icon, mail) => { const getIcon = (icon, mail) => {
if (icon) return icon if (icon) return icon
let defaultIcon = '!{ default_avatar ? `?d=${default_avatar}` : ''}' let defaultIcon = '!{ default_avatar ? `?d=${default_avatar}` : ''}'
let iconUrl = `https://gravatar.loli.net/avatar/${md5(mail.toLowerCase()) + defaultIcon}` let iconUrl = `https://gravatar.loli.net/avatar/${md5(mail.toLowerCase()) + defaultIcon}`
return iconUrl return iconUrl
} }
const getComment = ele => { const getComment = ele => {
const serverURL = '!{theme.valine.serverURLs || `https://${theme.valine.appId.substring(0,8)}.api.lncldglobal.com` }' const serverURL = '!{theme.valine.serverURLs || `https://${theme.valine.appId.substring(0,8)}.api.lncldglobal.com` }'
var settings = { var settings = {
"method": "GET", "method": "GET",
"headers": { "headers": {
"X-LC-Id": '!{theme.valine.appId}', "X-LC-Id": '!{theme.valine.appId}',
"X-LC-Key": '!{theme.valine.appKey}', "X-LC-Key": '!{theme.valine.appKey}',
"Content-Type": "application/json" "Content-Type": "application/json"
}, },
} }
fetch(`${serverURL}/1.1/classes/Comment?limit=!{theme.aside.card_newest_comments.limit}&order=-createdAt`,settings) fetch(`${serverURL}/1.1/classes/Comment?limit=!{newestCommentsLimit}&order=-createdAt`,settings)
.then(response => response.json()) .then(response => response.json())
.then(data => { .then(data => {
const valineArray = data.results.map(e => { const valineArray = data.results.map(e => {
return { return {
'avatar': getIcon(e.QQAvatar, e.mail), 'avatar': getIcon(e.QQAvatar, e.mail),
'content': changeContent(e.comment), 'content': changeContent(e.comment),
'nick': e.nick, 'nick': e.nick,
'url': e.url + '#' + e.objectId, 'url': e.url + '#' + e.objectId,
'date': e.updatedAt, 'date': e.updatedAt,
} }
}) })
btf.saveToLocal.set(keyName, JSON.stringify(valineArray), !{theme.aside.card_newest_comments.storage}/(60*24)) btf.saveToLocal.set(keyName, JSON.stringify(valineArray), !{theme.aside.card_newest_comments.storage}/(60*24))
generateHtml(valineArray, ele) generateHtml(valineArray, ele)
}).catch(e => { }).catch(e => {
console.error(e) console.error(e)
ele.textContent= "!{_p('aside.card_newest_comments.error')}" ele.textContent= "!{_p('aside.card_newest_comments.error')}"
}) })
} }
run(keyName, getComment) run(keyName, getComment)
}) })

View File

@@ -1,32 +1,32 @@
- const serverURL = theme.waline.serverURL.replace(/\/$/, '') - const serverURL = theme.waline.serverURL.replace(/\/$/, '')
!= partial("includes/third-party/newest-comments/common.pug", {}, { cache: true }) != partial("includes/third-party/newest-comments/common.pug", {}, { cache: true })
script. script.
window.addEventListener('load', () => { window.addEventListener('load', () => {
const keyName = 'waline-newest-comments' const keyName = 'waline-newest-comments'
const { changeContent, generateHtml, run } = window.newestComments const { changeContent, generateHtml, run } = window.newestComments
const getComment = async (ele) => { const getComment = async (ele) => {
try { try {
const res = await fetch('!{serverURL}/api/comment?type=recent&count=!{theme.aside.card_newest_comments.limit}', { method: 'GET' }) const res = await fetch('!{serverURL}/api/comment?type=recent&count=!{newestCommentsLimit}')
const result = await res.json() const result = await res.json()
const walineArray = result.data.map(e => { const walineArray = result.data.map(e => {
return { return {
'content': changeContent(e.comment), 'content': changeContent(e.comment),
'avatar': e.avatar, 'avatar': e.avatar,
'nick': e.nick, 'nick': e.nick,
'url': e.url + '#' + e.objectId, 'url': e.url + '#' + e.objectId,
'date': e.time || e.insertedAt 'date': e.time || e.insertedAt
} }
}) })
btf.saveToLocal.set(keyName, JSON.stringify(walineArray), !{theme.aside.card_newest_comments.storage}/(60*24)) btf.saveToLocal.set(keyName, JSON.stringify(walineArray), !{theme.aside.card_newest_comments.storage}/(60*24))
generateHtml(walineArray, ele) generateHtml(walineArray, ele)
} catch (err) { } catch (err) {
console.error(err) console.error(err)
ele.textContent= "!{_p('aside.card_newest_comments.error')}" ele.textContent= "!{_p('aside.card_newest_comments.error')}"
} }
} }
run(keyName, getComment) run(keyName, getComment)
}) })

View File

@@ -1,23 +0,0 @@
script.
(() => {
const panguFn = () => {
if (typeof pangu === 'object') pangu.autoSpacingPage()
else {
btf.getScript('!{url_for(theme.asset.pangu)}')
.then(() => {
pangu.autoSpacingPage()
})
}
}
const panguInit = () => {
if (!{theme.pangu.field === 'post'}){
GLOBAL_CONFIG_SITE.isPost && panguFn()
} else {
panguFn()
}
}
btf.addGlobalFn('pjaxComplete', panguInit, 'pangu')
document.addEventListener('DOMContentLoaded', panguInit)
})()

View File

@@ -1,63 +1,73 @@
- var pjaxExclude = 'a:not([target="_blank"])' - var pjaxExclude = 'a:not([target="_blank"])'
if theme.pjax.exclude if theme.pjax.exclude
each val in theme.pjax.exclude each val in theme.pjax.exclude
- pjaxExclude += `:not([href="${val}"])` - pjaxExclude += `:not([href="${val}"])`
- let pjaxSelectors = ['head > title', '#config-diff', '#body-wrap', '#rightside-config-hide', '#rightside-config-show', '.js-pjax'] - let pjaxSelectors = ['head > title', '#config-diff', '#body-wrap', '#rightside-config-hide', '#rightside-config-show', '.js-pjax']
- let choose = theme.comments.use - let choose = theme.comments.use
if choose if choose
if theme.Open_Graph_meta.enable && (choose.includes('Livere') || choose.includes('Utterances') || choose.includes('Giscus')) if choose.includes('Livere') || choose.includes('Utterances') || choose.includes('Giscus')
- pjaxSelectors.unshift('meta[property="og:image"]', 'meta[property="og:title"]', 'meta[property="og:url"]') - pjaxSelectors.unshift('link[rel="canonical"]')
if choose.includes('Utterances') || choose.includes('Giscus') if theme.Open_Graph_meta.enable
- pjaxSelectors.unshift('link[rel="canonical"]') - pjaxSelectors.unshift('meta[property="og:image"]', 'meta[property="og:title"]', 'meta[property="og:url"]', 'meta[property="og:description"]')
else
script(src=url_for(theme.asset.pjax)) - pjaxSelectors.unshift('meta[name="description"]')
script.
(() => { script(src=url_for(theme.asset.pjax) defer)
const pjaxSelectors = !{JSON.stringify(pjaxSelectors)} script.
document.addEventListener('DOMContentLoaded', () => {
window.pjax = new Pjax({ const pjaxSelectors = !{JSON.stringify(pjaxSelectors)}
elements: '!{pjaxExclude}',
selectors: pjaxSelectors, window.pjax = new Pjax({
cacheBust: false, elements: '!{pjaxExclude}',
analytics: !{theme.google_analytics ? true : false}, selectors: pjaxSelectors,
scrollRestoration: false cacheBust: false,
}) analytics: !{theme.google_analytics ? true : false},
scrollRestoration: false
const triggerPjaxFn = (val) => { })
if (!val) return
Object.values(val).forEach(fn => fn()) const triggerPjaxFn = (val) => {
} if (!val) return
Object.values(val).forEach(fn => {
document.addEventListener('pjax:send', () => { try {
// removeEventListener fn()
btf.removeGlobalFnEvent('pjaxSendOnce') } catch (err) {
btf.removeGlobalFnEvent('themeChange') console.debug('Pjax callback failed:', err)
}
// reset readmode })
const $bodyClassList = document.body.classList }
if ($bodyClassList.contains('read-mode')) $bodyClassList.remove('read-mode')
document.addEventListener('pjax:send', () => {
triggerPjaxFn(window.globalFn.pjaxSend) // removeEventListener
}) btf.removeGlobalFnEvent('pjaxSendOnce')
btf.removeGlobalFnEvent('themeChange')
document.addEventListener('pjax:complete', () => {
btf.removeGlobalFnEvent('pjaxCompleteOnce') // reset readmode
document.querySelectorAll('script[data-pjax]').forEach(item => { const $bodyClassList = document.body.classList
const newScript = document.createElement('script') if ($bodyClassList.contains('read-mode')) $bodyClassList.remove('read-mode')
const content = item.text || item.textContent || item.innerHTML || ""
Array.from(item.attributes).forEach(attr => newScript.setAttribute(attr.name, attr.value)) triggerPjaxFn(window.globalFn.pjaxSend)
newScript.appendChild(document.createTextNode(content)) })
item.parentNode.replaceChild(newScript, item)
}) document.addEventListener('pjax:complete', () => {
btf.removeGlobalFnEvent('pjaxCompleteOnce')
triggerPjaxFn(window.globalFn.pjaxComplete) document.querySelectorAll('script[data-pjax]').forEach(item => {
}) const newScript = document.createElement('script')
const content = item.text || item.textContent || item.innerHTML || ""
document.addEventListener('pjax:error', e => { Array.from(item.attributes).forEach(attr => newScript.setAttribute(attr.name, attr.value))
if (e.request.status === 404) { newScript.appendChild(document.createTextNode(content))
pjax.loadUrl('!{url_for("/404.html")}') item.parentNode.replaceChild(newScript, item)
} })
})
})() triggerPjaxFn(window.globalFn.pjaxComplete)
})
document.addEventListener('pjax:error', e => {
if (e.request.status === 404) {
!{theme.error_404 && theme.error_404.enable}
? pjax.loadUrl('!{url_for("/404.html")}')
: window.location.href = e.request.responseURL
}
})
})

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