Files
NodeBB/public/src/admin/general/dashboard.js

588 lines
17 KiB
JavaScript
Raw Normal View History

2017-02-18 01:56:23 -07:00
'use strict';
2017-08-30 10:50:28 -06:00
define('admin/general/dashboard', ['semver', 'Chart', 'translator', 'benchpress'], function (semver, Chart, translator, Benchpress) {
2016-07-29 19:31:48 +00:00
var Admin = {};
var intervals = {
rooms: false,
2017-02-17 19:31:21 -07:00
graphs: false,
};
2016-07-29 19:31:48 +00:00
var isMobile = false;
var graphData = {
rooms: {},
2017-02-17 19:31:21 -07:00
traffic: {},
};
2016-07-29 19:31:48 +00:00
var currentGraph = {
units: 'hours',
2017-02-17 19:31:21 -07:00
until: undefined,
};
2015-07-08 16:35:59 -04:00
var DEFAULTS = {
roomInterval: 10000,
graphInterval: 15000,
2017-02-17 19:31:21 -07:00
realtimeInterval: 1500,
2015-07-08 16:35:59 -04:00
};
2017-02-18 01:27:46 -07:00
2017-02-18 18:55:33 -07:00
var usedTopicColors = [];
2017-02-18 15:05:36 -07:00
$(window).on('action:ajaxify.start', function () {
2016-03-09 21:36:20 +02:00
clearInterval(intervals.rooms);
clearInterval(intervals.graphs);
intervals.rooms = null;
intervals.graphs = null;
graphData.rooms = null;
graphData.traffic = null;
usedTopicColors.length = 0;
});
Admin.init = function () {
app.enterRoom('admin');
2014-10-17 23:14:17 -04:00
isMobile = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent);
2015-05-29 16:47:12 -04:00
$('[data-toggle="tooltip"]').tooltip();
2015-07-14 17:03:28 -04:00
2015-07-08 16:35:59 -04:00
setupRealtimeButton();
2017-03-13 20:20:41 -07:00
setupGraphs(function () {
socket.emit('admin.rooms.getAll', Admin.updateRoomUsage);
initiateDashboard();
});
2018-04-09 11:50:30 -04:00
setupFullscreen();
};
Admin.updateRoomUsage = function (err, data) {
if (err) {
return app.alertError(err.message);
}
2015-07-08 16:54:39 -04:00
if (JSON.stringify(graphData.rooms) === JSON.stringify(data)) {
return;
}
graphData.rooms = data;
var html = '<div class="text-center pull-left">'
+ '<span class="formatted-number">' + data.onlineRegisteredCount + '</span>'
+ '<div class="stat">[[admin/general/dashboard:active-users.users]]</div>'
+ '</div>'
+ '<div class="text-center pull-left">'
+ '<span class="formatted-number">' + data.onlineGuestCount + '</span>'
+ '<div class="stat">[[admin/general/dashboard:active-users.guests]]</div>'
+ '</div>'
+ '<div class="text-center pull-left">'
+ '<span class="formatted-number">' + (data.onlineRegisteredCount + data.onlineGuestCount) + '</span>'
+ '<div class="stat">[[admin/general/dashboard:active-users.total]]</div>'
+ '</div>'
+ '<div class="text-center pull-left">'
+ '<span class="formatted-number">' + data.socketCount + '</span>'
+ '<div class="stat">[[admin/general/dashboard:active-users.connections]]</div>'
+ '</div>';
updateRegisteredGraph(data.onlineRegisteredCount, data.onlineGuestCount);
updatePresenceGraph(data.users);
2017-10-18 14:49:42 -04:00
updateTopicsGraph(data.topTenTopics);
$('#active-users').translateHtml(html);
};
var graphs = {
traffic: null,
registered: null,
presence: null,
2017-02-17 19:31:21 -07:00
topics: null,
};
2017-02-18 01:56:23 -07:00
var topicColors = ['#bf616a', '#5B90BF', '#d08770', '#ebcb8b', '#a3be8c', '#96b5b4', '#8fa1b3', '#b48ead', '#ab7967', '#46BFBD'];
2017-02-18 18:55:33 -07:00
/* eslint-disable */
// from chartjs.org
function lighten(col, amt) {
var usePound = false;
if (col[0] === '#') {
col = col.slice(1);
usePound = true;
}
2017-02-18 01:31:47 -07:00
var num = parseInt(col, 16);
var r = (num >> 16) + amt;
if (r > 255) r = 255;
else if (r < 0) r = 0;
var b = ((num >> 8) & 0x00FF) + amt;
if (b > 255) b = 255;
else if (b < 0) b = 0;
var g = (num & 0x0000FF) + amt;
if (g > 255) g = 255;
else if (g < 0) g = 0;
2017-02-18 01:56:23 -07:00
return (usePound ? '#' : '') + (g | (b << 8) | (r << 16)).toString(16);
}
2017-02-18 18:55:33 -07:00
/* eslint-enable */
2017-03-13 20:20:41 -07:00
function setupGraphs(callback) {
callback = callback || function () {};
2017-02-17 20:20:42 -07:00
var trafficCanvas = document.getElementById('analytics-traffic');
var registeredCanvas = document.getElementById('analytics-registered');
var presenceCanvas = document.getElementById('analytics-presence');
var topicsCanvas = document.getElementById('analytics-topics');
var trafficCtx = trafficCanvas.getContext('2d');
var registeredCtx = registeredCanvas.getContext('2d');
var presenceCtx = presenceCanvas.getContext('2d');
var topicsCtx = topicsCanvas.getContext('2d');
var trafficLabels = utils.getHoursArray();
if (isMobile) {
2016-08-09 07:54:58 -07:00
Chart.defaults.global.tooltips.enabled = false;
}
var t = translator.Translator.create();
Promise.all([
t.translateKey('admin/general/dashboard:graphs.page-views', []),
2018-10-24 11:24:37 -04:00
t.translateKey('admin/general/dashboard:graphs.page-views-registered', []),
t.translateKey('admin/general/dashboard:graphs.page-views-guest', []),
t.translateKey('admin/general/dashboard:graphs.page-views-bot', []),
t.translateKey('admin/general/dashboard:graphs.unique-visitors', []),
t.translateKey('admin/general/dashboard:graphs.registered-users', []),
t.translateKey('admin/general/dashboard:graphs.anonymous-users', []),
t.translateKey('admin/general/dashboard:on-categories', []),
t.translateKey('admin/general/dashboard:reading-posts', []),
t.translateKey('admin/general/dashboard:browsing-topics', []),
t.translateKey('admin/general/dashboard:recent', []),
t.translateKey('admin/general/dashboard:unread', []),
]).then(function (translations) {
var data = {
labels: trafficLabels,
datasets: [
{
label: translations[0],
2017-02-18 01:56:23 -07:00
backgroundColor: 'rgba(220,220,220,0.2)',
borderColor: 'rgba(220,220,220,1)',
pointBackgroundColor: 'rgba(220,220,220,1)',
pointHoverBackgroundColor: '#fff',
pointBorderColor: '#fff',
pointHoverBorderColor: 'rgba(220,220,220,1)',
2017-02-18 01:31:47 -07:00
data: [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
},
{
label: translations[1],
2018-10-24 11:55:44 -04:00
backgroundColor: '#ab464233',
2018-10-24 11:39:40 -04:00
borderColor: '#ab4642',
pointBackgroundColor: '#ab4642',
2018-10-24 12:12:17 -04:00
pointHoverBackgroundColor: '#ab4642',
2018-10-24 11:24:37 -04:00
pointBorderColor: '#fff',
2018-10-24 11:39:40 -04:00
pointHoverBorderColor: '#ab4642',
2018-10-24 11:24:37 -04:00
data: [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
},
{
label: translations[2],
2018-10-24 11:55:44 -04:00
backgroundColor: '#ba8baf33',
2018-10-24 11:39:40 -04:00
borderColor: '#ba8baf',
pointBackgroundColor: '#ba8baf',
2018-10-24 12:12:17 -04:00
pointHoverBackgroundColor: '#ba8baf',
2018-10-24 11:24:37 -04:00
pointBorderColor: '#fff',
2018-10-24 11:39:40 -04:00
pointHoverBorderColor: '#ba8baf',
2018-10-24 11:24:37 -04:00
data: [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
},
{
label: translations[3],
2018-10-24 11:55:44 -04:00
backgroundColor: '#f7ca8833',
2018-10-24 11:39:40 -04:00
borderColor: '#f7ca88',
pointBackgroundColor: '#f7ca88',
2018-10-24 12:12:17 -04:00
pointHoverBackgroundColor: '#f7ca88',
2018-10-24 11:24:37 -04:00
pointBorderColor: '#fff',
2018-10-24 11:39:40 -04:00
pointHoverBorderColor: '#f7ca88',
2018-10-24 11:24:37 -04:00
data: [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
},
{
label: translations[4],
2017-02-18 01:56:23 -07:00
backgroundColor: 'rgba(151,187,205,0.2)',
borderColor: 'rgba(151,187,205,1)',
pointBackgroundColor: 'rgba(151,187,205,1)',
2017-09-14 15:57:15 -04:00
pointHoverBackgroundColor: 'rgba(151,187,205,1)',
2017-02-18 01:56:23 -07:00
pointBorderColor: '#fff',
pointHoverBorderColor: 'rgba(151,187,205,1)',
2017-02-18 01:31:47 -07:00
data: [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0],
2017-02-17 19:31:21 -07:00
},
],
};
trafficCanvas.width = $(trafficCanvas).parent().width();
2018-01-08 13:20:56 -05:00
data.datasets[0].yAxisID = 'left-y-axis';
2018-10-24 11:46:39 -04:00
data.datasets[1].yAxisID = 'left-y-axis';
data.datasets[2].yAxisID = 'left-y-axis';
data.datasets[3].yAxisID = 'left-y-axis';
data.datasets[4].yAxisID = 'right-y-axis';
2018-01-08 13:20:56 -05:00
graphs.traffic = new Chart(trafficCtx, {
type: 'line',
data: data,
options: {
responsive: true,
legend: {
2018-10-24 12:51:14 -04:00
display: true,
},
scales: {
yAxes: [{
2018-01-08 13:20:56 -05:00
id: 'left-y-axis',
ticks: {
beginAtZero: true,
},
type: 'linear',
position: 'left',
2018-03-07 12:26:26 -05:00
scaleLabel: {
display: true,
labelString: translations[0],
},
2018-01-08 13:20:56 -05:00
}, {
id: 'right-y-axis',
ticks: {
2017-02-17 19:31:21 -07:00
beginAtZero: true,
2018-01-08 13:20:56 -05:00
suggestedMax: 10,
2017-02-17 19:31:21 -07:00
},
2018-01-08 13:20:56 -05:00
type: 'linear',
position: 'right',
2018-03-07 12:26:26 -05:00
scaleLabel: {
display: true,
2018-10-24 11:24:37 -04:00
labelString: translations[4],
2018-03-07 12:26:26 -05:00
},
2017-02-17 19:31:21 -07:00
}],
},
2017-08-02 16:17:24 -04:00
tooltips: {
mode: 'x',
},
2017-02-17 19:31:21 -07:00
},
});
2017-02-18 01:27:46 -07:00
graphs.registered = new Chart(registeredCtx, {
type: 'doughnut',
data: {
2018-10-24 17:32:22 -04:00
labels: translations.slice(5, 7),
datasets: [{
data: [1, 1],
2017-02-18 01:56:23 -07:00
backgroundColor: ['#F7464A', '#46BFBD'],
hoverBackgroundColor: ['#FF5A5E', '#5AD3D1'],
2017-02-17 19:31:21 -07:00
}],
},
options: {
responsive: true,
legend: {
2017-02-17 19:31:21 -07:00
display: false,
},
},
});
graphs.presence = new Chart(presenceCtx, {
type: 'doughnut',
data: {
2018-10-24 17:32:22 -04:00
labels: translations.slice(7, 12),
datasets: [{
data: [1, 1, 1, 1, 1],
2017-02-18 01:56:23 -07:00
backgroundColor: ['#F7464A', '#46BFBD', '#FDB45C', '#949FB1', '#9FB194'],
hoverBackgroundColor: ['#FF5A5E', '#5AD3D1', '#FFC870', '#A8B3C5', '#A8B3C5'],
2017-02-17 19:31:21 -07:00
}],
},
options: {
responsive: true,
legend: {
2017-02-17 19:31:21 -07:00
display: false,
},
},
});
2017-02-18 01:27:46 -07:00
graphs.topics = new Chart(topicsCtx, {
type: 'doughnut',
data: {
labels: [],
datasets: [{
data: [],
backgroundColor: [],
2017-02-17 19:31:21 -07:00
hoverBackgroundColor: [],
}],
},
options: {
responsive: true,
legend: {
2017-02-17 19:31:21 -07:00
display: false,
},
},
});
updateTrafficGraph();
$(window).on('resize', adjustPieCharts);
adjustPieCharts();
$('[data-action="updateGraph"]:not([data-units="custom"])').on('click', function () {
2017-05-11 16:53:30 -04:00
var until = new Date();
var amount = $(this).attr('data-amount');
2017-05-24 21:18:23 -04:00
if ($(this).attr('data-units') === 'days') {
until.setHours(0, 0, 0, 0);
}
2017-05-11 16:53:30 -04:00
until = until.getTime();
updateTrafficGraph($(this).attr('data-units'), until, amount);
$('[data-action="updateGraph"]').removeClass('active');
$(this).addClass('active');
require(['translator'], function (translator) {
translator.translate('[[admin/general/dashboard:page-views-custom]]', function (translated) {
$('[data-action="updateGraph"][data-units="custom"]').text(translated);
});
});
});
$('[data-action="updateGraph"][data-units="custom"]').on('click', function () {
var targetEl = $(this);
2017-08-30 10:50:28 -06:00
Benchpress.parse('admin/partials/pageviews-range-select', {}, function (html) {
var modal = bootbox.dialog({
title: '[[admin/general/dashboard:page-views-custom]]',
message: html,
buttons: {
submit: {
label: '[[global:search]]',
className: 'btn-primary',
callback: submit,
},
},
}).on('shown.bs.modal', function () {
var date = new Date();
var today = date.toISOString().substr(0, 10);
date.setDate(date.getDate() - 1);
var yesterday = date.toISOString().substr(0, 10);
modal.find('#startRange').val(targetEl.attr('data-startRange') || yesterday);
modal.find('#endRange').val(targetEl.attr('data-endRange') || today);
});
function submit() {
// NEED TO ADD VALIDATION HERE FOR YYYY-MM-DD
var formData = modal.find('form').serializeObject();
var validRegexp = /\d{4}-\d{2}-\d{2}/;
// Input validation
if (!formData.startRange && !formData.endRange) {
// No range? Assume last 30 days
updateTrafficGraph('days');
$('[data-action="updateGraph"]').removeClass('active');
$('[data-action="updateGraph"][data-units="days"]').addClass('active');
return;
} else if (!validRegexp.test(formData.startRange) || !validRegexp.test(formData.endRange)) {
// Invalid Input
modal.find('.alert-danger').removeClass('hidden');
return false;
}
var until = new Date(formData.endRange);
until.setDate(until.getDate() + 1);
until = until.getTime();
var amount = (until - new Date(formData.startRange).getTime()) / (1000 * 60 * 60 * 24);
updateTrafficGraph('days', until, amount);
$('[data-action="updateGraph"]').removeClass('active');
targetEl.addClass('active');
// Update "custom range" label
targetEl.attr('data-startRange', formData.startRange);
targetEl.attr('data-endRange', formData.endRange);
targetEl.html(formData.startRange + ' &ndash; ' + formData.endRange);
}
});
});
2017-01-12 14:22:30 -05:00
socket.emit('admin.rooms.getAll', Admin.updateRoomUsage);
2017-01-12 14:22:30 -05:00
initiateDashboard();
2017-03-13 20:20:41 -07:00
callback();
});
}
function adjustPieCharts() {
$('.pie-chart.legend-up').each(function () {
var $this = $(this);
if ($this.width() < 320) {
$this.addClass('compact');
} else {
$this.removeClass('compact');
}
});
}
function updateTrafficGraph(units, until, amount) {
// until and amount are optional
2014-10-19 16:13:10 -04:00
if (!app.isFocused) {
return;
}
socket.emit('admin.analytics.get', {
graph: 'traffic',
units: units || 'hours',
2017-02-17 19:31:21 -07:00
until: until,
amount: amount,
}, function (err, data) {
2016-08-16 19:46:59 +02:00
if (err) {
return app.alertError(err.message);
}
2015-07-08 16:54:39 -04:00
if (JSON.stringify(graphData.traffic) === JSON.stringify(data)) {
return;
}
graphData.traffic = data;
if (units === 'days') {
graphs.traffic.data.xLabels = utils.getDaysArray(until, amount);
} else {
2016-07-29 19:31:48 +00:00
graphs.traffic.data.xLabels = utils.getHoursArray();
2017-05-11 16:53:30 -04:00
$('#pageViewsThirty').html(data.summary.thirty);
$('#pageViewsSeven').html(data.summary.seven);
$('#pageViewsPastDay').html(data.pastDay);
2017-05-11 16:53:30 -04:00
utils.addCommasToNumbers($('#pageViewsThirty'));
utils.addCommasToNumbers($('#pageViewsSeven'));
utils.addCommasToNumbers($('#pageViewsPastDay'));
}
2016-07-29 19:31:48 +00:00
graphs.traffic.data.datasets[0].data = data.pageviews;
2018-10-24 11:24:37 -04:00
graphs.traffic.data.datasets[1].data = data.pageviewsRegistered;
2018-10-24 11:34:57 -04:00
graphs.traffic.data.datasets[2].data = data.pageviewsGuest;
graphs.traffic.data.datasets[3].data = data.pageviewsBot;
2018-10-24 11:24:37 -04:00
graphs.traffic.data.datasets[4].data = data.uniqueVisitors;
2016-07-29 19:31:48 +00:00
graphs.traffic.data.labels = graphs.traffic.data.xLabels;
graphs.traffic.update();
currentGraph.units = units;
currentGraph.until = until;
currentGraph.amount = amount;
});
}
function updateRegisteredGraph(registered, anonymous) {
2016-07-29 19:31:48 +00:00
graphs.registered.data.datasets[0].data[0] = registered;
graphs.registered.data.datasets[0].data[1] = anonymous;
graphs.registered.update();
}
function updatePresenceGraph(users) {
2016-07-29 19:31:48 +00:00
graphs.presence.data.datasets[0].data[0] = users.categories;
graphs.presence.data.datasets[0].data[1] = users.topics;
graphs.presence.data.datasets[0].data[2] = users.category;
graphs.presence.data.datasets[0].data[3] = users.recent;
graphs.presence.data.datasets[0].data[4] = users.unread;
graphs.presence.update();
}
2017-02-18 01:27:46 -07:00
function updateTopicsGraph(topics) {
2017-10-18 14:49:42 -04:00
if (!topics.length) {
topics = [{
2018-06-02 15:54:28 -04:00
title: '[[admin/general/dashboard:no-users-browsing]]',
2017-10-18 14:49:42 -04:00
count: 1,
}];
}
2016-07-29 19:31:48 +00:00
graphs.topics.data.labels = [];
graphs.topics.data.datasets[0].data = [];
graphs.topics.data.datasets[0].backgroundColor = [];
graphs.topics.data.datasets[0].hoverBackgroundColor = [];
2017-02-18 01:27:46 -07:00
2017-10-18 14:49:42 -04:00
topics.forEach(function (topic, i) {
graphs.topics.data.labels.push(topic.title);
graphs.topics.data.datasets[0].data.push(topic.count);
2016-07-29 19:31:48 +00:00
graphs.topics.data.datasets[0].backgroundColor.push(topicColors[i]);
graphs.topics.data.datasets[0].hoverBackgroundColor.push(lighten(topicColors[i], 10));
2017-10-18 14:49:42 -04:00
});
2017-02-18 01:27:46 -07:00
function buildTopicsLegend() {
var legend = $('#topics-legend').html('');
2018-06-02 15:54:28 -04:00
var html = '';
2017-10-18 14:49:42 -04:00
topics.forEach(function (topic, i) {
var label = topic.count === '0' ? topic.title : '<a title="' + topic.title + '"href="' + RELATIVE_PATH + '/topic/' + topic.tid + '" target="_blank"> ' + topic.title + '</a>';
2017-02-18 01:27:46 -07:00
html += '<li>'
+ '<div style="background-color: ' + topicColors[i] + ';"></div>'
+ '<span>' + label + '</span>'
+ '</li>';
2017-10-18 14:49:42 -04:00
});
2018-06-02 15:54:28 -04:00
legend.translateHtml(html);
}
buildTopicsLegend();
graphs.topics.update();
}
2015-07-08 16:35:59 -04:00
function setupRealtimeButton() {
$('#toggle-realtime .fa').on('click', function () {
2015-07-08 16:35:59 -04:00
var $this = $(this);
if ($this.hasClass('fa-toggle-on')) {
$this.removeClass('fa-toggle-on').addClass('fa-toggle-off');
$this.parent().find('strong').html('OFF');
initiateDashboard(false);
} else {
$this.removeClass('fa-toggle-off').addClass('fa-toggle-on');
$this.parent().find('strong').html('ON');
initiateDashboard(true);
}
});
}
function initiateDashboard(realtime) {
clearInterval(intervals.rooms);
clearInterval(intervals.graphs);
intervals.rooms = setInterval(function () {
2015-07-08 16:35:59 -04:00
if (app.isFocused && app.isConnected) {
2015-11-04 17:43:43 -05:00
socket.emit('admin.rooms.getAll', Admin.updateRoomUsage);
2015-07-08 16:35:59 -04:00
}
}, realtime ? DEFAULTS.realtimeInterval : DEFAULTS.roomInterval);
intervals.graphs = setInterval(function () {
updateTrafficGraph(currentGraph.units, currentGraph.until, currentGraph.amount);
}, realtime ? DEFAULTS.realtimeInterval : DEFAULTS.graphInterval);
2015-07-08 16:35:59 -04:00
}
2018-04-09 11:50:30 -04:00
function setupFullscreen() {
var container = document.getElementById('analytics-traffic-container');
var $container = $(container);
var btn = $container.find('.fa-expand');
var fsMethod;
var exitMethod;
if (container.requestFullscreen) {
fsMethod = 'requestFullscreen';
exitMethod = 'exitFullscreen';
} else if (container.mozRequestFullScreen) {
fsMethod = 'mozRequestFullScreen';
exitMethod = 'mozCancelFullScreen';
} else if (container.webkitRequestFullscreen) {
fsMethod = 'webkitRequestFullscreen';
exitMethod = 'webkitCancelFullScreen';
} else if (container.msRequestFullscreen) {
fsMethod = 'msRequestFullscreen';
exitMethod = 'msCancelFullScreen';
}
if (fsMethod) {
btn.addClass('active');
btn.on('click', function () {
if ($container.hasClass('fullscreen')) {
document[exitMethod]();
$container.removeClass('fullscreen');
} else {
container[fsMethod]();
$container.addClass('fullscreen');
}
});
}
}
return Admin;
2016-07-29 12:55:59 -07:00
});