Sharkey/src/client/app/desktop/views/pages/admin/admin.chart.vue

404 lines
11 KiB
Vue
Raw Normal View History

2018-08-23 23:37:19 +03:00
<template>
<div class="card gkgckalzgidaygcxnugepioremxvxvpt">
<header>
<b>%i18n:@title%:</b>
<select v-model="chartType">
2018-08-24 00:41:53 +03:00
<optgroup label="%i18n:@users%">
2018-08-24 08:55:58 +03:00
<option value="users">%i18n:@charts.users%</option>
<option value="users-total">%i18n:@charts.users-total%</option>
2018-08-24 00:41:53 +03:00
</optgroup>
<optgroup label="%i18n:@notes%">
2018-08-24 08:55:58 +03:00
<option value="notes">%i18n:@charts.notes%</option>
<option value="local-notes">%i18n:@charts.local-notes%</option>
<option value="remote-notes">%i18n:@charts.remote-notes%</option>
<option value="notes-total">%i18n:@charts.notes-total%</option>
2018-08-24 00:41:53 +03:00
</optgroup>
<optgroup label="%i18n:@drive%">
2018-08-24 08:55:58 +03:00
<option value="drive-files">%i18n:@charts.drive-files%</option>
<option value="drive-files-total">%i18n:@charts.drive-files-total%</option>
<option value="drive">%i18n:@charts.drive%</option>
<option value="drive-total">%i18n:@charts.drive-total%</option>
2018-08-24 00:41:53 +03:00
</optgroup>
2018-08-23 23:37:19 +03:00
</select>
<div>
2018-08-24 08:55:58 +03:00
<span @click="span = 'day'" :class="{ active: span == 'day' }">%i18n:@per-day%</span> | <span @click="span = 'hour'" :class="{ active: span == 'hour' }">%i18n:@per-hour%</span>
2018-08-23 23:37:19 +03:00
</div>
</header>
2018-08-24 01:17:17 +03:00
<div>
<x-chart v-if="chart" :data="data[0]" :opts="data[1]"/>
</div>
2018-08-23 23:37:19 +03:00
</div>
</template>
<script lang="ts">
import Vue from 'vue';
import XChart from './admin.chart.chart.ts';
export default Vue.extend({
components: {
XChart
},
data() {
return {
chart: null,
2018-08-24 08:55:58 +03:00
chartType: 'notes',
2018-08-23 23:37:19 +03:00
span: 'hour'
};
},
computed: {
data(): any {
if (this.chart == null) return null;
switch (this.chartType) {
2018-08-24 08:55:58 +03:00
case 'users': return this.usersChart(false);
case 'users-total': return this.usersChart(true);
case 'notes': return this.notesChart('combined');
case 'local-notes': return this.notesChart('local');
case 'remote-notes': return this.notesChart('remote');
case 'notes-total': return this.notesTotalChart();
case 'drive': return this.driveChart(false);
case 'drive-total': return this.driveChart(true);
case 'drive-files': return this.driveFilesChart(false);
case 'drive-files-total': return this.driveFilesChart(true);
2018-08-23 23:37:19 +03:00
}
},
stats(): any[] {
return (
this.span == 'day' ? this.chart.perDay :
this.span == 'hour' ? this.chart.perHour :
null
);
}
},
created() {
(this as any).api('chart').then(chart => {
this.chart = chart;
});
},
2018-08-23 23:37:19 +03:00
methods: {
2018-08-24 08:55:58 +03:00
notesChart(type: string): any {
2018-08-23 23:37:19 +03:00
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
2018-08-24 08:55:58 +03:00
normal: type == 'local' ? x.notes.local.diffs.normal : type == 'remote' ? x.notes.remote.diffs.normal : x.notes.local.diffs.normal + x.notes.remote.diffs.normal,
reply: type == 'local' ? x.notes.local.diffs.reply : type == 'remote' ? x.notes.remote.diffs.reply : x.notes.local.diffs.reply + x.notes.remote.diffs.reply,
renote: type == 'local' ? x.notes.local.diffs.renote : type == 'remote' ? x.notes.remote.diffs.renote : x.notes.local.diffs.renote + x.notes.remote.diffs.renote,
all: type == 'local' ? x.notes.local.diff : type == 'remote' ? x.notes.remote.diff : x.notes.local.diff + x.notes.remote.diff
2018-08-23 23:37:19 +03:00
}));
return [{
datasets: [{
2018-08-24 00:41:53 +03:00
label: 'All',
fill: false,
borderColor: '#555',
borderWidth: 2,
2018-08-24 03:39:16 +03:00
borderDash: [4, 4],
2018-08-24 00:41:53 +03:00
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.all }))
}, {
2018-08-24 08:55:58 +03:00
label: 'Renotes',
2018-08-24 02:56:57 +03:00
fill: true,
2018-08-24 08:55:58 +03:00
backgroundColor: 'rgba(161, 222, 65, 0.1)',
borderColor: '#a1de41',
2018-08-23 23:37:19 +03:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-24 08:55:58 +03:00
data: data.map(x => ({ t: x.date, y: x.renote }))
2018-08-23 23:37:19 +03:00
}, {
label: 'Replies',
2018-08-24 02:56:57 +03:00
fill: true,
backgroundColor: 'rgba(247, 121, 108, 0.1)',
2018-08-23 23:37:19 +03:00
borderColor: '#f7796c',
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.reply }))
}, {
2018-08-24 08:55:58 +03:00
label: 'Normal',
2018-08-24 02:56:57 +03:00
fill: true,
2018-08-24 08:55:58 +03:00
backgroundColor: 'rgba(65, 221, 222, 0.1)',
borderColor: '#41ddde',
2018-08-23 23:37:19 +03:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-24 08:55:58 +03:00
data: data.map(x => ({ t: x.date, y: x.normal }))
2018-08-23 23:37:19 +03:00
}]
2018-08-24 08:55:58 +03:00
}, {
scales: {
yAxes: [{
ticks: {
callback: value => {
return Vue.filter('number')(value);
}
}
}]
},
tooltips: {
callbacks: {
label: (tooltipItem, data) => {
const label = data.datasets[tooltipItem.datasetIndex].label || '';
return `${label}: ${Vue.filter('number')(tooltipItem.yLabel)}`;
}
}
}
2018-08-23 23:37:19 +03:00
}];
},
2018-08-24 00:41:53 +03:00
2018-08-24 08:55:58 +03:00
notesTotalChart(): any {
2018-08-23 23:37:19 +03:00
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
2018-08-24 08:55:58 +03:00
localCount: x.notes.local.total,
remoteCount: x.notes.remote.total
2018-08-23 23:37:19 +03:00
}));
return [{
datasets: [{
2018-08-24 08:55:58 +03:00
label: 'Notes',
fill: false,
borderColor: '#555',
borderWidth: 2,
borderDash: [4, 4],
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.remoteCount + x.localCount }))
}, {
label: 'Remote Notes',
fill: true,
backgroundColor: 'rgba(65, 221, 222, 0.1)',
borderColor: '#41ddde',
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.remoteCount }))
}, {
label: 'Local Notes',
2018-08-24 01:17:17 +03:00
fill: true,
2018-08-24 02:56:57 +03:00
backgroundColor: 'rgba(246, 88, 79, 0.1)',
2018-08-23 23:37:19 +03:00
borderColor: '#f6584f',
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-24 08:55:58 +03:00
data: data.map(x => ({ t: x.date, y: x.localCount }))
2018-08-23 23:37:19 +03:00
}]
2018-08-24 08:55:58 +03:00
}, {
scales: {
yAxes: [{
ticks: {
callback: value => {
return Vue.filter('number')(value);
}
}
}]
},
tooltips: {
callbacks: {
label: (tooltipItem, data) => {
const label = data.datasets[tooltipItem.datasetIndex].label || '';
return `${label}: ${Vue.filter('number')(tooltipItem.yLabel)}`;
}
}
}
2018-08-23 23:37:19 +03:00
}];
},
2018-08-24 00:41:53 +03:00
2018-08-24 08:55:58 +03:00
usersChart(total: boolean): any {
2018-08-23 23:37:19 +03:00
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
2018-08-24 08:55:58 +03:00
localCount: total ? x.users.local.total : x.users.local.diff,
remoteCount: total ? x.users.remote.total : x.users.remote.diff
2018-08-23 23:37:19 +03:00
}));
return [{
datasets: [{
2018-08-24 08:55:58 +03:00
label: 'Users',
fill: false,
borderColor: '#555',
borderWidth: 2,
borderDash: [4, 4],
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.remoteCount + x.localCount }))
}, {
label: 'Remote Users',
fill: true,
backgroundColor: 'rgba(65, 221, 222, 0.1)',
borderColor: '#41ddde',
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.remoteCount }))
}, {
label: 'Local Users',
2018-08-24 01:17:17 +03:00
fill: true,
2018-08-24 02:56:57 +03:00
backgroundColor: 'rgba(246, 88, 79, 0.1)',
2018-08-23 23:37:19 +03:00
borderColor: '#f6584f',
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-24 08:55:58 +03:00
data: data.map(x => ({ t: x.date, y: x.localCount }))
2018-08-23 23:37:19 +03:00
}]
2018-08-24 08:55:58 +03:00
}, {
scales: {
yAxes: [{
ticks: {
callback: value => {
return Vue.filter('number')(value);
}
}
}]
},
tooltips: {
callbacks: {
label: (tooltipItem, data) => {
const label = data.datasets[tooltipItem.datasetIndex].label || '';
return `${label}: ${Vue.filter('number')(tooltipItem.yLabel)}`;
}
}
}
2018-08-23 23:37:19 +03:00
}];
},
2018-08-24 00:41:53 +03:00
2018-08-24 08:55:58 +03:00
driveChart(total: boolean): any {
2018-08-23 23:37:19 +03:00
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
2018-08-24 08:55:58 +03:00
localSize: total ? x.drive.local.totalSize : x.drive.local.diffSize,
remoteSize: total ? x.drive.remote.totalSize : x.drive.remote.diffSize
2018-08-23 23:37:19 +03:00
}));
return [{
datasets: [{
2018-08-24 08:55:58 +03:00
label: 'Drive Usage',
fill: false,
borderColor: '#555',
borderWidth: 2,
borderDash: [4, 4],
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.remoteSize + x.localSize }))
}, {
label: 'Remote Drive Usage',
fill: true,
backgroundColor: 'rgba(65, 221, 222, 0.1)',
borderColor: '#41ddde',
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.remoteSize }))
}, {
label: 'Local Drive Usage',
2018-08-24 01:17:17 +03:00
fill: true,
2018-08-24 02:56:57 +03:00
backgroundColor: 'rgba(246, 88, 79, 0.1)',
2018-08-23 23:37:19 +03:00
borderColor: '#f6584f',
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-24 08:55:58 +03:00
data: data.map(x => ({ t: x.date, y: x.localSize }))
2018-08-23 23:37:19 +03:00
}]
}, {
scales: {
yAxes: [{
ticks: {
2018-08-24 02:56:57 +03:00
callback: value => {
2018-08-23 23:37:19 +03:00
return Vue.filter('bytes')(value);
}
}
}]
2018-08-24 02:56:57 +03:00
},
tooltips: {
callbacks: {
2018-08-24 08:55:58 +03:00
label: (tooltipItem, data) => {
const label = data.datasets[tooltipItem.datasetIndex].label || '';
return `${label}: ${Vue.filter('bytes')(tooltipItem.yLabel)}`;
2018-08-24 02:56:57 +03:00
}
}
2018-08-23 23:37:19 +03:00
}
}];
2018-08-24 00:41:53 +03:00
},
2018-08-24 08:55:58 +03:00
driveFilesChart(total: boolean): any {
2018-08-24 00:41:53 +03:00
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
2018-08-24 08:55:58 +03:00
localCount: total ? x.drive.local.totalCount : x.drive.local.diffCount,
remoteCount: total ? x.drive.remote.totalCount : x.drive.remote.diffCount
2018-08-24 00:41:53 +03:00
}));
return [{
datasets: [{
2018-08-24 08:55:58 +03:00
label: 'Drive Files',
fill: false,
borderColor: '#555',
borderWidth: 2,
borderDash: [4, 4],
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.remoteCount + x.localCount }))
}, {
label: 'Remote Drive Files',
fill: true,
backgroundColor: 'rgba(65, 221, 222, 0.1)',
borderColor: '#41ddde',
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.remoteCount }))
}, {
label: 'Local Drive Files',
2018-08-24 03:39:16 +03:00
fill: true,
backgroundColor: 'rgba(246, 88, 79, 0.1)',
2018-08-24 00:41:53 +03:00
borderColor: '#f6584f',
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-24 08:55:58 +03:00
data: data.map(x => ({ t: x.date, y: x.localCount }))
2018-08-24 00:41:53 +03:00
}]
2018-08-24 08:55:58 +03:00
}, {
scales: {
yAxes: [{
ticks: {
callback: value => {
return Vue.filter('number')(value);
}
}
}]
},
tooltips: {
callbacks: {
label: (tooltipItem, data) => {
const label = data.datasets[tooltipItem.datasetIndex].label || '';
return `${label}: ${Vue.filter('number')(tooltipItem.yLabel)}`;
}
}
}
2018-08-24 00:41:53 +03:00
}];
2018-08-24 08:55:58 +03:00
}
2018-08-23 23:37:19 +03:00
}
});
</script>
<style lang="stylus" scoped>
@import '~const.styl'
.gkgckalzgidaygcxnugepioremxvxvpt
2018-08-24 02:56:57 +03:00
*
user-select none
2018-08-23 23:37:19 +03:00
> header
display flex
> b
margin-right 8px
> *:last-child
margin-left auto
2018-08-24 08:55:58 +03:00
*
&:not(.active)
color $theme-color
cursor pointer
2018-08-24 01:17:17 +03:00
> div
> *
display block
height 300px
2018-08-23 23:37:19 +03:00
</style>