mirror of
https://github.com/chartjs/Chart.js.git
synced 2026-03-05 16:04:03 +01:00
* Add support for description in fixtures * Update datasetController to draw active(s) last * Handle hiding of line by borderWidth * Disable filling of scatter charts by default * Make radar chart consistent with line
160 lines
4.1 KiB
JavaScript
160 lines
4.1 KiB
JavaScript
import DatasetController from '../core/core.datasetController';
|
|
import {valueOrDefault} from '../helpers/helpers.core';
|
|
import {isNumber} from '../helpers/helpers.math';
|
|
import {resolve} from '../helpers/helpers.options';
|
|
|
|
export default class LineController extends DatasetController {
|
|
|
|
update(mode) {
|
|
const me = this;
|
|
const meta = me._cachedMeta;
|
|
const line = meta.dataset;
|
|
const points = meta.data || [];
|
|
|
|
// Update Line
|
|
// In resize mode only point locations change, so no need to set the points or options.
|
|
if (mode !== 'resize') {
|
|
const properties = {
|
|
points,
|
|
options: me.resolveDatasetElementOptions()
|
|
};
|
|
|
|
me.updateElement(line, undefined, properties, mode);
|
|
}
|
|
|
|
// Update Points
|
|
me.updateElements(points, 0, mode);
|
|
}
|
|
|
|
updateElements(points, start, mode) {
|
|
const me = this;
|
|
const reset = mode === 'reset';
|
|
const {xScale, yScale, _stacked} = me._cachedMeta;
|
|
const firstOpts = me.resolveDataElementOptions(start, mode);
|
|
const sharedOptions = me.getSharedOptions(mode, points[start], firstOpts);
|
|
const includeOptions = me.includeOptions(mode, sharedOptions);
|
|
const spanGaps = valueOrDefault(me._config.spanGaps, me.chart.options.spanGaps);
|
|
const maxGapLength = isNumber(spanGaps) ? spanGaps : Number.POSITIVE_INFINITY;
|
|
let prevParsed;
|
|
|
|
for (let i = 0; i < points.length; ++i) {
|
|
const index = start + i;
|
|
const point = points[i];
|
|
const parsed = me.getParsed(index);
|
|
const x = xScale.getPixelForValue(parsed.x, index);
|
|
const y = reset ? yScale.getBasePixel() : yScale.getPixelForValue(_stacked ? me.applyStack(yScale, parsed) : parsed.y, index);
|
|
const properties = {
|
|
x,
|
|
y,
|
|
skip: isNaN(x) || isNaN(y),
|
|
stop: i > 0 && (parsed.x - prevParsed.x) > maxGapLength
|
|
};
|
|
|
|
if (includeOptions) {
|
|
properties.options = me.resolveDataElementOptions(index, mode);
|
|
}
|
|
|
|
me.updateElement(point, index, properties, mode);
|
|
|
|
prevParsed = parsed;
|
|
}
|
|
|
|
me.updateSharedOptions(sharedOptions, mode);
|
|
}
|
|
|
|
/**
|
|
* @param {boolean} [active]
|
|
* @protected
|
|
*/
|
|
resolveDatasetElementOptions(active) {
|
|
const me = this;
|
|
const config = me._config;
|
|
const options = me.chart.options;
|
|
const lineOptions = options.elements.line;
|
|
const values = super.resolveDatasetElementOptions(active);
|
|
const showLine = valueOrDefault(config.showLine, options.showLines);
|
|
|
|
// The default behavior of lines is to break at null values, according
|
|
// to https://github.com/chartjs/Chart.js/issues/2435#issuecomment-216718158
|
|
// This option gives lines the ability to span gaps
|
|
values.spanGaps = valueOrDefault(config.spanGaps, options.spanGaps);
|
|
values.tension = valueOrDefault(config.lineTension, lineOptions.tension);
|
|
values.stepped = resolve([config.stepped, lineOptions.stepped]);
|
|
|
|
if (!showLine) {
|
|
values.borderWidth = 0;
|
|
}
|
|
|
|
return values;
|
|
}
|
|
|
|
/**
|
|
* @protected
|
|
*/
|
|
getMaxOverflow() {
|
|
const me = this;
|
|
const meta = me._cachedMeta;
|
|
const border = meta.dataset.options.borderWidth || 0;
|
|
const data = meta.data || [];
|
|
if (!data.length) {
|
|
return border;
|
|
}
|
|
const firstPoint = data[0].size();
|
|
const lastPoint = data[data.length - 1].size();
|
|
return Math.max(border, firstPoint, lastPoint) / 2;
|
|
}
|
|
}
|
|
|
|
LineController.id = 'line';
|
|
|
|
/**
|
|
* @type {any}
|
|
*/
|
|
LineController.defaults = {
|
|
datasetElementType: 'line',
|
|
datasetElementOptions: [
|
|
'backgroundColor',
|
|
'borderCapStyle',
|
|
'borderColor',
|
|
'borderDash',
|
|
'borderDashOffset',
|
|
'borderJoinStyle',
|
|
'borderWidth',
|
|
'capBezierPoints',
|
|
'cubicInterpolationMode',
|
|
'fill'
|
|
],
|
|
|
|
dataElementType: 'point',
|
|
dataElementOptions: {
|
|
backgroundColor: 'pointBackgroundColor',
|
|
borderColor: 'pointBorderColor',
|
|
borderWidth: 'pointBorderWidth',
|
|
hitRadius: 'pointHitRadius',
|
|
hoverHitRadius: 'pointHitRadius',
|
|
hoverBackgroundColor: 'pointHoverBackgroundColor',
|
|
hoverBorderColor: 'pointHoverBorderColor',
|
|
hoverBorderWidth: 'pointHoverBorderWidth',
|
|
hoverRadius: 'pointHoverRadius',
|
|
pointStyle: 'pointStyle',
|
|
radius: 'pointRadius',
|
|
rotation: 'pointRotation'
|
|
},
|
|
|
|
showLines: true,
|
|
spanGaps: false,
|
|
|
|
hover: {
|
|
mode: 'index'
|
|
},
|
|
|
|
scales: {
|
|
_index_: {
|
|
type: 'category',
|
|
},
|
|
_value_: {
|
|
type: 'linear',
|
|
},
|
|
}
|
|
};
|