mirror of
https://github.com/discourse/discourse.git
synced 2025-01-10 00:44:14 +08:00
175 lines
4.3 KiB
JavaScript
175 lines
4.3 KiB
JavaScript
import { ajax } from "discourse/lib/ajax";
|
|
import AsyncReport from "admin/mixins/async-report";
|
|
import Report from "admin/models/report";
|
|
import { number } from 'discourse/lib/formatter';
|
|
import loadScript from "discourse/lib/load-script";
|
|
import { registerTooltip, unregisterTooltip } from "discourse/lib/tooltip";
|
|
|
|
function collapseWeekly(data, average) {
|
|
let aggregate = [];
|
|
let bucket, i;
|
|
let offset = data.length % 7;
|
|
for(i = offset; i < data.length; i++) {
|
|
|
|
if (bucket && (i % 7 === offset)) {
|
|
if (average) {
|
|
bucket.y = parseFloat((bucket.y / 7.0).toFixed(2));
|
|
}
|
|
aggregate.push(bucket);
|
|
bucket = null;
|
|
}
|
|
|
|
bucket = bucket || { x: data[i].x, y: 0 };
|
|
bucket.y += data[i].y;
|
|
}
|
|
return aggregate;
|
|
}
|
|
|
|
export default Ember.Component.extend(AsyncReport, {
|
|
classNames: ["chart", "dashboard-mini-chart"],
|
|
total: 0,
|
|
|
|
init() {
|
|
this._super();
|
|
|
|
this._colorsPool = ["rgb(0,136,204)", "rgb(235,83,148)"];
|
|
},
|
|
|
|
didRender() {
|
|
this._super();
|
|
registerTooltip($(this.element).find("[data-tooltip]"));
|
|
},
|
|
|
|
willDestroyElement() {
|
|
this._super();
|
|
unregisterTooltip($(this.element).find("[data-tooltip]"));
|
|
},
|
|
|
|
pickColorAtIndex(index) {
|
|
return this._colorsPool[index] || this._colorsPool[0];
|
|
},
|
|
|
|
fetchReport() {
|
|
this._super();
|
|
|
|
let payload = {
|
|
data: { cache: true, facets: ["prev_period"] }
|
|
};
|
|
|
|
if (this.get("startDate")) {
|
|
payload.data.start_date = this.get("startDate").locale('en').format('YYYY-MM-DD[T]HH:mm:ss.SSSZZ');
|
|
}
|
|
|
|
if (this.get("endDate")) {
|
|
payload.data.end_date = this.get("endDate").locale('en').format('YYYY-MM-DD[T]HH:mm:ss.SSSZZ');
|
|
}
|
|
|
|
if (this._chart) {
|
|
this._chart.destroy();
|
|
this._chart = null;
|
|
}
|
|
|
|
return Ember.RSVP.Promise.all(this.get("dataSources").map(dataSource => {
|
|
return ajax(dataSource, payload)
|
|
.then(response => {
|
|
this.get("reports").pushObject(this.loadReport(response.report));
|
|
});
|
|
}));
|
|
},
|
|
|
|
loadReport(report, previousReport) {
|
|
Report.fillMissingDates(report);
|
|
|
|
if (report.data && report.data.length > 40) {
|
|
report.data = collapseWeekly(report.data, report.average);
|
|
}
|
|
|
|
if (previousReport && previousReport.color.length) {
|
|
report.color = previousReport.color;
|
|
} else {
|
|
const dataSourceNameIndex = this.get("dataSourceNames").split(",").indexOf(report.type);
|
|
report.color = this.pickColorAtIndex(dataSourceNameIndex);
|
|
}
|
|
|
|
return Report.create(report);
|
|
},
|
|
|
|
renderReport() {
|
|
this._super();
|
|
|
|
Ember.run.schedule("afterRender", () => {
|
|
const $chartCanvas = this.$(".chart-canvas");
|
|
if (!$chartCanvas.length) return;
|
|
const context = $chartCanvas[0].getContext("2d");
|
|
|
|
const reportsForPeriod = this.get("reportsForPeriod");
|
|
|
|
const labels = Ember.makeArray(reportsForPeriod.get("firstObject.data")).map(d => d.x);
|
|
|
|
const data = {
|
|
labels,
|
|
datasets: reportsForPeriod.map(report => {
|
|
return {
|
|
data: Ember.makeArray(report.data).map(d => d.y),
|
|
backgroundColor: "rgba(200,220,240,0.3)",
|
|
borderColor: report.color
|
|
};
|
|
})
|
|
};
|
|
|
|
if (this._chart) {
|
|
this._chart.destroy();
|
|
this._chart = null;
|
|
}
|
|
|
|
loadScript("/javascripts/Chart.min.js").then(() => {
|
|
if (this._chart) {
|
|
this._chart.destroy();
|
|
}
|
|
this._chart = new window.Chart(context, this._buildChartConfig(data));
|
|
});
|
|
});
|
|
},
|
|
|
|
_buildChartConfig(data) {
|
|
return {
|
|
type: "line",
|
|
data,
|
|
options: {
|
|
tooltips: {
|
|
callbacks: {
|
|
title: (context) => moment(context[0].xLabel, "YYYY-MM-DD").format("LL")
|
|
}
|
|
},
|
|
legend: {
|
|
display: false
|
|
},
|
|
responsive: true,
|
|
maintainAspectRatio: false,
|
|
layout: {
|
|
padding: {
|
|
left: 0,
|
|
top: 0,
|
|
right: 0,
|
|
bottom: 0
|
|
}
|
|
},
|
|
scales: {
|
|
yAxes: [{
|
|
display: true,
|
|
ticks: { callback: (label) => number(label) }
|
|
}],
|
|
xAxes: [{
|
|
display: true,
|
|
gridLines: { display: false },
|
|
type: "time",
|
|
time: {
|
|
parser: "YYYY-MM-DD"
|
|
}
|
|
}],
|
|
}
|
|
},
|
|
};
|
|
}
|
|
});
|