hexo文章标签统计实现

近日浏览公司大神闪烁之狐的博客,发现他的关于页有着一个很漂亮的文章统计图,于是想给自己的博客也安排上。

效果预览

先附上我的效果图:https://jiangliuhong.top/statistics/

统计图

再附上大佬的效果图:https://blinkfox.github.io/about/

统计图2

实现思路

统计数据来源

  • 文章发布统计:利用site.posts获取文章信息
1
2
3
4
5
6
site.posts.forEach(function (post) {
var month = post.date.format('YYYY-MM');
if (monthMap.has(month)) {
monthMap.set(month, monthMap.get(month) + 1);
}
});
  • 文章分类统计:利用site.categories获取分类信息
1
2
3
site.categories.map(function(category) {
categoryArr.push({'name': category.name, 'value': category.length})
});
  • 标签统计:利用site.tags获取标签信息
1
2
3
site.tags.map(function(tag) {
tagArr.push({'name': tag.name, 'value': tag.length});
});

统计呈现方式

使用echarts呈现,具体参见echarts文档

新建layout

layout中新建statistics布局,然后在source目录新建statistics/index.md文件,在文件中指定layout

1
2
3
4
5
6
---
title: 我的统计
date: 20120-04-22 12:35:59
layout: statistics
---

具体实现

下面的实现访问为基于hexo-fluid主题实现

首先在layout目录下新建一个statistics.ejs文件,文件内容为:

1
2
3
4
5
6
7
8
9
<%
page.layout = "statistics"
page.banner_img = theme.statistics.banner_img
page.banner_img_height = theme.statistics.banner_img_height
page.banner_mask_alpha = theme.statistics.banner_mask_alpha
%>
<div class="mt-5 markdown-body">
<%- partial('_widget/mystatistics') %>
</div>

_widget目录下创建一个mystatistics.ejs,文件内容为:

需要注意的是,在下面源码中有这样一行src="<%- url_for(theme.my.libs.js.echarts) %>",这是指定了一个echarts.js文件地址,你可以前往官网https://echarts.apache.org/zh/download.html下载(同时也可以访问我的地址进行下载https://jiangliuhong.top/js/echarts/echarts.min.js),然后把这个修改成你的地址即可。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
<style type="text/css">
#posts-chart,
#categories-chart,
#tags-chart {
width: 100%;
height: 300px;
margin: 0.5rem auto;
padding: 0.5rem;
}
</style>

<div id="postCharts" class="post-charts">
<div class="title center-align center-align-title" data-aos="zoom-in-up">
文章统计图
</div>
<div class="row">
<div class="chart col s12 m6 l4" data-aos="zoom-in-up">
<div id="posts-chart"></div>
</div>
<div class="chart col s12 m6 l4" data-aos="zoom-in-up">
<div id="categories-chart"></div>
</div>
</div>
<div class="row">
<div class="chart col s12 m6 l4" data-aos="zoom-in-up">
<div id="tags-chart"></div>
</div>
</div>
</div>

<script type="text/javascript" src="<%- url_for(theme.my.libs.js.echarts) %>"></script>
<script>
let postsChart = echarts.init(document.getElementById('posts-chart'));
let categoriesChart = echarts.init(document.getElementById('categories-chart'));
let tagsChart = echarts.init(document.getElementById('tags-chart'));

<%
/* calculate postsOption data. */
var startDate = moment().subtract(1, 'years').startOf('month');
var endDate = moment().endOf('month');

var monthMap = new Map();
var dayTime = 3600 * 24 * 1000;
for (var time = startDate; time <= endDate; time += dayTime) {
var month = moment(time).format('YYYY-MM');
if (!monthMap.has(month)) {
monthMap.set(month, 0);
}
}

// post and count map.
site.posts.forEach(function (post) {
var month = post.date.format('YYYY-MM');
if (monthMap.has(month)) {
monthMap.set(month, monthMap.get(month) + 1);
}
});

// xAxis data and yAxis data.
var monthArr = JSON.stringify([...monthMap.keys()]);
var monthValueArr = JSON.stringify([...monthMap.values()]);
%>

let postsOption = {
title: {
text: '文章发布统计图',
top: -5,
x: 'center'
},
tooltip: {
trigger: 'axis'
},
xAxis: {
type: 'category',
data: <%- monthArr %>
},
yAxis: {
type: 'value',
},
series: [
{
name: '文章篇数',
type: 'line',
color: ['#6772e5'],
data: <%- monthValueArr %>,
markPoint: {
symbolSize: 45,
color: ['#fa755a','#3ecf8e','#82d3f4'],
data: [{
type: 'max',
itemStyle: {color: ['#3ecf8e']},
name: '最大值'
}, {
type: 'min',
itemStyle: {color: ['#fa755a']},
name: '最小值'
}]
},
markLine: {
itemStyle: {color: ['#ab47bc']},
data: [
{type: 'average', name: '<%- __("average") %>'}
]
}
}
]
};

<%
/* calculate categoriesOption data. */
var categoryArr = [];
site.categories.map(function(category) {
categoryArr.push({'name': category.name, 'value': category.length})
});

var categoryArrJson = JSON.stringify(categoryArr);
%>

let categoriesOption = {
title: {
text: '文章分类统计图',
top: -4,
x: 'center'
},
tooltip: {
trigger: 'item',
formatter: "{a} <br/>{b} : {c} ({d}%)"
},
series: [
{
name: '分类',
type: 'pie',
radius: '50%',
color: ['#6772e5', '#ff9e0f', '#fa755a', '#3ecf8e', '#82d3f4', '#ab47bc', '#525f7f', '#f51c47', '#26A69A'],
data: <%- categoryArrJson %>,
itemStyle: {
emphasis: {
shadowBlur: 10,
shadowOffsetX: 0,
shadowColor: 'rgba(0, 0, 0, 0.5)'
}
}
}
]
};


<%
/* calculate tagsOption data. */
// get all tags name and count,then order by length desc.
var tagArr = [];
site.tags.map(function(tag) {
tagArr.push({'name': tag.name, 'value': tag.length});
});
tagArr.sort((a, b) => {return b.value - a.value});

// get Top 10 tags name and count.
var tagNameArr = [];
var tagCountArr = [];
for (var i = 0, len = Math.min(tagArr.length, 10); i < len; i++) {
tagNameArr.push(tagArr[i].name);
tagCountArr.push(tagArr[i].value);
}

var tagNameArrJson = JSON.stringify(tagNameArr);
var tagCountArrJson = JSON.stringify(tagCountArr);

%>

let tagsOption = {
title: {
text: 'TOP10 标签统计图',
top: -5,
x: 'center'
},
tooltip: {},
xAxis: [
{
type: 'category',
axisLabel:{
interval:0,
rotate:40
},
data: <%- tagNameArrJson %>
}
],
yAxis: [
{
type: 'value'
}
],
series: [
{
name: 'TOP10标签统计',
type: 'bar',
color: ['#82d3f4'],
barWidth : 18,
data: <%- tagCountArrJson %>,
markPoint: {
symbolSize: 45,
data: [{
type: 'max',
itemStyle: {color: ['#3ecf8e']},
name: '最大值'
}, {
type: 'min',
itemStyle: {color: ['#fa755a']},
name: '最小值'
}],
},
markLine: {
itemStyle: {color: ['#ab47bc']},
data: [
{type: 'average', name: '平均值'}
]
}
}
]
};

// render the charts
postsChart.setOption(postsOption);
categoriesChart.setOption(categoriesOption);
tagsChart.setOption(tagsOption);
</script>