mirror of
https://github.com/zulip/zulip.git
synced 2025-10-23 04:52:12 +00:00
In this commit: Two new URLs are added, to make all realms accessible for server admins. One is for the stats page itself and another for getting chart data i.e. chart data API requests. For the above two new URLs corresponding two view functions are added.
717 lines
24 KiB
JavaScript
717 lines
24 KiB
JavaScript
var font_14pt = {
|
|
family: 'Source Sans Pro',
|
|
size: 14,
|
|
color: '#000000',
|
|
};
|
|
|
|
var last_full_update = Math.min();
|
|
|
|
// TODO: should take a dict of arrays and do it for all keys
|
|
function partial_sums(array) {
|
|
var accumulator = 0;
|
|
return array.map(function (o) {
|
|
accumulator += o;
|
|
return accumulator;
|
|
});
|
|
}
|
|
|
|
// Assumes date is a round number of hours
|
|
function floor_to_local_day(date) {
|
|
var date_copy = new Date(date.getTime());
|
|
date_copy.setHours(0);
|
|
return date_copy;
|
|
}
|
|
|
|
// Assumes date is a round number of hours
|
|
function floor_to_local_week(date) {
|
|
var date_copy = floor_to_local_day(date);
|
|
date_copy.setHours(-24 * date.getDay());
|
|
return date_copy;
|
|
}
|
|
|
|
function format_date(date, include_hour) {
|
|
var months = [i18n.t('January'), i18n.t('February'), i18n.t('March'), i18n.t('April'), i18n.t('May'), i18n.t('June'),
|
|
i18n.t('July'), i18n.t('August'), i18n.t('September'), i18n.t('October'), i18n.t('November'),
|
|
i18n.t('December')];
|
|
var month_str = months[date.getMonth()];
|
|
var year = date.getFullYear();
|
|
var day = date.getDate();
|
|
if (include_hour) {
|
|
var hour = date.getHours();
|
|
|
|
var str = hour >= 12 ? "PM" : "AM";
|
|
|
|
return month_str + " " + day + ", " + (hour % 12) + ":00" + str;
|
|
}
|
|
return month_str + ' ' + day + ', ' + year;
|
|
}
|
|
|
|
function update_last_full_update(end_times) {
|
|
if (end_times.length === 0) {
|
|
return;
|
|
}
|
|
|
|
last_full_update = Math.min(last_full_update, end_times[end_times.length - 1]);
|
|
var update_time = new Date(last_full_update * 1000);
|
|
var locale_date = update_time.toLocaleDateString('en-US', { year: 'numeric', month: 'long', day: 'numeric' });
|
|
var locale_time = update_time.toLocaleTimeString().replace(":00 ", " ");
|
|
|
|
$('#id_last_full_update').text(locale_time + " on " + locale_date);
|
|
$('#id_last_full_update').closest('.last-update').show();
|
|
}
|
|
|
|
$(function () {
|
|
$('span[data-toggle="tooltip"]').tooltip({
|
|
animation: false,
|
|
placement: 'top',
|
|
html: true,
|
|
trigger: 'manual',
|
|
});
|
|
$('#id_last_update_question_sign').hover(function () {
|
|
$('span[data-toggle="tooltip"]').tooltip('toggle');
|
|
});
|
|
});
|
|
|
|
|
|
function get_chart_data(data, callback) {
|
|
var url;
|
|
if (page_params.is_staff) {
|
|
url = '/json/analytics/chart_data/realm/' + page_params.stats_realm;
|
|
} else {
|
|
url = '/json/analytics/chart_data';
|
|
}
|
|
|
|
$.get({
|
|
url: url,
|
|
data: data,
|
|
idempotent: true,
|
|
success: function (data) {
|
|
callback(data);
|
|
update_last_full_update(data.end_times);
|
|
},
|
|
error: function (xhr) {
|
|
$('#id_stats_errors').show().text(JSON.parse(xhr.responseText).msg);
|
|
},
|
|
});
|
|
}
|
|
|
|
function populate_messages_sent_over_time(data) {
|
|
if (data.end_times.length === 0) {
|
|
// TODO: do something nicer here
|
|
return;
|
|
}
|
|
|
|
// Helper functions
|
|
function make_traces(dates, values, type, date_formatter) {
|
|
var text = dates.map(function (date) {
|
|
return date_formatter(date);
|
|
});
|
|
var common = { x: dates, type: type, hoverinfo: 'none', text: text };
|
|
return {
|
|
human: $.extend({ // 5062a0
|
|
name: i18n.t("Humans"), y: values.human, marker: {color: '#5f6ea0'}}, common),
|
|
bot: $.extend({ // a09b5f bbb56e
|
|
name: i18n.t("Bots"), y: values.bot, marker: {color: '#b7b867'}}, common),
|
|
me: $.extend({
|
|
name: i18n.t("Me"), y: values.me, marker: {color: '#be6d68'}}, common),
|
|
};
|
|
}
|
|
|
|
var layout = {
|
|
barmode: 'group',
|
|
width: 750,
|
|
height: 400,
|
|
margin: { l: 40, r: 0, b: 40, t: 0 },
|
|
xaxis: {
|
|
fixedrange: true,
|
|
rangeslider: { bordercolor: '#D8D8D8', borderwidth: 1 },
|
|
type: 'date',
|
|
},
|
|
yaxis: { fixedrange: true, rangemode: 'tozero' },
|
|
legend: {
|
|
x: 0.62, y: 1.12, orientation: 'h', font: font_14pt,
|
|
},
|
|
font: font_14pt,
|
|
};
|
|
|
|
function make_rangeselector(x, y, button1, button2) {
|
|
return { x: x, y: y,
|
|
buttons: [$.extend({stepmode: 'backward'}, button1),
|
|
$.extend({stepmode: 'backward'}, button2),
|
|
{step: 'all', label: 'All time'}] };
|
|
}
|
|
|
|
// This is also the cumulative rangeselector
|
|
var daily_rangeselector = make_rangeselector(
|
|
0.68, -0.62,
|
|
{count: 10, label: i18n.t('Last 10 days'), step: 'day'},
|
|
{count: 30, label: i18n.t('Last 30 days'), step: 'day'});
|
|
var weekly_rangeselector = make_rangeselector(
|
|
0.656, -0.62,
|
|
{count: 2, label: i18n.t('Last 2 months'), step: 'month'},
|
|
{count: 6, label: i18n.t('Last 6 months'), step: 'month'});
|
|
|
|
function add_hover_handler() {
|
|
document.getElementById('id_messages_sent_over_time').on('plotly_hover', function (data) {
|
|
$("#hoverinfo").show();
|
|
document.getElementById('hover_date').innerText =
|
|
data.points[0].data.text[data.points[0].pointNumber];
|
|
var values = [null, null, null];
|
|
data.points.forEach(function (trace) {
|
|
values[trace.curveNumber] = trace.y;
|
|
});
|
|
var hover_text_ids = ['hover_me', 'hover_human', 'hover_bot'];
|
|
var hover_value_ids = ['hover_me_value', 'hover_human_value', 'hover_bot_value'];
|
|
for (var i = 0; i < values.length; i += 1) {
|
|
if (values[i] !== null) {
|
|
document.getElementById(hover_text_ids[i]).style.display = 'inline';
|
|
document.getElementById(hover_value_ids[i]).style.display = 'inline';
|
|
document.getElementById(hover_value_ids[i]).innerText = values[i];
|
|
} else {
|
|
document.getElementById(hover_text_ids[i]).style.display = 'none';
|
|
document.getElementById(hover_value_ids[i]).style.display = 'none';
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
var start_dates = data.end_times.map(function (timestamp) {
|
|
// data.end_times are the ends of hour long intervals.
|
|
return new Date(timestamp*1000 - 60*60*1000);
|
|
});
|
|
|
|
function aggregate_data(aggregation) {
|
|
var start;
|
|
var is_boundary;
|
|
if (aggregation === 'day') {
|
|
start = floor_to_local_day(start_dates[0]);
|
|
is_boundary = function (date) {
|
|
return date.getHours() === 0;
|
|
};
|
|
} else if (aggregation === 'week') {
|
|
start = floor_to_local_week(start_dates[0]);
|
|
is_boundary = function (date) {
|
|
return date.getHours() === 0 && date.getDay() === 0;
|
|
};
|
|
}
|
|
var dates = [start];
|
|
var values = {human: [], bot: [], me: []};
|
|
var current = {human: 0, bot: 0, me: 0};
|
|
var i_init = 0;
|
|
if (is_boundary(start_dates[0])) {
|
|
current = {human: data.realm.human[0], bot: data.realm.bot[0], me: data.user.human[0]};
|
|
i_init = 1;
|
|
}
|
|
for (var i = i_init; i < start_dates.length; i += 1) {
|
|
if (is_boundary(start_dates[i])) {
|
|
dates.push(start_dates[i]);
|
|
values.human.push(current.human);
|
|
values.bot.push(current.bot);
|
|
values.me.push(current.me);
|
|
current = {human: 0, bot: 0, me: 0};
|
|
}
|
|
current.human += data.realm.human[i];
|
|
current.bot += data.realm.bot[i];
|
|
current.me += data.user.human[i];
|
|
}
|
|
values.human.push(current.human);
|
|
values.bot.push(current.bot);
|
|
values.me.push(current.me);
|
|
return {
|
|
dates: dates, values: values,
|
|
last_value_is_partial: !is_boundary(new Date(
|
|
start_dates[start_dates.length-1].getTime() + 60*60*1000))};
|
|
}
|
|
|
|
// Generate traces
|
|
var date_formatter = function (date) {
|
|
return format_date(date, true);
|
|
};
|
|
var values = {me: data.user.human, human: data.realm.human, bot: data.realm.bot};
|
|
|
|
var info = aggregate_data('day');
|
|
date_formatter = function (date) {
|
|
return format_date(date, false);
|
|
};
|
|
var last_day_is_partial = info.last_value_is_partial;
|
|
var daily_traces = make_traces(info.dates, info.values, 'bar', date_formatter);
|
|
|
|
info = aggregate_data('week');
|
|
date_formatter = function (date) {
|
|
return i18n.t("Week of __date__", {date: format_date(date, false)});
|
|
};
|
|
var last_week_is_partial = info.last_value_is_partial;
|
|
var weekly_traces = make_traces(info.dates, info.values, 'bar', date_formatter);
|
|
|
|
var dates = data.end_times.map(function (timestamp) {
|
|
return new Date(timestamp*1000);
|
|
});
|
|
values = {human: partial_sums(data.realm.human), bot: partial_sums(data.realm.bot),
|
|
me: partial_sums(data.user.human)};
|
|
date_formatter = function (date) {
|
|
return format_date(date, true);
|
|
};
|
|
var cumulative_traces = make_traces(dates, values, 'scatter', date_formatter);
|
|
|
|
// Functions to draw and interact with the plot
|
|
|
|
// We need to redraw plot entirely if switching from (the cumulative) line
|
|
// graph to any bar graph, since otherwise the rangeselector shows both (plotly bug)
|
|
var clicked_cumulative = false;
|
|
|
|
function draw_or_update_plot(rangeselector, traces, last_value_is_partial, initial_draw) {
|
|
$('#daily_button, #weekly_button, #cumulative_button').removeClass("selected");
|
|
$('#id_messages_sent_over_time > div').removeClass("spinner");
|
|
if (initial_draw) {
|
|
traces.human.visible = true;
|
|
traces.bot.visible = 'legendonly';
|
|
traces.me.visible = 'legendonly';
|
|
} else {
|
|
var plotDiv = document.getElementById('id_messages_sent_over_time');
|
|
traces.me.visible = plotDiv.data[0].visible;
|
|
traces.human.visible = plotDiv.data[1].visible;
|
|
traces.bot.visible = plotDiv.data[2].visible;
|
|
}
|
|
layout.xaxis.rangeselector = rangeselector;
|
|
if (clicked_cumulative || initial_draw) {
|
|
Plotly.newPlot('id_messages_sent_over_time',
|
|
[traces.me, traces.human, traces.bot], layout, {displayModeBar: false});
|
|
add_hover_handler();
|
|
} else {
|
|
Plotly.deleteTraces('id_messages_sent_over_time', [0, 1, 2]);
|
|
Plotly.addTraces('id_messages_sent_over_time', [traces.me, traces.human, traces.bot]);
|
|
Plotly.relayout('id_messages_sent_over_time', layout);
|
|
}
|
|
$('#id_messages_sent_over_time').attr('last_value_is_partial', last_value_is_partial);
|
|
}
|
|
|
|
// Click handlers for aggregation buttons
|
|
$('#daily_button').click(function () {
|
|
draw_or_update_plot(daily_rangeselector, daily_traces, last_day_is_partial, false);
|
|
$(this).addClass("selected");
|
|
clicked_cumulative = false;
|
|
});
|
|
|
|
$('#weekly_button').click(function () {
|
|
draw_or_update_plot(weekly_rangeselector, weekly_traces, last_week_is_partial, false);
|
|
$(this).addClass("selected");
|
|
clicked_cumulative = false;
|
|
});
|
|
|
|
$('#cumulative_button').click(function () {
|
|
clicked_cumulative = false;
|
|
draw_or_update_plot(daily_rangeselector, cumulative_traces, false, false);
|
|
$(this).addClass("selected");
|
|
clicked_cumulative = true;
|
|
});
|
|
|
|
// Initial drawing of plot
|
|
if (weekly_traces.human.x.length < 12) {
|
|
draw_or_update_plot(daily_rangeselector, daily_traces, last_day_is_partial, true);
|
|
$('#daily_button').addClass("selected");
|
|
} else {
|
|
draw_or_update_plot(weekly_rangeselector, weekly_traces, last_week_is_partial, true);
|
|
$('#weekly_button').addClass("selected");
|
|
}
|
|
}
|
|
|
|
get_chart_data(
|
|
{chart_name: 'messages_sent_over_time', min_length: '10'},
|
|
populate_messages_sent_over_time
|
|
);
|
|
|
|
function round_to_percentages(values, total) {
|
|
return values.map(function (x) {
|
|
if (x === total) {
|
|
return '100%';
|
|
}
|
|
if (x === 0) {
|
|
return '0%';
|
|
}
|
|
var unrounded = x/total*100;
|
|
|
|
var precision = Math.min(
|
|
6, // this is the max precision (two #, 4 decimal points; 99.9999%).
|
|
Math.max(
|
|
2, // the minimum amount of precision (40% or 6.0%).
|
|
Math.floor(-Math.log10(100 - unrounded)) + 3
|
|
)
|
|
);
|
|
|
|
return unrounded.toPrecision(precision) + '%';
|
|
});
|
|
}
|
|
|
|
// Last label will turn into "Other" if time_series data has a label not in labels
|
|
function compute_summary_chart_data(time_series_data, num_steps, labels_) {
|
|
var data = {};
|
|
var key;
|
|
for (key in time_series_data) {
|
|
if (!time_series_data.hasOwnProperty(key)) {
|
|
continue;
|
|
}
|
|
if (time_series_data[key].length < num_steps) {
|
|
num_steps = time_series_data[key].length;
|
|
}
|
|
var sum = 0;
|
|
for (var i=1; i<=num_steps; i+=1) {
|
|
sum += time_series_data[key][time_series_data[key].length-i];
|
|
}
|
|
data[key] = sum;
|
|
}
|
|
var labels = labels_.slice();
|
|
var values = [];
|
|
labels.forEach(function (label) {
|
|
if (data.hasOwnProperty(label)) {
|
|
values.push(data[label]);
|
|
delete data[label];
|
|
} else {
|
|
values.push(0);
|
|
}
|
|
});
|
|
if (!$.isEmptyObject(data)) {
|
|
labels[labels.length-1] = "Other";
|
|
for (key in data) {
|
|
if (data.hasOwnProperty(key)) {
|
|
values[labels.length-1] += data[key];
|
|
}
|
|
}
|
|
}
|
|
var total = values.reduce(function (a, b) { return a + b; }, 0);
|
|
return {
|
|
values: values,
|
|
labels: labels,
|
|
percentages: round_to_percentages(values, total),
|
|
total: total,
|
|
};
|
|
}
|
|
|
|
function populate_messages_sent_by_client(data) {
|
|
var layout = {
|
|
width: 750,
|
|
height: null, // set in draw_plot()
|
|
margin: { l: 3, r: 40, b: 40, t: 0 },
|
|
font: font_14pt,
|
|
xaxis: { range: null }, // set in draw_plot()
|
|
yaxis: { showticklabels: false },
|
|
showlegend: false,
|
|
};
|
|
|
|
// sort labels so that values are descending in the default view
|
|
var realm_month = compute_summary_chart_data(data.realm, 30, data.display_order.slice(0, 12));
|
|
var label_values = [];
|
|
for (var i=0; i<realm_month.values.length; i+=1) {
|
|
label_values.push({
|
|
label: realm_month.labels[i],
|
|
value: realm_month.labels[i] === "Other" ? -1 : realm_month.values[i],
|
|
});
|
|
}
|
|
label_values.sort(function (a, b) { return b.value - a.value; });
|
|
var labels = [];
|
|
label_values.forEach(function (item) { labels.push(item.label); });
|
|
|
|
function make_plot_data(time_series_data, num_steps) {
|
|
var plot_data = compute_summary_chart_data(time_series_data, num_steps, labels);
|
|
plot_data.values.reverse();
|
|
plot_data.labels.reverse();
|
|
plot_data.percentages.reverse();
|
|
var annotations = {values: [], labels: [], text: []};
|
|
for (var i=0; i<plot_data.values.length; i+=1) {
|
|
if (plot_data.values[i] > 0) {
|
|
annotations.values.push(plot_data.values[i]);
|
|
annotations.labels.push(plot_data.labels[i]);
|
|
annotations.text.push(' ' + plot_data.labels[i] + ' (' + plot_data.percentages[i] + ')');
|
|
}
|
|
}
|
|
return {
|
|
trace: {
|
|
x: plot_data.values,
|
|
y: plot_data.labels,
|
|
type: 'bar',
|
|
orientation: 'h',
|
|
sort: false,
|
|
textinfo: "text",
|
|
hoverinfo: "none",
|
|
marker: { color: '#537c5e' },
|
|
font: { family: 'Source Sans Pro', size: 18, color: '#000000' },
|
|
},
|
|
trace_annotations: {
|
|
x: annotations.values,
|
|
y: annotations.labels,
|
|
mode: 'text',
|
|
type: 'scatter',
|
|
textposition: 'middle right',
|
|
text: annotations.text,
|
|
},
|
|
};
|
|
}
|
|
|
|
var plot_data = {
|
|
realm: {
|
|
cumulative: make_plot_data(data.realm, data.end_times.length),
|
|
year: make_plot_data(data.realm, 365),
|
|
month: make_plot_data(data.realm, 30),
|
|
week: make_plot_data(data.realm, 7),
|
|
},
|
|
user: {
|
|
cumulative: make_plot_data(data.user, data.end_times.length),
|
|
year: make_plot_data(data.user, 365),
|
|
month: make_plot_data(data.user, 30),
|
|
week: make_plot_data(data.user, 7),
|
|
},
|
|
};
|
|
|
|
var user_button = 'realm';
|
|
var time_button;
|
|
if (data.end_times.length >= 30) {
|
|
time_button = 'month';
|
|
$('#messages_by_client_last_month_button').addClass("selected");
|
|
} else {
|
|
time_button = 'cumulative';
|
|
$('#messages_by_client_cumulative_button').addClass("selected");
|
|
}
|
|
|
|
if (data.end_times.length < 365) {
|
|
$("#pie_messages_sent_by_client button[data-time='year']").remove();
|
|
if (data.end_times.length < 30) {
|
|
$("#pie_messages_sent_by_client button[data-time='month']").remove();
|
|
if (data.end_times.length < 7) {
|
|
$("#pie_messages_sent_by_client button[data-time='week']").remove();
|
|
}
|
|
}
|
|
}
|
|
|
|
function draw_plot() {
|
|
$('#id_messages_sent_by_client > div').removeClass("spinner");
|
|
var data_ = plot_data[user_button][time_button];
|
|
layout.height = layout.margin.b + data_.trace.x.length * 30;
|
|
layout.xaxis.range = [0, Math.max.apply(null, data_.trace.x) * 1.3];
|
|
Plotly.newPlot('id_messages_sent_by_client',
|
|
[data_.trace, data_.trace_annotations],
|
|
layout,
|
|
{displayModeBar: false, staticPlot: true});
|
|
}
|
|
|
|
draw_plot();
|
|
|
|
// Click handlers
|
|
function set_user_button(button) {
|
|
$("#pie_messages_sent_by_client button[data-user]").removeClass("selected");
|
|
button.addClass("selected");
|
|
}
|
|
|
|
function set_time_button(button) {
|
|
$("#pie_messages_sent_by_client button[data-time]").removeClass("selected");
|
|
button.addClass("selected");
|
|
}
|
|
|
|
$("#pie_messages_sent_by_client button").click(function () {
|
|
if ($(this).attr("data-user")) {
|
|
set_user_button($(this));
|
|
user_button = $(this).attr("data-user");
|
|
}
|
|
if ($(this).attr("data-time")) {
|
|
set_time_button($(this));
|
|
time_button = $(this).attr("data-time");
|
|
}
|
|
draw_plot();
|
|
});
|
|
|
|
// handle links with @href started with '#' only
|
|
$(document).on('click', 'a[href^="#"]', function (e) {
|
|
// target element id
|
|
var id = $(this).attr('href');
|
|
// target element
|
|
var $id = $(id);
|
|
if ($id.length === 0) {
|
|
return;
|
|
}
|
|
// prevent standard hash navigation (avoid blinking in IE)
|
|
e.preventDefault();
|
|
var pos = $id.offset().top+$('.page-content')[0].scrollTop-50;
|
|
$('.page-content').animate({scrollTop: pos + "px"}, 500);
|
|
});
|
|
}
|
|
|
|
get_chart_data(
|
|
{chart_name: 'messages_sent_by_client', min_length: '10'},
|
|
populate_messages_sent_by_client
|
|
);
|
|
|
|
function populate_messages_sent_by_message_type(data) {
|
|
var layout = {
|
|
margin: { l: 90, r: 0, b: 0, t: 0 },
|
|
width: 750,
|
|
height: 300,
|
|
font: font_14pt,
|
|
};
|
|
|
|
function make_plot_data(time_series_data, num_steps) {
|
|
var plot_data = compute_summary_chart_data(time_series_data, num_steps, data.display_order);
|
|
var labels = [];
|
|
for (var i=0; i<plot_data.labels.length; i+=1) {
|
|
labels.push(plot_data.labels[i] + ' (' + plot_data.percentages[i] + ')');
|
|
}
|
|
return {
|
|
trace: {
|
|
values: plot_data.values,
|
|
labels: labels,
|
|
type: 'pie',
|
|
direction: 'clockwise',
|
|
rotation: -90,
|
|
sort: false,
|
|
textinfo: "text",
|
|
text: plot_data.labels.map(function () { return ''; }),
|
|
hoverinfo: "label+value",
|
|
pull: 0.05,
|
|
marker: {
|
|
colors: ['#68537c', '#be6d68', '#b3b348'],
|
|
},
|
|
},
|
|
total_str: "<b>Total messages:</b> " + plot_data.total.toString().
|
|
replace(/\B(?=(\d{3})+(?!\d))/g, ","),
|
|
};
|
|
}
|
|
|
|
var plot_data = {
|
|
realm: {
|
|
cumulative: make_plot_data(data.realm, data.end_times.length),
|
|
year: make_plot_data(data.realm, 365),
|
|
month: make_plot_data(data.realm, 30),
|
|
week: make_plot_data(data.realm, 7),
|
|
},
|
|
user: {
|
|
cumulative: make_plot_data(data.user, data.end_times.length),
|
|
year: make_plot_data(data.user, 365),
|
|
month: make_plot_data(data.user, 30),
|
|
week: make_plot_data(data.user, 7),
|
|
},
|
|
};
|
|
|
|
var user_button = 'realm';
|
|
var time_button;
|
|
if (data.end_times.length >= 30) {
|
|
time_button = 'month';
|
|
$('#messages_by_type_last_month_button').addClass("selected");
|
|
} else {
|
|
time_button = 'cumulative';
|
|
$('#messages_by_type_cumulative_button').addClass("selected");
|
|
}
|
|
var totaldiv = document.getElementById('pie_messages_sent_by_type_total');
|
|
|
|
if (data.end_times.length < 365) {
|
|
$("#pie_messages_sent_by_type button[data-time='year']").remove();
|
|
if (data.end_times.length < 30) {
|
|
$("#pie_messages_sent_by_type button[data-time='month']").remove();
|
|
if (data.end_times.length < 7) {
|
|
$("#pie_messages_sent_by_type button[data-time='week']").remove();
|
|
}
|
|
}
|
|
}
|
|
|
|
function draw_plot() {
|
|
$('#id_messages_sent_by_message_type > div').removeClass("spinner");
|
|
Plotly.newPlot('id_messages_sent_by_message_type',
|
|
[plot_data[user_button][time_button].trace],
|
|
layout,
|
|
{displayModeBar: false});
|
|
totaldiv.innerHTML = plot_data[user_button][time_button].total_str;
|
|
}
|
|
|
|
draw_plot();
|
|
|
|
// Click handlers
|
|
function set_user_button(button) {
|
|
$("#pie_messages_sent_by_type button[data-user]").removeClass("selected");
|
|
button.addClass("selected");
|
|
}
|
|
|
|
function set_time_button(button) {
|
|
$("#pie_messages_sent_by_type button[data-time]").removeClass("selected");
|
|
button.addClass("selected");
|
|
}
|
|
|
|
$("#pie_messages_sent_by_type button").click(function () {
|
|
if ($(this).attr("data-user")) {
|
|
set_user_button($(this));
|
|
user_button = $(this).attr("data-user");
|
|
}
|
|
if ($(this).attr("data-time")) {
|
|
set_time_button($(this));
|
|
time_button = $(this).attr("data-time");
|
|
}
|
|
draw_plot();
|
|
});
|
|
}
|
|
|
|
get_chart_data(
|
|
{chart_name: 'messages_sent_by_message_type', min_length: '10'},
|
|
populate_messages_sent_by_message_type
|
|
);
|
|
|
|
function populate_number_of_users(data) {
|
|
var layout = {
|
|
width: 750,
|
|
height: 370,
|
|
margin: {
|
|
l: 40, r: 0, b: 100, t: 20,
|
|
},
|
|
xaxis: {
|
|
fixedrange: true,
|
|
rangeselector: {
|
|
x: 0.808,
|
|
y: -0.33,
|
|
buttons: [
|
|
{
|
|
count: 30,
|
|
label: i18n.t('Last 30 days'),
|
|
step: 'day',
|
|
stepmode: 'backward',
|
|
},
|
|
{
|
|
step: 'all',
|
|
label: i18n.t('All time'),
|
|
},
|
|
],
|
|
},
|
|
},
|
|
yaxis: {
|
|
fixedrange: true,
|
|
rangemode: 'tozero',
|
|
},
|
|
font: font_14pt,
|
|
};
|
|
|
|
var end_dates = data.end_times.map(function (timestamp) {
|
|
return new Date(timestamp*1000);
|
|
});
|
|
|
|
var text = end_dates.map(format_date);
|
|
|
|
var trace = {
|
|
x: end_dates,
|
|
y: data.realm.human,
|
|
type: 'scatter',
|
|
name: "Active users",
|
|
hoverinfo: 'none',
|
|
text: text,
|
|
visible: true,
|
|
};
|
|
|
|
$('#id_number_of_users > div').removeClass("spinner");
|
|
|
|
Plotly.newPlot('id_number_of_users', [trace], layout, {displayModeBar: false});
|
|
|
|
document.getElementById('id_number_of_users').on('plotly_hover', function (data) {
|
|
$("#users_hover_info").show();
|
|
$("#users_hover_date").text(data.points[0].data.text[data.points[0].pointNumber]);
|
|
$("#users_hover_humans").css("display", "inline");
|
|
$("#users_hover_humans_value").text(data.points[0].y);
|
|
});
|
|
}
|
|
|
|
get_chart_data(
|
|
{chart_name: 'number_of_humans', min_length: '10'},
|
|
populate_number_of_users
|
|
);
|