2019-12-19 18:31:25 +01:00
|
|
|
/*
|
|
|
|
* μlogger
|
|
|
|
*
|
|
|
|
* Copyright(C) 2019 Bartek Fabiszewski (www.fabiszewski.net)
|
|
|
|
*
|
|
|
|
* This is free software; you can redistribute it and/or modify it under
|
|
|
|
* the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2019-12-19 22:20:55 +01:00
|
|
|
import { lang as $ } from './initializer.js';
|
2019-12-19 18:31:25 +01:00
|
|
|
import Chartist from 'chartist'
|
|
|
|
import ViewModel from './viewmodel.js';
|
|
|
|
import ctAxisTitle from 'chartist-plugin-axistitle';
|
2020-04-14 19:54:28 +02:00
|
|
|
import uObserve from './observe.js';
|
2019-12-19 18:31:25 +01:00
|
|
|
import uUtils from './utils.js';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {Object} PlotPoint
|
|
|
|
* @property {number} x
|
|
|
|
* @property {number} y
|
|
|
|
*/
|
|
|
|
/**
|
|
|
|
* @typedef {PlotPoint[]} PlotData
|
|
|
|
*/
|
|
|
|
|
|
|
|
// FIXME: Chartist is not suitable for large data sets
|
|
|
|
const LARGE_DATA = 1000;
|
|
|
|
export default class ChartViewModel extends ViewModel {
|
|
|
|
/**
|
|
|
|
* @param {uState} state
|
|
|
|
*/
|
|
|
|
constructor(state) {
|
|
|
|
super({
|
|
|
|
pointSelected: null,
|
|
|
|
chartVisible: false,
|
|
|
|
buttonVisible: false,
|
2020-01-30 23:15:25 +01:00
|
|
|
onChartToggle: null,
|
|
|
|
onMenuToggle: null
|
2019-12-19 18:31:25 +01:00
|
|
|
});
|
|
|
|
this.state = state;
|
|
|
|
/** @type {PlotData} */
|
|
|
|
this.data = [];
|
|
|
|
/** @type {?Chartist.Line} */
|
|
|
|
this.chart = null;
|
|
|
|
/** @type {?NodeListOf<SVGLineElement>} */
|
|
|
|
this.chartPoints = null;
|
|
|
|
/** @type {HTMLDivElement} */
|
|
|
|
this.chartElement = document.querySelector('#chart');
|
|
|
|
/** @type {HTMLDivElement} */
|
|
|
|
this.chartContainer = this.chartElement.parentElement;
|
|
|
|
/** @type {HTMLAnchorElement} */
|
|
|
|
this.buttonElement = document.querySelector('#altitudes');
|
|
|
|
}
|
|
|
|
|
2019-12-22 20:00:23 +01:00
|
|
|
/**
|
|
|
|
* @return {ChartViewModel}
|
|
|
|
*/
|
2019-12-19 18:31:25 +01:00
|
|
|
init() {
|
|
|
|
this.chartSetup();
|
|
|
|
this.setObservers();
|
|
|
|
this.bindAll();
|
2019-12-22 20:00:23 +01:00
|
|
|
return this;
|
2019-12-19 18:31:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
chartSetup() {
|
|
|
|
uUtils.addCss('css/chartist.min.css', 'chartist_css');
|
|
|
|
this.chart = new Chartist.Line(this.chartElement, {
|
|
|
|
series: [ this.data ]
|
|
|
|
}, {
|
|
|
|
lineSmooth: true,
|
|
|
|
showArea: true,
|
|
|
|
axisX: {
|
|
|
|
type: Chartist.AutoScaleAxis,
|
|
|
|
onlyInteger: true,
|
|
|
|
showLabel: false
|
|
|
|
},
|
|
|
|
plugins: [
|
|
|
|
ctAxisTitle({
|
|
|
|
axisY: {
|
2019-12-19 22:20:55 +01:00
|
|
|
axisTitle: `${$._('altitude')} (${$.unit('unitDistance')})`,
|
2019-12-19 18:31:25 +01:00
|
|
|
axisClass: 'ct-axis-title',
|
|
|
|
offset: {
|
|
|
|
x: 0,
|
|
|
|
y: 11
|
|
|
|
},
|
|
|
|
textAnchor: 'middle',
|
|
|
|
flipTitle: true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
]
|
|
|
|
});
|
|
|
|
|
|
|
|
this.chart.on('created', () => this.onCreated());
|
|
|
|
}
|
|
|
|
|
|
|
|
onCreated() {
|
|
|
|
if (this.data.length && this.data.length <= LARGE_DATA) {
|
|
|
|
this.chartPoints = document.querySelectorAll('.ct-series .ct-point');
|
|
|
|
const len = this.chartPoints.length;
|
|
|
|
for (let id = 0; id < len; id++) {
|
|
|
|
this.chartPoints[id].addEventListener('click', () => {
|
|
|
|
this.model.pointSelected = id;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
setObservers() {
|
|
|
|
this.state.onChanged('currentTrack', (track) => {
|
2020-04-14 19:54:28 +02:00
|
|
|
if (track) {
|
|
|
|
uObserve.observe(track, 'positions', () => {
|
|
|
|
this.onTrackUpdate(track, true);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
this.onTrackUpdate(track);
|
2019-12-19 18:31:25 +01:00
|
|
|
});
|
|
|
|
this.onChanged('buttonVisible', (visible) => this.renderButton(visible));
|
|
|
|
this.onChanged('chartVisible', (visible) => this.renderContainer(visible));
|
|
|
|
this.model.onChartToggle = () => {
|
|
|
|
this.model.chartVisible = !this.model.chartVisible;
|
|
|
|
};
|
2020-01-30 23:15:25 +01:00
|
|
|
this.model.onMenuToggle = () => {
|
2020-04-14 21:07:54 +02:00
|
|
|
if (this.model.chartVisible) {
|
|
|
|
this.chart.update();
|
|
|
|
}
|
2020-01-30 23:15:25 +01:00
|
|
|
};
|
2019-12-19 18:31:25 +01:00
|
|
|
}
|
|
|
|
|
2020-04-14 19:54:28 +02:00
|
|
|
/**
|
|
|
|
* @param {?uTrack} track
|
|
|
|
* @param {boolean=} update
|
|
|
|
*/
|
|
|
|
onTrackUpdate(track, update = false) {
|
|
|
|
this.render(track, update);
|
|
|
|
this.model.buttonVisible = !!track && track.hasPlotData;
|
|
|
|
}
|
|
|
|
|
2019-12-19 18:31:25 +01:00
|
|
|
/**
|
|
|
|
* @param {boolean} isVisible
|
|
|
|
*/
|
|
|
|
renderContainer(isVisible) {
|
|
|
|
if (isVisible) {
|
|
|
|
this.chartContainer.style.display = 'block';
|
2020-04-14 19:54:28 +02:00
|
|
|
this.render(this.state.currentTrack);
|
2019-12-19 18:31:25 +01:00
|
|
|
} else {
|
|
|
|
this.chartContainer.style.display = 'none';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {boolean} isVisible
|
|
|
|
*/
|
|
|
|
renderButton(isVisible) {
|
|
|
|
if (isVisible) {
|
|
|
|
this.buttonElement.style.visibility = 'visible';
|
|
|
|
} else {
|
|
|
|
this.buttonElement.style.visibility = 'hidden';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-14 19:54:28 +02:00
|
|
|
/**
|
|
|
|
* @param {?uTrack} track
|
|
|
|
* @param {boolean=} update
|
|
|
|
*/
|
|
|
|
render(track, update = false) {
|
2019-12-19 18:31:25 +01:00
|
|
|
let data = [];
|
2020-04-14 19:54:28 +02:00
|
|
|
if (track && track.hasPlotData && this.model.chartVisible) {
|
|
|
|
data = track.plotData;
|
2019-12-19 18:31:25 +01:00
|
|
|
} else {
|
|
|
|
this.model.chartVisible = false;
|
|
|
|
}
|
2020-04-14 19:54:28 +02:00
|
|
|
if (update || this.data !== data) {
|
|
|
|
console.log(`Chart${update ? ' forced' : ''} update (${data.length})`);
|
2019-12-19 18:31:25 +01:00
|
|
|
this.data = data;
|
|
|
|
const options = {
|
|
|
|
lineSmooth: (data.length <= LARGE_DATA)
|
|
|
|
};
|
|
|
|
this.chart.update({ series: [ data ] }, options, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {number} pointId
|
|
|
|
* @param {string} $className
|
|
|
|
*/
|
|
|
|
pointAddClass(pointId, $className) {
|
|
|
|
if (this.model.chartVisible && this.chartPoints.length > pointId) {
|
|
|
|
const point = this.chartPoints[pointId];
|
|
|
|
point.classList.add($className);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} $className
|
|
|
|
*/
|
|
|
|
pointsRemoveClass($className) {
|
2019-12-22 14:57:25 +01:00
|
|
|
if (this.model.chartVisible && this.chartPoints) {
|
|
|
|
this.chartPoints.forEach((el) => el.classList.remove($className));
|
|
|
|
}
|
2019-12-19 18:31:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {number} pointId
|
|
|
|
*/
|
|
|
|
onPointOver(pointId) {
|
|
|
|
this.pointAddClass(pointId, 'ct-point-hilight');
|
|
|
|
}
|
|
|
|
|
|
|
|
onPointOut() {
|
|
|
|
this.pointsRemoveClass('ct-point-hilight');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {number} pointId
|
|
|
|
*/
|
|
|
|
onPointSelect(pointId) {
|
|
|
|
this.pointAddClass(pointId, 'ct-point-selected');
|
|
|
|
}
|
|
|
|
|
|
|
|
onPointUnselect() {
|
|
|
|
this.pointsRemoveClass('ct-point-selected');
|
|
|
|
}
|
|
|
|
}
|