1
0
Fork 0
arangodb/html/admin/js/graphViewer/graph/forceLayouter.js

155 lines
4.1 KiB
JavaScript

/*jslint indent: 2, nomen: true, maxlen: 100, white: true plusplus: true */
/*global _, d3*/
////////////////////////////////////////////////////////////////////////////////
/// @brief Graph functionality
///
/// @file
///
/// DISCLAIMER
///
/// Copyright 2010-2012 triagens GmbH, Cologne, Germany
///
/// Licensed under the Apache License, Version 2.0 (the "License");
/// you may not use this file except in compliance with the License.
/// You may obtain a copy of the License at
///
/// http://www.apache.org/licenses/LICENSE-2.0
///
/// Unless required by applicable law or agreed to in writing, software
/// distributed under the License is distributed on an "AS IS" BASIS,
/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
/// See the License for the specific language governing permissions and
/// limitations under the License.
///
/// Copyright holder is triAGENS GmbH, Cologne, Germany
///
/// @author Michael Hackstein
/// @author Copyright 2011-2013, triAGENS GmbH, Cologne, Germany
////////////////////////////////////////////////////////////////////////////////
/*
* example config:
* {
* nodes: nodes,
* links: links,
*
* (optional)
* width: width,
* height: height,
* distance: distance,
* gravity: gravity,
* onUpdate: callback
* }
*/
function ForceLayouter(config) {
"use strict";
var self = this,
force = d3.layout.force(),
/*
distance = config.distance || 240, // 80
gravity = config.gravity || 0.01, // 0.08
charge = config.charge || -1000, // -240
*/
defaultCharge = config.charge || -600,
defaultDistance = config.distance || 80,
gravity = config.gravity || 0.01,
distance = function(d) {
var res = 0;
if (d.source._isCommunity) {
res += d.source.getDistance(defaultDistance);
} else {
res += defaultDistance;
}
if (d.target._isCommunity) {
res += d.target.getDistance(defaultDistance);
} else {
res += defaultDistance;
}
return res;
},
charge = function(d) {
if (d._isCommunity) {
return d.getCharge(defaultCharge);
}
return defaultCharge;
},
onUpdate = config.onUpdate || function () {},
width = config.width || 880,
height = config.height || 680,
parseConfig = function(config) {
if (config.distance) {
defaultDistance = config.distance;
}
if (config.gravity) {
force.gravity(config.gravity);
}
if (config.charge) {
defaultCharge = config.charge;
}
};
if (config.nodes === undefined) {
throw "No nodes defined";
}
if (config.links === undefined) {
throw "No links defined";
}
// Set up the force
force.nodes(config.nodes); // Set nodes
force.links(config.links); // Set edges
force.size([width, height]); // Set width and height
force.linkDistance(distance); // Set distance between nodes
force.gravity(gravity); // Set gravity
force.charge(charge); // Set charge
force.on("tick", function(){}); // Bind tick function
self.start = function() {
force.start(); // Start Force computation
};
self.stop = function() {
force.stop(); // Stop Force computation
};
self.drag = force.drag;
self.setCombinedUpdateFunction = function(nodeShaper, edgeShaper, additional) {
if (additional !== undefined) {
onUpdate = function() {
if (force.alpha() < 0.1) {
nodeShaper.updateNodes();
edgeShaper.updateEdges();
additional();
if (force.alpha() < 0.05) {
self.stop();
}
}
};
force.on("tick", onUpdate);
} else {
onUpdate = function() {
if (force.alpha() < 0.1) {
nodeShaper.updateNodes();
edgeShaper.updateEdges();
if (force.alpha() < 0.05) {
self.stop();
}
}
};
force.on("tick", onUpdate);
}
};
self.changeTo = function(config) {
parseConfig(config);
};
self.changeWidth = function(w) {
width = w;
force.size([width, height]); // Set width and height
};
}