diff --git a/cytoscape-expand-collapse.js b/cytoscape-expand-collapse.js
index 268aef3..67bc84e 100644
--- a/cytoscape-expand-collapse.js
+++ b/cytoscape-expand-collapse.js
@@ -30,8 +30,8 @@ module.exports = function (params) {
init: function () {
var self = this;
var opts = params;
- var $container = $(this);
- var cy;
+ var $container = this;
+ var cy = this.cytoscape('get');
var $canvas = $('');
$container.append($canvas);
@@ -314,6 +314,7 @@ module.exports = function (params) {
$container.data('cyexpandcollapse', data);
},
unbind: function () {
+ var cy = this.cytoscape('get');
cy.off('mouseover', 'node', eMouseOver)
.off('mouseout tapdragout', 'node', eMouseOut)
.off('position', 'node', ePosition)
@@ -336,6 +337,7 @@ module.exports = function (params) {
return $(this);
};
+
},{"./debounce":3}],3:[function(_dereq_,module,exports){
var debounce = (function () {
/**
@@ -1524,4 +1526,4 @@ module.exports = function (cy) {
},{}]},{},[6])(6)
});
-//# sourceMappingURL=data:application/json;charset:utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/boundingBoxUtilities.js","src/cueUtilities.js","src/debounce.js","src/elementUtilities.js","src/expandCollapseUtilities.js","src/index.js","src/undoRedoUtilities.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3TA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7OA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACplBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","var boundingBoxUtilities = {\n  equalBoundingBoxes: function(bb1, bb2){\n      return bb1.x1 == bb2.x1 && bb1.x2 == bb2.x2 && bb1.y1 == bb2.y1 && bb1.y2 == bb2.y2;\n  },\n  getUnion: function(bb1, bb2){\n      var union = {\n      x1: Math.min(bb1.x1, bb2.x1),\n      x2: Math.max(bb1.x2, bb2.x2),\n      y1: Math.min(bb1.y1, bb2.y1),\n      y2: Math.max(bb1.y2, bb2.y2),\n    };\n\n    union.w = union.x2 - union.x1;\n    union.h = union.y2 - union.y1;\n\n    return union;\n  }\n};\n\nmodule.exports = boundingBoxUtilities;","var debounce = require('./debounce');\n\nmodule.exports = function (params) {\n  var fn = params;\n\n  var eMouseOver, eMouseOut, ePosition, eRemove, eTap, eZoom, eAdd, eFree;\n  var functions = {\n    init: function () {\n      var self = this;\n      var opts = params;\n      var $container = $(this);\n      var cy;\n      var $canvas = $('<canvas></canvas>');\n\n      $container.append($canvas);\n\n      var _sizeCanvas = debounce(function () {\n        $canvas\n          .attr('height', $container.height())\n          .attr('width', $container.width())\n          .css({\n            'position': 'absolute',\n            'top': 0,\n            'left': 0,\n            'z-index': '999'\n          })\n        ;\n\n        setTimeout(function () {\n          var canvasBb = $canvas.offset();\n          var containerBb = $container.offset();\n\n          $canvas\n            .css({\n              'top': -(canvasBb.top - containerBb.top),\n              'left': -(canvasBb.left - containerBb.left)\n            })\n          ;\n\n          // refresh the cues on canvas resize\n          if(cy){\n            clearDraws(true);\n          }\n        }, 0);\n\n      }, 250);\n\n      function sizeCanvas() {\n        _sizeCanvas();\n      }\n\n      sizeCanvas();\n\n      $(window).bind('resize', function () {\n        sizeCanvas();\n      });\n\n      var ctx = $canvas[0].getContext('2d');\n\n      // write options to data\n      var data = $container.data('cyexpandcollapse');\n      if (data == null) {\n        data = {};\n      }\n      data.options = opts;\n\n      var optCache;\n\n      function options() {\n        return optCache || (optCache = $container.data('cyexpandcollapse').options);\n      }\n\n      function clearDraws(keepExpandCues) {\n\n        var w = $container.width();\n        var h = $container.height();\n\n        ctx.clearRect(0, 0, w, h);\n\n        if (keepExpandCues) {\n          var collapsedNodes = cy.nodes('[expanded-collapsed=\"collapsed\"]');\n          for (var i = 0; i < collapsedNodes.length; i++) {\n            drawExpandCollapseCue(collapsedNodes[i]);\n          }\n        }\n      }\n\n      function clearNodeDraw(node) {\n\n        var x = node._private.data.expandcollapseRenderedStartX;\n        var y = node._private.data.expandcollapseRenderedStartY;\n        var s = node._private.data.expandcollapseRenderedCueSize;\n\n        if (node.data('expanded-collapsed') === 'collapsed') {\n          drawExpandCollapseCue(node);\n        }\n        ctx.clearRect(x, y, s, s);\n      }\n\n      function drawExpandCollapseCue(node) {\n        var cy = node.cy();\n        var children = node.children();\n        var collapsedChildren = node._private.data.collapsedChildren;\n        var hasChildren = children != null && children.length > 0;\n        //check if the expand or collapse cue is to be drawn\n        if (!hasChildren && collapsedChildren == null) {\n          return;\n        }\n\n        var expandedOrcollapsed = node.data('expanded-collapsed');\n\n        //Draw expand-collapse rectangles\n        var rectSize = options().expandCollapseCueSize;\n        var lineSize = options().expandCollapseCueLineSize;\n        var diff;\n\n        rectSize = rectSize * cy.zoom();\n        lineSize = lineSize * cy.zoom();\n        diff = (rectSize - lineSize) / 2;\n\n        var expandcollapseStartX;\n        var expandcollapseStartY;\n        var expandcollapseEndX;\n        var expandcollapseEndY;\n        var expandcollapseRectSize;\n\n        var expandcollapseCenterX;\n        var expandcollapseCenterY;\n\n        if (options().expandCollapseCuePosition === 'top-left') {\n          var p = node.renderedPosition();\n          var w = node.renderedOuterWidth();\n          var h = node.renderedOuterHeight();\n\n          expandcollapseCenterX = p.x - w / 2 - rectSize / 4 + rectSize / 2;\n          expandcollapseCenterY = p.y - h / 2 - rectSize / 4 + rectSize / 2;\n        } else {\n          var option = options().expandCollapseCuePosition;\n          var cueCenter = typeof option === 'function' ? option.call(this, node) : option;\n          var expandcollapseCenter = elementUtilities.convertToRenderedPosition(cueCenter);\n\n          expandcollapseCenterX = expandcollapseCenter.x;\n          expandcollapseCenterY = expandcollapseCenter.y;\n        }\n\n        expandcollapseStartX = expandcollapseCenterX - rectSize / 2;\n        expandcollapseStartY = expandcollapseCenterY - rectSize / 2;\n        expandcollapseEndX = expandcollapseStartX + rectSize;\n        expandcollapseEndY = expandcollapseStartY + rectSize;\n        expandcollapseRectSize = rectSize;\n\n        // Draw expand/collapse cue if specified use image else draw it\n        if (expandedOrcollapsed === 'expanded' && options().expandCueImage) {\n          var img=new Image();\n          img.src = options().expandCueImage;\n          ctx.drawImage(img, expandcollapseCenterX, expandcollapseCenterY, rectSize, rectSize);\n        }\n        else if (expandedOrcollapsed === 'collapsed' && options().collapseCueImage) {\n          var img=new Image();\n          img.src = options().collapseCueImage;\n          ctx.drawImage(img, expandcollapseCenterX, expandcollapseCenterY, rectSize, rectSize);\n        }\n        else {\n          var oldFillStyle = ctx.fillStyle;\n          var oldWidth = ctx.lineWidth;\n          var oldStrokeStyle = ctx.strokeStyle;\n\n          ctx.fillStyle = \"black\";\n          ctx.strokeStyle = \"black\";\n\n          ctx.ellipse(expandcollapseCenterX, expandcollapseCenterY, rectSize / 2, rectSize / 2, 0, 0, 2 * Math.PI);\n          ctx.fill();\n\n          ctx.beginPath();\n\n          ctx.strokeStyle = \"white\";\n          ctx.lineWidth = 2.6 * cy.zoom();\n\n          ctx.moveTo(expandcollapseStartX + diff, expandcollapseStartY + rectSize / 2);\n          ctx.lineTo(expandcollapseStartX + lineSize + diff, expandcollapseStartY + rectSize / 2);\n\n          if (expandedOrcollapsed == 'collapsed') {\n            ctx.moveTo(expandcollapseStartX + rectSize / 2, expandcollapseStartY + diff);\n            ctx.lineTo(expandcollapseStartX + rectSize / 2, expandcollapseStartY + lineSize + diff);\n          }\n\n          ctx.closePath();\n          ctx.stroke();\n\n          ctx.strokeStyle = oldStrokeStyle;\n          ctx.fillStyle = oldFillStyle;\n          ctx.lineWidth = oldWidth;\n        }\n\n        node._private.data.expandcollapseRenderedStartX = expandcollapseStartX;\n        node._private.data.expandcollapseRenderedStartY = expandcollapseStartY;\n        node._private.data.expandcollapseRenderedCueSize = expandcollapseRectSize;\n      }\n\n      $container.cytoscape(function (e) {\n        cy = this;\n        clearDraws(true);\n\n        cy.bind('zoom pan', eZoom = function () {\n          clearDraws(true);\n        });\n\n\n        cy.on('mouseover', 'node', eMouseOver = function (e) {\n          var node = this;\n\n          // remove old handle\n          clearDraws(true);\n\n          // add new handle\n          drawExpandCollapseCue(node);\n\n          var lastPosition = {};\n\n        });\n\n        cy.on('mouseout tapdragout', 'node', eMouseOut = function (e) {\n\n          clearDraws(true);\n\n        });\n\n        cy.on('position', 'node', ePosition = function () {\n          var node = this;\n\n          clearDraws(true);\n        });\n\n        cy.on('remove', 'node', eRemove = function () {\n          var node = this;\n          clearNodeDraw(node);\n        });\n        \n        cy.on('add', 'node', eAdd = function () {\n          var node = this;\n          drawExpandCollapseCue(node);\n        });\n        \n        cy.on('free', 'node', eFree = function () {\n          var node = this;\n          \n          clearDraws(true);\n        });\n        \n        var ur;\n        cy.on('tap', 'node', eTap = function (event) {\n          var node = this;\n\n          var expandcollapseRenderedStartX = node._private.data.expandcollapseRenderedStartX;\n          var expandcollapseRenderedStartY = node._private.data.expandcollapseRenderedStartY;\n          var expandcollapseRenderedRectSize = node._private.data.expandcollapseRenderedCueSize;\n          var expandcollapseRenderedEndX = expandcollapseRenderedStartX + expandcollapseRenderedRectSize;\n          var expandcollapseRenderedEndY = expandcollapseRenderedStartY + expandcollapseRenderedRectSize;\n\n          var cyRenderedPosX = event.cyRenderedPosition.x;\n          var cyRenderedPosY = event.cyRenderedPosition.y;\n          var factor = (options().expandCollapseCueSensitivity - 1) / 2;\n\n          if (cyRenderedPosX >= expandcollapseRenderedStartX - expandcollapseRenderedRectSize * factor\n            && cyRenderedPosX <= expandcollapseRenderedEndX + expandcollapseRenderedRectSize * factor\n            && cyRenderedPosY >= expandcollapseRenderedStartY - expandcollapseRenderedRectSize * factor\n            && cyRenderedPosY <= expandcollapseRenderedEndY + expandcollapseRenderedRectSize * factor) {\n            if(opts.undoable && !ur)\n              ur = cy.undoRedo({\n                defaultActions: false\n              });\n            if(node.isCollapsible())\n              if (opts.undoable)\n                ur.do(\"collapse\", {\n                  nodes: node,\n                  options: opts\n                });\n              else\n                node.collapse(opts);\n            else if(node.isExpandable())\n              if (opts.undoable)\n                ur.do(\"expand\", {\n                  nodes: node,\n                  options: opts\n                });\n              else\n                node.expand(opts);\n          }\n        });\n      });\n\n      $container.data('cyexpandcollapse', data);\n    },\n    unbind: function () {\n        cy.off('mouseover', 'node', eMouseOver)\n          .off('mouseout tapdragout', 'node', eMouseOut)\n          .off('position', 'node', ePosition)\n          .off('remove', 'node', eRemove)\n          .off('tap', 'node', eTap)\n          .off('add', 'node', eAdd)\n          .off('free', 'node', eFree);\n\n        cy.unbind(\"zoom pan\", eZoom);\n    }\n  };\n\n  if (functions[fn]) {\n    return functions[fn].apply(this, Array.prototype.slice.call(arguments, 1));\n  } else if (typeof fn == 'object' || !fn) {\n    return functions.init.apply(this, arguments);\n  } else {\n    $.error('No such function `' + fn + '` for cytoscape.js-expand-collapse');\n  }\n\n  return $(this);\n};","var debounce = (function () {\n  /**\n   * lodash 3.1.1 (Custom Build) <https://lodash.com/>\n   * Build: `lodash modern modularize exports=\"npm\" -o ./`\n   * Copyright 2012-2015 The Dojo Foundation <http://dojofoundation.org/>\n   * Based on Underscore.js 1.8.3 <http://underscorejs.org/LICENSE>\n   * Copyright 2009-2015 Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors\n   * Available under MIT license <https://lodash.com/license>\n   */\n  /** Used as the `TypeError` message for \"Functions\" methods. */\n  var FUNC_ERROR_TEXT = 'Expected a function';\n\n  /* Native method references for those with the same name as other `lodash` methods. */\n  var nativeMax = Math.max,\n          nativeNow = Date.now;\n\n  /**\n   * Gets the number of milliseconds that have elapsed since the Unix epoch\n   * (1 January 1970 00:00:00 UTC).\n   *\n   * @static\n   * @memberOf _\n   * @category Date\n   * @example\n   *\n   * _.defer(function(stamp) {\n   *   console.log(_.now() - stamp);\n   * }, _.now());\n   * // => logs the number of milliseconds it took for the deferred function to be invoked\n   */\n  var now = nativeNow || function () {\n    return new Date().getTime();\n  };\n\n  /**\n   * Creates a debounced function that delays invoking `func` until after `wait`\n   * milliseconds have elapsed since the last time the debounced function was\n   * invoked. The debounced function comes with a `cancel` method to cancel\n   * delayed invocations. Provide an options object to indicate that `func`\n   * should be invoked on the leading and/or trailing edge of the `wait` timeout.\n   * Subsequent calls to the debounced function return the result of the last\n   * `func` invocation.\n   *\n   * **Note:** If `leading` and `trailing` options are `true`, `func` is invoked\n   * on the trailing edge of the timeout only if the the debounced function is\n   * invoked more than once during the `wait` timeout.\n   *\n   * See [David Corbacho's article](http://drupalmotion.com/article/debounce-and-throttle-visual-explanation)\n   * for details over the differences between `_.debounce` and `_.throttle`.\n   *\n   * @static\n   * @memberOf _\n   * @category Function\n   * @param {Function} func The function to debounce.\n   * @param {number} [wait=0] The number of milliseconds to delay.\n   * @param {Object} [options] The options object.\n   * @param {boolean} [options.leading=false] Specify invoking on the leading\n   *  edge of the timeout.\n   * @param {number} [options.maxWait] The maximum time `func` is allowed to be\n   *  delayed before it's invoked.\n   * @param {boolean} [options.trailing=true] Specify invoking on the trailing\n   *  edge of the timeout.\n   * @returns {Function} Returns the new debounced function.\n   * @example\n   *\n   * // avoid costly calculations while the window size is in flux\n   * jQuery(window).on('resize', _.debounce(calculateLayout, 150));\n   *\n   * // invoke `sendMail` when the click event is fired, debouncing subsequent calls\n   * jQuery('#postbox').on('click', _.debounce(sendMail, 300, {\n   *   'leading': true,\n   *   'trailing': false\n   * }));\n   *\n   * // ensure `batchLog` is invoked once after 1 second of debounced calls\n   * var source = new EventSource('/stream');\n   * jQuery(source).on('message', _.debounce(batchLog, 250, {\n   *   'maxWait': 1000\n   * }));\n   *\n   * // cancel a debounced call\n   * var todoChanges = _.debounce(batchLog, 1000);\n   * Object.observe(models.todo, todoChanges);\n   *\n   * Object.observe(models, function(changes) {\n   *   if (_.find(changes, { 'user': 'todo', 'type': 'delete'})) {\n   *     todoChanges.cancel();\n   *   }\n   * }, ['delete']);\n   *\n   * // ...at some point `models.todo` is changed\n   * models.todo.completed = true;\n   *\n   * // ...before 1 second has passed `models.todo` is deleted\n   * // which cancels the debounced `todoChanges` call\n   * delete models.todo;\n   */\n  function debounce(func, wait, options) {\n    var args,\n            maxTimeoutId,\n            result,\n            stamp,\n            thisArg,\n            timeoutId,\n            trailingCall,\n            lastCalled = 0,\n            maxWait = false,\n            trailing = true;\n\n    if (typeof func != 'function') {\n      throw new TypeError(FUNC_ERROR_TEXT);\n    }\n    wait = wait < 0 ? 0 : (+wait || 0);\n    if (options === true) {\n      var leading = true;\n      trailing = false;\n    } else if (isObject(options)) {\n      leading = !!options.leading;\n      maxWait = 'maxWait' in options && nativeMax(+options.maxWait || 0, wait);\n      trailing = 'trailing' in options ? !!options.trailing : trailing;\n    }\n\n    function cancel() {\n      if (timeoutId) {\n        clearTimeout(timeoutId);\n      }\n      if (maxTimeoutId) {\n        clearTimeout(maxTimeoutId);\n      }\n      lastCalled = 0;\n      maxTimeoutId = timeoutId = trailingCall = undefined;\n    }\n\n    function complete(isCalled, id) {\n      if (id) {\n        clearTimeout(id);\n      }\n      maxTimeoutId = timeoutId = trailingCall = undefined;\n      if (isCalled) {\n        lastCalled = now();\n        result = func.apply(thisArg, args);\n        if (!timeoutId && !maxTimeoutId) {\n          args = thisArg = undefined;\n        }\n      }\n    }\n\n    function delayed() {\n      var remaining = wait - (now() - stamp);\n      if (remaining <= 0 || remaining > wait) {\n        complete(trailingCall, maxTimeoutId);\n      } else {\n        timeoutId = setTimeout(delayed, remaining);\n      }\n    }\n\n    function maxDelayed() {\n      complete(trailing, timeoutId);\n    }\n\n    function debounced() {\n      args = arguments;\n      stamp = now();\n      thisArg = this;\n      trailingCall = trailing && (timeoutId || !leading);\n\n      if (maxWait === false) {\n        var leadingCall = leading && !timeoutId;\n      } else {\n        if (!maxTimeoutId && !leading) {\n          lastCalled = stamp;\n        }\n        var remaining = maxWait - (stamp - lastCalled),\n                isCalled = remaining <= 0 || remaining > maxWait;\n\n        if (isCalled) {\n          if (maxTimeoutId) {\n            maxTimeoutId = clearTimeout(maxTimeoutId);\n          }\n          lastCalled = stamp;\n          result = func.apply(thisArg, args);\n        }\n        else if (!maxTimeoutId) {\n          maxTimeoutId = setTimeout(maxDelayed, remaining);\n        }\n      }\n      if (isCalled && timeoutId) {\n        timeoutId = clearTimeout(timeoutId);\n      }\n      else if (!timeoutId && wait !== maxWait) {\n        timeoutId = setTimeout(delayed, wait);\n      }\n      if (leadingCall) {\n        isCalled = true;\n        result = func.apply(thisArg, args);\n      }\n      if (isCalled && !timeoutId && !maxTimeoutId) {\n        args = thisArg = undefined;\n      }\n      return result;\n    }\n\n    debounced.cancel = cancel;\n    return debounced;\n  }\n\n  /**\n   * Checks if `value` is the [language type](https://es5.github.io/#x8) of `Object`.\n   * (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`)\n   *\n   * @static\n   * @memberOf _\n   * @category Lang\n   * @param {*} value The value to check.\n   * @returns {boolean} Returns `true` if `value` is an object, else `false`.\n   * @example\n   *\n   * _.isObject({});\n   * // => true\n   *\n   * _.isObject([1, 2, 3]);\n   * // => true\n   *\n   * _.isObject(1);\n   * // => false\n   */\n  function isObject(value) {\n    // Avoid a V8 JIT bug in Chrome 19-20.\n    // See https://code.google.com/p/v8/issues/detail?id=2291 for more details.\n    var type = typeof value;\n    return !!value && (type == 'object' || type == 'function');\n  }\n\n  return debounce;\n\n})();\n\nmodule.exports = debounce;","function elementUtilities(cy) {\n return {\n  moveNodes: function (positionDiff, nodes, notCalcTopMostNodes) {\n    var topMostNodes = notCalcTopMostNodes ? nodes : this.getTopMostNodes(nodes);\n    for (var i = 0; i < topMostNodes.length; i++) {\n      var node = topMostNodes[i];\n      var oldX = node.position(\"x\");\n      var oldY = node.position(\"y\");\n      node.position({\n        x: oldX + positionDiff.x,\n        y: oldY + positionDiff.y\n      });\n      var children = node.children();\n      this.moveNodes(positionDiff, children, true);\n    }\n  },\n  getTopMostNodes: function (nodes) {//*//\n    var nodesMap = {};\n    for (var i = 0; i < nodes.length; i++) {\n      nodesMap[nodes[i].id()] = true;\n    }\n    var roots = nodes.filter(function (i, ele) {\n      var parent = ele.parent()[0];\n      while (parent != null) {\n        if (nodesMap[parent.id()]) {\n          return false;\n        }\n        parent = parent.parent()[0];\n      }\n      return true;\n    });\n\n    return roots;\n  },\n  rearrange: function (layoutBy) {\n    if (typeof layoutBy === \"function\") {\n      layoutBy();\n    } else if (layoutBy != null) {\n      cy.layout(layoutBy);\n    }\n  },\n  convertToRenderedPosition: function (modelPosition) {\n    var pan = cy.pan();\n    var zoom = cy.zoom();\n\n    var x = modelPosition.x * zoom + pan.x;\n    var y = modelPosition.y * zoom + pan.y;\n\n    return {\n      x: x,\n      y: y\n    };\n  }\n };\n}\n\nmodule.exports = elementUtilities;\n","var boundingBoxUtilities = require('./boundingBoxUtilities');\n\n// Expand collapse utilities\nfunction expandCollapseUtilities(cy) {\nvar elementUtilities = require('./elementUtilities')(cy);\nreturn {\n  //the number of nodes moving animatedly after expand operation\n  animatedlyMovingNodeCount: 0,\n  //A funtion basicly expanding a node it is to be called when a node is expanded anyway\n  expandNodeBaseFunction: function (node, triggerLayout, single, layoutBy) {//*//\n    //check how the position of the node is changed\n    var positionDiff = {\n      x: node.position('x') - node.data('position-before-collapse').x,\n      y: node.position('y') - node.data('position-before-collapse').y\n    };\n\n    node.removeData(\"infoLabel\");\n    node.data('expanded-collapsed', 'expanded');\n\n    node.trigger(\"beforeExpand\");\n    node._private.data.collapsedChildren.restore();\n    this.repairEdges(node);\n    node._private.data.collapsedChildren = null;\n    node.trigger(\"afterExpand\");\n\n\n    elementUtilities.moveNodes(positionDiff, node.children());\n    node.removeData('position-before-collapse');\n\n    if (single)\n      this.endOperation(layoutBy);\n    // refreshPaddings();\n   /* if (triggerLayout)\n      elementUtilities.rearrange(layoutBy);*/\n  },\n  simpleCollapseGivenNodes: function (nodes) {//*//\n    nodes.data(\"collapse\", true);\n    var roots = elementUtilities.getTopMostNodes(nodes);\n    for (var i = 0; i < roots.length; i++) {\n      var root = roots[i];\n      this.collapseBottomUp(root);\n    }\n    return nodes;\n  },\n  simpleExpandGivenNodes: function (nodes, applyFishEyeViewToEachNode) {//*//\n    nodes.data(\"expand\", true);\n    var roots = elementUtilities.getTopMostNodes(nodes);\n    for (var i = 0; i < roots.length; i++) {\n      var root = roots[i];\n      this.expandTopDown(root, applyFishEyeViewToEachNode);\n    }\n    return nodes;\n  },\n  simpleExpandAllNodes: function (nodes, applyFishEyeViewToEachNode) {//*//\n    if (nodes === undefined) {\n      nodes = cy.nodes();\n    }\n    var orphans;\n    orphans = elementUtilities.getTopMostNodes(nodes);\n    var expandStack = [];\n    for (var i = 0; i < orphans.length; i++) {\n      var root = orphans[i];\n      this.expandAllTopDown(root, expandStack, applyFishEyeViewToEachNode);\n    }\n    return expandStack;\n  },\n  endOperation: function (layoutBy) {\n    var self = this;\n    cy.ready(function () {\n      elementUtilities.rearrange(layoutBy);\n    });\n  },\n  expandAllNodes: function (nodes, options) {//*//\n    var expandedStack = this.simpleExpandAllNodes(nodes, options.fisheye);\n\n    this.endOperation(options.layoutBy);\n\n    //elementUtilities.rearrange(options.layoutBy);\n\n    /*\n     * return the nodes to undo the operation\n     */\n    return expandedStack;\n  },\n  expandAllTopDown: function (root, expandStack, applyFishEyeViewToEachNode) {//*//\n    if (root._private.data.collapsedChildren != null) {\n      expandStack.push(root);\n      this.simpleExpandNode(root, applyFishEyeViewToEachNode);\n    }\n    var children = root.children();\n    for (var i = 0; i < children.length; i++) {\n      var node = children[i];\n      this.expandAllTopDown(node, expandStack, applyFishEyeViewToEachNode);\n    }\n  },\n  //Expand the given nodes perform incremental layout after expandation\n  expandGivenNodes: function (nodes, options) {//*//\n    if (nodes.length === 1) {\n      this.expandNode(nodes[0], options.fisheye, options.animate, options.layoutBy);\n\n    } else {\n      this.simpleExpandGivenNodes(nodes, options.fisheye);\n      this.endOperation(options.layoutBy);\n\n      //elementUtilities.rearrange(options.layoutBy);\n    }\n\n    /*\n     * return the nodes to undo the operation\n     */\n    return nodes;\n  },\n  //collapse the given nodes then make incremental layout\n  collapseGivenNodes: function (nodes, options) {//*//\n    this.simpleCollapseGivenNodes(nodes, options);\n\n    this.endOperation(options.layoutBy);\n    //elementUtilities.rearrange(options.layoutBy);\n\n    /*\n     * return the nodes to undo the operation\n     */\n    return nodes;\n  },\n  //collapse the nodes in bottom up order starting from the root\n  collapseBottomUp: function (root) {//*//\n    var children = root.children();\n    for (var i = 0; i < children.length; i++) {\n      var node = children[i];\n      this.collapseBottomUp(node);\n    }\n    //If the root is a compound node to be collapsed then collapse it\n    if (root.data(\"collapse\") && root.children().length > 0) {\n      this.simpleCollapseNode(root);\n      root.removeData(\"collapse\");\n    }\n  },\n  //expand the nodes in top down order starting from the root\n  expandTopDown: function (root, applyFishEyeViewToEachNode) {//*//\n    if (root.data(\"expand\") && root._private.data.collapsedChildren != null) {\n      this.simpleExpandNode(root, applyFishEyeViewToEachNode);\n      root.removeData(\"expand\");\n    }\n    var children = root.children();\n    for (var i = 0; i < children.length; i++) {\n      var node = children[i];\n      this.expandTopDown(node);\n    }\n  },\n  expandNode: function (node, fisheye, animate, layoutBy) {\n    if (node._private.data.collapsedChildren != null) {\n      this.simpleExpandNode(node, fisheye, true, animate, layoutBy);\n\n      /*\n       * return the node to undo the operation\n       */\n      return node;\n    }\n  },\n  convertToModelPosition: function (renderedPosition) {\n    var pan = cy.pan();\n    var zoom = cy.zoom();\n\n    var x = (renderedPosition.x - pan.x) / zoom;\n    var y = (renderedPosition.y - pan.y) / zoom;\n\n    return {\n      x: x,\n      y: y\n    };\n  },\n  /*\n   *\n   * This method expands the given node\n   * without making incremental layout\n   * after expand operation it will be simply\n   * used to undo the collapse operation\n   */\n  simpleExpandNode: function (node, applyFishEyeViewToEachNode, singleNotSimple, animate, layoutBy) {//*//\n    var self = this;\n\n    var commonExpandOperation = function (node, applyFishEyeViewToEachNode, singleNotSimple, animate, layoutBy) {\n      if (applyFishEyeViewToEachNode) {\n\n        node.data('width-before-fisheye', node.data('size-before-collapse').w);\n        node.data('height-before-fisheye', node.data('size-before-collapse').h);\n\n        self.fishEyeViewExpandGivenNode(node, singleNotSimple, node, animate, layoutBy);\n      }\n\n      if (!singleNotSimple || !applyFishEyeViewToEachNode || !animate) {\n        self.expandNodeBaseFunction(node, singleNotSimple, singleNotSimple, layoutBy); //*****\n      }\n    };\n\n    if (node._private.data.collapsedChildren != null) {\n      this.storeWidthHeight(node);\n      if (applyFishEyeViewToEachNode && singleNotSimple) {\n        var topLeftPosition = this.convertToModelPosition({x: 0, y: 0});\n        var bottomRightPosition = this.convertToModelPosition({x: cy.width(), y: cy.height()});\n        var padding = 80;\n        var bb = {\n          x1: topLeftPosition.x,\n          x2: bottomRightPosition.x,\n          y1: topLeftPosition.y,\n          y2: bottomRightPosition.y\n        };\n\n        var nodeBB = {\n          x1: node.position('x') - node.data('size-before-collapse').w / 2 - padding,\n          x2: node.position('x') + node.data('size-before-collapse').w / 2 + padding,\n          y1: node.position('y') - node.data('size-before-collapse').h / 2 - padding,\n          y2: node.position('y') + node.data('size-before-collapse').h / 2 + padding\n        };\n\n        var unionBB = boundingBoxUtilities.getUnion(nodeBB, bb);\n        var animating = false;\n\n        if (!boundingBoxUtilities.equalBoundingBoxes(unionBB, bb)) {\n          var viewPort = cy.getFitViewport(unionBB, 10);\n          var self = this;\n          animating = animate;\n          if (animate) {\n            cy.animate({\n              pan: viewPort.pan,\n              zoom: viewPort.zoom,\n              complete: function () {\n                commonExpandOperation(node, applyFishEyeViewToEachNode, singleNotSimple, animate, layoutBy);\n              }\n            }, {\n              duration: 1000\n            });\n          }\n          else {\n            cy.zoom(viewPort.zoom);\n            cy.pan(viewPort.pan);\n          }\n        }\n        if (!animating) {\n          commonExpandOperation(node, applyFishEyeViewToEachNode, singleNotSimple, animate, layoutBy);\n        }\n      }\n      else {\n        commonExpandOperation(node, applyFishEyeViewToEachNode, singleNotSimple, animate, layoutBy);\n      }\n\n      //return the node to undo the operation\n      return node;\n    }\n  },\n  //collapse the given node without making incremental layout\n  simpleCollapseNode: function (node) {//*//\n    if (node._private.data.collapsedChildren == null) {\n      node.data('position-before-collapse', {\n        x: node.position().x,\n        y: node.position().y\n      });\n\n      node.data('size-before-collapse', {\n        w: node.outerWidth(),\n        h: node.outerHeight()\n      });\n\n      node.children().unselect();\n      node.children().connectedEdges().unselect();\n\n      node.data('expanded-collapsed', 'collapsed');\n\n      var children = node.children();\n\n      node.trigger(\"beforeCollapse\");\n      this.barrowEdgesOfcollapsedChildren(node);\n\n      this.removeChildren(node, node);\n\n      node.trigger(\"afterCollapse\");\n      \n      node.position(node.data('position-before-collapse'));\n\n      //return the node to undo the operation\n      return node;\n    }\n  },\n  storeWidthHeight: function (node) {//*//\n    if (node != null) {\n      node.data('x-before-fisheye', this.xPositionInParent(node));\n      node.data('y-before-fisheye', this.yPositionInParent(node));\n      node.data('width-before-fisheye', node.outerWidth());\n      node.data('height-before-fisheye', node.outerHeight());\n\n      if (node.parent()[0] != null) {\n        this.storeWidthHeight(node.parent()[0]);\n      }\n    }\n\n  },\n  fishEyeViewExpandGivenNode: function (node, singleNotSimple, nodeToExpand, animate, layoutBy) {//*//\n    var siblings = this.getSiblings(node);\n\n    var x_a = this.xPositionInParent(node);\n    var y_a = this.yPositionInParent(node);\n\n    var d_x_left = Math.abs((node.data('width-before-fisheye') - node.outerWidth()) / 2);\n    var d_x_right = Math.abs((node.data('width-before-fisheye') - node.outerWidth()) / 2);\n    var d_y_upper = Math.abs((node.data('height-before-fisheye') - node.outerHeight()) / 2);\n    var d_y_lower = Math.abs((node.data('height-before-fisheye') - node.outerHeight()) / 2);\n\n    var abs_diff_on_x = Math.abs(node.data('x-before-fisheye') - x_a);\n    var abs_diff_on_y = Math.abs(node.data('y-before-fisheye') - y_a);\n\n    // Center went to LEFT\n    if (node.data('x-before-fisheye') > x_a) {\n      d_x_left = d_x_left + abs_diff_on_x;\n      d_x_right = d_x_right - abs_diff_on_x;\n    }\n    // Center went to RIGHT\n    else {\n      d_x_left = d_x_left - abs_diff_on_x;\n      d_x_right = d_x_right + abs_diff_on_x;\n    }\n\n    // Center went to UP\n    if (node.data('y-before-fisheye') > y_a) {\n      d_y_upper = d_y_upper + abs_diff_on_y;\n      d_y_lower = d_y_lower - abs_diff_on_y;\n    }\n    // Center went to DOWN\n    else {\n      d_y_upper = d_y_upper - abs_diff_on_y;\n      d_y_lower = d_y_lower + abs_diff_on_y;\n    }\n\n    var xPosInParentSibling = [];\n    var yPosInParentSibling = [];\n\n    for (var i = 0; i < siblings.length; i++) {\n      xPosInParentSibling.push(this.xPositionInParent(siblings[i]));\n      yPosInParentSibling.push(this.yPositionInParent(siblings[i]));\n    }\n\n    for (var i = 0; i < siblings.length; i++) {\n      var sibling = siblings[i];\n\n      var x_b = xPosInParentSibling[i];\n      var y_b = yPosInParentSibling[i];\n\n      var slope = (y_b - y_a) / (x_b - x_a);\n\n      var d_x = 0;\n      var d_y = 0;\n      var T_x = 0;\n      var T_y = 0;\n\n      // Current sibling is on the LEFT\n      if (x_a > x_b) {\n        d_x = d_x_left;\n      }\n      // Current sibling is on the RIGHT\n      else {\n        d_x = d_x_right;\n      }\n      // Current sibling is on the UPPER side\n      if (y_a > y_b) {\n        d_y = d_y_upper;\n      }\n      // Current sibling is on the LOWER side\n      else {\n        d_y = d_y_lower;\n      }\n\n      if (isFinite(slope)) {\n        T_x = Math.min(d_x, (d_y / Math.abs(slope)));\n      }\n\n      if (slope !== 0) {\n        T_y = Math.min(d_y, (d_x * Math.abs(slope)));\n      }\n\n      if (x_a > x_b) {\n        T_x = -1 * T_x;\n      }\n\n      if (y_a > y_b) {\n        T_y = -1 * T_y;\n      }\n\n      this.fishEyeViewMoveNode(sibling, T_x, T_y, nodeToExpand, singleNotSimple, animate, layoutBy);\n    }\n\n    if (siblings.length == 0) {\n      this.expandNodeBaseFunction(nodeToExpand, singleNotSimple, true, layoutBy);\n    }\n\n    if (node.parent()[0] != null) {\n      this.fishEyeViewExpandGivenNode(node.parent()[0], singleNotSimple, nodeToExpand, animate, layoutBy);\n    }\n\n    return node;\n  },\n  getSiblings: function (node) {//*//\n    var siblings;\n\n    if (node.parent()[0] == null) {\n      siblings = cy.collection();\n      var orphans = cy.nodes().orphans();\n\n      for (var i = 0; i < orphans.length; i++) {\n        if (orphans[i] != node) {\n          siblings = siblings.add(orphans[i]);\n        }\n      }\n    } else {\n      siblings = node.siblings();\n    }\n\n    return siblings;\n  },\n  /*\n   * Move node operation specialized for fish eye view expand operation\n   * Moves the node by moving its descandents. Movement is animated if singleNotSimple flag is truthy.\n   */\n  fishEyeViewMoveNode: function (node, T_x, T_y, nodeToExpand, singleNotSimple, animate, layoutBy) {//*//\n    var childrenList = node.children();\n    var self = this;\n\n    if (childrenList.length == 0) {\n      var newPosition = {x: node.position('x') + T_x, y: node.position('y') + T_y};\n      if (!singleNotSimple || !animate) {\n        node.position(newPosition);\n      }\n      else {\n        this.animatedlyMovingNodeCount++;\n        node.animate({\n          position: newPosition,\n          complete: function () {\n            self.animatedlyMovingNodeCount--;\n            if (self.animatedlyMovingNodeCount > 0 || nodeToExpand.data('expanded-collapsed') === 'expanded') {\n\n              return;\n            }\n\n            self.expandNodeBaseFunction(nodeToExpand, singleNotSimple, true, layoutBy);\n\n          }\n        }, {\n          duration: 1000\n        });\n      }\n    }\n    else {\n\n      for (var i = 0; i < childrenList.length; i++) {\n        this.fishEyeViewMoveNode(childrenList[i], T_x, T_y, nodeToExpand, singleNotSimple, animate, layoutBy);\n      }\n    }\n  },\n  xPositionInParent: function (node) {//*//\n    var parent = node.parent()[0];\n    var x_a = 0.0;\n\n    // Given node is not a direct child of the the root graph\n    if (parent != null) {\n      x_a = node.relativePosition('x') + (parent.width() / 2);\n    }\n    // Given node is a direct child of the the root graph\n\n    else {\n      x_a = node.position('x');\n    }\n\n    return x_a;\n  },\n  yPositionInParent: function (node) {//*//\n    var parent = node.parent()[0];\n\n    var y_a = 0.0;\n\n    // Given node is not a direct child of the the root graph\n    if (parent != null) {\n      y_a = node.relativePosition('y') + (parent.height() / 2);\n    }\n    // Given node is a direct child of the the root graph\n\n    else {\n      y_a = node.position('y');\n    }\n\n    return y_a;\n  },\n  /*\n   * for all children of the node parameter call this method\n   * with the same root parameter,\n   * remove the child and add the removed child to the collapsedchildren data\n   * of the root to restore them in the case of expandation\n   * root._private.data.collapsedChildren keeps the nodes to restore when the\n   * root is expanded\n   */\n  removeChildren: function (node, root) {//*//\n    var children = node.children();\n    for (var i = 0; i < children.length; i++) {\n      var child = children[i];\n      this.removeChildren(child, root);\n      var removedChild = child.remove();\n      if (root._private.data.collapsedChildren == null) {\n        root._private.data.collapsedChildren = removedChild;\n      }\n      else {\n        root._private.data.collapsedChildren = root._private.data.collapsedChildren.union(removedChild);\n      }\n    }\n  },\n  isMetaEdge: function(edge) {\n    return edge.hasClass(\"meta\");\n  },\n  barrowEdgesOfcollapsedChildren: function(node) {\n    var relatedNodes = node.descendants();\n    var edges = relatedNodes.edgesWith(cy.nodes().not(relatedNodes.union(node)));\n    \n    var relatedNodeMap = {};\n    \n    relatedNodes.each(function(i, ele) {\n      relatedNodeMap[ele.id()] = true;\n    });\n    \n    for (var i = 0; i < edges.length; i++) {\n      var edge = edges[i];\n      var source = edge.source();\n      var target = edge.target();\n      \n      if (!this.isMetaEdge(edge)) { // is original\n        var originalEndsData = {\n          source: source,\n          target: target\n        };\n        \n        edge.addClass(\"meta\");\n        edge.data('originalEnds', originalEndsData);\n      }\n      \n      edge.move({\n        target: !relatedNodeMap[target.id()] ? target.id() : node.id(),\n        source: !relatedNodeMap[source.id()] ? source.id() : node.id()\n      });\n    }\n  },\n  findNewEnd: function(node) {\n    var current = node;\n    \n    while( !current.inside() ) {\n      current = current.parent();\n    }\n    \n    return current;\n  },\n  repairEdges: function(node) {\n    var connectedMetaEdges = node.connectedEdges('.meta');\n    \n    for (var i = 0; i < connectedMetaEdges.length; i++) {\n      var edge = connectedMetaEdges[i];\n      var originalEnds = edge.data('originalEnds');\n      var currentSrcId = edge.data('source');\n      var currentTgtId = edge.data('target');\n      \n      if ( currentSrcId === node.id() ) {\n        edge = edge.move({\n          source: this.findNewEnd(originalEnds.source).id()\n        });\n      } else {\n        edge = edge.move({\n          target: this.findNewEnd(originalEnds.target).id()\n        });\n      }\n      \n      if ( edge.data('source') === originalEnds.source.id() && edge.data('target') === originalEnds.target.id() ) {\n        edge.removeClass('meta');\n        edge.removeData('originalEnds');\n      }\n    }\n  },\n  /*node is an outer node of root\n   if root is not it's anchestor\n   and it is not the root itself*/\n  isOuterNode: function (node, root) {//*//\n    var temp = node;\n    while (temp != null) {\n      if (temp == root) {\n        return false;\n      }\n      temp = temp.parent()[0];\n    }\n    return true;\n  }\n}\n};\n\nmodule.exports = expandCollapseUtilities;\n",";\n(function ($$, $) {\n  'use strict';\n\n  // registers the extension on a cytoscape lib ref\n  var register = function (cytoscape, $) {\n\n    if (!cytoscape) {\n      return;\n    } // can't register if cytoscape unspecified\n\n    var expandCollapseUtilities = require('./expandCollapseUtilities');\n    var undoRedoUtilities = require('./undoRedoUtilities');\n    var elementUtilities = require('./elementUtilities');\n    $.fn.cytoscapeExpandCollapse = require(\"./cueUtilities\");\n\n    var options = {\n      layoutBy: null, // for rearrange after expand/collapse. It's just layout options or whole layout function. Choose your side!\n      fisheye: true, // whether to perform fisheye view after expand/collapse you can specify a function too\n      animate: true, // whether to animate on drawing changes you can specify a function too\n      ready: function () { }, // callback when expand/collapse initialized\n      undoable: true, // and if undoRedoExtension exists,\n\n      cueEnabled: true, // Whether cues are enabled\n      expandCollapseCuePosition: 'top-left', // default cue position is top left you can specify a function per node too\n      expandCollapseCueSize: 12, // size of expand-collapse cue\n      expandCollapseCueLineSize: 8, // size of lines used for drawing plus-minus icons\n      expandCueImage: undefined, // image of expand icon if undefined draw regular expand cue\n      collapseCueImage: undefined, // image of collapse icon if undefined draw regular collapse cue\n      expandCollapseCueSensitivity: 1 // sensitivity of expand-collapse cues\n    };\n\n    function setOptions(from) {\n      var tempOpts = {};\n      for (var key in options)\n        tempOpts[key] = options[key];\n\n      for (var key in from)\n        if (tempOpts.hasOwnProperty(key))\n          tempOpts[key] = from[key];\n      return tempOpts;\n    }\n    \n    // evaluate some specific options in case of they are specified as functions to be dynamically changed\n    function evalOptions(options) {\n      var animate = typeof options.animate === 'function' ? options.animate.call() : options.animate;\n      var fisheye = typeof options.fisheye === 'function' ? options.fisheye.call() : options.fisheye;\n      \n      options.animate = animate;\n      options.fisheye = fisheye;\n    }\n\n\n    // cy.expandCollapse()\n    cytoscape(\"core\", \"expandCollapse\", function (opts) {\n      var cy = this;\n      options = setOptions(opts);\n\n      // All parent nodes are expanded on load\n      cy.nodes(':parent').data('expanded-collapsed', 'expanded');\n      undoRedoUtilities(cy);\n      \n      if(options.cueEnabled)\n        $(cy.container()).cytoscapeExpandCollapse(options);\n      else\n        $(cy.container()).cytoscapeExpandCollapse(\"unbind\");\n\n\n      options.ready();\n\n\n      return cy;\n    });\n    \n    // set functions\n    \n    // set all options at once\n    cytoscape(\"core\", \"setExpandCollapseOptions\", function (opts) {\n      options = opts;\n    });\n    \n    // set the option whose name is given\n    cytoscape(\"core\", \"setExpandCollapseOption\", function (name, value) {\n      options[name] = value;\n    });\n\n    // Collection functions\n\n    // eles.collapse(options)\n    cytoscape('collection', 'collapse', function (opts) {\n      var eles = this.collapsibleNodes();\n      var tempOptions = setOptions(opts);\n      evalOptions(tempOptions);\n\n      return expandCollapseUtilities(this.cy()).collapseGivenNodes(eles, tempOptions);\n    });\n\n    // eles.collapseAll(options)\n    cytoscape('collection', 'collapseRecursively', function (opts) {\n      var eles = this.collapsibleNodes();\n      var tempOptions = setOptions(opts);\n      evalOptions(tempOptions);\n\n      return eles.union(eles.descendants()).collapse(tempOptions);\n    });\n\n    // eles.expand(options)\n    cytoscape('collection', 'expand', function (opts) {\n      var eles = this.expandableNodes();\n      var tempOptions = setOptions(opts);\n      evalOptions(tempOptions);\n\n      return expandCollapseUtilities(this.cy()).expandGivenNodes(eles, tempOptions);\n    });\n\n    // eles.expandAll(options)\n    cytoscape('collection', 'expandRecursively', function (opts) {\n      var eles = this.expandableNodes();\n      var tempOptions = setOptions(opts);\n      evalOptions(tempOptions);\n\n      return expandCollapseUtilities(this.cy()).expandAllNodes(eles, tempOptions);\n    });\n\n\n    // Core functions\n\n    // cy.collapseAll(options)\n    cytoscape('core', 'collapseAll', function (opts) {\n      var cy = this;\n      var tempOptions = setOptions(opts);\n      evalOptions(tempOptions);\n\n      return cy.collapsibleNodes().collapseRecursively(tempOptions);\n    });\n\n    // cy.expandAll(options)\n    cytoscape('core', 'expandAll', function (opts) {\n      var cy = this;\n      var tempOptions = setOptions(opts);\n      evalOptions(tempOptions);\n\n      return cy.expandableNodes().expandRecursively(tempOptions);\n    });\n\n\n    // Utility functions\n\n    // ele.isCollapsible()\n    cytoscape('collection', 'isExpandable', function () {\n      var ele = this;\n\n      return (ele.data(\"expanded-collapsed\") === \"collapsed\");\n    });\n\n    // ele.isExpandable()\n    cytoscape('collection', 'isCollapsible', function () {\n      var ele = this;\n      return !ele.isExpandable() && ele.isParent();\n    });\n\n    // eles.collapsed()\n    cytoscape('collection', 'collapsibleNodes', function () {\n      var eles = this;\n\n      return eles.filter(function (i, ele) {\n        return ele.isCollapsible();\n      });\n    });\n\n    // eles.expanded()\n    cytoscape('collection', 'expandableNodes', function () {\n      var eles = this;\n\n      return eles.filter(function (i, ele) {\n        return ele.isExpandable();\n      });\n    });\n    // eles.collapsed()\n    cytoscape('core', 'collapsibleNodes', function () {\n      var cy = this;\n\n      return cy.nodes().collapsibleNodes();\n    });\n\n    // eles.expanded()\n    cytoscape('core', 'expandableNodes', function () {\n      var cy = this;\n\n      return cy.nodes().expandableNodes();\n    });\n  };\n\n  if (typeof module !== 'undefined' && module.exports) { // expose as a commonjs module\n    module.exports = register;\n  }\n\n  if (typeof define !== 'undefined' && define.amd) { // expose as an amd/requirejs module\n    define('cytoscape-expand-collapse', function () {\n      return register;\n    });\n  }\n\n    if (typeof cytoscape !== 'undefined' && typeof jQuery !== 'undefined') { // expose to global cytoscape (i.e. window.cytoscape)\n      register(cytoscape, jQuery);\n  }\n\n})();\n","module.exports = function (cy) {\n  if (cy.undoRedo == null)\n    return;\n\n  var ur = cy.undoRedo({}, true);\n\n  function getEles(_eles) {\n    return (typeof _eles === \"string\") ? cy.$(_eles) : _eles;\n  }\n\n  function getNodePositionsAndSizes() {\n    var positionsAndSizes = {};\n    var nodes = cy.nodes();\n\n    for (var i = 0; i < nodes.length; i++) {\n      var ele = nodes[i];\n      positionsAndSizes[ele.id()] = {\n        width: ele.width(),\n        height: ele.height(),\n        x: ele.position(\"x\"),\n        y: ele.position(\"y\")\n      };\n    }\n\n    return positionsAndSizes;\n  }\n\n  function returnToPositionsAndSizes(nodesData) {\n    var currentPositionsAndSizes = {};\n    cy.nodes().positions(function (i, ele) {\n      currentPositionsAndSizes[ele.id()] = {\n        width: ele.width(),\n        height: ele.height(),\n        x: ele.position(\"x\"),\n        y: ele.position(\"y\")\n      };\n      var data = nodesData[ele.id()];\n      ele._private.data.width = data.width;\n      ele._private.data.height = data.height;\n      return {\n        x: data.x,\n        y: data.y\n      };\n    });\n\n    return currentPositionsAndSizes;\n  }\n\n  var secondTimeOpts = {\n    layoutBy: null,\n    animate: false,\n    fisheye: false\n  };\n\n  function doIt(func) {\n    return function (args) {\n      var result = {};\n      var nodes = getEles(args.nodes);\n      if (args.firstTime) {\n        result.oldData = getNodePositionsAndSizes();\n        result.nodes = func.indexOf(\"All\") > 0 ? cy[func](args.options) : nodes[func](args.options);\n      } else {\n        result.oldData = getNodePositionsAndSizes();\n        result.nodes = func.indexOf(\"All\") > 0 ? cy[func](secondTimeOpts) : cy.collection(nodes)[func](secondTimeOpts);\n        returnToPositionsAndSizes(args.oldData);\n      }\n\n      return result;\n    };\n  }\n\n  var actions = [\"collapse\", \"collapseRecursively\", \"collapseAll\", \"expand\", \"expandRecursively\", \"expandAll\"];\n\n  for (var i = 0; i < actions.length; i++) {\n    ur.action(actions[i], doIt(actions[i]), doIt(actions[(i + 3) % 6]));\n  }\n\n};\n"]}
+//# sourceMappingURL=data:application/json;charset:utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/boundingBoxUtilities.js","src/cueUtilities.js","src/debounce.js","src/elementUtilities.js","src/expandCollapseUtilities.js","src/index.js","src/undoRedoUtilities.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7TA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7OA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACplBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","var boundingBoxUtilities = {\n  equalBoundingBoxes: function(bb1, bb2){\n      return bb1.x1 == bb2.x1 && bb1.x2 == bb2.x2 && bb1.y1 == bb2.y1 && bb1.y2 == bb2.y2;\n  },\n  getUnion: function(bb1, bb2){\n      var union = {\n      x1: Math.min(bb1.x1, bb2.x1),\n      x2: Math.max(bb1.x2, bb2.x2),\n      y1: Math.min(bb1.y1, bb2.y1),\n      y2: Math.max(bb1.y2, bb2.y2),\n    };\n\n    union.w = union.x2 - union.x1;\n    union.h = union.y2 - union.y1;\n\n    return union;\n  }\n};\n\nmodule.exports = boundingBoxUtilities;","var debounce = require('./debounce');\n\nmodule.exports = function (params) {\n  var fn = params;\n\n  var eMouseOver, eMouseOut, ePosition, eRemove, eTap, eZoom, eAdd, eFree;\n  var functions = {\n    init: function () {\n      var self = this;\n      var opts = params;\n      var $container = this;\n      var cy = this.cytoscape('get');\n      var $canvas = $('<canvas></canvas>');\n\n      $container.append($canvas);\n\n      var _sizeCanvas = debounce(function () {\n        $canvas\n          .attr('height', $container.height())\n          .attr('width', $container.width())\n          .css({\n            'position': 'absolute',\n            'top': 0,\n            'left': 0,\n            'z-index': '999'\n          })\n        ;\n\n        setTimeout(function () {\n          var canvasBb = $canvas.offset();\n          var containerBb = $container.offset();\n\n          $canvas\n            .css({\n              'top': -(canvasBb.top - containerBb.top),\n              'left': -(canvasBb.left - containerBb.left)\n            })\n          ;\n\n          // refresh the cues on canvas resize\n          if(cy){\n            clearDraws(true);\n          }\n        }, 0);\n\n      }, 250);\n\n      function sizeCanvas() {\n        _sizeCanvas();\n      }\n\n      sizeCanvas();\n\n      $(window).bind('resize', function () {\n        sizeCanvas();\n      });\n\n      var ctx = $canvas[0].getContext('2d');\n\n      // write options to data\n      var data = $container.data('cyexpandcollapse');\n      if (data == null) {\n        data = {};\n      }\n      data.options = opts;\n\n      var optCache;\n\n      function options() {\n        return optCache || (optCache = $container.data('cyexpandcollapse').options);\n      }\n\n      function clearDraws(keepExpandCues) {\n\n        var w = $container.width();\n        var h = $container.height();\n\n        ctx.clearRect(0, 0, w, h);\n\n        if (keepExpandCues) {\n          var collapsedNodes = cy.nodes('[expanded-collapsed=\"collapsed\"]');\n          for (var i = 0; i < collapsedNodes.length; i++) {\n            drawExpandCollapseCue(collapsedNodes[i]);\n          }\n        }\n      }\n\n      function clearNodeDraw(node) {\n\n        var x = node._private.data.expandcollapseRenderedStartX;\n        var y = node._private.data.expandcollapseRenderedStartY;\n        var s = node._private.data.expandcollapseRenderedCueSize;\n\n        if (node.data('expanded-collapsed') === 'collapsed') {\n          drawExpandCollapseCue(node);\n        }\n        ctx.clearRect(x, y, s, s);\n      }\n\n      function drawExpandCollapseCue(node) {\n        var cy = node.cy();\n        var children = node.children();\n        var collapsedChildren = node._private.data.collapsedChildren;\n        var hasChildren = children != null && children.length > 0;\n        //check if the expand or collapse cue is to be drawn\n        if (!hasChildren && collapsedChildren == null) {\n          return;\n        }\n\n        var expandedOrcollapsed = node.data('expanded-collapsed');\n\n        //Draw expand-collapse rectangles\n        var rectSize = options().expandCollapseCueSize;\n        var lineSize = options().expandCollapseCueLineSize;\n        var diff;\n\n        rectSize = rectSize * cy.zoom();\n        lineSize = lineSize * cy.zoom();\n        diff = (rectSize - lineSize) / 2;\n\n        var expandcollapseStartX;\n        var expandcollapseStartY;\n        var expandcollapseEndX;\n        var expandcollapseEndY;\n        var expandcollapseRectSize;\n\n        var expandcollapseCenterX;\n        var expandcollapseCenterY;\n\n        if (options().expandCollapseCuePosition === 'top-left') {\n          var p = node.renderedPosition();\n          var w = node.renderedOuterWidth();\n          var h = node.renderedOuterHeight();\n\n          expandcollapseCenterX = p.x - w / 2 - rectSize / 4 + rectSize / 2;\n          expandcollapseCenterY = p.y - h / 2 - rectSize / 4 + rectSize / 2;\n        } else {\n          var option = options().expandCollapseCuePosition;\n          var cueCenter = typeof option === 'function' ? option.call(this, node) : option;\n          var expandcollapseCenter = elementUtilities.convertToRenderedPosition(cueCenter);\n\n          expandcollapseCenterX = expandcollapseCenter.x;\n          expandcollapseCenterY = expandcollapseCenter.y;\n        }\n\n        expandcollapseStartX = expandcollapseCenterX - rectSize / 2;\n        expandcollapseStartY = expandcollapseCenterY - rectSize / 2;\n        expandcollapseEndX = expandcollapseStartX + rectSize;\n        expandcollapseEndY = expandcollapseStartY + rectSize;\n        expandcollapseRectSize = rectSize;\n\n        // Draw expand/collapse cue if specified use image else draw it\n        if (expandedOrcollapsed === 'expanded' && options().expandCueImage) {\n          var img=new Image();\n          img.src = options().expandCueImage;\n          ctx.drawImage(img, expandcollapseCenterX, expandcollapseCenterY, rectSize, rectSize);\n        }\n        else if (expandedOrcollapsed === 'collapsed' && options().collapseCueImage) {\n          var img=new Image();\n          img.src = options().collapseCueImage;\n          ctx.drawImage(img, expandcollapseCenterX, expandcollapseCenterY, rectSize, rectSize);\n        }\n        else {\n          var oldFillStyle = ctx.fillStyle;\n          var oldWidth = ctx.lineWidth;\n          var oldStrokeStyle = ctx.strokeStyle;\n\n          ctx.fillStyle = \"black\";\n          ctx.strokeStyle = \"black\";\n\n          ctx.ellipse(expandcollapseCenterX, expandcollapseCenterY, rectSize / 2, rectSize / 2, 0, 0, 2 * Math.PI);\n          ctx.fill();\n\n          ctx.beginPath();\n\n          ctx.strokeStyle = \"white\";\n          ctx.lineWidth = 2.6 * cy.zoom();\n\n          ctx.moveTo(expandcollapseStartX + diff, expandcollapseStartY + rectSize / 2);\n          ctx.lineTo(expandcollapseStartX + lineSize + diff, expandcollapseStartY + rectSize / 2);\n\n          if (expandedOrcollapsed == 'collapsed') {\n            ctx.moveTo(expandcollapseStartX + rectSize / 2, expandcollapseStartY + diff);\n            ctx.lineTo(expandcollapseStartX + rectSize / 2, expandcollapseStartY + lineSize + diff);\n          }\n\n          ctx.closePath();\n          ctx.stroke();\n\n          ctx.strokeStyle = oldStrokeStyle;\n          ctx.fillStyle = oldFillStyle;\n          ctx.lineWidth = oldWidth;\n        }\n\n        node._private.data.expandcollapseRenderedStartX = expandcollapseStartX;\n        node._private.data.expandcollapseRenderedStartY = expandcollapseStartY;\n        node._private.data.expandcollapseRenderedCueSize = expandcollapseRectSize;\n      }\n\n      $container.cytoscape(function (e) {\n        cy = this;\n        clearDraws(true);\n\n        cy.bind('zoom pan', eZoom = function () {\n          clearDraws(true);\n        });\n\n\n        cy.on('mouseover', 'node', eMouseOver = function (e) {\n          var node = this;\n\n          // remove old handle\n          clearDraws(true);\n\n          // add new handle\n          drawExpandCollapseCue(node);\n\n          var lastPosition = {};\n\n        });\n\n        cy.on('mouseout tapdragout', 'node', eMouseOut = function (e) {\n\n          clearDraws(true);\n\n        });\n\n        cy.on('position', 'node', ePosition = function () {\n          var node = this;\n\n          clearDraws(true);\n        });\n\n        cy.on('remove', 'node', eRemove = function () {\n          var node = this;\n          clearNodeDraw(node);\n        });\n        \n        cy.on('add', 'node', eAdd = function () {\n          var node = this;\n          drawExpandCollapseCue(node);\n        });\n        \n        cy.on('free', 'node', eFree = function () {\n          var node = this;\n          \n          clearDraws(true);\n        });\n        \n        var ur;\n        cy.on('tap', 'node', eTap = function (event) {\n          var node = this;\n\n          var expandcollapseRenderedStartX = node._private.data.expandcollapseRenderedStartX;\n          var expandcollapseRenderedStartY = node._private.data.expandcollapseRenderedStartY;\n          var expandcollapseRenderedRectSize = node._private.data.expandcollapseRenderedCueSize;\n          var expandcollapseRenderedEndX = expandcollapseRenderedStartX + expandcollapseRenderedRectSize;\n          var expandcollapseRenderedEndY = expandcollapseRenderedStartY + expandcollapseRenderedRectSize;\n\n          var cyRenderedPosX = event.cyRenderedPosition.x;\n          var cyRenderedPosY = event.cyRenderedPosition.y;\n          var factor = (options().expandCollapseCueSensitivity - 1) / 2;\n\n          if (cyRenderedPosX >= expandcollapseRenderedStartX - expandcollapseRenderedRectSize * factor\n            && cyRenderedPosX <= expandcollapseRenderedEndX + expandcollapseRenderedRectSize * factor\n            && cyRenderedPosY >= expandcollapseRenderedStartY - expandcollapseRenderedRectSize * factor\n            && cyRenderedPosY <= expandcollapseRenderedEndY + expandcollapseRenderedRectSize * factor) {\n            if(opts.undoable && !ur)\n              ur = cy.undoRedo({\n                defaultActions: false\n              });\n            if(node.isCollapsible())\n              if (opts.undoable)\n                ur.do(\"collapse\", {\n                  nodes: node,\n                  options: opts\n                });\n              else\n                node.collapse(opts);\n            else if(node.isExpandable())\n              if (opts.undoable)\n                ur.do(\"expand\", {\n                  nodes: node,\n                  options: opts\n                });\n              else\n                node.expand(opts);\n          }\n        });\n      });\n\n      $container.data('cyexpandcollapse', data);\n    },\n    unbind: function () {\n        var cy = this.cytoscape('get');\n        cy.off('mouseover', 'node', eMouseOver)\n          .off('mouseout tapdragout', 'node', eMouseOut)\n          .off('position', 'node', ePosition)\n          .off('remove', 'node', eRemove)\n          .off('tap', 'node', eTap)\n          .off('add', 'node', eAdd)\n          .off('free', 'node', eFree);\n\n        cy.unbind(\"zoom pan\", eZoom);\n    }\n  };\n\n  if (functions[fn]) {\n    return functions[fn].apply(this, Array.prototype.slice.call(arguments, 1));\n  } else if (typeof fn == 'object' || !fn) {\n    return functions.init.apply(this, arguments);\n  } else {\n    $.error('No such function `' + fn + '` for cytoscape.js-expand-collapse');\n  }\n\n  return $(this);\n};\n","var debounce = (function () {\n  /**\n   * lodash 3.1.1 (Custom Build) <https://lodash.com/>\n   * Build: `lodash modern modularize exports=\"npm\" -o ./`\n   * Copyright 2012-2015 The Dojo Foundation <http://dojofoundation.org/>\n   * Based on Underscore.js 1.8.3 <http://underscorejs.org/LICENSE>\n   * Copyright 2009-2015 Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors\n   * Available under MIT license <https://lodash.com/license>\n   */\n  /** Used as the `TypeError` message for \"Functions\" methods. */\n  var FUNC_ERROR_TEXT = 'Expected a function';\n\n  /* Native method references for those with the same name as other `lodash` methods. */\n  var nativeMax = Math.max,\n          nativeNow = Date.now;\n\n  /**\n   * Gets the number of milliseconds that have elapsed since the Unix epoch\n   * (1 January 1970 00:00:00 UTC).\n   *\n   * @static\n   * @memberOf _\n   * @category Date\n   * @example\n   *\n   * _.defer(function(stamp) {\n   *   console.log(_.now() - stamp);\n   * }, _.now());\n   * // => logs the number of milliseconds it took for the deferred function to be invoked\n   */\n  var now = nativeNow || function () {\n    return new Date().getTime();\n  };\n\n  /**\n   * Creates a debounced function that delays invoking `func` until after `wait`\n   * milliseconds have elapsed since the last time the debounced function was\n   * invoked. The debounced function comes with a `cancel` method to cancel\n   * delayed invocations. Provide an options object to indicate that `func`\n   * should be invoked on the leading and/or trailing edge of the `wait` timeout.\n   * Subsequent calls to the debounced function return the result of the last\n   * `func` invocation.\n   *\n   * **Note:** If `leading` and `trailing` options are `true`, `func` is invoked\n   * on the trailing edge of the timeout only if the the debounced function is\n   * invoked more than once during the `wait` timeout.\n   *\n   * See [David Corbacho's article](http://drupalmotion.com/article/debounce-and-throttle-visual-explanation)\n   * for details over the differences between `_.debounce` and `_.throttle`.\n   *\n   * @static\n   * @memberOf _\n   * @category Function\n   * @param {Function} func The function to debounce.\n   * @param {number} [wait=0] The number of milliseconds to delay.\n   * @param {Object} [options] The options object.\n   * @param {boolean} [options.leading=false] Specify invoking on the leading\n   *  edge of the timeout.\n   * @param {number} [options.maxWait] The maximum time `func` is allowed to be\n   *  delayed before it's invoked.\n   * @param {boolean} [options.trailing=true] Specify invoking on the trailing\n   *  edge of the timeout.\n   * @returns {Function} Returns the new debounced function.\n   * @example\n   *\n   * // avoid costly calculations while the window size is in flux\n   * jQuery(window).on('resize', _.debounce(calculateLayout, 150));\n   *\n   * // invoke `sendMail` when the click event is fired, debouncing subsequent calls\n   * jQuery('#postbox').on('click', _.debounce(sendMail, 300, {\n   *   'leading': true,\n   *   'trailing': false\n   * }));\n   *\n   * // ensure `batchLog` is invoked once after 1 second of debounced calls\n   * var source = new EventSource('/stream');\n   * jQuery(source).on('message', _.debounce(batchLog, 250, {\n   *   'maxWait': 1000\n   * }));\n   *\n   * // cancel a debounced call\n   * var todoChanges = _.debounce(batchLog, 1000);\n   * Object.observe(models.todo, todoChanges);\n   *\n   * Object.observe(models, function(changes) {\n   *   if (_.find(changes, { 'user': 'todo', 'type': 'delete'})) {\n   *     todoChanges.cancel();\n   *   }\n   * }, ['delete']);\n   *\n   * // ...at some point `models.todo` is changed\n   * models.todo.completed = true;\n   *\n   * // ...before 1 second has passed `models.todo` is deleted\n   * // which cancels the debounced `todoChanges` call\n   * delete models.todo;\n   */\n  function debounce(func, wait, options) {\n    var args,\n            maxTimeoutId,\n            result,\n            stamp,\n            thisArg,\n            timeoutId,\n            trailingCall,\n            lastCalled = 0,\n            maxWait = false,\n            trailing = true;\n\n    if (typeof func != 'function') {\n      throw new TypeError(FUNC_ERROR_TEXT);\n    }\n    wait = wait < 0 ? 0 : (+wait || 0);\n    if (options === true) {\n      var leading = true;\n      trailing = false;\n    } else if (isObject(options)) {\n      leading = !!options.leading;\n      maxWait = 'maxWait' in options && nativeMax(+options.maxWait || 0, wait);\n      trailing = 'trailing' in options ? !!options.trailing : trailing;\n    }\n\n    function cancel() {\n      if (timeoutId) {\n        clearTimeout(timeoutId);\n      }\n      if (maxTimeoutId) {\n        clearTimeout(maxTimeoutId);\n      }\n      lastCalled = 0;\n      maxTimeoutId = timeoutId = trailingCall = undefined;\n    }\n\n    function complete(isCalled, id) {\n      if (id) {\n        clearTimeout(id);\n      }\n      maxTimeoutId = timeoutId = trailingCall = undefined;\n      if (isCalled) {\n        lastCalled = now();\n        result = func.apply(thisArg, args);\n        if (!timeoutId && !maxTimeoutId) {\n          args = thisArg = undefined;\n        }\n      }\n    }\n\n    function delayed() {\n      var remaining = wait - (now() - stamp);\n      if (remaining <= 0 || remaining > wait) {\n        complete(trailingCall, maxTimeoutId);\n      } else {\n        timeoutId = setTimeout(delayed, remaining);\n      }\n    }\n\n    function maxDelayed() {\n      complete(trailing, timeoutId);\n    }\n\n    function debounced() {\n      args = arguments;\n      stamp = now();\n      thisArg = this;\n      trailingCall = trailing && (timeoutId || !leading);\n\n      if (maxWait === false) {\n        var leadingCall = leading && !timeoutId;\n      } else {\n        if (!maxTimeoutId && !leading) {\n          lastCalled = stamp;\n        }\n        var remaining = maxWait - (stamp - lastCalled),\n                isCalled = remaining <= 0 || remaining > maxWait;\n\n        if (isCalled) {\n          if (maxTimeoutId) {\n            maxTimeoutId = clearTimeout(maxTimeoutId);\n          }\n          lastCalled = stamp;\n          result = func.apply(thisArg, args);\n        }\n        else if (!maxTimeoutId) {\n          maxTimeoutId = setTimeout(maxDelayed, remaining);\n        }\n      }\n      if (isCalled && timeoutId) {\n        timeoutId = clearTimeout(timeoutId);\n      }\n      else if (!timeoutId && wait !== maxWait) {\n        timeoutId = setTimeout(delayed, wait);\n      }\n      if (leadingCall) {\n        isCalled = true;\n        result = func.apply(thisArg, args);\n      }\n      if (isCalled && !timeoutId && !maxTimeoutId) {\n        args = thisArg = undefined;\n      }\n      return result;\n    }\n\n    debounced.cancel = cancel;\n    return debounced;\n  }\n\n  /**\n   * Checks if `value` is the [language type](https://es5.github.io/#x8) of `Object`.\n   * (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`)\n   *\n   * @static\n   * @memberOf _\n   * @category Lang\n   * @param {*} value The value to check.\n   * @returns {boolean} Returns `true` if `value` is an object, else `false`.\n   * @example\n   *\n   * _.isObject({});\n   * // => true\n   *\n   * _.isObject([1, 2, 3]);\n   * // => true\n   *\n   * _.isObject(1);\n   * // => false\n   */\n  function isObject(value) {\n    // Avoid a V8 JIT bug in Chrome 19-20.\n    // See https://code.google.com/p/v8/issues/detail?id=2291 for more details.\n    var type = typeof value;\n    return !!value && (type == 'object' || type == 'function');\n  }\n\n  return debounce;\n\n})();\n\nmodule.exports = debounce;","function elementUtilities(cy) {\n return {\n  moveNodes: function (positionDiff, nodes, notCalcTopMostNodes) {\n    var topMostNodes = notCalcTopMostNodes ? nodes : this.getTopMostNodes(nodes);\n    for (var i = 0; i < topMostNodes.length; i++) {\n      var node = topMostNodes[i];\n      var oldX = node.position(\"x\");\n      var oldY = node.position(\"y\");\n      node.position({\n        x: oldX + positionDiff.x,\n        y: oldY + positionDiff.y\n      });\n      var children = node.children();\n      this.moveNodes(positionDiff, children, true);\n    }\n  },\n  getTopMostNodes: function (nodes) {//*//\n    var nodesMap = {};\n    for (var i = 0; i < nodes.length; i++) {\n      nodesMap[nodes[i].id()] = true;\n    }\n    var roots = nodes.filter(function (i, ele) {\n      var parent = ele.parent()[0];\n      while (parent != null) {\n        if (nodesMap[parent.id()]) {\n          return false;\n        }\n        parent = parent.parent()[0];\n      }\n      return true;\n    });\n\n    return roots;\n  },\n  rearrange: function (layoutBy) {\n    if (typeof layoutBy === \"function\") {\n      layoutBy();\n    } else if (layoutBy != null) {\n      cy.layout(layoutBy);\n    }\n  },\n  convertToRenderedPosition: function (modelPosition) {\n    var pan = cy.pan();\n    var zoom = cy.zoom();\n\n    var x = modelPosition.x * zoom + pan.x;\n    var y = modelPosition.y * zoom + pan.y;\n\n    return {\n      x: x,\n      y: y\n    };\n  }\n };\n}\n\nmodule.exports = elementUtilities;\n","var boundingBoxUtilities = require('./boundingBoxUtilities');\n\n// Expand collapse utilities\nfunction expandCollapseUtilities(cy) {\nvar elementUtilities = require('./elementUtilities')(cy);\nreturn {\n  //the number of nodes moving animatedly after expand operation\n  animatedlyMovingNodeCount: 0,\n  //A funtion basicly expanding a node it is to be called when a node is expanded anyway\n  expandNodeBaseFunction: function (node, triggerLayout, single, layoutBy) {//*//\n    //check how the position of the node is changed\n    var positionDiff = {\n      x: node.position('x') - node.data('position-before-collapse').x,\n      y: node.position('y') - node.data('position-before-collapse').y\n    };\n\n    node.removeData(\"infoLabel\");\n    node.data('expanded-collapsed', 'expanded');\n\n    node.trigger(\"beforeExpand\");\n    node._private.data.collapsedChildren.restore();\n    this.repairEdges(node);\n    node._private.data.collapsedChildren = null;\n    node.trigger(\"afterExpand\");\n\n\n    elementUtilities.moveNodes(positionDiff, node.children());\n    node.removeData('position-before-collapse');\n\n    if (single)\n      this.endOperation(layoutBy);\n    // refreshPaddings();\n   /* if (triggerLayout)\n      elementUtilities.rearrange(layoutBy);*/\n  },\n  simpleCollapseGivenNodes: function (nodes) {//*//\n    nodes.data(\"collapse\", true);\n    var roots = elementUtilities.getTopMostNodes(nodes);\n    for (var i = 0; i < roots.length; i++) {\n      var root = roots[i];\n      this.collapseBottomUp(root);\n    }\n    return nodes;\n  },\n  simpleExpandGivenNodes: function (nodes, applyFishEyeViewToEachNode) {//*//\n    nodes.data(\"expand\", true);\n    var roots = elementUtilities.getTopMostNodes(nodes);\n    for (var i = 0; i < roots.length; i++) {\n      var root = roots[i];\n      this.expandTopDown(root, applyFishEyeViewToEachNode);\n    }\n    return nodes;\n  },\n  simpleExpandAllNodes: function (nodes, applyFishEyeViewToEachNode) {//*//\n    if (nodes === undefined) {\n      nodes = cy.nodes();\n    }\n    var orphans;\n    orphans = elementUtilities.getTopMostNodes(nodes);\n    var expandStack = [];\n    for (var i = 0; i < orphans.length; i++) {\n      var root = orphans[i];\n      this.expandAllTopDown(root, expandStack, applyFishEyeViewToEachNode);\n    }\n    return expandStack;\n  },\n  endOperation: function (layoutBy) {\n    var self = this;\n    cy.ready(function () {\n      elementUtilities.rearrange(layoutBy);\n    });\n  },\n  expandAllNodes: function (nodes, options) {//*//\n    var expandedStack = this.simpleExpandAllNodes(nodes, options.fisheye);\n\n    this.endOperation(options.layoutBy);\n\n    //elementUtilities.rearrange(options.layoutBy);\n\n    /*\n     * return the nodes to undo the operation\n     */\n    return expandedStack;\n  },\n  expandAllTopDown: function (root, expandStack, applyFishEyeViewToEachNode) {//*//\n    if (root._private.data.collapsedChildren != null) {\n      expandStack.push(root);\n      this.simpleExpandNode(root, applyFishEyeViewToEachNode);\n    }\n    var children = root.children();\n    for (var i = 0; i < children.length; i++) {\n      var node = children[i];\n      this.expandAllTopDown(node, expandStack, applyFishEyeViewToEachNode);\n    }\n  },\n  //Expand the given nodes perform incremental layout after expandation\n  expandGivenNodes: function (nodes, options) {//*//\n    if (nodes.length === 1) {\n      this.expandNode(nodes[0], options.fisheye, options.animate, options.layoutBy);\n\n    } else {\n      this.simpleExpandGivenNodes(nodes, options.fisheye);\n      this.endOperation(options.layoutBy);\n\n      //elementUtilities.rearrange(options.layoutBy);\n    }\n\n    /*\n     * return the nodes to undo the operation\n     */\n    return nodes;\n  },\n  //collapse the given nodes then make incremental layout\n  collapseGivenNodes: function (nodes, options) {//*//\n    this.simpleCollapseGivenNodes(nodes, options);\n\n    this.endOperation(options.layoutBy);\n    //elementUtilities.rearrange(options.layoutBy);\n\n    /*\n     * return the nodes to undo the operation\n     */\n    return nodes;\n  },\n  //collapse the nodes in bottom up order starting from the root\n  collapseBottomUp: function (root) {//*//\n    var children = root.children();\n    for (var i = 0; i < children.length; i++) {\n      var node = children[i];\n      this.collapseBottomUp(node);\n    }\n    //If the root is a compound node to be collapsed then collapse it\n    if (root.data(\"collapse\") && root.children().length > 0) {\n      this.simpleCollapseNode(root);\n      root.removeData(\"collapse\");\n    }\n  },\n  //expand the nodes in top down order starting from the root\n  expandTopDown: function (root, applyFishEyeViewToEachNode) {//*//\n    if (root.data(\"expand\") && root._private.data.collapsedChildren != null) {\n      this.simpleExpandNode(root, applyFishEyeViewToEachNode);\n      root.removeData(\"expand\");\n    }\n    var children = root.children();\n    for (var i = 0; i < children.length; i++) {\n      var node = children[i];\n      this.expandTopDown(node);\n    }\n  },\n  expandNode: function (node, fisheye, animate, layoutBy) {\n    if (node._private.data.collapsedChildren != null) {\n      this.simpleExpandNode(node, fisheye, true, animate, layoutBy);\n\n      /*\n       * return the node to undo the operation\n       */\n      return node;\n    }\n  },\n  convertToModelPosition: function (renderedPosition) {\n    var pan = cy.pan();\n    var zoom = cy.zoom();\n\n    var x = (renderedPosition.x - pan.x) / zoom;\n    var y = (renderedPosition.y - pan.y) / zoom;\n\n    return {\n      x: x,\n      y: y\n    };\n  },\n  /*\n   *\n   * This method expands the given node\n   * without making incremental layout\n   * after expand operation it will be simply\n   * used to undo the collapse operation\n   */\n  simpleExpandNode: function (node, applyFishEyeViewToEachNode, singleNotSimple, animate, layoutBy) {//*//\n    var self = this;\n\n    var commonExpandOperation = function (node, applyFishEyeViewToEachNode, singleNotSimple, animate, layoutBy) {\n      if (applyFishEyeViewToEachNode) {\n\n        node.data('width-before-fisheye', node.data('size-before-collapse').w);\n        node.data('height-before-fisheye', node.data('size-before-collapse').h);\n\n        self.fishEyeViewExpandGivenNode(node, singleNotSimple, node, animate, layoutBy);\n      }\n\n      if (!singleNotSimple || !applyFishEyeViewToEachNode || !animate) {\n        self.expandNodeBaseFunction(node, singleNotSimple, singleNotSimple, layoutBy); //*****\n      }\n    };\n\n    if (node._private.data.collapsedChildren != null) {\n      this.storeWidthHeight(node);\n      if (applyFishEyeViewToEachNode && singleNotSimple) {\n        var topLeftPosition = this.convertToModelPosition({x: 0, y: 0});\n        var bottomRightPosition = this.convertToModelPosition({x: cy.width(), y: cy.height()});\n        var padding = 80;\n        var bb = {\n          x1: topLeftPosition.x,\n          x2: bottomRightPosition.x,\n          y1: topLeftPosition.y,\n          y2: bottomRightPosition.y\n        };\n\n        var nodeBB = {\n          x1: node.position('x') - node.data('size-before-collapse').w / 2 - padding,\n          x2: node.position('x') + node.data('size-before-collapse').w / 2 + padding,\n          y1: node.position('y') - node.data('size-before-collapse').h / 2 - padding,\n          y2: node.position('y') + node.data('size-before-collapse').h / 2 + padding\n        };\n\n        var unionBB = boundingBoxUtilities.getUnion(nodeBB, bb);\n        var animating = false;\n\n        if (!boundingBoxUtilities.equalBoundingBoxes(unionBB, bb)) {\n          var viewPort = cy.getFitViewport(unionBB, 10);\n          var self = this;\n          animating = animate;\n          if (animate) {\n            cy.animate({\n              pan: viewPort.pan,\n              zoom: viewPort.zoom,\n              complete: function () {\n                commonExpandOperation(node, applyFishEyeViewToEachNode, singleNotSimple, animate, layoutBy);\n              }\n            }, {\n              duration: 1000\n            });\n          }\n          else {\n            cy.zoom(viewPort.zoom);\n            cy.pan(viewPort.pan);\n          }\n        }\n        if (!animating) {\n          commonExpandOperation(node, applyFishEyeViewToEachNode, singleNotSimple, animate, layoutBy);\n        }\n      }\n      else {\n        commonExpandOperation(node, applyFishEyeViewToEachNode, singleNotSimple, animate, layoutBy);\n      }\n\n      //return the node to undo the operation\n      return node;\n    }\n  },\n  //collapse the given node without making incremental layout\n  simpleCollapseNode: function (node) {//*//\n    if (node._private.data.collapsedChildren == null) {\n      node.data('position-before-collapse', {\n        x: node.position().x,\n        y: node.position().y\n      });\n\n      node.data('size-before-collapse', {\n        w: node.outerWidth(),\n        h: node.outerHeight()\n      });\n\n      node.children().unselect();\n      node.children().connectedEdges().unselect();\n\n      node.data('expanded-collapsed', 'collapsed');\n\n      var children = node.children();\n\n      node.trigger(\"beforeCollapse\");\n      this.barrowEdgesOfcollapsedChildren(node);\n\n      this.removeChildren(node, node);\n\n      node.trigger(\"afterCollapse\");\n      \n      node.position(node.data('position-before-collapse'));\n\n      //return the node to undo the operation\n      return node;\n    }\n  },\n  storeWidthHeight: function (node) {//*//\n    if (node != null) {\n      node.data('x-before-fisheye', this.xPositionInParent(node));\n      node.data('y-before-fisheye', this.yPositionInParent(node));\n      node.data('width-before-fisheye', node.outerWidth());\n      node.data('height-before-fisheye', node.outerHeight());\n\n      if (node.parent()[0] != null) {\n        this.storeWidthHeight(node.parent()[0]);\n      }\n    }\n\n  },\n  fishEyeViewExpandGivenNode: function (node, singleNotSimple, nodeToExpand, animate, layoutBy) {//*//\n    var siblings = this.getSiblings(node);\n\n    var x_a = this.xPositionInParent(node);\n    var y_a = this.yPositionInParent(node);\n\n    var d_x_left = Math.abs((node.data('width-before-fisheye') - node.outerWidth()) / 2);\n    var d_x_right = Math.abs((node.data('width-before-fisheye') - node.outerWidth()) / 2);\n    var d_y_upper = Math.abs((node.data('height-before-fisheye') - node.outerHeight()) / 2);\n    var d_y_lower = Math.abs((node.data('height-before-fisheye') - node.outerHeight()) / 2);\n\n    var abs_diff_on_x = Math.abs(node.data('x-before-fisheye') - x_a);\n    var abs_diff_on_y = Math.abs(node.data('y-before-fisheye') - y_a);\n\n    // Center went to LEFT\n    if (node.data('x-before-fisheye') > x_a) {\n      d_x_left = d_x_left + abs_diff_on_x;\n      d_x_right = d_x_right - abs_diff_on_x;\n    }\n    // Center went to RIGHT\n    else {\n      d_x_left = d_x_left - abs_diff_on_x;\n      d_x_right = d_x_right + abs_diff_on_x;\n    }\n\n    // Center went to UP\n    if (node.data('y-before-fisheye') > y_a) {\n      d_y_upper = d_y_upper + abs_diff_on_y;\n      d_y_lower = d_y_lower - abs_diff_on_y;\n    }\n    // Center went to DOWN\n    else {\n      d_y_upper = d_y_upper - abs_diff_on_y;\n      d_y_lower = d_y_lower + abs_diff_on_y;\n    }\n\n    var xPosInParentSibling = [];\n    var yPosInParentSibling = [];\n\n    for (var i = 0; i < siblings.length; i++) {\n      xPosInParentSibling.push(this.xPositionInParent(siblings[i]));\n      yPosInParentSibling.push(this.yPositionInParent(siblings[i]));\n    }\n\n    for (var i = 0; i < siblings.length; i++) {\n      var sibling = siblings[i];\n\n      var x_b = xPosInParentSibling[i];\n      var y_b = yPosInParentSibling[i];\n\n      var slope = (y_b - y_a) / (x_b - x_a);\n\n      var d_x = 0;\n      var d_y = 0;\n      var T_x = 0;\n      var T_y = 0;\n\n      // Current sibling is on the LEFT\n      if (x_a > x_b) {\n        d_x = d_x_left;\n      }\n      // Current sibling is on the RIGHT\n      else {\n        d_x = d_x_right;\n      }\n      // Current sibling is on the UPPER side\n      if (y_a > y_b) {\n        d_y = d_y_upper;\n      }\n      // Current sibling is on the LOWER side\n      else {\n        d_y = d_y_lower;\n      }\n\n      if (isFinite(slope)) {\n        T_x = Math.min(d_x, (d_y / Math.abs(slope)));\n      }\n\n      if (slope !== 0) {\n        T_y = Math.min(d_y, (d_x * Math.abs(slope)));\n      }\n\n      if (x_a > x_b) {\n        T_x = -1 * T_x;\n      }\n\n      if (y_a > y_b) {\n        T_y = -1 * T_y;\n      }\n\n      this.fishEyeViewMoveNode(sibling, T_x, T_y, nodeToExpand, singleNotSimple, animate, layoutBy);\n    }\n\n    if (siblings.length == 0) {\n      this.expandNodeBaseFunction(nodeToExpand, singleNotSimple, true, layoutBy);\n    }\n\n    if (node.parent()[0] != null) {\n      this.fishEyeViewExpandGivenNode(node.parent()[0], singleNotSimple, nodeToExpand, animate, layoutBy);\n    }\n\n    return node;\n  },\n  getSiblings: function (node) {//*//\n    var siblings;\n\n    if (node.parent()[0] == null) {\n      siblings = cy.collection();\n      var orphans = cy.nodes().orphans();\n\n      for (var i = 0; i < orphans.length; i++) {\n        if (orphans[i] != node) {\n          siblings = siblings.add(orphans[i]);\n        }\n      }\n    } else {\n      siblings = node.siblings();\n    }\n\n    return siblings;\n  },\n  /*\n   * Move node operation specialized for fish eye view expand operation\n   * Moves the node by moving its descandents. Movement is animated if singleNotSimple flag is truthy.\n   */\n  fishEyeViewMoveNode: function (node, T_x, T_y, nodeToExpand, singleNotSimple, animate, layoutBy) {//*//\n    var childrenList = node.children();\n    var self = this;\n\n    if (childrenList.length == 0) {\n      var newPosition = {x: node.position('x') + T_x, y: node.position('y') + T_y};\n      if (!singleNotSimple || !animate) {\n        node.position(newPosition);\n      }\n      else {\n        this.animatedlyMovingNodeCount++;\n        node.animate({\n          position: newPosition,\n          complete: function () {\n            self.animatedlyMovingNodeCount--;\n            if (self.animatedlyMovingNodeCount > 0 || nodeToExpand.data('expanded-collapsed') === 'expanded') {\n\n              return;\n            }\n\n            self.expandNodeBaseFunction(nodeToExpand, singleNotSimple, true, layoutBy);\n\n          }\n        }, {\n          duration: 1000\n        });\n      }\n    }\n    else {\n\n      for (var i = 0; i < childrenList.length; i++) {\n        this.fishEyeViewMoveNode(childrenList[i], T_x, T_y, nodeToExpand, singleNotSimple, animate, layoutBy);\n      }\n    }\n  },\n  xPositionInParent: function (node) {//*//\n    var parent = node.parent()[0];\n    var x_a = 0.0;\n\n    // Given node is not a direct child of the the root graph\n    if (parent != null) {\n      x_a = node.relativePosition('x') + (parent.width() / 2);\n    }\n    // Given node is a direct child of the the root graph\n\n    else {\n      x_a = node.position('x');\n    }\n\n    return x_a;\n  },\n  yPositionInParent: function (node) {//*//\n    var parent = node.parent()[0];\n\n    var y_a = 0.0;\n\n    // Given node is not a direct child of the the root graph\n    if (parent != null) {\n      y_a = node.relativePosition('y') + (parent.height() / 2);\n    }\n    // Given node is a direct child of the the root graph\n\n    else {\n      y_a = node.position('y');\n    }\n\n    return y_a;\n  },\n  /*\n   * for all children of the node parameter call this method\n   * with the same root parameter,\n   * remove the child and add the removed child to the collapsedchildren data\n   * of the root to restore them in the case of expandation\n   * root._private.data.collapsedChildren keeps the nodes to restore when the\n   * root is expanded\n   */\n  removeChildren: function (node, root) {//*//\n    var children = node.children();\n    for (var i = 0; i < children.length; i++) {\n      var child = children[i];\n      this.removeChildren(child, root);\n      var removedChild = child.remove();\n      if (root._private.data.collapsedChildren == null) {\n        root._private.data.collapsedChildren = removedChild;\n      }\n      else {\n        root._private.data.collapsedChildren = root._private.data.collapsedChildren.union(removedChild);\n      }\n    }\n  },\n  isMetaEdge: function(edge) {\n    return edge.hasClass(\"meta\");\n  },\n  barrowEdgesOfcollapsedChildren: function(node) {\n    var relatedNodes = node.descendants();\n    var edges = relatedNodes.edgesWith(cy.nodes().not(relatedNodes.union(node)));\n    \n    var relatedNodeMap = {};\n    \n    relatedNodes.each(function(i, ele) {\n      relatedNodeMap[ele.id()] = true;\n    });\n    \n    for (var i = 0; i < edges.length; i++) {\n      var edge = edges[i];\n      var source = edge.source();\n      var target = edge.target();\n      \n      if (!this.isMetaEdge(edge)) { // is original\n        var originalEndsData = {\n          source: source,\n          target: target\n        };\n        \n        edge.addClass(\"meta\");\n        edge.data('originalEnds', originalEndsData);\n      }\n      \n      edge.move({\n        target: !relatedNodeMap[target.id()] ? target.id() : node.id(),\n        source: !relatedNodeMap[source.id()] ? source.id() : node.id()\n      });\n    }\n  },\n  findNewEnd: function(node) {\n    var current = node;\n    \n    while( !current.inside() ) {\n      current = current.parent();\n    }\n    \n    return current;\n  },\n  repairEdges: function(node) {\n    var connectedMetaEdges = node.connectedEdges('.meta');\n    \n    for (var i = 0; i < connectedMetaEdges.length; i++) {\n      var edge = connectedMetaEdges[i];\n      var originalEnds = edge.data('originalEnds');\n      var currentSrcId = edge.data('source');\n      var currentTgtId = edge.data('target');\n      \n      if ( currentSrcId === node.id() ) {\n        edge = edge.move({\n          source: this.findNewEnd(originalEnds.source).id()\n        });\n      } else {\n        edge = edge.move({\n          target: this.findNewEnd(originalEnds.target).id()\n        });\n      }\n      \n      if ( edge.data('source') === originalEnds.source.id() && edge.data('target') === originalEnds.target.id() ) {\n        edge.removeClass('meta');\n        edge.removeData('originalEnds');\n      }\n    }\n  },\n  /*node is an outer node of root\n   if root is not it's anchestor\n   and it is not the root itself*/\n  isOuterNode: function (node, root) {//*//\n    var temp = node;\n    while (temp != null) {\n      if (temp == root) {\n        return false;\n      }\n      temp = temp.parent()[0];\n    }\n    return true;\n  }\n}\n};\n\nmodule.exports = expandCollapseUtilities;\n",";\n(function ($$, $) {\n  'use strict';\n\n  // registers the extension on a cytoscape lib ref\n  var register = function (cytoscape, $) {\n\n    if (!cytoscape) {\n      return;\n    } // can't register if cytoscape unspecified\n\n    var expandCollapseUtilities = require('./expandCollapseUtilities');\n    var undoRedoUtilities = require('./undoRedoUtilities');\n    var elementUtilities = require('./elementUtilities');\n    $.fn.cytoscapeExpandCollapse = require(\"./cueUtilities\");\n\n    var options = {\n      layoutBy: null, // for rearrange after expand/collapse. It's just layout options or whole layout function. Choose your side!\n      fisheye: true, // whether to perform fisheye view after expand/collapse you can specify a function too\n      animate: true, // whether to animate on drawing changes you can specify a function too\n      ready: function () { }, // callback when expand/collapse initialized\n      undoable: true, // and if undoRedoExtension exists,\n\n      cueEnabled: true, // Whether cues are enabled\n      expandCollapseCuePosition: 'top-left', // default cue position is top left you can specify a function per node too\n      expandCollapseCueSize: 12, // size of expand-collapse cue\n      expandCollapseCueLineSize: 8, // size of lines used for drawing plus-minus icons\n      expandCueImage: undefined, // image of expand icon if undefined draw regular expand cue\n      collapseCueImage: undefined, // image of collapse icon if undefined draw regular collapse cue\n      expandCollapseCueSensitivity: 1 // sensitivity of expand-collapse cues\n    };\n\n    function setOptions(from) {\n      var tempOpts = {};\n      for (var key in options)\n        tempOpts[key] = options[key];\n\n      for (var key in from)\n        if (tempOpts.hasOwnProperty(key))\n          tempOpts[key] = from[key];\n      return tempOpts;\n    }\n    \n    // evaluate some specific options in case of they are specified as functions to be dynamically changed\n    function evalOptions(options) {\n      var animate = typeof options.animate === 'function' ? options.animate.call() : options.animate;\n      var fisheye = typeof options.fisheye === 'function' ? options.fisheye.call() : options.fisheye;\n      \n      options.animate = animate;\n      options.fisheye = fisheye;\n    }\n\n\n    // cy.expandCollapse()\n    cytoscape(\"core\", \"expandCollapse\", function (opts) {\n      var cy = this;\n      options = setOptions(opts);\n\n      // All parent nodes are expanded on load\n      cy.nodes(':parent').data('expanded-collapsed', 'expanded');\n      undoRedoUtilities(cy);\n      \n      if(options.cueEnabled)\n        $(cy.container()).cytoscapeExpandCollapse(options);\n      else\n        $(cy.container()).cytoscapeExpandCollapse(\"unbind\");\n\n\n      options.ready();\n\n\n      return cy;\n    });\n    \n    // set functions\n    \n    // set all options at once\n    cytoscape(\"core\", \"setExpandCollapseOptions\", function (opts) {\n      options = opts;\n    });\n    \n    // set the option whose name is given\n    cytoscape(\"core\", \"setExpandCollapseOption\", function (name, value) {\n      options[name] = value;\n    });\n\n    // Collection functions\n\n    // eles.collapse(options)\n    cytoscape('collection', 'collapse', function (opts) {\n      var eles = this.collapsibleNodes();\n      var tempOptions = setOptions(opts);\n      evalOptions(tempOptions);\n\n      return expandCollapseUtilities(this.cy()).collapseGivenNodes(eles, tempOptions);\n    });\n\n    // eles.collapseAll(options)\n    cytoscape('collection', 'collapseRecursively', function (opts) {\n      var eles = this.collapsibleNodes();\n      var tempOptions = setOptions(opts);\n      evalOptions(tempOptions);\n\n      return eles.union(eles.descendants()).collapse(tempOptions);\n    });\n\n    // eles.expand(options)\n    cytoscape('collection', 'expand', function (opts) {\n      var eles = this.expandableNodes();\n      var tempOptions = setOptions(opts);\n      evalOptions(tempOptions);\n\n      return expandCollapseUtilities(this.cy()).expandGivenNodes(eles, tempOptions);\n    });\n\n    // eles.expandAll(options)\n    cytoscape('collection', 'expandRecursively', function (opts) {\n      var eles = this.expandableNodes();\n      var tempOptions = setOptions(opts);\n      evalOptions(tempOptions);\n\n      return expandCollapseUtilities(this.cy()).expandAllNodes(eles, tempOptions);\n    });\n\n\n    // Core functions\n\n    // cy.collapseAll(options)\n    cytoscape('core', 'collapseAll', function (opts) {\n      var cy = this;\n      var tempOptions = setOptions(opts);\n      evalOptions(tempOptions);\n\n      return cy.collapsibleNodes().collapseRecursively(tempOptions);\n    });\n\n    // cy.expandAll(options)\n    cytoscape('core', 'expandAll', function (opts) {\n      var cy = this;\n      var tempOptions = setOptions(opts);\n      evalOptions(tempOptions);\n\n      return cy.expandableNodes().expandRecursively(tempOptions);\n    });\n\n\n    // Utility functions\n\n    // ele.isCollapsible()\n    cytoscape('collection', 'isExpandable', function () {\n      var ele = this;\n\n      return (ele.data(\"expanded-collapsed\") === \"collapsed\");\n    });\n\n    // ele.isExpandable()\n    cytoscape('collection', 'isCollapsible', function () {\n      var ele = this;\n      return !ele.isExpandable() && ele.isParent();\n    });\n\n    // eles.collapsed()\n    cytoscape('collection', 'collapsibleNodes', function () {\n      var eles = this;\n\n      return eles.filter(function (i, ele) {\n        return ele.isCollapsible();\n      });\n    });\n\n    // eles.expanded()\n    cytoscape('collection', 'expandableNodes', function () {\n      var eles = this;\n\n      return eles.filter(function (i, ele) {\n        return ele.isExpandable();\n      });\n    });\n    // eles.collapsed()\n    cytoscape('core', 'collapsibleNodes', function () {\n      var cy = this;\n\n      return cy.nodes().collapsibleNodes();\n    });\n\n    // eles.expanded()\n    cytoscape('core', 'expandableNodes', function () {\n      var cy = this;\n\n      return cy.nodes().expandableNodes();\n    });\n  };\n\n  if (typeof module !== 'undefined' && module.exports) { // expose as a commonjs module\n    module.exports = register;\n  }\n\n  if (typeof define !== 'undefined' && define.amd) { // expose as an amd/requirejs module\n    define('cytoscape-expand-collapse', function () {\n      return register;\n    });\n  }\n\n    if (typeof cytoscape !== 'undefined' && typeof jQuery !== 'undefined') { // expose to global cytoscape (i.e. window.cytoscape)\n      register(cytoscape, jQuery);\n  }\n\n})();\n","module.exports = function (cy) {\n  if (cy.undoRedo == null)\n    return;\n\n  var ur = cy.undoRedo({}, true);\n\n  function getEles(_eles) {\n    return (typeof _eles === \"string\") ? cy.$(_eles) : _eles;\n  }\n\n  function getNodePositionsAndSizes() {\n    var positionsAndSizes = {};\n    var nodes = cy.nodes();\n\n    for (var i = 0; i < nodes.length; i++) {\n      var ele = nodes[i];\n      positionsAndSizes[ele.id()] = {\n        width: ele.width(),\n        height: ele.height(),\n        x: ele.position(\"x\"),\n        y: ele.position(\"y\")\n      };\n    }\n\n    return positionsAndSizes;\n  }\n\n  function returnToPositionsAndSizes(nodesData) {\n    var currentPositionsAndSizes = {};\n    cy.nodes().positions(function (i, ele) {\n      currentPositionsAndSizes[ele.id()] = {\n        width: ele.width(),\n        height: ele.height(),\n        x: ele.position(\"x\"),\n        y: ele.position(\"y\")\n      };\n      var data = nodesData[ele.id()];\n      ele._private.data.width = data.width;\n      ele._private.data.height = data.height;\n      return {\n        x: data.x,\n        y: data.y\n      };\n    });\n\n    return currentPositionsAndSizes;\n  }\n\n  var secondTimeOpts = {\n    layoutBy: null,\n    animate: false,\n    fisheye: false\n  };\n\n  function doIt(func) {\n    return function (args) {\n      var result = {};\n      var nodes = getEles(args.nodes);\n      if (args.firstTime) {\n        result.oldData = getNodePositionsAndSizes();\n        result.nodes = func.indexOf(\"All\") > 0 ? cy[func](args.options) : nodes[func](args.options);\n      } else {\n        result.oldData = getNodePositionsAndSizes();\n        result.nodes = func.indexOf(\"All\") > 0 ? cy[func](secondTimeOpts) : cy.collection(nodes)[func](secondTimeOpts);\n        returnToPositionsAndSizes(args.oldData);\n      }\n\n      return result;\n    };\n  }\n\n  var actions = [\"collapse\", \"collapseRecursively\", \"collapseAll\", \"expand\", \"expandRecursively\", \"expandAll\"];\n\n  for (var i = 0; i < actions.length; i++) {\n    ur.action(actions[i], doIt(actions[i]), doIt(actions[(i + 3) % 6]));\n  }\n\n};\n"]}
diff --git a/src/cueUtilities.js b/src/cueUtilities.js
index b21b510..05b2517 100644
--- a/src/cueUtilities.js
+++ b/src/cueUtilities.js
@@ -8,8 +8,8 @@ module.exports = function (params) {
init: function () {
var self = this;
var opts = params;
- var $container = $(this);
- var cy;
+ var $container = this;
+ var cy = this.cytoscape('get');
var $canvas = $('');
$container.append($canvas);
@@ -292,6 +292,7 @@ module.exports = function (params) {
$container.data('cyexpandcollapse', data);
},
unbind: function () {
+ var cy = this.cytoscape('get');
cy.off('mouseover', 'node', eMouseOver)
.off('mouseout tapdragout', 'node', eMouseOut)
.off('position', 'node', ePosition)
@@ -313,4 +314,4 @@ module.exports = function (params) {
}
return $(this);
-};
\ No newline at end of file
+};