hexo-theme-shokax 0.2.4 → 0.2.5-beta1
Sign up to get free protection for your applications and to get access to all the features.
- package/README.md +7 -2
- package/README_en.MD +44 -13
- package/_config.yml +11 -0
- package/layout/_mixin/postmeta.pug +0 -1
- package/layout/_partials/post/post.pug +13 -0
- package/package.json +2 -2
- package/scripts/helpers/summary_ai.js +96 -0
- package/source/css/_common/components/tags/tabs.styl +10 -3
- package/source/js/_app/library.js +2 -2
- package/source/js/_app/page.js +2 -2
- package/source/js/_app/player.js +5 -5
package/README.md
CHANGED
@@ -25,15 +25,17 @@ shokaX的社区资源导航和插件仓库为[awesome-shokaX](https://github.com
|
|
25
25
|
| PWA支持 | ✅ | JSD拆分 | ✅ |
|
26
26
|
| 注入API | ✅ | 社区插件系统 | ✅ |
|
27
27
|
| 自定义字体 | ✅* | 自定义样式 | ✅* |
|
28
|
-
| 多种评论系统支持 | ✅ |
|
28
|
+
| 多种评论系统支持 | ✅ | AI生成文章概括 | 🔬 |
|
29
29
|
| 底部备案号 | ✅ | 自定义页尾 | ✅* |
|
30
30
|
| CSS渐变封面 | ✅ | typescript支持 | ✅ |
|
31
31
|
|
32
32
|
备注:
|
33
33
|
- *: 需要使用注入API实现
|
34
|
+
- 🔬: 实验中,可能存在问题
|
34
35
|
|
35
36
|
|
36
37
|
## 🔧 如何安装?
|
38
|
+
注意: 本项目需要 node.js 18.x 或更高版本才能运行 \
|
37
39
|
建议使用[ShokaX-CLI](https://github.com/zkz098/shokaX-CLI) ,执行下列命令即可:
|
38
40
|
```bash
|
39
41
|
npm i shokax-cli --location=global
|
@@ -58,7 +60,7 @@ github仓库建议通过右边的 releases 下载,步骤为:
|
|
58
60
|
- [Easy hexo](https://easyhexo.com/)
|
59
61
|
|
60
62
|
# 许可证
|
61
|
-
许可证: GPL 3 \
|
63
|
+
许可证: GPL 3 or later \
|
62
64
|
[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2Fzkz098%2Fhexo-theme-shokaX.svg?type=large)](https://app.fossa.com/projects/git%2Bgithub.com%2Fzkz098%2Fhexo-theme-shokaX?ref=badge_large)
|
63
65
|
|
64
66
|
## 特别说明
|
@@ -77,3 +79,6 @@ GPL许可证主要目的是限制修改后的分发行为,避免未经许可
|
|
77
79
|
|
78
80
|
## 特别鸣谢
|
79
81
|
[<img src="https://resources.jetbrains.com/storage/products/company/brand/logos/jb_beam.png" width="25%">](https://jb.gg/OpenSourceSupport)
|
82
|
+
|
83
|
+
## 其他信息
|
84
|
+
![Star history chart](https://api.star-history.com/svg?repos=theme-shoka-x/hexo-theme-shokaX&type=Date)
|
package/README_en.MD
CHANGED
@@ -1,25 +1,56 @@
|
|
1
|
+
If the repository address you are visiting is zkz098/hexo-theme-shokaX, please switch to the latest address: theme-shoka-x/hexo-theme-shokaX.
|
2
|
+
|
1
3
|
# hexo-theme-shokaX
|
2
4
|
[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2Fzkz098%2Fhexo-theme-shokaX.svg?type=shield)](https://app.fossa.com/projects/git%2Bgithub.com%2Fzkz098%2Fhexo-theme-shokaX?ref=badge_shield)
|
3
|
-
![LICENSE]( https://img.shields.io/github/license/
|
4
|
-
![stars](https://img.shields.io/github/stars/
|
5
|
+
![LICENSE]( https://img.shields.io/github/license/theme-shoka-x/hexo-theme-shokaX)
|
6
|
+
![stars](https://img.shields.io/github/stars/theme-shoka-x/hexo-theme-shokaX)
|
5
7
|
![version](https://shields.io/npm/v/hexo-theme-shokax)
|
8
|
+
![build](https://img.shields.io/github/actions/workflow/status/theme-shoka-x/hexo-theme-shokaX/build-theme.yml)
|
9
|
+
|
10
|
+
Language: [简体中文](./README.md) | English \
|
11
|
+
This project is a secondary development version of shoka (spiritual sequel), dedicated to improving performance and optimizing modding experience.
|
12
|
+
The reason for its birth is that shoka has not been updated for two years, with a large backlog of bugs and feature requests.
|
13
|
+
|
14
|
+
The community resource navigation and plugin repository for shokaX is awesome-shokaX.
|
15
|
+
|
16
|
+
## 💬 Differences with shoka
|
17
|
+
The original shoka used javascript+Native+nunjucks technology, while shokaX uses typescript+Vue 3+Pug technology and has changed a lot of hard-to-access CDN links.
|
18
|
+
|
19
|
+
## ✨ Feature List
|
20
|
+
| Feature Name | Implementation Status | Feature Name | Implementation Status |
|
21
|
+
|:-------------------------------:|:---------------------:|:------------------------------:|:---------------------:|
|
22
|
+
| PWA Support | ✅ | JSD Splitting | ✅ |
|
23
|
+
| Injection API | ✅ | Community Plugin System | ✅ |
|
24
|
+
| Custom Fonts | ✅* | Custom Styles | ✅* |
|
25
|
+
| Multiple Comment System Support | ✅ | User Behavior Analysis Support | ✅ |
|
26
|
+
| Record Number at the Bottom | ✅ | Custom Footer | ✅* |
|
27
|
+
| CSS Gradient Cover | ✅ | Typescript Support | ✅ |
|
6
28
|
|
7
|
-
|
8
|
-
|
9
|
-
This project is in a period of intensive development, but the github repository version is basically available. \
|
10
|
-
Please refer to the wiki for secondary development and common problems.
|
11
|
-
Starting with `0.0.2-alpha2`, `lantern` and `qweather` have been migrated as plugins.
|
12
|
-
The plugin system is complete, see [awesome-shokaX](https://github.com/zkz098/awesome-shokaX) for how to use it.
|
29
|
+
Remarks:
|
30
|
+
- *: Requires implementation using Injection API.
|
13
31
|
|
14
|
-
## How to
|
15
|
-
[ShokaX-CLI](https://github.com/zkz098/shokaX-CLI)
|
32
|
+
## 🔧 How to Install?
|
33
|
+
It is recommended to use [ShokaX-CLI](https://github.com/zkz098/shokaX-CLI) and execute the following command:
|
16
34
|
```bash
|
17
35
|
npm i shokax-cli --location=global
|
18
|
-
# hexo init
|
19
|
-
SXC install
|
36
|
+
# hexo init initializes the environment
|
37
|
+
SXC install shokaX
|
20
38
|
```
|
39
|
+
[Click here](https://docs.kaitaku.xyz/guide/#%E9%85%8D%E7%BD%AE%E4%B8%BB%E9%A2%98) for the next configuration steps.
|
40
|
+
|
41
|
+
It is recommended to download the GitHub repository from the Releases on the right-hand side, as follows:
|
42
|
+
|
43
|
+
- Click on the Latest version in Releases
|
44
|
+
- Download the Source code(zip) in Assets
|
45
|
+
- Unzip to use as the theme
|
21
46
|
|
22
47
|
# License
|
23
|
-
|
48
|
+
license: GPL 3 or later
|
24
49
|
[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2Fzkz098%2Fhexo-theme-shokaX.svg?type=large)](https://app.fossa.com/projects/git%2Bgithub.com%2Fzkz098%2Fhexo-theme-shokaX?ref=badge_large)
|
25
50
|
|
51
|
+
## Contributors
|
52
|
+
|
53
|
+
[![](https://contributors-img.web.app/image?repo=zkz098/hexo-theme-shokaX)](https://github.com/zkz098/hexo-theme-shokaX/graphs/contributors)
|
54
|
+
|
55
|
+
## Special thanks
|
56
|
+
[<img src="https://resources.jetbrains.com/storage/products/company/brand/logos/jb_beam.png" width="25%">](https://jb.gg/OpenSourceSupport)
|
package/_config.yml
CHANGED
@@ -198,6 +198,17 @@ giscus:
|
|
198
198
|
commentTheme:
|
199
199
|
lang:
|
200
200
|
|
201
|
+
summary:
|
202
|
+
enable: false
|
203
|
+
introduce: "我是基于ChatGPT-turbo-3.5实现的AI助手,在此网站上负责整理和概括文章" # AI自我介绍
|
204
|
+
mode: openai # openai/custom
|
205
|
+
openai:
|
206
|
+
remote: "https://api.openai.com"
|
207
|
+
apikey: "key"
|
208
|
+
custom:
|
209
|
+
remote: "http://localhost:8000"
|
210
|
+
|
211
|
+
|
201
212
|
# Social Links
|
202
213
|
# Usage: `Key: permalink || icon || color`
|
203
214
|
# Key is the link label showing to end users.
|
@@ -12,6 +12,19 @@ article(itemscope itemtype="http://schema.org/Article" class="post block" lang=t
|
|
12
12
|
div(class="gallery" itemscope itemtype="http://schema.org/ImageGallery")
|
13
13
|
each photo in post.photos
|
14
14
|
img(data-src=_image_url(photo, post.path) itemprop="contentUrl")
|
15
|
+
if theme.summary.enable && page.layout === 'post'
|
16
|
+
div(class='tabs' id='summary')
|
17
|
+
div(class="show-btn")
|
18
|
+
div(class="nav")
|
19
|
+
ul(class="special")
|
20
|
+
div(class="tab" data-id="summary" data-title="自我介绍")
|
21
|
+
p
|
22
|
+
!= get_introduce()
|
23
|
+
div(class="tab active" data-id="summary" data-title="文章概括")
|
24
|
+
p
|
25
|
+
!= get_summary(page)
|
26
|
+
|
27
|
+
|
15
28
|
!= post.content
|
16
29
|
if post.tags && post.tags.length
|
17
30
|
div(class="tags")
|
package/package.json
CHANGED
@@ -1,6 +1,6 @@
|
|
1
1
|
{
|
2
2
|
"name": "hexo-theme-shokax",
|
3
|
-
"version": "0.2.
|
3
|
+
"version": "0.2.5-beta1",
|
4
4
|
"description": "a hexo theme based on shoka",
|
5
5
|
"main": "index.js",
|
6
6
|
"repository": "https://github.com/zkz098/hexo-theme-shokaX",
|
@@ -39,7 +39,7 @@
|
|
39
39
|
"typescript": "^5.0.4",
|
40
40
|
"vue": "^3.2.47",
|
41
41
|
"vuepress": "2.0.0-beta.61",
|
42
|
-
"vuepress-plugin-sitemap2": "2.0.0-beta.
|
42
|
+
"vuepress-plugin-sitemap2": "2.0.0-beta.206"
|
43
43
|
},
|
44
44
|
"dependencies": {
|
45
45
|
"js-yaml": "^4.1.0",
|
@@ -0,0 +1,96 @@
|
|
1
|
+
"use strict";
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
4
|
+
};
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
6
|
+
const node_fs_1 = __importDefault(require("node:fs"));
|
7
|
+
function getContent(post) {
|
8
|
+
return post?.raw ?? post?._content ?? post.content;
|
9
|
+
}
|
10
|
+
let db;
|
11
|
+
function postMessage(path, content, dbPath, startMessage) {
|
12
|
+
if (node_fs_1.default.existsSync('summary.json')) {
|
13
|
+
db = JSON.parse(node_fs_1.default.readFileSync('summary.json'));
|
14
|
+
}
|
15
|
+
else {
|
16
|
+
db = {};
|
17
|
+
}
|
18
|
+
const config = hexo.theme.config.summary;
|
19
|
+
if (config.enable) {
|
20
|
+
if (typeof db?.[path] !== 'undefined' && typeof db?.[path]?.[dbPath] !== 'undefined') {
|
21
|
+
return db[path][dbPath];
|
22
|
+
}
|
23
|
+
else {
|
24
|
+
if (typeof db?.[path] === 'undefined') {
|
25
|
+
db[path] = {};
|
26
|
+
}
|
27
|
+
else {
|
28
|
+
db[path][dbPath] = '';
|
29
|
+
}
|
30
|
+
}
|
31
|
+
if (config.mode === 'openai') {
|
32
|
+
const request = () => {
|
33
|
+
fetch(`${config.openai.remote}/v1/chat/completions`, {
|
34
|
+
method: 'POST',
|
35
|
+
headers: requestHeaders,
|
36
|
+
body: JSON.stringify(requestBody)
|
37
|
+
}).then((response) => {
|
38
|
+
if (!response.ok) {
|
39
|
+
throw Error('ERROR: Failed to get summary from Openai API');
|
40
|
+
}
|
41
|
+
response.json().then((data) => {
|
42
|
+
const summary = data.choices[0].message.content;
|
43
|
+
try {
|
44
|
+
db[path][dbPath] = summary;
|
45
|
+
}
|
46
|
+
catch (e) {
|
47
|
+
db ??= {};
|
48
|
+
db[path] ??= {};
|
49
|
+
db[path][dbPath] ??= '';
|
50
|
+
console.log(db[path]);
|
51
|
+
db[path][dbPath] = summary;
|
52
|
+
}
|
53
|
+
node_fs_1.default.writeFileSync('summary.json', JSON.stringify(db));
|
54
|
+
if (node_fs_1.default.existsSync('requested.lock')) {
|
55
|
+
node_fs_1.default.unlinkSync('requested.lock');
|
56
|
+
}
|
57
|
+
return summary;
|
58
|
+
});
|
59
|
+
});
|
60
|
+
};
|
61
|
+
const checkTime = () => {
|
62
|
+
if (node_fs_1.default.existsSync('request.lock')) {
|
63
|
+
if (node_fs_1.default.existsSync('requested.lock')) {
|
64
|
+
setTimeout(checkTime, 1000 * 10);
|
65
|
+
return;
|
66
|
+
}
|
67
|
+
node_fs_1.default.writeFileSync('requested.lock', '');
|
68
|
+
setTimeout(request, 1000 * 20);
|
69
|
+
node_fs_1.default.unlinkSync('request.lock');
|
70
|
+
}
|
71
|
+
else {
|
72
|
+
node_fs_1.default.writeFileSync('request.lock', '');
|
73
|
+
request();
|
74
|
+
}
|
75
|
+
};
|
76
|
+
const requestHeaders = {
|
77
|
+
'Content-Type': 'application/json',
|
78
|
+
Authorization: `Bearer ${config.openai.apikey}`
|
79
|
+
};
|
80
|
+
const requestBody = {
|
81
|
+
model: 'gpt-3.5-turbo',
|
82
|
+
messages: [{ role: 'user', content: `${startMessage} ${content}` }],
|
83
|
+
temperature: 0.7
|
84
|
+
};
|
85
|
+
checkTime();
|
86
|
+
}
|
87
|
+
else {
|
88
|
+
}
|
89
|
+
}
|
90
|
+
}
|
91
|
+
hexo.extend.helper.register('get_summary', (post) => {
|
92
|
+
return postMessage(post.path, getContent(post), 'summary', '请为下述文章提供一份200字以内的概括,使用中文回答且尽可能简洁: ');
|
93
|
+
});
|
94
|
+
hexo.extend.helper.register('get_introduce', () => {
|
95
|
+
return hexo.theme.config.summary.introduce;
|
96
|
+
});
|
@@ -14,6 +14,13 @@
|
|
14
14
|
overflow-x: auto;
|
15
15
|
}
|
16
16
|
|
17
|
+
ul.special::before {
|
18
|
+
font-family: ic;
|
19
|
+
content: "\e652";
|
20
|
+
font-size: x-large;
|
21
|
+
padding-left: 1em;
|
22
|
+
}
|
23
|
+
|
17
24
|
li {
|
18
25
|
position: relative;
|
19
26
|
cursor: pointer;
|
@@ -27,20 +34,20 @@
|
|
27
34
|
position: absolute;
|
28
35
|
left: 50%;
|
29
36
|
right: 50%;
|
30
|
-
top:
|
37
|
+
top: 2em;
|
31
38
|
bottom: 0;
|
32
39
|
transition: all .2s ease-in-out;
|
33
40
|
width: auto;
|
34
41
|
height: auto;
|
35
42
|
background: none;
|
36
|
-
border-radius:
|
43
|
+
border-radius: 10px;
|
37
44
|
border-bottom: .125rem solid transparent;
|
38
45
|
}
|
39
46
|
|
40
47
|
&.active::before {
|
41
48
|
left: 0;
|
42
49
|
right: 0;
|
43
|
-
border-bottom-color: var(--
|
50
|
+
border-bottom-color: var(--primary-color);
|
44
51
|
}
|
45
52
|
}
|
46
53
|
}
|
@@ -61,7 +61,7 @@ Object.assign(HTMLElement.prototype, {
|
|
61
61
|
}
|
62
62
|
return this.getBoundingClientRect().width;
|
63
63
|
},
|
64
|
-
|
64
|
+
getTop: function () {
|
65
65
|
return this.getBoundingClientRect().top;
|
66
66
|
},
|
67
67
|
left: function () {
|
@@ -296,7 +296,7 @@ const pageScroll = function (target, offset, complete) {
|
|
296
296
|
targets: typeof offset === 'number' ? target.parentNode : document.scrollingElement,
|
297
297
|
duration: 500,
|
298
298
|
easing: 'easeInOutQuad',
|
299
|
-
scrollTop: offset || (typeof target === 'number' ? target : (target ? target.
|
299
|
+
scrollTop: offset || (typeof target === 'number' ? target : (target ? target.getTop() + document.documentElement.scrollTop - siteNavHeight : 0)),
|
300
300
|
complete: function () {
|
301
301
|
complete && complete();
|
302
302
|
}
|
package/source/js/_app/page.js
CHANGED
@@ -72,7 +72,7 @@ const postFancybox = function (p) {
|
|
72
72
|
$dom.each(p + ' p.gallery', function (element) {
|
73
73
|
const box = document.createElement('div');
|
74
74
|
box.className = 'gallery';
|
75
|
-
box.attr('data-height', element.attr('data-height') || 220);
|
75
|
+
box.attr('data-height', String(element.attr('data-height') || 220));
|
76
76
|
box.innerHTML = element.innerHTML.replace(/<br>/g, '');
|
77
77
|
element.parentNode.insertBefore(box, element);
|
78
78
|
element.remove();
|
@@ -405,7 +405,7 @@ const tabFormat = function () {
|
|
405
405
|
target.addClass('active');
|
406
406
|
});
|
407
407
|
box.appendChild(element);
|
408
|
-
element.attr('data-ready', true);
|
408
|
+
element.attr('data-ready', String(true));
|
409
409
|
});
|
410
410
|
};
|
411
411
|
const loadComments = function () {
|
package/source/js/_app/player.js
CHANGED
@@ -425,7 +425,7 @@ const mediaPlayer = function (t, config) {
|
|
425
425
|
t.player = {
|
426
426
|
_id: utils.random(999999),
|
427
427
|
group: true,
|
428
|
-
load:
|
428
|
+
load: (newList) => {
|
429
429
|
let d = '';
|
430
430
|
if (newList && newList.length > 0) {
|
431
431
|
if (this.options.rawList !== newList) {
|
@@ -483,7 +483,7 @@ const mediaPlayer = function (t, config) {
|
|
483
483
|
}
|
484
484
|
});
|
485
485
|
},
|
486
|
-
mode:
|
486
|
+
mode: () => {
|
487
487
|
const total = playlist.data.length;
|
488
488
|
if (!total || playlist.errnum === total) {
|
489
489
|
return;
|
@@ -496,7 +496,7 @@ const mediaPlayer = function (t, config) {
|
|
496
496
|
}
|
497
497
|
playlist.index = index;
|
498
498
|
};
|
499
|
-
const random =
|
499
|
+
const random = () => {
|
500
500
|
const p = utils.random(total);
|
501
501
|
if (playlist.index !== p) {
|
502
502
|
playlist.index = p;
|
@@ -523,7 +523,7 @@ const mediaPlayer = function (t, config) {
|
|
523
523
|
}
|
524
524
|
this.init();
|
525
525
|
},
|
526
|
-
switch:
|
526
|
+
switch: (index) => {
|
527
527
|
if (typeof index === 'number' &&
|
528
528
|
index !== playlist.index &&
|
529
529
|
playlist.current() &&
|
@@ -555,7 +555,7 @@ const mediaPlayer = function (t, config) {
|
|
555
555
|
this.play();
|
556
556
|
}
|
557
557
|
},
|
558
|
-
play:
|
558
|
+
play: () => {
|
559
559
|
NOWPLAYING && NOWPLAYING.player.pause();
|
560
560
|
if (playlist.current().error) {
|
561
561
|
this.mode();
|