let util = require("../../../../util"); let Node = require("../Node").default; /** * A Cluster is a special Node that allows a group of Nodes positioned closely together * to be represented by a single Cluster Node. * * @extends Node */ class Cluster extends Node { /** * @param {Object} options * @param {Object} body * @param {Array.}imagelist * @param {Array} grouplist * @param {Object} globalOptions * @param {Object} defaultOptions Global default options for nodes */ constructor(options, body, imagelist, grouplist, globalOptions, defaultOptions) { super(options, body, imagelist, grouplist, globalOptions, defaultOptions); this.isCluster = true; this.containedNodes = {}; this.containedEdges = {}; } /** * Transfer child cluster data to current and disconnect the child cluster. * * Please consult the header comment in 'Clustering.js' for the fields set here. * * @param {string|number} childClusterId id of child cluster to open */ _openChildCluster(childClusterId) { let childCluster = this.body.nodes[childClusterId]; if (this.containedNodes[childClusterId] === undefined) { throw new Error('node with id: ' + childClusterId + ' not in current cluster'); } if (!childCluster.isCluster) { throw new Error('node with id: ' + childClusterId + ' is not a cluster'); } // Disconnect child cluster from current cluster delete this.containedNodes[childClusterId]; util.forEach(childCluster.edges, (edge) => { delete this.containedEdges[edge.id]; }); // Transfer nodes and edges util.forEach(childCluster.containedNodes, (node, nodeId) => { this.containedNodes[nodeId] = node; }); childCluster.containedNodes = {}; util.forEach(childCluster.containedEdges, (edge, edgeId) => { this.containedEdges[edgeId] = edge; }); childCluster.containedEdges = {}; // Transfer edges within cluster edges which are clustered util.forEach(childCluster.edges, (clusterEdge) => { util.forEach(this.edges, (parentClusterEdge) => { // Assumption: a clustered edge can only be present in a single clustering edge // Not tested here let index = parentClusterEdge.clusteringEdgeReplacingIds.indexOf(clusterEdge.id); if (index === -1) return; util.forEach(clusterEdge.clusteringEdgeReplacingIds, (srcId) => { parentClusterEdge.clusteringEdgeReplacingIds.push(srcId); // Maintain correct bookkeeping for transferred edge this.body.edges[srcId].edgeReplacedById = parentClusterEdge.id; }); // Remove cluster edge from parent cluster edge parentClusterEdge.clusteringEdgeReplacingIds.splice(index, 1); }); }); childCluster.edges = []; } } export default Cluster;