vis.js is a dynamic, browser-based visualization library

68 lines
1.9 KiB

  1. class HierarchicalRepulsionSolver {
  2. constructor(body, physicsBody, options) {
  3. this.body = body;
  4. this.physicsBody = physicsBody;
  5. this.setOptions(options);
  6. }
  7. setOptions(options) {
  8. this.options = options;
  9. }
  10. /**
  11. * Calculate the forces the nodes apply on each other based on a repulsion field.
  12. * This field is linearly approximated.
  13. *
  14. * @private
  15. */
  16. solve() {
  17. var dx, dy, distance, fx, fy, repulsingForce, node1, node2, i, j;
  18. var nodes = this.body.nodes;
  19. var nodeIndices = this.physicsBody.physicsNodeIndices;
  20. var forces = this.physicsBody.forces;
  21. // repulsing forces between nodes
  22. var nodeDistance = this.options.nodeDistance;
  23. // we loop from i over all but the last entree in the array
  24. // j loops from i+1 to the last. This way we do not double count any of the indices, nor i === j
  25. for (i = 0; i < nodeIndices.length - 1; i++) {
  26. node1 = nodes[nodeIndices[i]];
  27. for (j = i + 1; j < nodeIndices.length; j++) {
  28. node2 = nodes[nodeIndices[j]];
  29. // nodes only affect nodes on their level
  30. if (node1.level === node2.level) {
  31. dx = node2.x - node1.x;
  32. dy = node2.y - node1.y;
  33. distance = Math.sqrt(dx * dx + dy * dy);
  34. var steepness = 0.05;
  35. if (distance < nodeDistance) {
  36. repulsingForce = -Math.pow(steepness * distance, 2) + Math.pow(steepness * nodeDistance, 2);
  37. }
  38. else {
  39. repulsingForce = 0;
  40. }
  41. // normalize force with
  42. if (distance === 0) {
  43. distance = 0.01;
  44. }
  45. else {
  46. repulsingForce = repulsingForce / distance;
  47. }
  48. fx = dx * repulsingForce;
  49. fy = dy * repulsingForce;
  50. forces[node1.id].x -= fx;
  51. forces[node1.id].y -= fy;
  52. forces[node2.id].x += fx;
  53. forces[node2.id].y += fy;
  54. }
  55. }
  56. }
  57. }
  58. }
  59. export default HierarchicalRepulsionSolver;