1
1
mirror of https://github.com/kokonect-link/cherrypick synced 2024-12-24 19:48:21 +09:00
cherrypick/src/client/app/desktop/views/components/charts.vue

731 lines
19 KiB
Vue
Raw Normal View History

2018-08-24 05:37:19 +09:00
<template>
<div class="gkgckalzgidaygcxnugepioremxvxvpt">
2018-08-24 05:37:19 +09:00
<header>
<b>%i18n:@title%:</b>
<select v-model="chartType">
2018-08-24 06:41:53 +09:00
<optgroup label="%i18n:@users%">
2018-08-24 14:55:58 +09:00
<option value="users">%i18n:@charts.users%</option>
<option value="users-total">%i18n:@charts.users-total%</option>
2018-08-24 06:41:53 +09:00
</optgroup>
<optgroup label="%i18n:@notes%">
2018-08-24 14:55:58 +09: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 06:41:53 +09:00
</optgroup>
<optgroup label="%i18n:@drive%">
2018-08-24 14:55:58 +09: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 06:41:53 +09:00
</optgroup>
2018-09-15 05:40:58 +09:00
<optgroup label="%i18n:@network%">
<option value="network-requests">%i18n:@charts.network-requests%</option>
<option value="network-time">%i18n:@charts.network-time%</option>
<option value="network-usage">%i18n:@charts.network-usage%</option>
</optgroup>
2018-08-24 05:37:19 +09:00
</select>
<div>
2018-08-24 14:55:58 +09: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-24 05:37:19 +09:00
</div>
</header>
2018-08-24 07:17:17 +09:00
<div>
<x-chart v-if="chart" :data="data[0]" :opts="data[1]"/>
</div>
2018-08-24 05:37:19 +09:00
</div>
</template>
<script lang="ts">
import Vue from 'vue';
import XChart from './charts.chart.ts';
2018-08-24 05:37:19 +09:00
2018-08-25 15:25:16 +09:00
const colors = {
local: 'rgb(246, 88, 79)',
remote: 'rgb(65, 221, 222)',
localPlus: 'rgb(52, 178, 118)',
remotePlus: 'rgb(158, 255, 209)',
localMinus: 'rgb(255, 97, 74)',
2018-09-15 05:40:58 +09:00
remoteMinus: 'rgb(255, 149, 134)',
incoming: 'rgb(52, 178, 118)',
outgoing: 'rgb(255, 97, 74)',
2018-08-25 15:25:16 +09:00
};
const rgba = (color: string): string => {
return color.replace('rgb', 'rgba').replace(')', ', 0.1)');
};
2018-08-24 05:37:19 +09:00
export default Vue.extend({
components: {
XChart
},
2018-08-25 15:25:16 +09:00
2018-08-24 05:37:19 +09:00
data() {
return {
chart: null,
2018-08-24 14:55:58 +09:00
chartType: 'notes',
2018-08-24 05:37:19 +09:00
span: 'hour'
};
},
2018-08-25 15:25:16 +09:00
2018-08-24 05:37:19 +09:00
computed: {
data(): any {
if (this.chart == null) return null;
switch (this.chartType) {
2018-08-24 14:55:58 +09: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();
2018-08-25 08:35:41 +09:00
case 'drive': return this.driveChart();
case 'drive-total': return this.driveTotalChart();
case 'drive-files': return this.driveFilesChart();
case 'drive-files-total': return this.driveFilesTotalChart();
2018-09-15 05:40:58 +09:00
case 'network-requests': return this.networkRequestsChart();
case 'network-time': return this.networkTimeChart();
case 'network-usage': return this.networkUsageChart();
2018-08-24 05:37:19 +09:00
}
},
2018-08-25 15:25:16 +09:00
2018-08-24 05:37:19 +09:00
stats(): any[] {
2018-10-21 07:10:35 +09:00
const now = new Date();
const y = now.getFullYear();
const m = now.getMonth();
const d = now.getDate();
const h = now.getHours();
const stats =
2018-08-24 05:37:19 +09:00
this.span == 'day' ? this.chart.perDay :
this.span == 'hour' ? this.chart.perHour :
2018-10-21 07:10:35 +09:00
null;
stats.forEach((s, i) => {
s.date =
this.span == 'day' ? new Date(y, m, d - i) :
this.span == 'hour' ? new Date(y, m, d, h - i) :
null;
});
return stats;
2018-08-24 05:37:19 +09:00
}
},
2018-08-25 15:25:16 +09:00
2018-10-21 09:20:11 +09:00
async created() {
const limit = 35;
const [perHour, perDay] = await Promise.all([Promise.all([
(this as any).api('charts/users', { limit: limit, span: 'hour' }),
(this as any).api('charts/notes', { limit: limit, span: 'hour' }),
(this as any).api('charts/drive', { limit: limit, span: 'hour' }),
(this as any).api('charts/network', { limit: limit, span: 'hour' })
]), Promise.all([
(this as any).api('charts/users', { limit: limit, span: 'day' }),
(this as any).api('charts/notes', { limit: limit, span: 'day' }),
(this as any).api('charts/drive', { limit: limit, span: 'day' }),
(this as any).api('charts/network', { limit: limit, span: 'day' })
])]);
const chart = {
perHour: [],
perDay: []
};
for (let i = 0; i < limit; i++) {
chart.perHour.push({
users: perHour[0][i],
notes: perHour[1][i],
drive: perHour[2][i],
network: perHour[3][i]
});
chart.perDay.push({
users: perDay[0][i],
notes: perDay[1][i],
drive: perDay[2][i],
network: perDay[3][i]
});
}
this.chart = chart;
},
2018-08-25 15:25:16 +09:00
2018-08-24 05:37:19 +09:00
methods: {
2018-08-24 14:55:58 +09:00
notesChart(type: string): any {
2018-08-24 05:37:19 +09:00
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
2018-08-24 14:55:58 +09: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,
2018-08-25 08:35:41 +09:00
all: type == 'local' ? (x.notes.local.inc + -x.notes.local.dec) : type == 'remote' ? (x.notes.remote.inc + -x.notes.remote.dec) : (x.notes.local.inc + -x.notes.local.dec) + (x.notes.remote.inc + -x.notes.remote.dec)
2018-08-24 05:37:19 +09:00
}));
return [{
datasets: [{
2018-08-24 06:41:53 +09:00
label: 'All',
fill: false,
borderColor: '#555',
borderWidth: 2,
2018-08-24 09:39:16 +09:00
borderDash: [4, 4],
2018-08-24 06:41:53 +09:00
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.all }))
}, {
2018-08-24 14:55:58 +09:00
label: 'Renotes',
2018-08-24 08:56:57 +09:00
fill: true,
2018-08-24 14:55:58 +09:00
backgroundColor: 'rgba(161, 222, 65, 0.1)',
borderColor: '#a1de41',
2018-08-24 05:37:19 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-24 14:55:58 +09:00
data: data.map(x => ({ t: x.date, y: x.renote }))
2018-08-24 05:37:19 +09:00
}, {
label: 'Replies',
2018-08-24 08:56:57 +09:00
fill: true,
backgroundColor: 'rgba(247, 121, 108, 0.1)',
2018-08-24 05:37:19 +09:00
borderColor: '#f7796c',
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.reply }))
}, {
2018-08-24 14:55:58 +09:00
label: 'Normal',
2018-08-24 08:56:57 +09:00
fill: true,
2018-08-24 14:55:58 +09:00
backgroundColor: 'rgba(65, 221, 222, 0.1)',
borderColor: '#41ddde',
2018-08-24 05:37:19 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-24 14:55:58 +09:00
data: data.map(x => ({ t: x.date, y: x.normal }))
2018-08-24 05:37:19 +09:00
}]
2018-08-24 14:55:58 +09: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 05:37:19 +09:00
}];
},
2018-08-24 06:41:53 +09:00
2018-08-24 14:55:58 +09:00
notesTotalChart(): any {
2018-08-24 05:37:19 +09:00
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
2018-08-24 14:55:58 +09:00
localCount: x.notes.local.total,
remoteCount: x.notes.remote.total
2018-08-24 05:37:19 +09:00
}));
return [{
datasets: [{
2018-08-25 08:35:41 +09:00
label: 'Combined',
2018-08-24 14:55:58 +09:00
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 }))
}, {
2018-08-25 15:25:16 +09:00
label: 'Local',
2018-08-24 14:55:58 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.local),
borderColor: colors.local,
2018-08-24 14:55:58 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.localCount }))
2018-08-24 14:55:58 +09:00
}, {
2018-08-25 15:25:16 +09:00
label: 'Remote',
2018-08-24 07:17:17 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.remote),
borderColor: colors.remote,
2018-08-24 05:37:19 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.remoteCount }))
2018-08-24 05:37:19 +09:00
}]
2018-08-24 14:55:58 +09: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 05:37:19 +09:00
}];
},
2018-08-24 06:41:53 +09:00
2018-08-24 14:55:58 +09:00
usersChart(total: boolean): any {
2018-08-24 05:37:19 +09:00
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
2018-08-25 08:35:41 +09:00
localCount: total ? x.users.local.total : (x.users.local.inc + -x.users.local.dec),
remoteCount: total ? x.users.remote.total : (x.users.remote.inc + -x.users.remote.dec)
2018-08-24 05:37:19 +09:00
}));
return [{
datasets: [{
2018-08-25 08:35:41 +09:00
label: 'Combined',
2018-08-24 14:55:58 +09:00
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 }))
}, {
2018-08-25 15:25:16 +09:00
label: 'Local',
2018-08-24 14:55:58 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.local),
borderColor: colors.local,
2018-08-24 14:55:58 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.localCount }))
2018-08-24 14:55:58 +09:00
}, {
2018-08-25 15:25:16 +09:00
label: 'Remote',
2018-08-24 07:17:17 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.remote),
borderColor: colors.remote,
2018-08-24 05:37:19 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.remoteCount }))
2018-08-24 05:37:19 +09:00
}]
2018-08-24 14:55:58 +09: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 05:37:19 +09:00
}];
},
2018-08-24 06:41:53 +09:00
2018-08-25 08:35:41 +09:00
driveChart(): any {
2018-08-24 05:37:19 +09:00
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
2018-08-25 08:35:41 +09:00
localInc: x.drive.local.incSize,
localDec: -x.drive.local.decSize,
remoteInc: x.drive.remote.incSize,
remoteDec: -x.drive.remote.decSize,
2018-08-24 05:37:19 +09:00
}));
return [{
datasets: [{
2018-08-25 08:35:41 +09:00
label: 'All',
fill: false,
borderColor: '#555',
borderWidth: 2,
borderDash: [4, 4],
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.localInc + x.localDec + x.remoteInc + x.remoteDec }))
}, {
2018-08-25 15:25:16 +09:00
label: 'Local +',
2018-08-25 08:35:41 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.localPlus),
borderColor: colors.localPlus,
2018-08-25 08:35:41 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.localInc }))
2018-08-25 08:35:41 +09:00
}, {
2018-08-25 15:25:16 +09:00
label: 'Local -',
2018-08-25 08:35:41 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.localMinus),
borderColor: colors.localMinus,
2018-08-25 08:35:41 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.localDec }))
2018-08-25 08:35:41 +09:00
}, {
2018-08-25 15:25:16 +09:00
label: 'Remote +',
2018-08-25 08:35:41 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.remotePlus),
borderColor: colors.remotePlus,
2018-08-25 08:35:41 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.remoteInc }))
2018-08-25 08:35:41 +09:00
}, {
2018-08-25 15:25:16 +09:00
label: 'Remote -',
2018-08-25 08:35:41 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.remoteMinus),
borderColor: colors.remoteMinus,
2018-08-25 08:35:41 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.remoteDec }))
2018-08-25 08:35:41 +09:00
}]
}, {
scales: {
yAxes: [{
ticks: {
callback: value => {
2018-08-25 12:14:36 +09:00
return Vue.filter('bytes')(value, 1);
2018-08-25 08:35:41 +09:00
}
}
}]
},
tooltips: {
callbacks: {
label: (tooltipItem, data) => {
const label = data.datasets[tooltipItem.datasetIndex].label || '';
2018-08-25 12:14:36 +09:00
return `${label}: ${Vue.filter('bytes')(tooltipItem.yLabel, 1)}`;
2018-08-25 08:35:41 +09:00
}
}
}
}];
},
driveTotalChart(): any {
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
localSize: x.drive.local.totalSize,
remoteSize: x.drive.remote.totalSize
}));
return [{
datasets: [{
label: 'Combined',
2018-08-24 14:55:58 +09:00
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 }))
}, {
2018-08-25 15:25:16 +09:00
label: 'Local',
2018-08-24 14:55:58 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.local),
borderColor: colors.local,
2018-08-24 14:55:58 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.localSize }))
2018-08-24 14:55:58 +09:00
}, {
2018-08-25 15:25:16 +09:00
label: 'Remote',
2018-08-24 07:17:17 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.remote),
borderColor: colors.remote,
2018-08-24 05:37:19 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.remoteSize }))
2018-08-24 05:37:19 +09:00
}]
}, {
scales: {
yAxes: [{
ticks: {
2018-08-24 08:56:57 +09:00
callback: value => {
2018-08-25 19:34:54 +09:00
return Vue.filter('bytes')(value, 1);
2018-08-24 05:37:19 +09:00
}
}
}]
2018-08-24 08:56:57 +09:00
},
tooltips: {
callbacks: {
2018-08-24 14:55:58 +09:00
label: (tooltipItem, data) => {
const label = data.datasets[tooltipItem.datasetIndex].label || '';
2018-08-25 19:34:54 +09:00
return `${label}: ${Vue.filter('bytes')(tooltipItem.yLabel, 1)}`;
2018-08-24 08:56:57 +09:00
}
}
2018-08-24 05:37:19 +09:00
}
}];
2018-08-24 06:41:53 +09:00
},
2018-08-25 08:35:41 +09:00
driveFilesChart(): any {
2018-08-24 06:41:53 +09:00
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
2018-08-25 08:35:41 +09:00
localInc: x.drive.local.incCount,
localDec: -x.drive.local.decCount,
remoteInc: x.drive.remote.incCount,
remoteDec: -x.drive.remote.decCount
2018-08-24 06:41:53 +09:00
}));
return [{
datasets: [{
2018-08-25 08:35:41 +09:00
label: 'All',
2018-08-24 14:55:58 +09:00
fill: false,
borderColor: '#555',
borderWidth: 2,
borderDash: [4, 4],
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 08:35:41 +09:00
data: data.map(x => ({ t: x.date, y: x.localInc + x.localDec + x.remoteInc + x.remoteDec }))
}, {
2018-08-25 15:25:16 +09:00
label: 'Local +',
2018-08-25 08:35:41 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.localPlus),
borderColor: colors.localPlus,
2018-08-25 08:35:41 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.localInc }))
2018-08-25 08:35:41 +09:00
}, {
2018-08-25 15:25:16 +09:00
label: 'Local -',
2018-08-25 08:35:41 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.localMinus),
borderColor: colors.localMinus,
2018-08-25 08:35:41 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.localDec }))
2018-08-25 08:35:41 +09:00
}, {
2018-08-25 15:25:16 +09:00
label: 'Remote +',
2018-08-25 08:35:41 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.remotePlus),
borderColor: colors.remotePlus,
2018-08-25 08:35:41 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.remoteInc }))
2018-08-25 08:35:41 +09:00
}, {
2018-08-25 15:25:16 +09:00
label: 'Remote -',
2018-08-25 08:35:41 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.remoteMinus),
borderColor: colors.remoteMinus,
2018-08-25 08:35:41 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.remoteDec }))
2018-08-25 08:35:41 +09: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)}`;
}
}
}
}];
},
driveFilesTotalChart(): any {
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
localCount: x.drive.local.totalCount,
remoteCount: x.drive.remote.totalCount,
}));
return [{
datasets: [{
label: 'Combined',
fill: false,
borderColor: '#555',
borderWidth: 2,
borderDash: [4, 4],
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.localCount + x.remoteCount }))
2018-08-24 14:55:58 +09:00
}, {
2018-08-25 15:25:16 +09:00
label: 'Local',
2018-08-24 14:55:58 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.local),
borderColor: colors.local,
2018-08-24 14:55:58 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.localCount }))
2018-08-24 14:55:58 +09:00
}, {
2018-08-25 15:25:16 +09:00
label: 'Remote',
2018-08-24 09:39:16 +09:00
fill: true,
2018-08-25 15:25:16 +09:00
backgroundColor: rgba(colors.remote),
borderColor: colors.remote,
2018-08-24 06:41:53 +09:00
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-08-25 15:25:16 +09:00
data: data.map(x => ({ t: x.date, y: x.remoteCount }))
2018-08-24 06:41:53 +09:00
}]
2018-08-24 14:55:58 +09: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 06:41:53 +09:00
}];
2018-09-15 05:40:58 +09:00
},
networkRequestsChart(): any {
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
2018-10-21 07:10:35 +09:00
incoming: x.network.incomingRequests
2018-09-15 05:40:58 +09:00
}));
return [{
datasets: [{
2018-10-21 07:10:35 +09:00
label: 'Incoming',
2018-09-15 05:40:58 +09:00
fill: true,
backgroundColor: rgba(colors.localPlus),
borderColor: colors.localPlus,
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
2018-10-21 09:20:11 +09:00
data: data.map(x => ({ t: x.date, y: x.incoming }))
2018-09-15 05:40:58 +09:00
}]
}];
},
networkTimeChart(): any {
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
2018-10-21 09:20:11 +09:00
time: x.network.incomingRequests != 0 ? (x.network.totalTime / x.network.incomingRequests) : 0,
2018-09-15 05:40:58 +09:00
}));
return [{
datasets: [{
label: 'Avg time (ms)',
fill: true,
backgroundColor: rgba(colors.localPlus),
borderColor: colors.localPlus,
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.time }))
}]
}];
},
networkUsageChart(): any {
const data = this.stats.slice().reverse().map(x => ({
date: new Date(x.date),
incoming: x.network.incomingBytes,
outgoing: x.network.outgoingBytes
}));
return [{
datasets: [{
label: 'Incoming',
fill: true,
backgroundColor: rgba(colors.incoming),
borderColor: colors.incoming,
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.incoming }))
}, {
label: 'Outgoing',
fill: true,
backgroundColor: rgba(colors.outgoing),
borderColor: colors.outgoing,
borderWidth: 2,
pointBackgroundColor: '#fff',
lineTension: 0,
data: data.map(x => ({ t: x.date, y: x.outgoing }))
}]
}, {
scales: {
yAxes: [{
ticks: {
callback: value => {
return Vue.filter('bytes')(value, 1);
}
}
}]
},
tooltips: {
callbacks: {
label: (tooltipItem, data) => {
const label = data.datasets[tooltipItem.datasetIndex].label || '';
return `${label}: ${Vue.filter('bytes')(tooltipItem.yLabel, 1)}`;
}
}
}
}];
},
2018-08-24 05:37:19 +09:00
}
});
</script>
<style lang="stylus" scoped>
2018-09-26 20:19:35 +09:00
2018-08-24 05:37:19 +09:00
.gkgckalzgidaygcxnugepioremxvxvpt
padding 32px
background #fff
box-shadow 0 2px 8px rgba(#000, 0.1)
2018-08-24 08:56:57 +09:00
*
user-select none
2018-08-24 05:37:19 +09:00
> header
display flex
margin 0 0 1em 0
padding 0 0 8px 0
font-size 1em
color #555
border-bottom solid 1px #eee
2018-08-24 05:37:19 +09:00
> b
margin-right 8px
> *:last-child
margin-left auto
2018-08-24 14:55:58 +09:00
*
&:not(.active)
2018-09-26 20:19:35 +09:00
color var(--primary)
2018-08-24 14:55:58 +09:00
cursor pointer
2018-08-24 07:17:17 +09:00
> div
> *
display block
2018-09-17 02:56:57 +09:00
height 350px
2018-08-24 07:17:17 +09:00
2018-08-24 05:37:19 +09:00
</style>