highcharts
Version:
JavaScript charting framework
181 lines (180 loc) • 5.92 kB
JavaScript
/* *
*
* (c) 2010-2024 Grzegorz Blachlinski, Sebastian Bochan
*
* License: www.highcharts.com/license
*
* !!!!!!! SOURCE GETS TRANSPILED BY TYPESCRIPT. EDIT TS FILE ONLY. !!!!!!!
*
* */
;
import GraphLayout from '../GraphLayoutComposition.js';
import PackedBubbleIntegration from './PackedBubbleIntegration.js';
import ReingoldFruchtermanLayout from '../Networkgraph/ReingoldFruchtermanLayout.js';
import U from '../../Core/Utilities.js';
const { addEvent, pick } = U;
/* *
*
* Functions
*
* */
/**
* @private
*/
function chartGetSelectedParentNodes() {
const allSeries = this.series, selectedParentsNodes = [];
allSeries.forEach((series) => {
if (series.parentNode && series.parentNode.selected) {
selectedParentsNodes.push(series.parentNode);
}
});
return selectedParentsNodes;
}
/**
* Remove accumulated data points to redistribute all of them again
* (i.e after hiding series by legend)
* @private
*/
function onChartBeforeRedraw() {
if (this.allDataPoints) {
delete this.allDataPoints;
}
}
/* *
*
* Class
*
* */
class PackedBubbleLayout extends ReingoldFruchtermanLayout {
constructor() {
/* *
*
* Static Functions
*
* */
super(...arguments);
this.index = NaN;
this.nodes = [];
this.series = [];
}
static compose(ChartClass) {
ReingoldFruchtermanLayout.compose(ChartClass);
GraphLayout.integrations.packedbubble = PackedBubbleIntegration;
GraphLayout.layouts.packedbubble = PackedBubbleLayout;
const chartProto = ChartClass.prototype;
if (!chartProto.getSelectedParentNodes) {
addEvent(ChartClass, 'beforeRedraw', onChartBeforeRedraw);
chartProto.getSelectedParentNodes = chartGetSelectedParentNodes;
}
}
/* *
*
* Functions
*
* */
beforeStep() {
if (this.options.marker) {
this.series.forEach((series) => {
if (series) {
series.calculateParentRadius();
}
});
}
}
// #14439, new stable check.
isStable() {
const tempDiff = Math.abs(this.prevSystemTemperature -
this.systemTemperature);
const upScaledTemperature = 10 * this.systemTemperature /
Math.sqrt(this.nodes.length);
return Math.abs(upScaledTemperature) < 1 &&
tempDiff < 0.00001 ||
this.temperature <= 0;
}
setCircularPositions() {
const layout = this, box = layout.box, nodes = layout.nodes, nodesLength = nodes.length + 1, angle = 2 * Math.PI / nodesLength, radius = layout.options.initialPositionRadius;
let centerX, centerY, index = 0;
for (const node of nodes) {
if (layout.options.splitSeries &&
!node.isParentNode) {
centerX = node.series.parentNode.plotX;
centerY = node.series.parentNode.plotY;
}
else {
centerX = box.width / 2;
centerY = box.height / 2;
}
node.plotX = node.prevX = pick(node.plotX, centerX +
radius * Math.cos(node.index || index * angle));
node.plotY = node.prevY = pick(node.plotY, centerY +
radius * Math.sin(node.index || index * angle));
node.dispX = 0;
node.dispY = 0;
index++;
}
}
repulsiveForces() {
const layout = this, bubblePadding = layout.options.bubblePadding, nodes = layout.nodes;
let force, distanceR, distanceXY;
nodes.forEach((node) => {
node.degree = node.mass;
node.neighbours = 0;
nodes.forEach((repNode) => {
force = 0;
if (
// Node cannot repulse itself:
node !== repNode &&
// Only close nodes affect each other:
// Not dragged:
!node.fixedPosition &&
(layout.options.seriesInteraction ||
node.series === repNode.series)) {
distanceXY = layout.getDistXY(node, repNode);
distanceR = (layout.vectorLength(distanceXY) -
(node.marker.radius +
repNode.marker.radius +
bubblePadding));
// TODO padding configurable
if (distanceR < 0) {
node.degree += 0.01;
node.neighbours++;
force = layout.repulsiveForce(-distanceR / Math.sqrt(node.neighbours), layout.k, node, repNode);
}
layout.force('repulsive', node, force * repNode.mass, distanceXY, repNode, distanceR);
}
});
});
}
applyLimitBox(node, box) {
const layout = this, factor = 0.01;
let distanceXY, distanceR;
// `parentNodeLimit` should be used together with seriesInteraction:
// false
if (layout.options.splitSeries &&
!node.isParentNode &&
layout.options.parentNodeLimit) {
distanceXY = layout.getDistXY(node, node.series.parentNode);
distanceR = (node.series.parentNodeRadius -
node.marker.radius -
layout.vectorLength(distanceXY));
if (distanceR < 0 &&
distanceR > -2 * node.marker.radius) {
node.plotX -= distanceXY.x * factor;
node.plotY -= distanceXY.y * factor;
}
}
super.applyLimitBox(node, box);
}
}
/* *
*
* Registry
*
* */
GraphLayout.layouts.packedbubble = PackedBubbleLayout;
/* *
*
* Default Export
*
* */
export default PackedBubbleLayout;