From 5e61c5529acf81c5c4a3524f6c1b82341c692e49 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Hasan=20Balc=C4=B1?= Date: Mon, 30 Mar 2020 13:45:56 +0300 Subject: [PATCH 1/6] Update demo file --- demo.html | 31 ------------------------------- 1 file changed, 31 deletions(-) diff --git a/demo.html b/demo.html index 3385461..11e8ba7 100644 --- a/demo.html +++ b/demo.html @@ -104,37 +104,6 @@ background: white; } - /*/* creating color boxes of highlighting colors*/ - /*.topnav input[type="color"] { - float: left; - width: 50px; - height: 30px; - margin: 5px; - border: 2px solid white; - border-style: solid; - cursor: pointer; - } -*/ - /*.topnav input[type="color"]:hover - { - border: 2px solid black; - }*/ - - .topnav button { - display: inline-block; - background-color: #7b38d8; - border-radius: 10px; - border: 4px double #cccccc; - color: #eeeeee; - text-align: center; - font-size: 28px; - padding: 20px; - width: 200px; - transition: all 0.5s; - cursor: pointer; - margin-left: 10px; - } - .topnav c { float: left; width: 30px; From f1bd372cd600a25ba453e5e7328772d90420a1ee Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Hasan=20Balc=C4=B1?= Date: Mon, 30 Mar 2020 13:49:23 +0300 Subject: [PATCH 2/6] Update package.json --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 1b6a425..7ac535f 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "cytoscape-view-utilities", - "version": "3.0.0", + "version": "4.0.0", "description": "Package of view utilities for cytoscape.js", "main": "src/index.js", "spm": { From 5a0d771a07b0a749ae8897f583c4d2cf31e696c4 Mon Sep 17 00:00:00 2001 From: canbax Date: Tue, 7 Apr 2020 17:29:42 +0300 Subject: [PATCH 3/6] use batch for faster hightlight/removeHighlights --- src/view-utilities.js | 29 +++++++++++++++-------------- 1 file changed, 15 insertions(+), 14 deletions(-) diff --git a/src/view-utilities.js b/src/view-utilities.js index 2ea1b4f..da1a480 100644 --- a/src/view-utilities.js +++ b/src/view-utilities.js @@ -34,13 +34,15 @@ var viewUtilities = function (cy, options) { // Helper functions for internal usage (not to be exposed) function highlight(eles, idx) { - for (var i = 0; i < options.highlightStyles.length; i++) { - var className = getCyClassName4Idx(i); - eles.removeClass(className); - } - var className = getCyClassName4Idx(idx); - eles.addClass(className); - eles.unselect(); + cy.batch(() => { + for (var i = 0; i < options.highlightStyles.length; i++) { + var className = getCyClassName4Idx(i); + eles.removeClass(className); + } + var className = getCyClassName4Idx(idx); + eles.addClass(className); + eles.unselect(); + }); } function getWithNeighbors(eles) { @@ -123,14 +125,13 @@ var viewUtilities = function (cy, options) { if (eles == null || eles.length == null) { eles = cy.elements(); } - - for (var i = 0; i < options.highlightStyles.length; i++) { - var className = getCyClassName4Idx(i); - eles.removeClass(className); - eles.removeData(className); - } + cy.batch(() => { + for (var i = 0; i < options.highlightStyles.length; i++) { + var className = getCyClassName4Idx(i); + eles.removeClass(className); + } + }); return eles; - // TODO check if remove data is needed here }; // Indicates if the ele is highlighted From 7650b26dcbc87a790e070d059044f82862e8fc92 Mon Sep 17 00:00:00 2001 From: canbax Date: Mon, 13 Apr 2020 15:06:27 +0300 Subject: [PATCH 4/6] functionality to remove highlight style --- README.md | 4 +++ cytoscape-view-utilities.js | 48 +++++++++++++++++-------------- src/view-utilities.js | 57 ++++++++++++++++++++----------------- 3 files changed, 62 insertions(+), 47 deletions(-) diff --git a/README.md b/README.md index a60ef24..541e715 100644 --- a/README.md +++ b/README.md @@ -102,6 +102,10 @@ Changes the style specified with `idx`. @param edgeStyle — [cytoscape style](https://js.cytoscape.org/#style) for edges
Adds a new style to the `highlightStyles` array. +`instance.removeHighlightStyle(styleIdx): void`
+@param styleIdx — index of the style to delete (0 based)
+Removes the style from `highlightStyles` array. + ## Default Options ``` highlightStyles: [], diff --git a/cytoscape-view-utilities.js b/cytoscape-view-utilities.js index 4bcab50..e614ad0 100644 --- a/cytoscape-view-utilities.js +++ b/cytoscape-view-utilities.js @@ -246,10 +246,12 @@ module.exports = function (cy, ur, viewUtilities) { },{}],3:[function(_dereq_,module,exports){ var viewUtilities = function (cy, options) { + var classNames4Styles = []; init(); function init() { // add provided styles for (var i = 0; i < options.highlightStyles.length; i++) { + classNames4Styles.push('__highligtighted__' + i); updateCyStyle(i); } @@ -267,26 +269,22 @@ var viewUtilities = function (cy, options) { } function updateCyStyle(classIdx) { - var className = getCyClassName4Idx(classIdx); + var className = classNames4Styles[classIdx]; var cssNode = options.highlightStyles[classIdx].node; var cssEdge = options.highlightStyles[classIdx].edge; cy.style().selector('node.' + className).css(cssNode).update(); cy.style().selector('edge.' + className).css(cssEdge).update(); } - function getCyClassName4Idx(i) { - return '__highligtighted__' + i; - } - // Helper functions for internal usage (not to be exposed) function highlight(eles, idx) { + cy.startBatch(); for (var i = 0; i < options.highlightStyles.length; i++) { - var className = getCyClassName4Idx(i); - eles.removeClass(className); + eles.removeClass(classNames4Styles[i]); } - var className = getCyClassName4Idx(idx); - eles.addClass(className); + eles.addClass(classNames4Styles[idx]); eles.unselect(); + cy.endBatch(); } function getWithNeighbors(eles) { @@ -353,7 +351,7 @@ var viewUtilities = function (cy, options) { highlight(eles, idx); // Use the helper here return eles; }; - + instance.getHighlightStyles = function () { return options.highlightStyles; }; @@ -366,25 +364,22 @@ var viewUtilities = function (cy, options) { // Remove highlights from eles. // If eles is not defined considers cy.elements() instance.removeHighlights = function (eles) { + cy.startBatch(); if (eles == null || eles.length == null) { eles = cy.elements(); } - for (var i = 0; i < options.highlightStyles.length; i++) { - var className = getCyClassName4Idx(i); - eles.removeClass(className); - eles.removeData(className); + eles.removeClass(classNames4Styles[i]); } + cy.endBatch(); return eles; - // TODO check if remove data is needed here }; // Indicates if the ele is highlighted instance.isHighlighted = function (ele) { var isHigh = false; for (var i = 0; i < options.highlightStyles.length; i++) { - var className = getCyClassName4Idx(i); - if (ele.is('.' + className + ':visible')) { + if (ele.is('.' + classNames4Styles[i] + ':visible')) { isHigh = true; } } @@ -401,16 +396,27 @@ var viewUtilities = function (cy, options) { instance.addHighlightStyle = function (nodeStyle, edgeStyle) { var o = { node: nodeStyle, edge: edgeStyle }; options.highlightStyles.push(o); + var s = '__highligtighted__' + (options.highlightStyles.length - 1); + classNames4Styles.push(s); updateCyStyle(options.highlightStyles.length - 1); addSelectionStyles(); }; - instance.getAllHighlightClasses = function() { + instance.removeHighlightStyle = function (styleIdx) { + if (styleIdx < 0 || styleIdx > options.highlightStyles.length - 1) { + return; + } + cy.elements().removeClass(classNames4Styles[styleIdx]); + options.highlightStyles.splice(styleIdx, 1); + classNames4Styles.splice(styleIdx, 1); + }; + + instance.getAllHighlightClasses = function () { var a = []; for (var i = 0; i < options.highlightStyles.length; i++) { - a.push(getCyClassName4Idx(i)); + a.push(classNames4Styles[i]); } - return a; + return classNames4Styles; }; //Zoom selected Nodes @@ -553,4 +559,4 @@ module.exports = viewUtilities; },{}]},{},[1])(1) }); -//# sourceMappingURL=data:application/json;charset:utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/index.js","src/undo-redo.js","src/view-utilities.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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})",";\r\n(function () {\r\n  'use strict';\r\n\r\n  // registers the extension on a cytoscape lib ref\r\n  var register = function (cytoscape) {\r\n\r\n    if (!cytoscape) {\r\n      return;\r\n    } // can't register if cytoscape unspecified\r\n\r\n    var options = {\r\n      highlightStyles: [],\r\n      selectStyles: {},\r\n      setVisibilityOnHide: false, // whether to set visibility on hide/show\r\n      setDisplayOnHide: true, // whether to set display on hide/show\r\n      zoomAnimationDuration: 1500, //default duration for zoom animation speed\r\n      neighbor: function (node) { // return desired neighbors of tapheld node\r\n        return false;\r\n      },\r\n      neighborSelectTime: 500 //ms, time to taphold to select desired neighbors\r\n    };\r\n\r\n    var undoRedo = require(\"./undo-redo\");\r\n    var viewUtilities = require(\"./view-utilities\");\r\n\r\n    cytoscape('core', 'viewUtilities', function (opts) {\r\n      var cy = this;\r\n\r\n      function getScratch(eleOrCy) {\r\n        if (!eleOrCy.scratch(\"_viewUtilities\")) {\r\n          eleOrCy.scratch(\"_viewUtilities\", {});\r\n        }\r\n\r\n        return eleOrCy.scratch(\"_viewUtilities\");\r\n      }\r\n      \r\n      // If 'get' is given as the param then return the extension instance\r\n      if (opts === 'get') {\r\n        return getScratch(cy).instance;\r\n      }\r\n      \r\n      /**\r\n      * Deep copy or merge objects - replacement for jQuery deep extend\r\n      * Taken from http://youmightnotneedjquery.com/#deep_extend\r\n      * and bug related to deep copy of Arrays is fixed.\r\n      * Usage:Object.extend({}, objA, objB)\r\n      */\r\n      function extendOptions(out) {\r\n        out = out || {};\r\n\r\n        for (var i = 1; i < arguments.length; i++) {\r\n          var obj = arguments[i];\r\n\r\n          if (!obj)\r\n            continue;\r\n\r\n          for (var key in obj) {\r\n            if (obj.hasOwnProperty(key)) {\r\n              if (Array.isArray(obj[key])) {\r\n                out[key] = obj[key].slice();\r\n              } else if (typeof obj[key] === 'object') {\r\n                out[key] = extendOptions(out[key], obj[key]);\r\n              } else {\r\n                out[key] = obj[key];\r\n              }\r\n            }\r\n          }\r\n        }\r\n\r\n        return out;\r\n      };\r\n\r\n      options = extendOptions({}, options, opts);\r\n\r\n      // create a view utilities instance\r\n      var instance = viewUtilities(cy, options);\r\n\r\n      if (cy.undoRedo) {\r\n        var ur = cy.undoRedo(null, true);\r\n        undoRedo(cy, ur, instance);\r\n      }\r\n\r\n      // set the instance on the scratch pad\r\n      getScratch(cy).instance = instance;\r\n\r\n      if (!getScratch(cy).initialized) {\r\n        getScratch(cy).initialized = true;\r\n\r\n        var shiftKeyDown = false;\r\n        document.addEventListener('keydown', function(event){\r\n          if(event.key == \"Shift\") {\r\n            shiftKeyDown = true;\r\n          }\r\n        });\r\n        document.addEventListener('keyup', function(event){\r\n          if(event.key == \"Shift\") {\r\n            shiftKeyDown = false;\r\n          }\r\n        });\r\n        //Select the desired neighbors after taphold-and-free\r\n        cy.on('taphold', 'node', function(event){\r\n          var target = event.target || event.cyTarget;\r\n          var tapheld = false;\r\n          var neighborhood;\r\n          var timeout = setTimeout(function(){\r\n            if(shiftKeyDown){\r\n              cy.elements().unselect();\r\n              neighborhood = options.neighbor(target);\r\n              if(neighborhood)\r\n                neighborhood.select();\r\n              target.lock();\r\n              tapheld = true;\r\n            }\r\n          }, options.neighborSelectTime - 500);\r\n          cy.on('free', 'node', function(){\r\n            var targetTapheld = event.target || event.cyTarget;\r\n            if(target == targetTapheld && tapheld === true){\r\n              tapheld = false;\r\n              if(neighborhood)\r\n                neighborhood.select();\r\n              target.unlock();\r\n            }\r\n            else{\r\n              clearTimeout(timeout);\r\n            }\r\n          });\r\n          cy.on('drag', 'node', function(){\r\n            var targetDragged = event.target || event.cyTarget;\r\n            if(target == targetDragged && tapheld === false){\r\n              clearTimeout(timeout);\r\n            }\r\n          })\r\n        });\r\n      }\r\n\r\n      // return the instance of extension\r\n      return getScratch(cy).instance;\r\n    });\r\n\r\n  };\r\n\r\n  if (typeof module !== 'undefined' && module.exports) { // expose as a commonjs module\r\n    module.exports = register;\r\n  }\r\n\r\n  if (typeof define !== 'undefined' && define.amd) { // expose as an amd/requirejs module\r\n    define('cytoscape-view-utilities', function () {\r\n      return register;\r\n    });\r\n  }\r\n\r\n  if (typeof cytoscape !== 'undefined') { // expose to global cytoscape (i.e. window.cytoscape)\r\n    register(cytoscape);\r\n  }\r\n\r\n})();\r\n","// Registers ur actions related to highlight\r\nfunction highlightUR(cy, ur, viewUtilities) {\r\n  function getStatus(eles) {\r\n    eles = eles ? eles : cy.elements();\r\n    var classes = viewUtilities.getAllHighlightClasses();\r\n    var r = [];\r\n    for (var i = 0; i < classes.length; i++) {\r\n      r.push(eles.filter(`.${classes[i]}:visible`))\r\n    }\r\n    var selector = classes.map(x => '.' + x).join(',');\r\n    // last element of array is elements which are not highlighted by any style\r\n    r.push(eles.filter(\":visible\").not(selector));\r\n    \r\n    return r;\r\n  }\r\n\r\n  function generalUndo(args) {\r\n    var current = args.current;\r\n    var r = [];\r\n    for (var i = 0; i < args.length - 1; i++) {\r\n      r.push(viewUtilities.highlight(args[i], i));\r\n    }\r\n    // last element is for not highlighted by any style\r\n    r.push(viewUtilities.removeHighlights(args[args.length - 1]));\r\n\r\n    r['current'] = current;\r\n    return r;\r\n  }\r\n\r\n  function generalRedo(args) {\r\n    var current = args.current;\r\n    var r = [];\r\n    for (var i = 0; i < current.length - 1; i++) {\r\n      r.push(viewUtilities.highlight(current[i], i));\r\n    }\r\n    // last element is for not highlighted by any style\r\n    r.push(viewUtilities.removeHighlights(current[current.length - 1]));\r\n\r\n    r['current'] = current;\r\n    return r;\r\n  }\r\n\r\n  function generateDoFunc(func) {\r\n    return function (args) {\r\n      var res = getStatus();\r\n      if (args.firstTime)\r\n        viewUtilities[func](args.eles, args.idx);\r\n      else\r\n        generalRedo(args);\r\n\r\n      res.current = getStatus();\r\n\r\n      return res;\r\n    };\r\n  }\r\n\r\n  ur.action(\"highlightNeighbors\", generateDoFunc(\"highlightNeighbors\"), generalUndo);\r\n  ur.action(\"highlight\", generateDoFunc(\"highlight\"), generalUndo);\r\n  ur.action(\"removeHighlights\", generateDoFunc(\"removeHighlights\"), generalUndo);\r\n}\r\n\r\n// Registers ur actions related to hide/show\r\nfunction hideShowUR(cy, ur, viewUtilities) {\r\n  function urShow(eles) {\r\n    return viewUtilities.show(eles);\r\n  }\r\n\r\n  function urHide(eles) {\r\n    return viewUtilities.hide(eles);\r\n  }\r\n\r\n  function urShowHiddenNeighbors(eles) {\r\n    return viewUtilities.showHiddenNeighbors(eles);\r\n  }\r\n\r\n  ur.action(\"show\", urShow, urHide);\r\n  ur.action(\"hide\", urHide, urShow);\r\n  ur.action(\"showHiddenNeighbors\",urShowHiddenNeighbors, urHide);\r\n}\r\n\r\nmodule.exports = function (cy, ur, viewUtilities) {\r\n  highlightUR(cy, ur, viewUtilities);\r\n  hideShowUR(cy, ur, viewUtilities);\r\n};\r\n","var viewUtilities = function (cy, options) {\r\n\r\n  init();\r\n  function init() {\r\n    // add provided styles\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      updateCyStyle(i);\r\n    }\r\n\r\n    // add styles for selected\r\n    addSelectionStyles();\r\n  }\r\n\r\n  function addSelectionStyles() {\r\n    if (options.selectStyles.node) {\r\n      cy.style().selector('node:selected').css(options.selectStyles.node).update();\r\n    }\r\n    if (options.selectStyles.edge) {\r\n      cy.style().selector('edge:selected').css(options.selectStyles.edge).update();\r\n    }\r\n  }\r\n\r\n  function updateCyStyle(classIdx) {\r\n    var className = getCyClassName4Idx(classIdx);\r\n    var cssNode = options.highlightStyles[classIdx].node;\r\n    var cssEdge = options.highlightStyles[classIdx].edge;\r\n    cy.style().selector('node.' + className).css(cssNode).update();\r\n    cy.style().selector('edge.' + className).css(cssEdge).update();\r\n  }\r\n\r\n  function getCyClassName4Idx(i) {\r\n    return '__highligtighted__' + i;\r\n  }\r\n\r\n  // Helper functions for internal usage (not to be exposed)\r\n  function highlight(eles, idx) {\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      var className = getCyClassName4Idx(i);\r\n      eles.removeClass(className);\r\n    }\r\n    var className = getCyClassName4Idx(idx);\r\n    eles.addClass(className);\r\n    eles.unselect();\r\n  }\r\n\r\n  function getWithNeighbors(eles) {\r\n    return eles.add(eles.descendants()).closedNeighborhood();\r\n  }\r\n  // the instance to be returned\r\n  var instance = {};\r\n\r\n  // Section hide-show\r\n  // hide given eles\r\n  instance.hide = function (eles) {\r\n    //eles = eles.filter(\"node\")\r\n    eles = eles.filter(\":visible\");\r\n    eles = eles.union(eles.connectedEdges());\r\n\r\n    eles.unselect();\r\n\r\n    if (options.setVisibilityOnHide) {\r\n      eles.css('visibility', 'hidden');\r\n    }\r\n\r\n    if (options.setDisplayOnHide) {\r\n      eles.css('display', 'none');\r\n    }\r\n\r\n    return eles;\r\n  };\r\n\r\n  // unhide given eles\r\n  instance.show = function (eles) {\r\n    eles = eles.not(\":visible\");\r\n\r\n    var connectedEdges = eles.connectedEdges(function (edge) {\r\n\r\n      if ((edge.source().visible() || eles.contains(edge.source())) && (edge.target().visible() || eles.contains(edge.target()))) {\r\n        return true;\r\n      } else {\r\n        return false;\r\n      }\r\n\r\n    });\r\n    eles = eles.union(connectedEdges);\r\n\r\n    eles.unselect();\r\n\r\n    if (options.setVisibilityOnHide) {\r\n      eles.css('visibility', 'visible');\r\n    }\r\n\r\n    if (options.setDisplayOnHide) {\r\n      eles.css('display', 'element');\r\n    }\r\n\r\n    return eles;\r\n  };\r\n\r\n  // Section highlight\r\n  instance.showHiddenNeighbors = function (eles) {\r\n    return this.show(getWithNeighbors(eles));\r\n  };\r\n\r\n  // Highlights eles\r\n  instance.highlight = function (eles, idx = 0) {\r\n    highlight(eles, idx); // Use the helper here\r\n    return eles;\r\n  };\r\n \r\n  instance.getHighlightStyles = function () {\r\n    return options.highlightStyles;\r\n  };\r\n\r\n  // Highlights eles' neighborhood\r\n  instance.highlightNeighbors = function (eles, idx = 0) {\r\n    return this.highlight(getWithNeighbors(eles), idx);\r\n  };\r\n\r\n  // Remove highlights from eles.\r\n  // If eles is not defined considers cy.elements()\r\n  instance.removeHighlights = function (eles) {\r\n    if (eles == null || eles.length == null) {\r\n      eles = cy.elements();\r\n    }\r\n\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      var className = getCyClassName4Idx(i);\r\n      eles.removeClass(className);\r\n      eles.removeData(className);\r\n    }\r\n    return eles;\r\n    // TODO check if remove data is needed here\r\n  };\r\n\r\n  // Indicates if the ele is highlighted\r\n  instance.isHighlighted = function (ele) {\r\n    var isHigh = false;\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      var className = getCyClassName4Idx(i);\r\n      if (ele.is('.' + className + ':visible')) {\r\n        isHigh = true;\r\n      }\r\n    }\r\n    return isHigh;\r\n  };\r\n\r\n  instance.changeHighlightStyle = function (idx, nodeStyle, edgeStyle) {\r\n    options.highlightStyles[idx].node = nodeStyle;\r\n    options.highlightStyles[idx].edge = edgeStyle;\r\n    updateCyStyle(idx);\r\n    addSelectionStyles();\r\n  };\r\n\r\n  instance.addHighlightStyle = function (nodeStyle, edgeStyle) {\r\n    var o = { node: nodeStyle, edge: edgeStyle };\r\n    options.highlightStyles.push(o);\r\n    updateCyStyle(options.highlightStyles.length - 1);\r\n    addSelectionStyles();\r\n  };\r\n\r\n  instance.getAllHighlightClasses = function() {\r\n    var a = [];\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      a.push(getCyClassName4Idx(i));\r\n    }\r\n    return a;\r\n  };\r\n\r\n  //Zoom selected Nodes\r\n  instance.zoomToSelected = function (eles) {\r\n    var boundingBox = eles.boundingBox();\r\n    var diff_x = Math.abs(boundingBox.x1 - boundingBox.x2);\r\n    var diff_y = Math.abs(boundingBox.y1 - boundingBox.y2);\r\n    var padding;\r\n    if (diff_x >= 200 || diff_y >= 200) {\r\n      padding = 50;\r\n    }\r\n    else {\r\n      padding = (cy.width() < cy.height()) ?\r\n        ((200 - diff_x) / 2 * cy.width() / 200) : ((200 - diff_y) / 2 * cy.height() / 200);\r\n    }\r\n\r\n    cy.animate({\r\n      fit: {\r\n        eles: eles,\r\n        padding: padding\r\n      }\r\n    }, {\r\n      duration: options.zoomAnimationDuration\r\n    });\r\n    return eles;\r\n  };\r\n\r\n  //Marquee Zoom\r\n  var tabStartHandler;\r\n  var tabEndHandler;\r\n\r\n  instance.enableMarqueeZoom = function (callback) {\r\n\r\n    var shiftKeyDown = false;\r\n    var rect_start_pos_x, rect_start_pos_y, rect_end_pos_x, rect_end_pos_y;\r\n    //Make the cy unselectable\r\n    cy.autounselectify(true);\r\n\r\n    document.addEventListener('keydown', function (event) {\r\n      if (event.key == \"Shift\") {\r\n        shiftKeyDown = true;\r\n      }\r\n    });\r\n    document.addEventListener('keyup', function (event) {\r\n      if (event.key == \"Shift\") {\r\n        shiftKeyDown = false;\r\n      }\r\n    });\r\n\r\n    cy.one('tapstart', tabStartHandler = function (event) {\r\n      if (shiftKeyDown == true) {\r\n        rect_start_pos_x = event.position.x;\r\n        rect_start_pos_y = event.position.y;\r\n        rect_end_pos_x = undefined;\r\n      }\r\n    });\r\n    cy.one('tapend', tabEndHandler = function (event) {\r\n      rect_end_pos_x = event.position.x;\r\n      rect_end_pos_y = event.position.y;\r\n      //check whether corners of rectangle is undefined\r\n      //abort marquee zoom if one corner is undefined\r\n      if (rect_start_pos_x == undefined || rect_end_pos_x == undefined) {\r\n        cy.autounselectify(false);\r\n        if (callback) {\r\n          callback();\r\n        }\r\n        return;\r\n      }\r\n      //Reoder rectangle positions\r\n      //Top left of the rectangle (rect_start_pos_x, rect_start_pos_y)\r\n      //right bottom of the rectangle (rect_end_pos_x, rect_end_pos_y)\r\n      if (rect_start_pos_x > rect_end_pos_x) {\r\n        var temp = rect_start_pos_x;\r\n        rect_start_pos_x = rect_end_pos_x;\r\n        rect_end_pos_x = temp;\r\n      }\r\n      if (rect_start_pos_y > rect_end_pos_y) {\r\n        var temp = rect_start_pos_y;\r\n        rect_start_pos_y = rect_end_pos_y;\r\n        rect_end_pos_y = temp;\r\n      }\r\n\r\n      //Extend sides of selected rectangle to 200px if less than 100px\r\n      if (rect_end_pos_x - rect_start_pos_x < 200) {\r\n        var extendPx = (200 - (rect_end_pos_x - rect_start_pos_x)) / 2;\r\n        rect_start_pos_x -= extendPx;\r\n        rect_end_pos_x += extendPx;\r\n      }\r\n      if (rect_end_pos_y - rect_start_pos_y < 200) {\r\n        var extendPx = (200 - (rect_end_pos_y - rect_start_pos_y)) / 2;\r\n        rect_start_pos_y -= extendPx;\r\n        rect_end_pos_y += extendPx;\r\n      }\r\n\r\n      //Check whether rectangle intersects with bounding box of the graph\r\n      //if not abort marquee zoom\r\n      if ((rect_start_pos_x > cy.elements().boundingBox().x2)\r\n        || (rect_end_pos_x < cy.elements().boundingBox().x1)\r\n        || (rect_start_pos_y > cy.elements().boundingBox().y2)\r\n        || (rect_end_pos_y < cy.elements().boundingBox().y1)) {\r\n        cy.autounselectify(false);\r\n        if (callback) {\r\n          callback();\r\n        }\r\n        return;\r\n      }\r\n\r\n      //Calculate zoom level\r\n      var zoomLevel = Math.min(cy.width() / (Math.abs(rect_end_pos_x - rect_start_pos_x)),\r\n        cy.height() / Math.abs(rect_end_pos_y - rect_start_pos_y));\r\n\r\n      var diff_x = cy.width() / 2 - (cy.pan().x + zoomLevel * (rect_start_pos_x + rect_end_pos_x) / 2);\r\n      var diff_y = cy.height() / 2 - (cy.pan().y + zoomLevel * (rect_start_pos_y + rect_end_pos_y) / 2);\r\n\r\n      cy.animate({\r\n        panBy: { x: diff_x, y: diff_y },\r\n        zoom: zoomLevel,\r\n        duration: options.zoomAnimationDuration,\r\n        complete: function () {\r\n          if (callback) {\r\n            callback();\r\n          }\r\n          cy.autounselectify(false);\r\n        }\r\n      });\r\n    });\r\n  };\r\n\r\n  instance.disableMarqueeZoom = function () {\r\n    cy.off('tapstart', tabStartHandler);\r\n    cy.off('tapend', tabEndHandler);\r\n    cy.autounselectify(false);\r\n  };\r\n\r\n  // return the instance\r\n  return instance;\r\n};\r\n\r\nmodule.exports = viewUtilities;\r\n"]} +//# sourceMappingURL=data:application/json;charset:utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/index.js","src/undo-redo.js","src/view-utilities.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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})",";\r\n(function () {\r\n  'use strict';\r\n\r\n  // registers the extension on a cytoscape lib ref\r\n  var register = function (cytoscape) {\r\n\r\n    if (!cytoscape) {\r\n      return;\r\n    } // can't register if cytoscape unspecified\r\n\r\n    var options = {\r\n      highlightStyles: [],\r\n      selectStyles: {},\r\n      setVisibilityOnHide: false, // whether to set visibility on hide/show\r\n      setDisplayOnHide: true, // whether to set display on hide/show\r\n      zoomAnimationDuration: 1500, //default duration for zoom animation speed\r\n      neighbor: function (node) { // return desired neighbors of tapheld node\r\n        return false;\r\n      },\r\n      neighborSelectTime: 500 //ms, time to taphold to select desired neighbors\r\n    };\r\n\r\n    var undoRedo = require(\"./undo-redo\");\r\n    var viewUtilities = require(\"./view-utilities\");\r\n\r\n    cytoscape('core', 'viewUtilities', function (opts) {\r\n      var cy = this;\r\n\r\n      function getScratch(eleOrCy) {\r\n        if (!eleOrCy.scratch(\"_viewUtilities\")) {\r\n          eleOrCy.scratch(\"_viewUtilities\", {});\r\n        }\r\n\r\n        return eleOrCy.scratch(\"_viewUtilities\");\r\n      }\r\n      \r\n      // If 'get' is given as the param then return the extension instance\r\n      if (opts === 'get') {\r\n        return getScratch(cy).instance;\r\n      }\r\n      \r\n      /**\r\n      * Deep copy or merge objects - replacement for jQuery deep extend\r\n      * Taken from http://youmightnotneedjquery.com/#deep_extend\r\n      * and bug related to deep copy of Arrays is fixed.\r\n      * Usage:Object.extend({}, objA, objB)\r\n      */\r\n      function extendOptions(out) {\r\n        out = out || {};\r\n\r\n        for (var i = 1; i < arguments.length; i++) {\r\n          var obj = arguments[i];\r\n\r\n          if (!obj)\r\n            continue;\r\n\r\n          for (var key in obj) {\r\n            if (obj.hasOwnProperty(key)) {\r\n              if (Array.isArray(obj[key])) {\r\n                out[key] = obj[key].slice();\r\n              } else if (typeof obj[key] === 'object') {\r\n                out[key] = extendOptions(out[key], obj[key]);\r\n              } else {\r\n                out[key] = obj[key];\r\n              }\r\n            }\r\n          }\r\n        }\r\n\r\n        return out;\r\n      };\r\n\r\n      options = extendOptions({}, options, opts);\r\n\r\n      // create a view utilities instance\r\n      var instance = viewUtilities(cy, options);\r\n\r\n      if (cy.undoRedo) {\r\n        var ur = cy.undoRedo(null, true);\r\n        undoRedo(cy, ur, instance);\r\n      }\r\n\r\n      // set the instance on the scratch pad\r\n      getScratch(cy).instance = instance;\r\n\r\n      if (!getScratch(cy).initialized) {\r\n        getScratch(cy).initialized = true;\r\n\r\n        var shiftKeyDown = false;\r\n        document.addEventListener('keydown', function(event){\r\n          if(event.key == \"Shift\") {\r\n            shiftKeyDown = true;\r\n          }\r\n        });\r\n        document.addEventListener('keyup', function(event){\r\n          if(event.key == \"Shift\") {\r\n            shiftKeyDown = false;\r\n          }\r\n        });\r\n        //Select the desired neighbors after taphold-and-free\r\n        cy.on('taphold', 'node', function(event){\r\n          var target = event.target || event.cyTarget;\r\n          var tapheld = false;\r\n          var neighborhood;\r\n          var timeout = setTimeout(function(){\r\n            if(shiftKeyDown){\r\n              cy.elements().unselect();\r\n              neighborhood = options.neighbor(target);\r\n              if(neighborhood)\r\n                neighborhood.select();\r\n              target.lock();\r\n              tapheld = true;\r\n            }\r\n          }, options.neighborSelectTime - 500);\r\n          cy.on('free', 'node', function(){\r\n            var targetTapheld = event.target || event.cyTarget;\r\n            if(target == targetTapheld && tapheld === true){\r\n              tapheld = false;\r\n              if(neighborhood)\r\n                neighborhood.select();\r\n              target.unlock();\r\n            }\r\n            else{\r\n              clearTimeout(timeout);\r\n            }\r\n          });\r\n          cy.on('drag', 'node', function(){\r\n            var targetDragged = event.target || event.cyTarget;\r\n            if(target == targetDragged && tapheld === false){\r\n              clearTimeout(timeout);\r\n            }\r\n          })\r\n        });\r\n      }\r\n\r\n      // return the instance of extension\r\n      return getScratch(cy).instance;\r\n    });\r\n\r\n  };\r\n\r\n  if (typeof module !== 'undefined' && module.exports) { // expose as a commonjs module\r\n    module.exports = register;\r\n  }\r\n\r\n  if (typeof define !== 'undefined' && define.amd) { // expose as an amd/requirejs module\r\n    define('cytoscape-view-utilities', function () {\r\n      return register;\r\n    });\r\n  }\r\n\r\n  if (typeof cytoscape !== 'undefined') { // expose to global cytoscape (i.e. window.cytoscape)\r\n    register(cytoscape);\r\n  }\r\n\r\n})();\r\n","// Registers ur actions related to highlight\r\nfunction highlightUR(cy, ur, viewUtilities) {\r\n  function getStatus(eles) {\r\n    eles = eles ? eles : cy.elements();\r\n    var classes = viewUtilities.getAllHighlightClasses();\r\n    var r = [];\r\n    for (var i = 0; i < classes.length; i++) {\r\n      r.push(eles.filter(`.${classes[i]}:visible`))\r\n    }\r\n    var selector = classes.map(x => '.' + x).join(',');\r\n    // last element of array is elements which are not highlighted by any style\r\n    r.push(eles.filter(\":visible\").not(selector));\r\n    \r\n    return r;\r\n  }\r\n\r\n  function generalUndo(args) {\r\n    var current = args.current;\r\n    var r = [];\r\n    for (var i = 0; i < args.length - 1; i++) {\r\n      r.push(viewUtilities.highlight(args[i], i));\r\n    }\r\n    // last element is for not highlighted by any style\r\n    r.push(viewUtilities.removeHighlights(args[args.length - 1]));\r\n\r\n    r['current'] = current;\r\n    return r;\r\n  }\r\n\r\n  function generalRedo(args) {\r\n    var current = args.current;\r\n    var r = [];\r\n    for (var i = 0; i < current.length - 1; i++) {\r\n      r.push(viewUtilities.highlight(current[i], i));\r\n    }\r\n    // last element is for not highlighted by any style\r\n    r.push(viewUtilities.removeHighlights(current[current.length - 1]));\r\n\r\n    r['current'] = current;\r\n    return r;\r\n  }\r\n\r\n  function generateDoFunc(func) {\r\n    return function (args) {\r\n      var res = getStatus();\r\n      if (args.firstTime)\r\n        viewUtilities[func](args.eles, args.idx);\r\n      else\r\n        generalRedo(args);\r\n\r\n      res.current = getStatus();\r\n\r\n      return res;\r\n    };\r\n  }\r\n\r\n  ur.action(\"highlightNeighbors\", generateDoFunc(\"highlightNeighbors\"), generalUndo);\r\n  ur.action(\"highlight\", generateDoFunc(\"highlight\"), generalUndo);\r\n  ur.action(\"removeHighlights\", generateDoFunc(\"removeHighlights\"), generalUndo);\r\n}\r\n\r\n// Registers ur actions related to hide/show\r\nfunction hideShowUR(cy, ur, viewUtilities) {\r\n  function urShow(eles) {\r\n    return viewUtilities.show(eles);\r\n  }\r\n\r\n  function urHide(eles) {\r\n    return viewUtilities.hide(eles);\r\n  }\r\n\r\n  function urShowHiddenNeighbors(eles) {\r\n    return viewUtilities.showHiddenNeighbors(eles);\r\n  }\r\n\r\n  ur.action(\"show\", urShow, urHide);\r\n  ur.action(\"hide\", urHide, urShow);\r\n  ur.action(\"showHiddenNeighbors\",urShowHiddenNeighbors, urHide);\r\n}\r\n\r\nmodule.exports = function (cy, ur, viewUtilities) {\r\n  highlightUR(cy, ur, viewUtilities);\r\n  hideShowUR(cy, ur, viewUtilities);\r\n};\r\n","var viewUtilities = function (cy, options) {\r\n\r\n  var classNames4Styles = [];\r\n  init();\r\n  function init() {\r\n    // add provided styles\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      classNames4Styles.push('__highligtighted__' + i);\r\n      updateCyStyle(i);\r\n    }\r\n\r\n    // add styles for selected\r\n    addSelectionStyles();\r\n  }\r\n\r\n  function addSelectionStyles() {\r\n    if (options.selectStyles.node) {\r\n      cy.style().selector('node:selected').css(options.selectStyles.node).update();\r\n    }\r\n    if (options.selectStyles.edge) {\r\n      cy.style().selector('edge:selected').css(options.selectStyles.edge).update();\r\n    }\r\n  }\r\n\r\n  function updateCyStyle(classIdx) {\r\n    var className = classNames4Styles[classIdx];\r\n    var cssNode = options.highlightStyles[classIdx].node;\r\n    var cssEdge = options.highlightStyles[classIdx].edge;\r\n    cy.style().selector('node.' + className).css(cssNode).update();\r\n    cy.style().selector('edge.' + className).css(cssEdge).update();\r\n  }\r\n\r\n  // Helper functions for internal usage (not to be exposed)\r\n  function highlight(eles, idx) {\r\n    cy.startBatch();\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      eles.removeClass(classNames4Styles[i]);\r\n    }\r\n    eles.addClass(classNames4Styles[idx]);\r\n    eles.unselect();\r\n    cy.endBatch();\r\n  }\r\n\r\n  function getWithNeighbors(eles) {\r\n    return eles.add(eles.descendants()).closedNeighborhood();\r\n  }\r\n  // the instance to be returned\r\n  var instance = {};\r\n\r\n  // Section hide-show\r\n  // hide given eles\r\n  instance.hide = function (eles) {\r\n    //eles = eles.filter(\"node\")\r\n    eles = eles.filter(\":visible\");\r\n    eles = eles.union(eles.connectedEdges());\r\n\r\n    eles.unselect();\r\n\r\n    if (options.setVisibilityOnHide) {\r\n      eles.css('visibility', 'hidden');\r\n    }\r\n\r\n    if (options.setDisplayOnHide) {\r\n      eles.css('display', 'none');\r\n    }\r\n\r\n    return eles;\r\n  };\r\n\r\n  // unhide given eles\r\n  instance.show = function (eles) {\r\n    eles = eles.not(\":visible\");\r\n\r\n    var connectedEdges = eles.connectedEdges(function (edge) {\r\n\r\n      if ((edge.source().visible() || eles.contains(edge.source())) && (edge.target().visible() || eles.contains(edge.target()))) {\r\n        return true;\r\n      } else {\r\n        return false;\r\n      }\r\n\r\n    });\r\n    eles = eles.union(connectedEdges);\r\n\r\n    eles.unselect();\r\n\r\n    if (options.setVisibilityOnHide) {\r\n      eles.css('visibility', 'visible');\r\n    }\r\n\r\n    if (options.setDisplayOnHide) {\r\n      eles.css('display', 'element');\r\n    }\r\n\r\n    return eles;\r\n  };\r\n\r\n  // Section highlight\r\n  instance.showHiddenNeighbors = function (eles) {\r\n    return this.show(getWithNeighbors(eles));\r\n  };\r\n\r\n  // Highlights eles\r\n  instance.highlight = function (eles, idx = 0) {\r\n    highlight(eles, idx); // Use the helper here\r\n    return eles;\r\n  };\r\n\r\n  instance.getHighlightStyles = function () {\r\n    return options.highlightStyles;\r\n  };\r\n\r\n  // Highlights eles' neighborhood\r\n  instance.highlightNeighbors = function (eles, idx = 0) {\r\n    return this.highlight(getWithNeighbors(eles), idx);\r\n  };\r\n\r\n  // Remove highlights from eles.\r\n  // If eles is not defined considers cy.elements()\r\n  instance.removeHighlights = function (eles) {\r\n    cy.startBatch();\r\n    if (eles == null || eles.length == null) {\r\n      eles = cy.elements();\r\n    }\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      eles.removeClass(classNames4Styles[i]);\r\n    }\r\n    cy.endBatch();\r\n    return eles;\r\n  };\r\n\r\n  // Indicates if the ele is highlighted\r\n  instance.isHighlighted = function (ele) {\r\n    var isHigh = false;\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      if (ele.is('.' + classNames4Styles[i] + ':visible')) {\r\n        isHigh = true;\r\n      }\r\n    }\r\n    return isHigh;\r\n  };\r\n\r\n  instance.changeHighlightStyle = function (idx, nodeStyle, edgeStyle) {\r\n    options.highlightStyles[idx].node = nodeStyle;\r\n    options.highlightStyles[idx].edge = edgeStyle;\r\n    updateCyStyle(idx);\r\n    addSelectionStyles();\r\n  };\r\n\r\n  instance.addHighlightStyle = function (nodeStyle, edgeStyle) {\r\n    var o = { node: nodeStyle, edge: edgeStyle };\r\n    options.highlightStyles.push(o);\r\n    var s = '__highligtighted__' + (options.highlightStyles.length - 1);\r\n    classNames4Styles.push(s);\r\n    updateCyStyle(options.highlightStyles.length - 1);\r\n    addSelectionStyles();\r\n  };\r\n\r\n  instance.removeHighlightStyle = function (styleIdx) {\r\n    if (styleIdx < 0 || styleIdx > options.highlightStyles.length - 1) {\r\n      return;\r\n    }\r\n    cy.elements().removeClass(classNames4Styles[styleIdx]);\r\n    options.highlightStyles.splice(styleIdx, 1);\r\n    classNames4Styles.splice(styleIdx, 1);\r\n  };\r\n\r\n  instance.getAllHighlightClasses = function () {\r\n    var a = [];\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      a.push(classNames4Styles[i]);\r\n    }\r\n    return classNames4Styles;\r\n  };\r\n\r\n  //Zoom selected Nodes\r\n  instance.zoomToSelected = function (eles) {\r\n    var boundingBox = eles.boundingBox();\r\n    var diff_x = Math.abs(boundingBox.x1 - boundingBox.x2);\r\n    var diff_y = Math.abs(boundingBox.y1 - boundingBox.y2);\r\n    var padding;\r\n    if (diff_x >= 200 || diff_y >= 200) {\r\n      padding = 50;\r\n    }\r\n    else {\r\n      padding = (cy.width() < cy.height()) ?\r\n        ((200 - diff_x) / 2 * cy.width() / 200) : ((200 - diff_y) / 2 * cy.height() / 200);\r\n    }\r\n\r\n    cy.animate({\r\n      fit: {\r\n        eles: eles,\r\n        padding: padding\r\n      }\r\n    }, {\r\n      duration: options.zoomAnimationDuration\r\n    });\r\n    return eles;\r\n  };\r\n\r\n  //Marquee Zoom\r\n  var tabStartHandler;\r\n  var tabEndHandler;\r\n\r\n  instance.enableMarqueeZoom = function (callback) {\r\n\r\n    var shiftKeyDown = false;\r\n    var rect_start_pos_x, rect_start_pos_y, rect_end_pos_x, rect_end_pos_y;\r\n    //Make the cy unselectable\r\n    cy.autounselectify(true);\r\n\r\n    document.addEventListener('keydown', function (event) {\r\n      if (event.key == \"Shift\") {\r\n        shiftKeyDown = true;\r\n      }\r\n    });\r\n    document.addEventListener('keyup', function (event) {\r\n      if (event.key == \"Shift\") {\r\n        shiftKeyDown = false;\r\n      }\r\n    });\r\n\r\n    cy.one('tapstart', tabStartHandler = function (event) {\r\n      if (shiftKeyDown == true) {\r\n        rect_start_pos_x = event.position.x;\r\n        rect_start_pos_y = event.position.y;\r\n        rect_end_pos_x = undefined;\r\n      }\r\n    });\r\n    cy.one('tapend', tabEndHandler = function (event) {\r\n      rect_end_pos_x = event.position.x;\r\n      rect_end_pos_y = event.position.y;\r\n      //check whether corners of rectangle is undefined\r\n      //abort marquee zoom if one corner is undefined\r\n      if (rect_start_pos_x == undefined || rect_end_pos_x == undefined) {\r\n        cy.autounselectify(false);\r\n        if (callback) {\r\n          callback();\r\n        }\r\n        return;\r\n      }\r\n      //Reoder rectangle positions\r\n      //Top left of the rectangle (rect_start_pos_x, rect_start_pos_y)\r\n      //right bottom of the rectangle (rect_end_pos_x, rect_end_pos_y)\r\n      if (rect_start_pos_x > rect_end_pos_x) {\r\n        var temp = rect_start_pos_x;\r\n        rect_start_pos_x = rect_end_pos_x;\r\n        rect_end_pos_x = temp;\r\n      }\r\n      if (rect_start_pos_y > rect_end_pos_y) {\r\n        var temp = rect_start_pos_y;\r\n        rect_start_pos_y = rect_end_pos_y;\r\n        rect_end_pos_y = temp;\r\n      }\r\n\r\n      //Extend sides of selected rectangle to 200px if less than 100px\r\n      if (rect_end_pos_x - rect_start_pos_x < 200) {\r\n        var extendPx = (200 - (rect_end_pos_x - rect_start_pos_x)) / 2;\r\n        rect_start_pos_x -= extendPx;\r\n        rect_end_pos_x += extendPx;\r\n      }\r\n      if (rect_end_pos_y - rect_start_pos_y < 200) {\r\n        var extendPx = (200 - (rect_end_pos_y - rect_start_pos_y)) / 2;\r\n        rect_start_pos_y -= extendPx;\r\n        rect_end_pos_y += extendPx;\r\n      }\r\n\r\n      //Check whether rectangle intersects with bounding box of the graph\r\n      //if not abort marquee zoom\r\n      if ((rect_start_pos_x > cy.elements().boundingBox().x2)\r\n        || (rect_end_pos_x < cy.elements().boundingBox().x1)\r\n        || (rect_start_pos_y > cy.elements().boundingBox().y2)\r\n        || (rect_end_pos_y < cy.elements().boundingBox().y1)) {\r\n        cy.autounselectify(false);\r\n        if (callback) {\r\n          callback();\r\n        }\r\n        return;\r\n      }\r\n\r\n      //Calculate zoom level\r\n      var zoomLevel = Math.min(cy.width() / (Math.abs(rect_end_pos_x - rect_start_pos_x)),\r\n        cy.height() / Math.abs(rect_end_pos_y - rect_start_pos_y));\r\n\r\n      var diff_x = cy.width() / 2 - (cy.pan().x + zoomLevel * (rect_start_pos_x + rect_end_pos_x) / 2);\r\n      var diff_y = cy.height() / 2 - (cy.pan().y + zoomLevel * (rect_start_pos_y + rect_end_pos_y) / 2);\r\n\r\n      cy.animate({\r\n        panBy: { x: diff_x, y: diff_y },\r\n        zoom: zoomLevel,\r\n        duration: options.zoomAnimationDuration,\r\n        complete: function () {\r\n          if (callback) {\r\n            callback();\r\n          }\r\n          cy.autounselectify(false);\r\n        }\r\n      });\r\n    });\r\n  };\r\n\r\n  instance.disableMarqueeZoom = function () {\r\n    cy.off('tapstart', tabStartHandler);\r\n    cy.off('tapend', tabEndHandler);\r\n    cy.autounselectify(false);\r\n  };\r\n\r\n  // return the instance\r\n  return instance;\r\n};\r\n\r\nmodule.exports = viewUtilities;\r\n"]} diff --git a/src/view-utilities.js b/src/view-utilities.js index da1a480..9e25215 100644 --- a/src/view-utilities.js +++ b/src/view-utilities.js @@ -1,9 +1,11 @@ var viewUtilities = function (cy, options) { + var classNames4Styles = []; init(); function init() { // add provided styles for (var i = 0; i < options.highlightStyles.length; i++) { + classNames4Styles.push('__highligtighted__' + i); updateCyStyle(i); } @@ -21,28 +23,22 @@ var viewUtilities = function (cy, options) { } function updateCyStyle(classIdx) { - var className = getCyClassName4Idx(classIdx); + var className = classNames4Styles[classIdx]; var cssNode = options.highlightStyles[classIdx].node; var cssEdge = options.highlightStyles[classIdx].edge; cy.style().selector('node.' + className).css(cssNode).update(); cy.style().selector('edge.' + className).css(cssEdge).update(); } - function getCyClassName4Idx(i) { - return '__highligtighted__' + i; - } - // Helper functions for internal usage (not to be exposed) function highlight(eles, idx) { - cy.batch(() => { - for (var i = 0; i < options.highlightStyles.length; i++) { - var className = getCyClassName4Idx(i); - eles.removeClass(className); - } - var className = getCyClassName4Idx(idx); - eles.addClass(className); - eles.unselect(); - }); + cy.startBatch(); + for (var i = 0; i < options.highlightStyles.length; i++) { + eles.removeClass(classNames4Styles[i]); + } + eles.addClass(classNames4Styles[idx]); + eles.unselect(); + cy.endBatch(); } function getWithNeighbors(eles) { @@ -109,7 +105,7 @@ var viewUtilities = function (cy, options) { highlight(eles, idx); // Use the helper here return eles; }; - + instance.getHighlightStyles = function () { return options.highlightStyles; }; @@ -122,15 +118,14 @@ var viewUtilities = function (cy, options) { // Remove highlights from eles. // If eles is not defined considers cy.elements() instance.removeHighlights = function (eles) { + cy.startBatch(); if (eles == null || eles.length == null) { eles = cy.elements(); } - cy.batch(() => { - for (var i = 0; i < options.highlightStyles.length; i++) { - var className = getCyClassName4Idx(i); - eles.removeClass(className); - } - }); + for (var i = 0; i < options.highlightStyles.length; i++) { + eles.removeClass(classNames4Styles[i]); + } + cy.endBatch(); return eles; }; @@ -138,8 +133,7 @@ var viewUtilities = function (cy, options) { instance.isHighlighted = function (ele) { var isHigh = false; for (var i = 0; i < options.highlightStyles.length; i++) { - var className = getCyClassName4Idx(i); - if (ele.is('.' + className + ':visible')) { + if (ele.is('.' + classNames4Styles[i] + ':visible')) { isHigh = true; } } @@ -156,16 +150,27 @@ var viewUtilities = function (cy, options) { instance.addHighlightStyle = function (nodeStyle, edgeStyle) { var o = { node: nodeStyle, edge: edgeStyle }; options.highlightStyles.push(o); + var s = '__highligtighted__' + (options.highlightStyles.length - 1); + classNames4Styles.push(s); updateCyStyle(options.highlightStyles.length - 1); addSelectionStyles(); }; - instance.getAllHighlightClasses = function() { + instance.removeHighlightStyle = function (styleIdx) { + if (styleIdx < 0 || styleIdx > options.highlightStyles.length - 1) { + return; + } + cy.elements().removeClass(classNames4Styles[styleIdx]); + options.highlightStyles.splice(styleIdx, 1); + classNames4Styles.splice(styleIdx, 1); + }; + + instance.getAllHighlightClasses = function () { var a = []; for (var i = 0; i < options.highlightStyles.length; i++) { - a.push(getCyClassName4Idx(i)); + a.push(classNames4Styles[i]); } - return a; + return classNames4Styles; }; //Zoom selected Nodes From fbe37da27d411f86b29b3ef68cce5d777e15ba47 Mon Sep 17 00:00:00 2001 From: canbax Date: Tue, 21 Apr 2020 13:05:51 +0300 Subject: [PATCH 5/6] ugurdogrusoz/visuall#202 fix --- cytoscape-view-utilities.js | 11 ++++++++--- src/view-utilities.js | 9 +++++++-- 2 files changed, 15 insertions(+), 5 deletions(-) diff --git a/cytoscape-view-utilities.js b/cytoscape-view-utilities.js index e614ad0..ae740cf 100644 --- a/cytoscape-view-utilities.js +++ b/cytoscape-view-utilities.js @@ -247,11 +247,15 @@ module.exports = function (cy, ur, viewUtilities) { var viewUtilities = function (cy, options) { var classNames4Styles = []; + // give a unique name for each unique style EVER added + var totStyleCnt = 0; init(); function init() { // add provided styles for (var i = 0; i < options.highlightStyles.length; i++) { - classNames4Styles.push('__highligtighted__' + i); + var s = '__highligtighted__' + totStyleCnt; + classNames4Styles.push(s); + totStyleCnt++; updateCyStyle(i); } @@ -396,8 +400,9 @@ var viewUtilities = function (cy, options) { instance.addHighlightStyle = function (nodeStyle, edgeStyle) { var o = { node: nodeStyle, edge: edgeStyle }; options.highlightStyles.push(o); - var s = '__highligtighted__' + (options.highlightStyles.length - 1); + var s = '__highligtighted__' + totStyleCnt; classNames4Styles.push(s); + totStyleCnt++; updateCyStyle(options.highlightStyles.length - 1); addSelectionStyles(); }; @@ -559,4 +564,4 @@ module.exports = viewUtilities; },{}]},{},[1])(1) }); -//# sourceMappingURL=data:application/json;charset:utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/index.js","src/undo-redo.js","src/view-utilities.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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})",";\r\n(function () {\r\n  'use strict';\r\n\r\n  // registers the extension on a cytoscape lib ref\r\n  var register = function (cytoscape) {\r\n\r\n    if (!cytoscape) {\r\n      return;\r\n    } // can't register if cytoscape unspecified\r\n\r\n    var options = {\r\n      highlightStyles: [],\r\n      selectStyles: {},\r\n      setVisibilityOnHide: false, // whether to set visibility on hide/show\r\n      setDisplayOnHide: true, // whether to set display on hide/show\r\n      zoomAnimationDuration: 1500, //default duration for zoom animation speed\r\n      neighbor: function (node) { // return desired neighbors of tapheld node\r\n        return false;\r\n      },\r\n      neighborSelectTime: 500 //ms, time to taphold to select desired neighbors\r\n    };\r\n\r\n    var undoRedo = require(\"./undo-redo\");\r\n    var viewUtilities = require(\"./view-utilities\");\r\n\r\n    cytoscape('core', 'viewUtilities', function (opts) {\r\n      var cy = this;\r\n\r\n      function getScratch(eleOrCy) {\r\n        if (!eleOrCy.scratch(\"_viewUtilities\")) {\r\n          eleOrCy.scratch(\"_viewUtilities\", {});\r\n        }\r\n\r\n        return eleOrCy.scratch(\"_viewUtilities\");\r\n      }\r\n      \r\n      // If 'get' is given as the param then return the extension instance\r\n      if (opts === 'get') {\r\n        return getScratch(cy).instance;\r\n      }\r\n      \r\n      /**\r\n      * Deep copy or merge objects - replacement for jQuery deep extend\r\n      * Taken from http://youmightnotneedjquery.com/#deep_extend\r\n      * and bug related to deep copy of Arrays is fixed.\r\n      * Usage:Object.extend({}, objA, objB)\r\n      */\r\n      function extendOptions(out) {\r\n        out = out || {};\r\n\r\n        for (var i = 1; i < arguments.length; i++) {\r\n          var obj = arguments[i];\r\n\r\n          if (!obj)\r\n            continue;\r\n\r\n          for (var key in obj) {\r\n            if (obj.hasOwnProperty(key)) {\r\n              if (Array.isArray(obj[key])) {\r\n                out[key] = obj[key].slice();\r\n              } else if (typeof obj[key] === 'object') {\r\n                out[key] = extendOptions(out[key], obj[key]);\r\n              } else {\r\n                out[key] = obj[key];\r\n              }\r\n            }\r\n          }\r\n        }\r\n\r\n        return out;\r\n      };\r\n\r\n      options = extendOptions({}, options, opts);\r\n\r\n      // create a view utilities instance\r\n      var instance = viewUtilities(cy, options);\r\n\r\n      if (cy.undoRedo) {\r\n        var ur = cy.undoRedo(null, true);\r\n        undoRedo(cy, ur, instance);\r\n      }\r\n\r\n      // set the instance on the scratch pad\r\n      getScratch(cy).instance = instance;\r\n\r\n      if (!getScratch(cy).initialized) {\r\n        getScratch(cy).initialized = true;\r\n\r\n        var shiftKeyDown = false;\r\n        document.addEventListener('keydown', function(event){\r\n          if(event.key == \"Shift\") {\r\n            shiftKeyDown = true;\r\n          }\r\n        });\r\n        document.addEventListener('keyup', function(event){\r\n          if(event.key == \"Shift\") {\r\n            shiftKeyDown = false;\r\n          }\r\n        });\r\n        //Select the desired neighbors after taphold-and-free\r\n        cy.on('taphold', 'node', function(event){\r\n          var target = event.target || event.cyTarget;\r\n          var tapheld = false;\r\n          var neighborhood;\r\n          var timeout = setTimeout(function(){\r\n            if(shiftKeyDown){\r\n              cy.elements().unselect();\r\n              neighborhood = options.neighbor(target);\r\n              if(neighborhood)\r\n                neighborhood.select();\r\n              target.lock();\r\n              tapheld = true;\r\n            }\r\n          }, options.neighborSelectTime - 500);\r\n          cy.on('free', 'node', function(){\r\n            var targetTapheld = event.target || event.cyTarget;\r\n            if(target == targetTapheld && tapheld === true){\r\n              tapheld = false;\r\n              if(neighborhood)\r\n                neighborhood.select();\r\n              target.unlock();\r\n            }\r\n            else{\r\n              clearTimeout(timeout);\r\n            }\r\n          });\r\n          cy.on('drag', 'node', function(){\r\n            var targetDragged = event.target || event.cyTarget;\r\n            if(target == targetDragged && tapheld === false){\r\n              clearTimeout(timeout);\r\n            }\r\n          })\r\n        });\r\n      }\r\n\r\n      // return the instance of extension\r\n      return getScratch(cy).instance;\r\n    });\r\n\r\n  };\r\n\r\n  if (typeof module !== 'undefined' && module.exports) { // expose as a commonjs module\r\n    module.exports = register;\r\n  }\r\n\r\n  if (typeof define !== 'undefined' && define.amd) { // expose as an amd/requirejs module\r\n    define('cytoscape-view-utilities', function () {\r\n      return register;\r\n    });\r\n  }\r\n\r\n  if (typeof cytoscape !== 'undefined') { // expose to global cytoscape (i.e. window.cytoscape)\r\n    register(cytoscape);\r\n  }\r\n\r\n})();\r\n","// Registers ur actions related to highlight\r\nfunction highlightUR(cy, ur, viewUtilities) {\r\n  function getStatus(eles) {\r\n    eles = eles ? eles : cy.elements();\r\n    var classes = viewUtilities.getAllHighlightClasses();\r\n    var r = [];\r\n    for (var i = 0; i < classes.length; i++) {\r\n      r.push(eles.filter(`.${classes[i]}:visible`))\r\n    }\r\n    var selector = classes.map(x => '.' + x).join(',');\r\n    // last element of array is elements which are not highlighted by any style\r\n    r.push(eles.filter(\":visible\").not(selector));\r\n    \r\n    return r;\r\n  }\r\n\r\n  function generalUndo(args) {\r\n    var current = args.current;\r\n    var r = [];\r\n    for (var i = 0; i < args.length - 1; i++) {\r\n      r.push(viewUtilities.highlight(args[i], i));\r\n    }\r\n    // last element is for not highlighted by any style\r\n    r.push(viewUtilities.removeHighlights(args[args.length - 1]));\r\n\r\n    r['current'] = current;\r\n    return r;\r\n  }\r\n\r\n  function generalRedo(args) {\r\n    var current = args.current;\r\n    var r = [];\r\n    for (var i = 0; i < current.length - 1; i++) {\r\n      r.push(viewUtilities.highlight(current[i], i));\r\n    }\r\n    // last element is for not highlighted by any style\r\n    r.push(viewUtilities.removeHighlights(current[current.length - 1]));\r\n\r\n    r['current'] = current;\r\n    return r;\r\n  }\r\n\r\n  function generateDoFunc(func) {\r\n    return function (args) {\r\n      var res = getStatus();\r\n      if (args.firstTime)\r\n        viewUtilities[func](args.eles, args.idx);\r\n      else\r\n        generalRedo(args);\r\n\r\n      res.current = getStatus();\r\n\r\n      return res;\r\n    };\r\n  }\r\n\r\n  ur.action(\"highlightNeighbors\", generateDoFunc(\"highlightNeighbors\"), generalUndo);\r\n  ur.action(\"highlight\", generateDoFunc(\"highlight\"), generalUndo);\r\n  ur.action(\"removeHighlights\", generateDoFunc(\"removeHighlights\"), generalUndo);\r\n}\r\n\r\n// Registers ur actions related to hide/show\r\nfunction hideShowUR(cy, ur, viewUtilities) {\r\n  function urShow(eles) {\r\n    return viewUtilities.show(eles);\r\n  }\r\n\r\n  function urHide(eles) {\r\n    return viewUtilities.hide(eles);\r\n  }\r\n\r\n  function urShowHiddenNeighbors(eles) {\r\n    return viewUtilities.showHiddenNeighbors(eles);\r\n  }\r\n\r\n  ur.action(\"show\", urShow, urHide);\r\n  ur.action(\"hide\", urHide, urShow);\r\n  ur.action(\"showHiddenNeighbors\",urShowHiddenNeighbors, urHide);\r\n}\r\n\r\nmodule.exports = function (cy, ur, viewUtilities) {\r\n  highlightUR(cy, ur, viewUtilities);\r\n  hideShowUR(cy, ur, viewUtilities);\r\n};\r\n","var viewUtilities = function (cy, options) {\r\n\r\n  var classNames4Styles = [];\r\n  init();\r\n  function init() {\r\n    // add provided styles\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      classNames4Styles.push('__highligtighted__' + i);\r\n      updateCyStyle(i);\r\n    }\r\n\r\n    // add styles for selected\r\n    addSelectionStyles();\r\n  }\r\n\r\n  function addSelectionStyles() {\r\n    if (options.selectStyles.node) {\r\n      cy.style().selector('node:selected').css(options.selectStyles.node).update();\r\n    }\r\n    if (options.selectStyles.edge) {\r\n      cy.style().selector('edge:selected').css(options.selectStyles.edge).update();\r\n    }\r\n  }\r\n\r\n  function updateCyStyle(classIdx) {\r\n    var className = classNames4Styles[classIdx];\r\n    var cssNode = options.highlightStyles[classIdx].node;\r\n    var cssEdge = options.highlightStyles[classIdx].edge;\r\n    cy.style().selector('node.' + className).css(cssNode).update();\r\n    cy.style().selector('edge.' + className).css(cssEdge).update();\r\n  }\r\n\r\n  // Helper functions for internal usage (not to be exposed)\r\n  function highlight(eles, idx) {\r\n    cy.startBatch();\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      eles.removeClass(classNames4Styles[i]);\r\n    }\r\n    eles.addClass(classNames4Styles[idx]);\r\n    eles.unselect();\r\n    cy.endBatch();\r\n  }\r\n\r\n  function getWithNeighbors(eles) {\r\n    return eles.add(eles.descendants()).closedNeighborhood();\r\n  }\r\n  // the instance to be returned\r\n  var instance = {};\r\n\r\n  // Section hide-show\r\n  // hide given eles\r\n  instance.hide = function (eles) {\r\n    //eles = eles.filter(\"node\")\r\n    eles = eles.filter(\":visible\");\r\n    eles = eles.union(eles.connectedEdges());\r\n\r\n    eles.unselect();\r\n\r\n    if (options.setVisibilityOnHide) {\r\n      eles.css('visibility', 'hidden');\r\n    }\r\n\r\n    if (options.setDisplayOnHide) {\r\n      eles.css('display', 'none');\r\n    }\r\n\r\n    return eles;\r\n  };\r\n\r\n  // unhide given eles\r\n  instance.show = function (eles) {\r\n    eles = eles.not(\":visible\");\r\n\r\n    var connectedEdges = eles.connectedEdges(function (edge) {\r\n\r\n      if ((edge.source().visible() || eles.contains(edge.source())) && (edge.target().visible() || eles.contains(edge.target()))) {\r\n        return true;\r\n      } else {\r\n        return false;\r\n      }\r\n\r\n    });\r\n    eles = eles.union(connectedEdges);\r\n\r\n    eles.unselect();\r\n\r\n    if (options.setVisibilityOnHide) {\r\n      eles.css('visibility', 'visible');\r\n    }\r\n\r\n    if (options.setDisplayOnHide) {\r\n      eles.css('display', 'element');\r\n    }\r\n\r\n    return eles;\r\n  };\r\n\r\n  // Section highlight\r\n  instance.showHiddenNeighbors = function (eles) {\r\n    return this.show(getWithNeighbors(eles));\r\n  };\r\n\r\n  // Highlights eles\r\n  instance.highlight = function (eles, idx = 0) {\r\n    highlight(eles, idx); // Use the helper here\r\n    return eles;\r\n  };\r\n\r\n  instance.getHighlightStyles = function () {\r\n    return options.highlightStyles;\r\n  };\r\n\r\n  // Highlights eles' neighborhood\r\n  instance.highlightNeighbors = function (eles, idx = 0) {\r\n    return this.highlight(getWithNeighbors(eles), idx);\r\n  };\r\n\r\n  // Remove highlights from eles.\r\n  // If eles is not defined considers cy.elements()\r\n  instance.removeHighlights = function (eles) {\r\n    cy.startBatch();\r\n    if (eles == null || eles.length == null) {\r\n      eles = cy.elements();\r\n    }\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      eles.removeClass(classNames4Styles[i]);\r\n    }\r\n    cy.endBatch();\r\n    return eles;\r\n  };\r\n\r\n  // Indicates if the ele is highlighted\r\n  instance.isHighlighted = function (ele) {\r\n    var isHigh = false;\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      if (ele.is('.' + classNames4Styles[i] + ':visible')) {\r\n        isHigh = true;\r\n      }\r\n    }\r\n    return isHigh;\r\n  };\r\n\r\n  instance.changeHighlightStyle = function (idx, nodeStyle, edgeStyle) {\r\n    options.highlightStyles[idx].node = nodeStyle;\r\n    options.highlightStyles[idx].edge = edgeStyle;\r\n    updateCyStyle(idx);\r\n    addSelectionStyles();\r\n  };\r\n\r\n  instance.addHighlightStyle = function (nodeStyle, edgeStyle) {\r\n    var o = { node: nodeStyle, edge: edgeStyle };\r\n    options.highlightStyles.push(o);\r\n    var s = '__highligtighted__' + (options.highlightStyles.length - 1);\r\n    classNames4Styles.push(s);\r\n    updateCyStyle(options.highlightStyles.length - 1);\r\n    addSelectionStyles();\r\n  };\r\n\r\n  instance.removeHighlightStyle = function (styleIdx) {\r\n    if (styleIdx < 0 || styleIdx > options.highlightStyles.length - 1) {\r\n      return;\r\n    }\r\n    cy.elements().removeClass(classNames4Styles[styleIdx]);\r\n    options.highlightStyles.splice(styleIdx, 1);\r\n    classNames4Styles.splice(styleIdx, 1);\r\n  };\r\n\r\n  instance.getAllHighlightClasses = function () {\r\n    var a = [];\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      a.push(classNames4Styles[i]);\r\n    }\r\n    return classNames4Styles;\r\n  };\r\n\r\n  //Zoom selected Nodes\r\n  instance.zoomToSelected = function (eles) {\r\n    var boundingBox = eles.boundingBox();\r\n    var diff_x = Math.abs(boundingBox.x1 - boundingBox.x2);\r\n    var diff_y = Math.abs(boundingBox.y1 - boundingBox.y2);\r\n    var padding;\r\n    if (diff_x >= 200 || diff_y >= 200) {\r\n      padding = 50;\r\n    }\r\n    else {\r\n      padding = (cy.width() < cy.height()) ?\r\n        ((200 - diff_x) / 2 * cy.width() / 200) : ((200 - diff_y) / 2 * cy.height() / 200);\r\n    }\r\n\r\n    cy.animate({\r\n      fit: {\r\n        eles: eles,\r\n        padding: padding\r\n      }\r\n    }, {\r\n      duration: options.zoomAnimationDuration\r\n    });\r\n    return eles;\r\n  };\r\n\r\n  //Marquee Zoom\r\n  var tabStartHandler;\r\n  var tabEndHandler;\r\n\r\n  instance.enableMarqueeZoom = function (callback) {\r\n\r\n    var shiftKeyDown = false;\r\n    var rect_start_pos_x, rect_start_pos_y, rect_end_pos_x, rect_end_pos_y;\r\n    //Make the cy unselectable\r\n    cy.autounselectify(true);\r\n\r\n    document.addEventListener('keydown', function (event) {\r\n      if (event.key == \"Shift\") {\r\n        shiftKeyDown = true;\r\n      }\r\n    });\r\n    document.addEventListener('keyup', function (event) {\r\n      if (event.key == \"Shift\") {\r\n        shiftKeyDown = false;\r\n      }\r\n    });\r\n\r\n    cy.one('tapstart', tabStartHandler = function (event) {\r\n      if (shiftKeyDown == true) {\r\n        rect_start_pos_x = event.position.x;\r\n        rect_start_pos_y = event.position.y;\r\n        rect_end_pos_x = undefined;\r\n      }\r\n    });\r\n    cy.one('tapend', tabEndHandler = function (event) {\r\n      rect_end_pos_x = event.position.x;\r\n      rect_end_pos_y = event.position.y;\r\n      //check whether corners of rectangle is undefined\r\n      //abort marquee zoom if one corner is undefined\r\n      if (rect_start_pos_x == undefined || rect_end_pos_x == undefined) {\r\n        cy.autounselectify(false);\r\n        if (callback) {\r\n          callback();\r\n        }\r\n        return;\r\n      }\r\n      //Reoder rectangle positions\r\n      //Top left of the rectangle (rect_start_pos_x, rect_start_pos_y)\r\n      //right bottom of the rectangle (rect_end_pos_x, rect_end_pos_y)\r\n      if (rect_start_pos_x > rect_end_pos_x) {\r\n        var temp = rect_start_pos_x;\r\n        rect_start_pos_x = rect_end_pos_x;\r\n        rect_end_pos_x = temp;\r\n      }\r\n      if (rect_start_pos_y > rect_end_pos_y) {\r\n        var temp = rect_start_pos_y;\r\n        rect_start_pos_y = rect_end_pos_y;\r\n        rect_end_pos_y = temp;\r\n      }\r\n\r\n      //Extend sides of selected rectangle to 200px if less than 100px\r\n      if (rect_end_pos_x - rect_start_pos_x < 200) {\r\n        var extendPx = (200 - (rect_end_pos_x - rect_start_pos_x)) / 2;\r\n        rect_start_pos_x -= extendPx;\r\n        rect_end_pos_x += extendPx;\r\n      }\r\n      if (rect_end_pos_y - rect_start_pos_y < 200) {\r\n        var extendPx = (200 - (rect_end_pos_y - rect_start_pos_y)) / 2;\r\n        rect_start_pos_y -= extendPx;\r\n        rect_end_pos_y += extendPx;\r\n      }\r\n\r\n      //Check whether rectangle intersects with bounding box of the graph\r\n      //if not abort marquee zoom\r\n      if ((rect_start_pos_x > cy.elements().boundingBox().x2)\r\n        || (rect_end_pos_x < cy.elements().boundingBox().x1)\r\n        || (rect_start_pos_y > cy.elements().boundingBox().y2)\r\n        || (rect_end_pos_y < cy.elements().boundingBox().y1)) {\r\n        cy.autounselectify(false);\r\n        if (callback) {\r\n          callback();\r\n        }\r\n        return;\r\n      }\r\n\r\n      //Calculate zoom level\r\n      var zoomLevel = Math.min(cy.width() / (Math.abs(rect_end_pos_x - rect_start_pos_x)),\r\n        cy.height() / Math.abs(rect_end_pos_y - rect_start_pos_y));\r\n\r\n      var diff_x = cy.width() / 2 - (cy.pan().x + zoomLevel * (rect_start_pos_x + rect_end_pos_x) / 2);\r\n      var diff_y = cy.height() / 2 - (cy.pan().y + zoomLevel * (rect_start_pos_y + rect_end_pos_y) / 2);\r\n\r\n      cy.animate({\r\n        panBy: { x: diff_x, y: diff_y },\r\n        zoom: zoomLevel,\r\n        duration: options.zoomAnimationDuration,\r\n        complete: function () {\r\n          if (callback) {\r\n            callback();\r\n          }\r\n          cy.autounselectify(false);\r\n        }\r\n      });\r\n    });\r\n  };\r\n\r\n  instance.disableMarqueeZoom = function () {\r\n    cy.off('tapstart', tabStartHandler);\r\n    cy.off('tapend', tabEndHandler);\r\n    cy.autounselectify(false);\r\n  };\r\n\r\n  // return the instance\r\n  return instance;\r\n};\r\n\r\nmodule.exports = viewUtilities;\r\n"]} +//# sourceMappingURL=data:application/json;charset:utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","src/index.js","src/undo-redo.js","src/view-utilities.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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})",";\r\n(function () {\r\n  'use strict';\r\n\r\n  // registers the extension on a cytoscape lib ref\r\n  var register = function (cytoscape) {\r\n\r\n    if (!cytoscape) {\r\n      return;\r\n    } // can't register if cytoscape unspecified\r\n\r\n    var options = {\r\n      highlightStyles: [],\r\n      selectStyles: {},\r\n      setVisibilityOnHide: false, // whether to set visibility on hide/show\r\n      setDisplayOnHide: true, // whether to set display on hide/show\r\n      zoomAnimationDuration: 1500, //default duration for zoom animation speed\r\n      neighbor: function (node) { // return desired neighbors of tapheld node\r\n        return false;\r\n      },\r\n      neighborSelectTime: 500 //ms, time to taphold to select desired neighbors\r\n    };\r\n\r\n    var undoRedo = require(\"./undo-redo\");\r\n    var viewUtilities = require(\"./view-utilities\");\r\n\r\n    cytoscape('core', 'viewUtilities', function (opts) {\r\n      var cy = this;\r\n\r\n      function getScratch(eleOrCy) {\r\n        if (!eleOrCy.scratch(\"_viewUtilities\")) {\r\n          eleOrCy.scratch(\"_viewUtilities\", {});\r\n        }\r\n\r\n        return eleOrCy.scratch(\"_viewUtilities\");\r\n      }\r\n      \r\n      // If 'get' is given as the param then return the extension instance\r\n      if (opts === 'get') {\r\n        return getScratch(cy).instance;\r\n      }\r\n      \r\n      /**\r\n      * Deep copy or merge objects - replacement for jQuery deep extend\r\n      * Taken from http://youmightnotneedjquery.com/#deep_extend\r\n      * and bug related to deep copy of Arrays is fixed.\r\n      * Usage:Object.extend({}, objA, objB)\r\n      */\r\n      function extendOptions(out) {\r\n        out = out || {};\r\n\r\n        for (var i = 1; i < arguments.length; i++) {\r\n          var obj = arguments[i];\r\n\r\n          if (!obj)\r\n            continue;\r\n\r\n          for (var key in obj) {\r\n            if (obj.hasOwnProperty(key)) {\r\n              if (Array.isArray(obj[key])) {\r\n                out[key] = obj[key].slice();\r\n              } else if (typeof obj[key] === 'object') {\r\n                out[key] = extendOptions(out[key], obj[key]);\r\n              } else {\r\n                out[key] = obj[key];\r\n              }\r\n            }\r\n          }\r\n        }\r\n\r\n        return out;\r\n      };\r\n\r\n      options = extendOptions({}, options, opts);\r\n\r\n      // create a view utilities instance\r\n      var instance = viewUtilities(cy, options);\r\n\r\n      if (cy.undoRedo) {\r\n        var ur = cy.undoRedo(null, true);\r\n        undoRedo(cy, ur, instance);\r\n      }\r\n\r\n      // set the instance on the scratch pad\r\n      getScratch(cy).instance = instance;\r\n\r\n      if (!getScratch(cy).initialized) {\r\n        getScratch(cy).initialized = true;\r\n\r\n        var shiftKeyDown = false;\r\n        document.addEventListener('keydown', function(event){\r\n          if(event.key == \"Shift\") {\r\n            shiftKeyDown = true;\r\n          }\r\n        });\r\n        document.addEventListener('keyup', function(event){\r\n          if(event.key == \"Shift\") {\r\n            shiftKeyDown = false;\r\n          }\r\n        });\r\n        //Select the desired neighbors after taphold-and-free\r\n        cy.on('taphold', 'node', function(event){\r\n          var target = event.target || event.cyTarget;\r\n          var tapheld = false;\r\n          var neighborhood;\r\n          var timeout = setTimeout(function(){\r\n            if(shiftKeyDown){\r\n              cy.elements().unselect();\r\n              neighborhood = options.neighbor(target);\r\n              if(neighborhood)\r\n                neighborhood.select();\r\n              target.lock();\r\n              tapheld = true;\r\n            }\r\n          }, options.neighborSelectTime - 500);\r\n          cy.on('free', 'node', function(){\r\n            var targetTapheld = event.target || event.cyTarget;\r\n            if(target == targetTapheld && tapheld === true){\r\n              tapheld = false;\r\n              if(neighborhood)\r\n                neighborhood.select();\r\n              target.unlock();\r\n            }\r\n            else{\r\n              clearTimeout(timeout);\r\n            }\r\n          });\r\n          cy.on('drag', 'node', function(){\r\n            var targetDragged = event.target || event.cyTarget;\r\n            if(target == targetDragged && tapheld === false){\r\n              clearTimeout(timeout);\r\n            }\r\n          })\r\n        });\r\n      }\r\n\r\n      // return the instance of extension\r\n      return getScratch(cy).instance;\r\n    });\r\n\r\n  };\r\n\r\n  if (typeof module !== 'undefined' && module.exports) { // expose as a commonjs module\r\n    module.exports = register;\r\n  }\r\n\r\n  if (typeof define !== 'undefined' && define.amd) { // expose as an amd/requirejs module\r\n    define('cytoscape-view-utilities', function () {\r\n      return register;\r\n    });\r\n  }\r\n\r\n  if (typeof cytoscape !== 'undefined') { // expose to global cytoscape (i.e. window.cytoscape)\r\n    register(cytoscape);\r\n  }\r\n\r\n})();\r\n","// Registers ur actions related to highlight\r\nfunction highlightUR(cy, ur, viewUtilities) {\r\n  function getStatus(eles) {\r\n    eles = eles ? eles : cy.elements();\r\n    var classes = viewUtilities.getAllHighlightClasses();\r\n    var r = [];\r\n    for (var i = 0; i < classes.length; i++) {\r\n      r.push(eles.filter(`.${classes[i]}:visible`))\r\n    }\r\n    var selector = classes.map(x => '.' + x).join(',');\r\n    // last element of array is elements which are not highlighted by any style\r\n    r.push(eles.filter(\":visible\").not(selector));\r\n    \r\n    return r;\r\n  }\r\n\r\n  function generalUndo(args) {\r\n    var current = args.current;\r\n    var r = [];\r\n    for (var i = 0; i < args.length - 1; i++) {\r\n      r.push(viewUtilities.highlight(args[i], i));\r\n    }\r\n    // last element is for not highlighted by any style\r\n    r.push(viewUtilities.removeHighlights(args[args.length - 1]));\r\n\r\n    r['current'] = current;\r\n    return r;\r\n  }\r\n\r\n  function generalRedo(args) {\r\n    var current = args.current;\r\n    var r = [];\r\n    for (var i = 0; i < current.length - 1; i++) {\r\n      r.push(viewUtilities.highlight(current[i], i));\r\n    }\r\n    // last element is for not highlighted by any style\r\n    r.push(viewUtilities.removeHighlights(current[current.length - 1]));\r\n\r\n    r['current'] = current;\r\n    return r;\r\n  }\r\n\r\n  function generateDoFunc(func) {\r\n    return function (args) {\r\n      var res = getStatus();\r\n      if (args.firstTime)\r\n        viewUtilities[func](args.eles, args.idx);\r\n      else\r\n        generalRedo(args);\r\n\r\n      res.current = getStatus();\r\n\r\n      return res;\r\n    };\r\n  }\r\n\r\n  ur.action(\"highlightNeighbors\", generateDoFunc(\"highlightNeighbors\"), generalUndo);\r\n  ur.action(\"highlight\", generateDoFunc(\"highlight\"), generalUndo);\r\n  ur.action(\"removeHighlights\", generateDoFunc(\"removeHighlights\"), generalUndo);\r\n}\r\n\r\n// Registers ur actions related to hide/show\r\nfunction hideShowUR(cy, ur, viewUtilities) {\r\n  function urShow(eles) {\r\n    return viewUtilities.show(eles);\r\n  }\r\n\r\n  function urHide(eles) {\r\n    return viewUtilities.hide(eles);\r\n  }\r\n\r\n  function urShowHiddenNeighbors(eles) {\r\n    return viewUtilities.showHiddenNeighbors(eles);\r\n  }\r\n\r\n  ur.action(\"show\", urShow, urHide);\r\n  ur.action(\"hide\", urHide, urShow);\r\n  ur.action(\"showHiddenNeighbors\",urShowHiddenNeighbors, urHide);\r\n}\r\n\r\nmodule.exports = function (cy, ur, viewUtilities) {\r\n  highlightUR(cy, ur, viewUtilities);\r\n  hideShowUR(cy, ur, viewUtilities);\r\n};\r\n","var viewUtilities = function (cy, options) {\r\n\r\n  var classNames4Styles = [];\r\n  // give a unique name for each unique style EVER added\r\n  var totStyleCnt = 0;\r\n  init();\r\n  function init() {\r\n    // add provided styles\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      var s = '__highligtighted__' + totStyleCnt;\r\n      classNames4Styles.push(s);\r\n      totStyleCnt++;\r\n      updateCyStyle(i);\r\n    }\r\n\r\n    // add styles for selected\r\n    addSelectionStyles();\r\n  }\r\n\r\n  function addSelectionStyles() {\r\n    if (options.selectStyles.node) {\r\n      cy.style().selector('node:selected').css(options.selectStyles.node).update();\r\n    }\r\n    if (options.selectStyles.edge) {\r\n      cy.style().selector('edge:selected').css(options.selectStyles.edge).update();\r\n    }\r\n  }\r\n\r\n  function updateCyStyle(classIdx) {\r\n    var className = classNames4Styles[classIdx];\r\n    var cssNode = options.highlightStyles[classIdx].node;\r\n    var cssEdge = options.highlightStyles[classIdx].edge;\r\n    cy.style().selector('node.' + className).css(cssNode).update();\r\n    cy.style().selector('edge.' + className).css(cssEdge).update();\r\n  }\r\n\r\n  // Helper functions for internal usage (not to be exposed)\r\n  function highlight(eles, idx) {\r\n    cy.startBatch();\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      eles.removeClass(classNames4Styles[i]);\r\n    }\r\n    eles.addClass(classNames4Styles[idx]);\r\n    eles.unselect();\r\n    cy.endBatch();\r\n  }\r\n\r\n  function getWithNeighbors(eles) {\r\n    return eles.add(eles.descendants()).closedNeighborhood();\r\n  }\r\n  // the instance to be returned\r\n  var instance = {};\r\n\r\n  // Section hide-show\r\n  // hide given eles\r\n  instance.hide = function (eles) {\r\n    //eles = eles.filter(\"node\")\r\n    eles = eles.filter(\":visible\");\r\n    eles = eles.union(eles.connectedEdges());\r\n\r\n    eles.unselect();\r\n\r\n    if (options.setVisibilityOnHide) {\r\n      eles.css('visibility', 'hidden');\r\n    }\r\n\r\n    if (options.setDisplayOnHide) {\r\n      eles.css('display', 'none');\r\n    }\r\n\r\n    return eles;\r\n  };\r\n\r\n  // unhide given eles\r\n  instance.show = function (eles) {\r\n    eles = eles.not(\":visible\");\r\n\r\n    var connectedEdges = eles.connectedEdges(function (edge) {\r\n\r\n      if ((edge.source().visible() || eles.contains(edge.source())) && (edge.target().visible() || eles.contains(edge.target()))) {\r\n        return true;\r\n      } else {\r\n        return false;\r\n      }\r\n\r\n    });\r\n    eles = eles.union(connectedEdges);\r\n\r\n    eles.unselect();\r\n\r\n    if (options.setVisibilityOnHide) {\r\n      eles.css('visibility', 'visible');\r\n    }\r\n\r\n    if (options.setDisplayOnHide) {\r\n      eles.css('display', 'element');\r\n    }\r\n\r\n    return eles;\r\n  };\r\n\r\n  // Section highlight\r\n  instance.showHiddenNeighbors = function (eles) {\r\n    return this.show(getWithNeighbors(eles));\r\n  };\r\n\r\n  // Highlights eles\r\n  instance.highlight = function (eles, idx = 0) {\r\n    highlight(eles, idx); // Use the helper here\r\n    return eles;\r\n  };\r\n\r\n  instance.getHighlightStyles = function () {\r\n    return options.highlightStyles;\r\n  };\r\n\r\n  // Highlights eles' neighborhood\r\n  instance.highlightNeighbors = function (eles, idx = 0) {\r\n    return this.highlight(getWithNeighbors(eles), idx);\r\n  };\r\n\r\n  // Remove highlights from eles.\r\n  // If eles is not defined considers cy.elements()\r\n  instance.removeHighlights = function (eles) {\r\n    cy.startBatch();\r\n    if (eles == null || eles.length == null) {\r\n      eles = cy.elements();\r\n    }\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      eles.removeClass(classNames4Styles[i]);\r\n    }\r\n    cy.endBatch();\r\n    return eles;\r\n  };\r\n\r\n  // Indicates if the ele is highlighted\r\n  instance.isHighlighted = function (ele) {\r\n    var isHigh = false;\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      if (ele.is('.' + classNames4Styles[i] + ':visible')) {\r\n        isHigh = true;\r\n      }\r\n    }\r\n    return isHigh;\r\n  };\r\n\r\n  instance.changeHighlightStyle = function (idx, nodeStyle, edgeStyle) {\r\n    options.highlightStyles[idx].node = nodeStyle;\r\n    options.highlightStyles[idx].edge = edgeStyle;\r\n    updateCyStyle(idx);\r\n    addSelectionStyles();\r\n  };\r\n\r\n  instance.addHighlightStyle = function (nodeStyle, edgeStyle) {\r\n    var o = { node: nodeStyle, edge: edgeStyle };\r\n    options.highlightStyles.push(o);\r\n    var s = '__highligtighted__' + totStyleCnt;\r\n    classNames4Styles.push(s);\r\n    totStyleCnt++;\r\n    updateCyStyle(options.highlightStyles.length - 1);\r\n    addSelectionStyles();\r\n  };\r\n\r\n  instance.removeHighlightStyle = function (styleIdx) {\r\n    if (styleIdx < 0 || styleIdx > options.highlightStyles.length - 1) {\r\n      return;\r\n    }\r\n    cy.elements().removeClass(classNames4Styles[styleIdx]);\r\n    options.highlightStyles.splice(styleIdx, 1);\r\n    classNames4Styles.splice(styleIdx, 1);\r\n  };\r\n\r\n  instance.getAllHighlightClasses = function () {\r\n    var a = [];\r\n    for (var i = 0; i < options.highlightStyles.length; i++) {\r\n      a.push(classNames4Styles[i]);\r\n    }\r\n    return classNames4Styles;\r\n  };\r\n\r\n  //Zoom selected Nodes\r\n  instance.zoomToSelected = function (eles) {\r\n    var boundingBox = eles.boundingBox();\r\n    var diff_x = Math.abs(boundingBox.x1 - boundingBox.x2);\r\n    var diff_y = Math.abs(boundingBox.y1 - boundingBox.y2);\r\n    var padding;\r\n    if (diff_x >= 200 || diff_y >= 200) {\r\n      padding = 50;\r\n    }\r\n    else {\r\n      padding = (cy.width() < cy.height()) ?\r\n        ((200 - diff_x) / 2 * cy.width() / 200) : ((200 - diff_y) / 2 * cy.height() / 200);\r\n    }\r\n\r\n    cy.animate({\r\n      fit: {\r\n        eles: eles,\r\n        padding: padding\r\n      }\r\n    }, {\r\n      duration: options.zoomAnimationDuration\r\n    });\r\n    return eles;\r\n  };\r\n\r\n  //Marquee Zoom\r\n  var tabStartHandler;\r\n  var tabEndHandler;\r\n\r\n  instance.enableMarqueeZoom = function (callback) {\r\n\r\n    var shiftKeyDown = false;\r\n    var rect_start_pos_x, rect_start_pos_y, rect_end_pos_x, rect_end_pos_y;\r\n    //Make the cy unselectable\r\n    cy.autounselectify(true);\r\n\r\n    document.addEventListener('keydown', function (event) {\r\n      if (event.key == \"Shift\") {\r\n        shiftKeyDown = true;\r\n      }\r\n    });\r\n    document.addEventListener('keyup', function (event) {\r\n      if (event.key == \"Shift\") {\r\n        shiftKeyDown = false;\r\n      }\r\n    });\r\n\r\n    cy.one('tapstart', tabStartHandler = function (event) {\r\n      if (shiftKeyDown == true) {\r\n        rect_start_pos_x = event.position.x;\r\n        rect_start_pos_y = event.position.y;\r\n        rect_end_pos_x = undefined;\r\n      }\r\n    });\r\n    cy.one('tapend', tabEndHandler = function (event) {\r\n      rect_end_pos_x = event.position.x;\r\n      rect_end_pos_y = event.position.y;\r\n      //check whether corners of rectangle is undefined\r\n      //abort marquee zoom if one corner is undefined\r\n      if (rect_start_pos_x == undefined || rect_end_pos_x == undefined) {\r\n        cy.autounselectify(false);\r\n        if (callback) {\r\n          callback();\r\n        }\r\n        return;\r\n      }\r\n      //Reoder rectangle positions\r\n      //Top left of the rectangle (rect_start_pos_x, rect_start_pos_y)\r\n      //right bottom of the rectangle (rect_end_pos_x, rect_end_pos_y)\r\n      if (rect_start_pos_x > rect_end_pos_x) {\r\n        var temp = rect_start_pos_x;\r\n        rect_start_pos_x = rect_end_pos_x;\r\n        rect_end_pos_x = temp;\r\n      }\r\n      if (rect_start_pos_y > rect_end_pos_y) {\r\n        var temp = rect_start_pos_y;\r\n        rect_start_pos_y = rect_end_pos_y;\r\n        rect_end_pos_y = temp;\r\n      }\r\n\r\n      //Extend sides of selected rectangle to 200px if less than 100px\r\n      if (rect_end_pos_x - rect_start_pos_x < 200) {\r\n        var extendPx = (200 - (rect_end_pos_x - rect_start_pos_x)) / 2;\r\n        rect_start_pos_x -= extendPx;\r\n        rect_end_pos_x += extendPx;\r\n      }\r\n      if (rect_end_pos_y - rect_start_pos_y < 200) {\r\n        var extendPx = (200 - (rect_end_pos_y - rect_start_pos_y)) / 2;\r\n        rect_start_pos_y -= extendPx;\r\n        rect_end_pos_y += extendPx;\r\n      }\r\n\r\n      //Check whether rectangle intersects with bounding box of the graph\r\n      //if not abort marquee zoom\r\n      if ((rect_start_pos_x > cy.elements().boundingBox().x2)\r\n        || (rect_end_pos_x < cy.elements().boundingBox().x1)\r\n        || (rect_start_pos_y > cy.elements().boundingBox().y2)\r\n        || (rect_end_pos_y < cy.elements().boundingBox().y1)) {\r\n        cy.autounselectify(false);\r\n        if (callback) {\r\n          callback();\r\n        }\r\n        return;\r\n      }\r\n\r\n      //Calculate zoom level\r\n      var zoomLevel = Math.min(cy.width() / (Math.abs(rect_end_pos_x - rect_start_pos_x)),\r\n        cy.height() / Math.abs(rect_end_pos_y - rect_start_pos_y));\r\n\r\n      var diff_x = cy.width() / 2 - (cy.pan().x + zoomLevel * (rect_start_pos_x + rect_end_pos_x) / 2);\r\n      var diff_y = cy.height() / 2 - (cy.pan().y + zoomLevel * (rect_start_pos_y + rect_end_pos_y) / 2);\r\n\r\n      cy.animate({\r\n        panBy: { x: diff_x, y: diff_y },\r\n        zoom: zoomLevel,\r\n        duration: options.zoomAnimationDuration,\r\n        complete: function () {\r\n          if (callback) {\r\n            callback();\r\n          }\r\n          cy.autounselectify(false);\r\n        }\r\n      });\r\n    });\r\n  };\r\n\r\n  instance.disableMarqueeZoom = function () {\r\n    cy.off('tapstart', tabStartHandler);\r\n    cy.off('tapend', tabEndHandler);\r\n    cy.autounselectify(false);\r\n  };\r\n\r\n  // return the instance\r\n  return instance;\r\n};\r\n\r\nmodule.exports = viewUtilities;\r\n"]} diff --git a/src/view-utilities.js b/src/view-utilities.js index 9e25215..47eb771 100644 --- a/src/view-utilities.js +++ b/src/view-utilities.js @@ -1,11 +1,15 @@ var viewUtilities = function (cy, options) { var classNames4Styles = []; + // give a unique name for each unique style EVER added + var totStyleCnt = 0; init(); function init() { // add provided styles for (var i = 0; i < options.highlightStyles.length; i++) { - classNames4Styles.push('__highligtighted__' + i); + var s = '__highligtighted__' + totStyleCnt; + classNames4Styles.push(s); + totStyleCnt++; updateCyStyle(i); } @@ -150,8 +154,9 @@ var viewUtilities = function (cy, options) { instance.addHighlightStyle = function (nodeStyle, edgeStyle) { var o = { node: nodeStyle, edge: edgeStyle }; options.highlightStyles.push(o); - var s = '__highligtighted__' + (options.highlightStyles.length - 1); + var s = '__highligtighted__' + totStyleCnt; classNames4Styles.push(s); + totStyleCnt++; updateCyStyle(options.highlightStyles.length - 1); addSelectionStyles(); }; From 7ff113c5c6d8de479905e93019d79f5a17f3fac1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Hasan=20Balc=C4=B1?= Date: Tue, 19 May 2020 16:44:40 +0300 Subject: [PATCH 6/6] Update LICENCE --- LICENSE | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/LICENSE b/LICENSE index 5d0faf1..0512bf5 100644 --- a/LICENSE +++ b/LICENSE @@ -1,6 +1,6 @@ MIT License -Copyright (c) 2017 iVis@Bilkent +Copyright (c) 2017 - present, iVis@Bilkent. Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal