mirror of
https://github.com/chartjs/Chart.js.git
synced 2026-03-05 07:54:04 +01:00
* Update dependencies * Fix privacy problems * color lib & consistency * Review update * getLabelCapacity * Review update
177 lines
3.8 KiB
JavaScript
177 lines
3.8 KiB
JavaScript
import DatasetController from '../core/core.datasetController';
|
|
import defaults from '../core/core.defaults';
|
|
import Point from '../elements/element.point';
|
|
import {resolve} from '../helpers/helpers.options';
|
|
|
|
defaults.set('bubble', {
|
|
animation: {
|
|
numbers: {
|
|
properties: ['x', 'y', 'borderWidth', 'radius']
|
|
}
|
|
},
|
|
scales: {
|
|
x: {
|
|
type: 'linear',
|
|
position: 'bottom'
|
|
},
|
|
y: {
|
|
type: 'linear',
|
|
position: 'left'
|
|
}
|
|
},
|
|
|
|
tooltips: {
|
|
callbacks: {
|
|
title() {
|
|
// Title doesn't make sense for scatter since we format the data as a point
|
|
return '';
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
export default class BubbleController extends DatasetController {
|
|
|
|
/**
|
|
* Parse array of objects
|
|
* @protected
|
|
*/
|
|
parseObjectData(meta, data, start, count) {
|
|
const {xScale, yScale} = meta;
|
|
const parsed = [];
|
|
let i, ilen, item;
|
|
for (i = start, ilen = start + count; i < ilen; ++i) {
|
|
item = data[i];
|
|
parsed.push({
|
|
x: xScale.parseObject(item, 'x', i),
|
|
y: yScale.parseObject(item, 'y', i),
|
|
_custom: item && item.r && +item.r
|
|
});
|
|
}
|
|
return parsed;
|
|
}
|
|
|
|
/**
|
|
* @protected
|
|
*/
|
|
getMaxOverflow() {
|
|
const me = this;
|
|
const meta = me._cachedMeta;
|
|
let i = (meta.data || []).length - 1;
|
|
let max = 0;
|
|
for (; i >= 0; --i) {
|
|
max = Math.max(max, me.getStyle(i, true).radius);
|
|
}
|
|
return max > 0 && max;
|
|
}
|
|
|
|
/**
|
|
* @protected
|
|
*/
|
|
getLabelAndValue(index) {
|
|
const me = this;
|
|
const meta = me._cachedMeta;
|
|
const {xScale, yScale} = meta;
|
|
const parsed = me.getParsed(index);
|
|
const x = xScale.getLabelForValue(parsed.x);
|
|
const y = yScale.getLabelForValue(parsed.y);
|
|
const r = parsed._custom;
|
|
|
|
return {
|
|
label: meta.label,
|
|
value: '(' + x + ', ' + y + (r ? ', ' + r : '') + ')'
|
|
};
|
|
}
|
|
|
|
update(mode) {
|
|
const me = this;
|
|
const points = me._cachedMeta.data;
|
|
|
|
// Update Points
|
|
me.updateElements(points, 0, mode);
|
|
}
|
|
|
|
updateElements(points, start, mode) {
|
|
const me = this;
|
|
const reset = mode === 'reset';
|
|
const {xScale, yScale} = me._cachedMeta;
|
|
const firstOpts = me.resolveDataElementOptions(start, mode);
|
|
const sharedOptions = me.getSharedOptions(mode, points[start], firstOpts);
|
|
const includeOptions = me.includeOptions(mode, sharedOptions);
|
|
|
|
for (let i = 0; i < points.length; i++) {
|
|
const point = points[i];
|
|
const index = start + i;
|
|
const parsed = !reset && me.getParsed(index);
|
|
const x = reset ? xScale.getPixelForDecimal(0.5) : xScale.getPixelForValue(parsed.x);
|
|
const y = reset ? yScale.getBasePixel() : yScale.getPixelForValue(parsed.y);
|
|
const properties = {
|
|
x,
|
|
y,
|
|
skip: isNaN(x) || isNaN(y)
|
|
};
|
|
|
|
if (includeOptions) {
|
|
properties.options = me.resolveDataElementOptions(i, mode);
|
|
|
|
if (reset) {
|
|
properties.options.radius = 0;
|
|
}
|
|
}
|
|
|
|
me.updateElement(point, index, properties, mode);
|
|
}
|
|
|
|
me.updateSharedOptions(sharedOptions, mode);
|
|
}
|
|
|
|
/**
|
|
* @protected
|
|
*/
|
|
resolveDataElementOptions(index, mode) {
|
|
const me = this;
|
|
const chart = me.chart;
|
|
const dataset = me.getDataset();
|
|
const parsed = me.getParsed(index);
|
|
let values = super.resolveDataElementOptions(index, mode);
|
|
|
|
// Scriptable options
|
|
const context = {
|
|
chart,
|
|
dataIndex: index,
|
|
dataset,
|
|
datasetIndex: me.index
|
|
};
|
|
|
|
// In case values were cached (and thus frozen), we need to clone the values
|
|
if (values.$shared) {
|
|
values = Object.assign({}, values, {$shared: false});
|
|
}
|
|
|
|
|
|
// Custom radius resolution
|
|
if (mode !== 'active') {
|
|
values.radius = 0;
|
|
}
|
|
values.radius += resolve([
|
|
parsed && parsed._custom,
|
|
me._config.radius,
|
|
chart.options.elements.point.radius
|
|
], context, index);
|
|
|
|
return values;
|
|
}
|
|
}
|
|
|
|
BubbleController.prototype.dataElementType = Point;
|
|
|
|
BubbleController.prototype.dataElementOptions = [
|
|
'backgroundColor',
|
|
'borderColor',
|
|
'borderWidth',
|
|
'hitRadius',
|
|
'radius',
|
|
'pointStyle',
|
|
'rotation'
|
|
];
|