213 lines
9.3 KiB
JavaScript
213 lines
9.3 KiB
JavaScript
|
/* *
|
||
|
*
|
||
|
* (c) 2010-2024 Torstein Honsi
|
||
|
*
|
||
|
* License: www.highcharts.com/license
|
||
|
*
|
||
|
* !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
|
||
|
*
|
||
|
* */
|
||
|
'use strict';
|
||
|
import ColumnSeries from './Column/ColumnSeries.js';
|
||
|
import H from '../Core/Globals.js';
|
||
|
const { composed } = H;
|
||
|
const { prototype: columnProto } = ColumnSeries;
|
||
|
import Series from '../Core/Series/Series.js';
|
||
|
const { prototype: seriesProto } = Series;
|
||
|
import U from '../Core/Utilities.js';
|
||
|
const { defined, pushUnique, stableSort } = U;
|
||
|
/* *
|
||
|
*
|
||
|
* Composition
|
||
|
*
|
||
|
* */
|
||
|
var OnSeriesComposition;
|
||
|
(function (OnSeriesComposition) {
|
||
|
/* *
|
||
|
*
|
||
|
* Declarations
|
||
|
*
|
||
|
* */
|
||
|
/* *
|
||
|
*
|
||
|
* Functions
|
||
|
*
|
||
|
* */
|
||
|
/**
|
||
|
* @private
|
||
|
*/
|
||
|
function compose(SeriesClass) {
|
||
|
if (pushUnique(composed, 'OnSeries')) {
|
||
|
const seriesProto = SeriesClass.prototype;
|
||
|
seriesProto.getPlotBox = getPlotBox;
|
||
|
seriesProto.translate = translate;
|
||
|
}
|
||
|
return SeriesClass;
|
||
|
}
|
||
|
OnSeriesComposition.compose = compose;
|
||
|
/**
|
||
|
* Override getPlotBox. If the onSeries option is valid, return the plot box
|
||
|
* of the onSeries, otherwise proceed as usual.
|
||
|
*
|
||
|
* @private
|
||
|
*/
|
||
|
function getPlotBox(name) {
|
||
|
return seriesProto.getPlotBox.call((this.options.onSeries &&
|
||
|
this.chart.get(this.options.onSeries)) || this, name);
|
||
|
}
|
||
|
OnSeriesComposition.getPlotBox = getPlotBox;
|
||
|
/**
|
||
|
* Extend the translate method by placing the point on the related series
|
||
|
*
|
||
|
* @private
|
||
|
*/
|
||
|
function translate() {
|
||
|
columnProto.translate.apply(this);
|
||
|
const series = this, options = series.options, chart = series.chart, points = series.points, optionsOnSeries = options.onSeries, onSeries = (optionsOnSeries &&
|
||
|
chart.get(optionsOnSeries)), step = onSeries && onSeries.options.step, onData = (onSeries && onSeries.points), inverted = chart.inverted, xAxis = series.xAxis, yAxis = series.yAxis;
|
||
|
let cursor = points.length - 1, point, lastPoint, onKey = options.onKey || 'y', i = onData && onData.length, xOffset = 0, leftPoint, lastX, rightPoint, currentDataGrouping, distanceRatio;
|
||
|
// Relate to a master series
|
||
|
if (onSeries && onSeries.visible && i) {
|
||
|
xOffset = (onSeries.pointXOffset || 0) + (onSeries.barW || 0) / 2;
|
||
|
currentDataGrouping = onSeries.currentDataGrouping;
|
||
|
lastX = (onData[i - 1].x +
|
||
|
(currentDataGrouping ? currentDataGrouping.totalRange : 0)); // #2374
|
||
|
// sort the data points
|
||
|
stableSort(points, (a, b) => (a.x - b.x));
|
||
|
onKey = 'plot' + onKey[0].toUpperCase() + onKey.substr(1);
|
||
|
while (i-- && points[cursor]) {
|
||
|
leftPoint = onData[i];
|
||
|
point = points[cursor];
|
||
|
point.y = leftPoint.y;
|
||
|
if (leftPoint.x <= point.x &&
|
||
|
typeof leftPoint[onKey] !== 'undefined') {
|
||
|
if (point.x <= lastX) { // #803
|
||
|
point.plotY = leftPoint[onKey];
|
||
|
// Interpolate between points, #666
|
||
|
if (leftPoint.x < point.x &&
|
||
|
!step) {
|
||
|
rightPoint = onData[i + 1];
|
||
|
if (rightPoint &&
|
||
|
typeof rightPoint[onKey] !== 'undefined') {
|
||
|
// If the series is spline, calculate Y of the
|
||
|
// point on the bezier line. #19264
|
||
|
if (defined(point.plotX) &&
|
||
|
onSeries.is('spline')) {
|
||
|
leftPoint = leftPoint;
|
||
|
rightPoint = rightPoint;
|
||
|
const p0 = [
|
||
|
leftPoint.plotX || 0,
|
||
|
leftPoint.plotY || 0
|
||
|
], p3 = [
|
||
|
rightPoint.plotX || 0,
|
||
|
rightPoint.plotY || 0
|
||
|
], p1 = (leftPoint.controlPoints?.high ||
|
||
|
p0), p2 = (rightPoint.controlPoints?.low ||
|
||
|
p3), pixelThreshold = 0.25, maxIterations = 100, calculateCoord = (t, key) => (
|
||
|
// The parametric formula for the
|
||
|
// cubic Bezier curve.
|
||
|
Math.pow(1 - t, 3) * p0[key] +
|
||
|
3 * (1 - t) * (1 - t) * t *
|
||
|
p1[key] + 3 * (1 - t) * t * t *
|
||
|
p2[key] + t * t * t * p3[key]);
|
||
|
let tMin = 0, tMax = 1, t;
|
||
|
// Find `t` of the parametric function of
|
||
|
// the bezier curve for the given `plotX`.
|
||
|
for (let i = 0; i < maxIterations; i++) {
|
||
|
const tMid = (tMin + tMax) / 2;
|
||
|
const xMid = calculateCoord(tMid, 0);
|
||
|
if (xMid === null) {
|
||
|
break;
|
||
|
}
|
||
|
if (Math.abs(xMid - point.plotX) < pixelThreshold) {
|
||
|
t = tMid;
|
||
|
break;
|
||
|
}
|
||
|
if (xMid < point.plotX) {
|
||
|
tMin = tMid;
|
||
|
}
|
||
|
else {
|
||
|
tMax = tMid;
|
||
|
}
|
||
|
}
|
||
|
if (defined(t)) {
|
||
|
point.plotY =
|
||
|
calculateCoord(t, 1);
|
||
|
point.y =
|
||
|
yAxis.toValue(point.plotY, true);
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
// The distance ratio, between 0 and 1
|
||
|
distanceRatio =
|
||
|
(point.x - leftPoint.x) /
|
||
|
(rightPoint.x - leftPoint.x);
|
||
|
point.plotY +=
|
||
|
distanceRatio *
|
||
|
// The plotY distance
|
||
|
(rightPoint[onKey] - leftPoint[onKey]);
|
||
|
point.y +=
|
||
|
distanceRatio *
|
||
|
(rightPoint.y - leftPoint.y);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
cursor--;
|
||
|
i++; // Check again for points in the same x position
|
||
|
if (cursor < 0) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
// Add plotY position and handle stacking
|
||
|
points.forEach((point, i) => {
|
||
|
let stackIndex;
|
||
|
point.plotX += xOffset; // #2049
|
||
|
// Undefined plotY means the point is either on axis, outside series
|
||
|
// range or hidden series. If the series is outside the range of the
|
||
|
// x axis it should fall through with an undefined plotY, but then
|
||
|
// we must remove the shapeArgs (#847). For inverted charts, we need
|
||
|
// to calculate position anyway, because series.invertGroups is not
|
||
|
// defined
|
||
|
if (typeof point.plotY === 'undefined' || inverted) {
|
||
|
if (point.plotX >= 0 &&
|
||
|
point.plotX <= xAxis.len) {
|
||
|
// We're inside xAxis range
|
||
|
if (inverted) {
|
||
|
point.plotY = xAxis.translate(point.x, 0, 1, 0, 1);
|
||
|
point.plotX = defined(point.y) ?
|
||
|
yAxis.translate(point.y, 0, 0, 0, 1) :
|
||
|
0;
|
||
|
}
|
||
|
else {
|
||
|
point.plotY = (xAxis.opposite ? 0 : series.yAxis.len) +
|
||
|
xAxis.offset; // For the windbarb demo
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
point.shapeArgs = {}; // 847
|
||
|
}
|
||
|
}
|
||
|
// If multiple flags appear at the same x, order them into a stack
|
||
|
lastPoint = points[i - 1];
|
||
|
if (lastPoint && lastPoint.plotX === point.plotX) {
|
||
|
if (typeof lastPoint.stackIndex === 'undefined') {
|
||
|
lastPoint.stackIndex = 0;
|
||
|
}
|
||
|
stackIndex = lastPoint.stackIndex + 1;
|
||
|
}
|
||
|
point.stackIndex = stackIndex; // #3639
|
||
|
});
|
||
|
this.onSeries = onSeries;
|
||
|
}
|
||
|
OnSeriesComposition.translate = translate;
|
||
|
})(OnSeriesComposition || (OnSeriesComposition = {}));
|
||
|
/* *
|
||
|
*
|
||
|
* Default Export
|
||
|
*
|
||
|
* */
|
||
|
export default OnSeriesComposition;
|