diff --git a/README.md b/README.md index 001b42f..3eb2feb 100644 --- a/README.md +++ b/README.md @@ -48,11 +48,7 @@ npm run build-dev 4. Choose how to annotate proteins (this will prob, because of https://github.com/MICommunity/ComplexViewer/issues/80): - myComplexViewer.setAnnotations("MI features"); //show features from MI data - myComplexViewer.setAnnotations("UniprotKB"); //show annotations from uniprot - myComplexViewer.setAnnotations("SuperFamily"); //from SuperFamily - myComplexViewer.setAnnotations("None"); //no annotations - myComplexViewer.setAnnotations("Interactor"); //colour bars and circles according to interactor (may help indicate stoichiometry) + // todo doc - see index.html for example, old way of doing should still be working 5. To change dataset without creating a new instance of the app, call the clear function, then call the readMijson() function with the new data: diff --git a/css/style.css b/css/style.css index da07ffc..50a24bf 100755 --- a/css/style.css +++ b/css/style.css @@ -57,54 +57,54 @@ } /*tables*/ -table { - width: 100%; - font-size: 0.9em; - left: auto; - position: relative; - margin-left: auto; - margin-right: auto; - border-collapse: collapse; - /* - background-color: rgba(255, 255, 255, 1); - */ - text-align: left; -} +/*table {*/ +/* width: 100%;*/ +/* font-size: 0.9em;*/ +/* left: auto;*/ +/* position: relative;*/ +/* margin-left: auto;*/ +/* margin-right: auto;*/ +/* border-collapse: collapse;*/ +/* !**/ +/* background-color: rgba(255, 255, 255, 1);*/ +/* *!*/ +/* text-align: left;*/ +/*}*/ -thead { - margin-top: 10px; - margin-bottom: 10px; -} +/*thead {*/ +/* margin-top: 10px;*/ +/* margin-bottom: 10px;*/ +/*}*/ -th { - padding-left: 5px; - padding-right: 5px; - vertical-align: middle; - height: 30px; -} +/*th {*/ +/* padding-left: 5px;*/ +/* padding-right: 5px;*/ +/* vertical-align: middle;*/ +/* height: 30px;*/ +/*}*/ -td { - padding-left: 5px; - padding-right: 5px; - vertical-align: middle; - text-align: left; -} +/*td {*/ +/* padding-left: 5px;*/ +/* padding-right: 5px;*/ +/* vertical-align: middle;*/ +/* text-align: left;*/ +/*}*/ -th:first-child { - padding-left: 10px; -} +/*th:first-child {*/ +/* padding-left: 10px;*/ +/*}*/ -th:last-child { - padding-right: 10px; -} +/*th:last-child {*/ +/* padding-right: 10px;*/ +/*}*/ -td:first-child { - padding-left: 10px; -} +/*td:first-child {*/ +/* padding-left: 10px;*/ +/*}*/ -td:last-child { - padding-right: 10px; -} +/*td:last-child {*/ +/* padding-right: 10px;*/ +/*}*/ /* fa */ @@ -152,19 +152,19 @@ body, input, select, textarea, button { color: #fff; } -.tableContainer a { - color: #091D42; - text-decoration: underline; - -webkit-transition: all 200ms ease-in-out; - -moz-transition: all 200ms ease-in-out; - -o-transition: all 200ms ease-in-out; - transition: all 200ms ease-in-out; -} +/*.tableContainer a {*/ +/* color: #091D42;*/ +/* text-decoration: underline;*/ +/* -webkit-transition: all 200ms ease-in-out;*/ +/* -moz-transition: all 200ms ease-in-out;*/ +/* -o-transition: all 200ms ease-in-out;*/ +/* transition: all 200ms ease-in-out;*/ +/*}*/ -.tableContainer a:hover { - color: #ffffff !important; - background-color: #091D42; -} +/*.tableContainer a:hover {*/ +/* color: #ffffff !important;*/ +/* background-color: #091D42;*/ +/*}*/ #main { @@ -184,49 +184,6 @@ body, input, select, textarea, button { padding-left: 10px; } - -/* -#networkDiv { - width:100%; - height:100%; - display:block; -} -*/ - -/*.bar rect { - fill: steelblue; -} - -.bar text { - fill: #fff; -} - -.resize path { - fill: #888; - stroke: #000; - stroke-width: 2px; -} - -.axis path, .axis line { - fill: none; - stroke: #000; - stroke-width: 1px; -} - -.panelInner th { - font-weight: 400; - color: #039; - height: 30px; -} - -.panelInner thead td { - color: #091D42; -} - -.panelInner td, .helpPanel td { - color: #777; -}*/ - .controls { background: #ecedf2; color: #091D42; @@ -240,33 +197,27 @@ body, input, select, textarea, button { /* margin-right: 0.5em;*/ /*}*/ -label.horizontalFlow input[type=number] { - margin-left: 0.2em; -} - -/* See http://jsfiddle.net/7jx02upg/1/ */ -label.horizontalFlow input[type=radio], label.horizontalFlow input[type=checkbox] { - /*vertical-align: normal;*/ -} +/*label.horizontalFlow input[type=number] {*/ +/* margin-left: 0.2em;*/ +/*}*/ -/* +/*!* See http://jsfiddle.net/7jx02upg/1/ *!*/ +/*label.horizontalFlow input[type=radio], label.horizontalFlow input[type=checkbox] {*/ +/* !*vertical-align: normal;*!*/ +/*}*/ -/* span.noBreak asks for no line-breaks internally */ +/*!* */ -span.noBreak select { - margin-left: 0.5em; -} +/*!* span.noBreak asks for no line-breaks internally *!*/ -/* and then put spaces after labels to give a known place for content to break */ +/*span.noBreak select {*/ +/* margin-left: 0.5em;*/ +/*}*/ -A.btn { - text-decoration: none; -} +/*!* and then put spaces after labels to give a known place for content to break *!*/ -/* For use with Split.js */ +/*A.btn {*/ +/* text-decoration: none;*/ +/*}*/ -.btn:disabled { - color: gray; - border-color: gray; -} diff --git a/data/index.js b/data/index.js index 2a10fc3..a6db25e 100644 --- a/data/index.js +++ b/data/index.js @@ -1,5 +1,10 @@ // eslint-disable-next-line no-unused-vars const exampleIndex = [ + { + "ac": "CPX-1920", + "name": "~CPX-1920", + "url": "https://www.ebi.ac.uk/complexportal/complex/CPX-1920", + }, { "ac": "EBI-9008420", "name": "Hemoglobin HbA complex", @@ -10,11 +15,6 @@ const exampleIndex = [ "name": "EBI-12598622", "url": "https://ebi-intact.github.io/intact-view/details/interaction/EBI-12598622", }, - { - "ac": "CPX-1920", - "name": "~CPX-1920", - "url": "https://www.ebi.ac.uk/complexportal/complex/CPX-1920", - }, { "ac": "EBI-4371590", "name": "EBI-4371590", diff --git a/dist/complexviewer.js b/dist/complexviewer.js index a0186e7..f63fd56 100644 --- a/dist/complexviewer.js +++ b/dist/complexviewer.js @@ -103,7 +103,7 @@ return /******/ (function(modules) { // webpackBootstrap /*! no static exports found */ /***/ (function(module, exports, __webpack_require__) { -eval("// Imports\nvar ___CSS_LOADER_API_IMPORT___ = __webpack_require__(/*! ../../node_modules/css-loader/dist/runtime/api.js */ \"./node_modules/css-loader/dist/runtime/api.js\");\nexports = ___CSS_LOADER_API_IMPORT___(false);\n// Module\nexports.push([module.i, \".protein {\\n cursor: crosshair;\\n}\\n\\n.link {\\n /*\\n cursor: crosshair;\\n */\\n}\\n\\n/*you need this to stop horrible looking flickering of text as you drag*/\\nsvg {\\n -webkit-user-select: none;\\n -moz-user-select: none;\\n user-select: none;\\n}\\n\\n.xlv_text {\\n /*font-weight: bold;*/\\n text-shadow: -1px -1px 0 white,\\n 1px -1px 0 white,\\n -1px 1px 0 white,\\n 1px 1px 0 white;\\n /* -2px -1px 0 white,\\n 2px -1px 0 white,\\n -2px 1px 0 white,\\n 2px 1px 0 white;*/\\n}\\n\\n.proteinLabel {\\n font-size: 10pt;\\n /*font-weight: bold;*/\\n}\\n\\n.custom-menu-margin {\\n padding: 20px;\\n display: none;\\n z-index: 10000;\\n position: absolute;\\n}\\n\\n.custom-menu {\\n overflow: hidden;\\n border: 1px solid #CCC;\\n white-space: nowrap;\\n /*\\n font-family: sans-serif;\\n*/\\n background: #FFF;\\n color: #333;\\n list-style: none;\\n padding: 0;\\n margin: 0;\\n pointer-events: all;\\n}\\n\\n.custom-menu li {\\n padding: 8px 12px;\\n cursor: pointer;\\n -webkit-user-select: none;\\n -moz-user-select: none;\\n user-select: none;\\n}\\n\\n.custom-menu li:hover {\\n background-color: #DEF;\\n}\\n\\n.barScale {\\n display: inline;\\n padding-left: 10px;\\n}\\n\", \"\"]);\n// Exports\nmodule.exports = exports;\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L2Nqcy5qcyEuL3NyYy9jc3MveGluZXQuY3NzLmpzIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vY29tcGxleHZpZXdlci8uL3NyYy9jc3MveGluZXQuY3NzP2Y0YzQiXSwic291cmNlc0NvbnRlbnQiOlsiLy8gSW1wb3J0c1xudmFyIF9fX0NTU19MT0FERVJfQVBJX0lNUE9SVF9fXyA9IHJlcXVpcmUoXCIuLi8uLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvYXBpLmpzXCIpO1xuZXhwb3J0cyA9IF9fX0NTU19MT0FERVJfQVBJX0lNUE9SVF9fXyhmYWxzZSk7XG4vLyBNb2R1bGVcbmV4cG9ydHMucHVzaChbbW9kdWxlLmlkLCBcIi5wcm90ZWluIHtcXG4gICAgY3Vyc29yOiBjcm9zc2hhaXI7XFxufVxcblxcbi5saW5rIHtcXG4gICAgLypcXG4gICAgICAgY3Vyc29yOiBjcm9zc2hhaXI7XFxuICAgICovXFxufVxcblxcbi8qeW91IG5lZWQgdGhpcyB0byBzdG9wIGhvcnJpYmxlIGxvb2tpbmcgZmxpY2tlcmluZyBvZiB0ZXh0IGFzIHlvdSBkcmFnKi9cXG5zdmcge1xcbiAgICAtd2Via2l0LXVzZXItc2VsZWN0OiBub25lO1xcbiAgICAtbW96LXVzZXItc2VsZWN0OiBub25lO1xcbiAgICB1c2VyLXNlbGVjdDogbm9uZTtcXG59XFxuXFxuLnhsdl90ZXh0IHtcXG4gICAgLypmb250LXdlaWdodDogYm9sZDsqL1xcbiAgICB0ZXh0LXNoYWRvdzogLTFweCAtMXB4IDAgd2hpdGUsXFxuICAgIDFweCAtMXB4IDAgd2hpdGUsXFxuICAgIC0xcHggMXB4IDAgd2hpdGUsXFxuICAgIDFweCAxcHggMCB3aGl0ZTtcXG4gICAgLyogICAtMnB4IC0xcHggMCB3aGl0ZSxcXG4gICAgICAgIDJweCAtMXB4IDAgd2hpdGUsXFxuICAgICAgICAtMnB4IDFweCAwIHdoaXRlLFxcbiAgICAgICAgMnB4IDFweCAwIHdoaXRlOyovXFxufVxcblxcbi5wcm90ZWluTGFiZWwge1xcbiAgICBmb250LXNpemU6IDEwcHQ7XFxuICAgIC8qZm9udC13ZWlnaHQ6IGJvbGQ7Ki9cXG59XFxuXFxuLmN1c3RvbS1tZW51LW1hcmdpbiB7XFxuICAgIHBhZGRpbmc6IDIwcHg7XFxuICAgIGRpc3BsYXk6IG5vbmU7XFxuICAgIHotaW5kZXg6IDEwMDAwO1xcbiAgICBwb3NpdGlvbjogYWJzb2x1dGU7XFxufVxcblxcbi5jdXN0b20tbWVudSB7XFxuICAgIG92ZXJmbG93OiBoaWRkZW47XFxuICAgIGJvcmRlcjogMXB4IHNvbGlkICNDQ0M7XFxuICAgIHdoaXRlLXNwYWNlOiBub3dyYXA7XFxuICAgIC8qXFxuICBmb250LWZhbWlseTogc2Fucy1zZXJpZjtcXG4qL1xcbiAgICBiYWNrZ3JvdW5kOiAjRkZGO1xcbiAgICBjb2xvcjogIzMzMztcXG4gICAgbGlzdC1zdHlsZTogbm9uZTtcXG4gICAgcGFkZGluZzogMDtcXG4gICAgbWFyZ2luOiAwO1xcbiAgICBwb2ludGVyLWV2ZW50czogYWxsO1xcbn1cXG5cXG4uY3VzdG9tLW1lbnUgbGkge1xcbiAgICBwYWRkaW5nOiA4cHggMTJweDtcXG4gICAgY3Vyc29yOiBwb2ludGVyO1xcbiAgICAtd2Via2l0LXVzZXItc2VsZWN0OiBub25lO1xcbiAgICAtbW96LXVzZXItc2VsZWN0OiBub25lO1xcbiAgICB1c2VyLXNlbGVjdDogbm9uZTtcXG59XFxuXFxuLmN1c3RvbS1tZW51IGxpOmhvdmVyIHtcXG4gICAgYmFja2dyb3VuZC1jb2xvcjogI0RFRjtcXG59XFxuXFxuLmJhclNjYWxlIHtcXG4gICAgZGlzcGxheTogaW5saW5lO1xcbiAgICBwYWRkaW5nLWxlZnQ6IDEwcHg7XFxufVxcblwiLCBcIlwiXSk7XG4vLyBFeHBvcnRzXG5tb2R1bGUuZXhwb3J0cyA9IGV4cG9ydHM7XG4iXSwibWFwcGluZ3MiOiJBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOyIsInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./node_modules/css-loader/dist/cjs.js!./src/css/xinet.css\n"); +eval("// Imports\nvar ___CSS_LOADER_API_IMPORT___ = __webpack_require__(/*! ../../node_modules/css-loader/dist/runtime/api.js */ \"./node_modules/css-loader/dist/runtime/api.js\");\nexports = ___CSS_LOADER_API_IMPORT___(false);\n// Module\nexports.push([module.i, \"/*you need this to stop horrible looking flickering of text as you drag*/\\nsvg {\\n -webkit-user-select: none;\\n -moz-user-select: none;\\n user-select: none;\\n}\\n\\n.highlight {\\n stroke: #ffff99;\\n}\\n\\n.link {\\n stroke-linecap: round;\\n stroke: black;\\n}\\n\\n.certain-link {\\n opacity: 0.6;\\n stroke-opacity: 0.6;\\n}\\n\\n.uncertain-link {\\n opacity: 0.2;\\n stroke-opacity: 0.6;\\n}\\n\\n.link-line {\\n stroke-width: 1;\\n}\\n\\n.link-highlight {\\n stroke-width: 10;\\n stroke-opacity: 0;\\n}\\n\\n.complex-outline {\\n stroke: white;\\n stroke-linejoin: round;\\n stroke-width: 7;\\n}\\n\\n.linked-complex {\\n stroke: black;\\n stroke-linejoin: round;\\n stroke-width: 1;\\n}\\n\\nfeature-link {\\n fill: black;\\n}\\n\\n.protein {\\n cursor: crosshair;\\n}\\n\\n/*todo - seperate out outline*/\\n.label {\\n font-size: 10pt;\\n /*font-weight: bold;*/\\n\\n /*color: #fff;*/\\n /*text-shadow: white 0px 0px 1px;*/\\n -webkit-font-smoothing: antialiased;\\n\\n /*text-shadow: #fff 0px 0px 1px, #fff 0px 0px 1px, #fff 0px 0px 1px,*/\\n /*#fff 0px 0px 1px, #fff 0px 0px 1px, #fff 0px 0px 1px;*/\\n\\n text-shadow: -1px -1px 0 white,\\n 1px -1px 0 white,\\n -1px 1px 0 white,\\n 1px 1px 0 white;\\n /*-2px -1px 0 white,*/\\n /* 2px -1px 0 white,*/\\n /* -2px 1px 0 white,*/\\n /* 2px 1px 0 white;*/\\n fill: black;\\n text-anchor: end;\\n}\\n\\n.tooltip{\\n text-anchor: start;\\n}\\n\\n.outline {\\n stroke: black;\\n stroke-width: 1;\\n stroke-opacity: 1;\\n fill: white;\\n fill-opacity: 1;\\n}\\n\\n.participant-highlight {\\n stroke-width: 5;\\n fill: none;\\n}\\n\\n/*for protein bar scale*/\\n.tick {\\n stroke: black;\\n}\\n\\n.tick-labels {\\n font-size: 8pt;\\n text-anchor: middle;\\n}\\n\\n/*not working right*/\\n.sequence {\\n font-family: 'Courier New', monospace;\\n font-size: 10px;\\n text-anchor: middle;\\n}\\n\\n.custom-menu-margin {\\n padding: 20px;\\n display: none;\\n z-index: 10000;\\n position: absolute;\\n}\\n\\n.custom-menu {\\n overflow: hidden;\\n border: 1px solid #CCC;\\n white-space: nowrap;\\n background: #FFF;\\n color: #333;\\n list-style: none;\\n padding: 0;\\n margin: 0;\\n pointer-events: all;\\n}\\n\\n.custom-menu li {\\n padding: 8px 12px;\\n cursor: pointer;\\n -webkit-user-select: none;\\n -moz-user-select: none;\\n user-select: none;\\n}\\n\\n.custom-menu li:hover {\\n background-color: #DEF;\\n}\\n\\n.barScale {\\n display: inline;\\n padding-left: 10px;\\n}\\n\\n.tooltip-background {\\n fill-opacity: 0.75;\\n stroke-opacity: 1;\\n stroke-width: 1;\\n}\\n\\n.tooltip-sub-background {\\n fill: white;\\n stroke: white;\\n opacity: 1;\\n stroke-width: 1;\\n}\", \"\"]);\n// Exports\nmodule.exports = exports;\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L2Nqcy5qcyEuL3NyYy9jc3MveGluZXQuY3NzLmpzIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vY29tcGxleHZpZXdlci8uL3NyYy9jc3MveGluZXQuY3NzP2Y0YzQiXSwic291cmNlc0NvbnRlbnQiOlsiLy8gSW1wb3J0c1xudmFyIF9fX0NTU19MT0FERVJfQVBJX0lNUE9SVF9fXyA9IHJlcXVpcmUoXCIuLi8uLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvYXBpLmpzXCIpO1xuZXhwb3J0cyA9IF9fX0NTU19MT0FERVJfQVBJX0lNUE9SVF9fXyhmYWxzZSk7XG4vLyBNb2R1bGVcbmV4cG9ydHMucHVzaChbbW9kdWxlLmlkLCBcIi8qeW91IG5lZWQgdGhpcyB0byBzdG9wIGhvcnJpYmxlIGxvb2tpbmcgZmxpY2tlcmluZyBvZiB0ZXh0IGFzIHlvdSBkcmFnKi9cXG5zdmcge1xcbiAgICAtd2Via2l0LXVzZXItc2VsZWN0OiBub25lO1xcbiAgICAtbW96LXVzZXItc2VsZWN0OiBub25lO1xcbiAgICB1c2VyLXNlbGVjdDogbm9uZTtcXG59XFxuXFxuLmhpZ2hsaWdodCB7XFxuICAgIHN0cm9rZTogI2ZmZmY5OTtcXG59XFxuXFxuLmxpbmsge1xcbiAgICBzdHJva2UtbGluZWNhcDogcm91bmQ7XFxuICAgIHN0cm9rZTogYmxhY2s7XFxufVxcblxcbi5jZXJ0YWluLWxpbmsge1xcbiAgICBvcGFjaXR5OiAwLjY7XFxuICAgIHN0cm9rZS1vcGFjaXR5OiAwLjY7XFxufVxcblxcbi51bmNlcnRhaW4tbGluayB7XFxuICAgIG9wYWNpdHk6IDAuMjtcXG4gICAgc3Ryb2tlLW9wYWNpdHk6IDAuNjtcXG59XFxuXFxuLmxpbmstbGluZSB7XFxuICAgIHN0cm9rZS13aWR0aDogMTtcXG59XFxuXFxuLmxpbmstaGlnaGxpZ2h0IHtcXG4gICAgc3Ryb2tlLXdpZHRoOiAxMDtcXG4gICAgc3Ryb2tlLW9wYWNpdHk6IDA7XFxufVxcblxcbi5jb21wbGV4LW91dGxpbmUge1xcbiAgICBzdHJva2U6IHdoaXRlO1xcbiAgICBzdHJva2UtbGluZWpvaW46IHJvdW5kO1xcbiAgICBzdHJva2Utd2lkdGg6IDc7XFxufVxcblxcbi5saW5rZWQtY29tcGxleCB7XFxuICAgIHN0cm9rZTogYmxhY2s7XFxuICAgIHN0cm9rZS1saW5lam9pbjogcm91bmQ7XFxuICAgIHN0cm9rZS13aWR0aDogMTtcXG59XFxuXFxuZmVhdHVyZS1saW5rIHtcXG4gICAgZmlsbDogYmxhY2s7XFxufVxcblxcbi5wcm90ZWluIHtcXG4gICAgY3Vyc29yOiBjcm9zc2hhaXI7XFxufVxcblxcbi8qdG9kbyAtIHNlcGVyYXRlIG91dCBvdXRsaW5lKi9cXG4ubGFiZWwge1xcbiAgICBmb250LXNpemU6IDEwcHQ7XFxuICAgIC8qZm9udC13ZWlnaHQ6IGJvbGQ7Ki9cXG5cXG4gICAgLypjb2xvcjogI2ZmZjsqL1xcbiAgICAvKnRleHQtc2hhZG93OiB3aGl0ZSAwcHggMHB4IDFweDsqL1xcbiAgICAtd2Via2l0LWZvbnQtc21vb3RoaW5nOiBhbnRpYWxpYXNlZDtcXG5cXG4gICAgLyp0ZXh0LXNoYWRvdzogI2ZmZiAwcHggMHB4IDFweCwgICAjZmZmIDBweCAwcHggMXB4LCAgICNmZmYgMHB4IDBweCAxcHgsKi9cXG4gICAgLyojZmZmIDBweCAwcHggMXB4LCAgICNmZmYgMHB4IDBweCAxcHgsICAgI2ZmZiAwcHggMHB4IDFweDsqL1xcblxcbiAgICB0ZXh0LXNoYWRvdzogLTFweCAtMXB4IDAgd2hpdGUsXFxuICAgIDFweCAtMXB4IDAgd2hpdGUsXFxuICAgIC0xcHggMXB4IDAgd2hpdGUsXFxuICAgIDFweCAxcHggMCB3aGl0ZTtcXG4gICAgICAgLyotMnB4IC0xcHggMCB3aGl0ZSwqL1xcbiAgICAgICAvKiAycHggLTFweCAwIHdoaXRlLCovXFxuICAgICAgIC8qIC0ycHggMXB4IDAgd2hpdGUsKi9cXG4gICAgICAgLyogMnB4IDFweCAwIHdoaXRlOyovXFxuICAgIGZpbGw6IGJsYWNrO1xcbiAgICB0ZXh0LWFuY2hvcjogZW5kO1xcbn1cXG5cXG4udG9vbHRpcHtcXG4gICAgdGV4dC1hbmNob3I6IHN0YXJ0O1xcbn1cXG5cXG4ub3V0bGluZSB7XFxuICAgIHN0cm9rZTogYmxhY2s7XFxuICAgIHN0cm9rZS13aWR0aDogMTtcXG4gICAgc3Ryb2tlLW9wYWNpdHk6IDE7XFxuICAgIGZpbGw6IHdoaXRlO1xcbiAgICBmaWxsLW9wYWNpdHk6IDE7XFxufVxcblxcbi5wYXJ0aWNpcGFudC1oaWdobGlnaHQge1xcbiAgICBzdHJva2Utd2lkdGg6IDU7XFxuICAgIGZpbGw6IG5vbmU7XFxufVxcblxcbi8qZm9yIHByb3RlaW4gYmFyIHNjYWxlKi9cXG4udGljayB7XFxuICAgIHN0cm9rZTogYmxhY2s7XFxufVxcblxcbi50aWNrLWxhYmVscyB7XFxuICAgIGZvbnQtc2l6ZTogOHB0O1xcbiAgICB0ZXh0LWFuY2hvcjogbWlkZGxlO1xcbn1cXG5cXG4vKm5vdCB3b3JraW5nIHJpZ2h0Ki9cXG4uc2VxdWVuY2Uge1xcbiAgICBmb250LWZhbWlseTogJ0NvdXJpZXIgTmV3JywgbW9ub3NwYWNlO1xcbiAgICBmb250LXNpemU6IDEwcHg7XFxuICAgIHRleHQtYW5jaG9yOiBtaWRkbGU7XFxufVxcblxcbi5jdXN0b20tbWVudS1tYXJnaW4ge1xcbiAgICBwYWRkaW5nOiAyMHB4O1xcbiAgICBkaXNwbGF5OiBub25lO1xcbiAgICB6LWluZGV4OiAxMDAwMDtcXG4gICAgcG9zaXRpb246IGFic29sdXRlO1xcbn1cXG5cXG4uY3VzdG9tLW1lbnUge1xcbiAgICBvdmVyZmxvdzogaGlkZGVuO1xcbiAgICBib3JkZXI6IDFweCBzb2xpZCAjQ0NDO1xcbiAgICB3aGl0ZS1zcGFjZTogbm93cmFwO1xcbiAgICBiYWNrZ3JvdW5kOiAjRkZGO1xcbiAgICBjb2xvcjogIzMzMztcXG4gICAgbGlzdC1zdHlsZTogbm9uZTtcXG4gICAgcGFkZGluZzogMDtcXG4gICAgbWFyZ2luOiAwO1xcbiAgICBwb2ludGVyLWV2ZW50czogYWxsO1xcbn1cXG5cXG4uY3VzdG9tLW1lbnUgbGkge1xcbiAgICBwYWRkaW5nOiA4cHggMTJweDtcXG4gICAgY3Vyc29yOiBwb2ludGVyO1xcbiAgICAtd2Via2l0LXVzZXItc2VsZWN0OiBub25lO1xcbiAgICAtbW96LXVzZXItc2VsZWN0OiBub25lO1xcbiAgICB1c2VyLXNlbGVjdDogbm9uZTtcXG59XFxuXFxuLmN1c3RvbS1tZW51IGxpOmhvdmVyIHtcXG4gICAgYmFja2dyb3VuZC1jb2xvcjogI0RFRjtcXG59XFxuXFxuLmJhclNjYWxlIHtcXG4gICAgZGlzcGxheTogaW5saW5lO1xcbiAgICBwYWRkaW5nLWxlZnQ6IDEwcHg7XFxufVxcblxcbi50b29sdGlwLWJhY2tncm91bmQge1xcbiAgICBmaWxsLW9wYWNpdHk6IDAuNzU7XFxuICAgIHN0cm9rZS1vcGFjaXR5OiAxO1xcbiAgICBzdHJva2Utd2lkdGg6IDE7XFxufVxcblxcbi50b29sdGlwLXN1Yi1iYWNrZ3JvdW5kIHtcXG4gICAgZmlsbDogd2hpdGU7XFxuICAgIHN0cm9rZTogd2hpdGU7XFxuICAgIG9wYWNpdHk6IDE7XFxuICAgIHN0cm9rZS13aWR0aDogMTtcXG59XCIsIFwiXCJdKTtcbi8vIEV4cG9ydHNcbm1vZHVsZS5leHBvcnRzID0gZXhwb3J0cztcbiJdLCJtYXBwaW5ncyI6IkFBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7Iiwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./node_modules/css-loader/dist/cjs.js!./src/css/xinet.css\n"); /***/ }), @@ -1130,7 +1130,7 @@ eval("var api = __webpack_require__(/*! ../../node_modules/style-loader/dist/run /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"fetchAnnotations\", function() { return fetchAnnotations; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"chooseColors\", function() { return chooseColors; });\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! d3 */ \"./node_modules/d3/d3.js\");\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(d3__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var d3_scale_chromatic__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! d3-scale-chromatic */ \"./node_modules/d3-scale-chromatic/src/index.js\");\n/* harmony import */ var _viz_interactor_annotation__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./viz/interactor/annotation */ \"./src/js/viz/interactor/annotation.js\");\n/* harmony import */ var _viz_sequence_datum__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./viz/sequence-datum */ \"./src/js/viz/sequence-datum.js\");\n\n\n\n\n\n\n//todo - cache annotations in memory\nfunction fetchAnnotations(annotationChoice, /*App*/ app, callback) {\n annotationChoice = annotationChoice.toUpperCase();\n // we only show annotations on proteins\n const proteins = Array.from(app.participants.values()).filter(function (value) {\n return value.type === \"protein\";\n });\n\n function clearHighlights(){\n for (let prot of proteins){\n prot.showHighlight(false);\n }\n }\n let protsAnnotated = 0;\n const molCount = proteins.length;\n\n if (annotationChoice === \"INTERACTOR\") {\n if (app.proteinCount < 21) {\n for (let prot of proteins) {\n const annotation = new _viz_interactor_annotation__WEBPACK_IMPORTED_MODULE_2__[\"Annotation\"](prot.json.label, new _viz_sequence_datum__WEBPACK_IMPORTED_MODULE_3__[\"SequenceDatum\"](null, 1 + \"-\" + prot.size));\n let annotations = prot.annotationSets.get(annotationChoice);\n if (typeof annotationSet === \"undefined\") {\n annotations = [];\n prot.annotationSets.set(annotationChoice, annotations);\n }\n annotations.push(annotation);\n }\n // app.annotationSetsShown.set(\"INTERACTOR\", true);\n } else {\n alert(\"Too many (> 20) - can't color by interactor.\");\n }\n callback();\n } else if (annotationChoice.toUpperCase() === \"SUPERFAMILY\") {\n for (let prot of proteins) {\n getSuperFamFeatures(prot, function () {\n protsAnnotated++;\n if (protsAnnotated === molCount) {\n // clearHighlights();\n callback();\n }\n });\n }\n } else if (annotationChoice.toUpperCase() === \"UNIPROTKB\") {\n for (let prot of proteins) {\n getUniProtFeatures(prot, function () {\n protsAnnotated++;\n if (protsAnnotated === molCount) {\n // clearHighlights();\n callback();\n }\n });\n }\n }\n}\n\nfunction extractUniprotAccession(id) {\n const uniprotAccRegex = new RegExp(\"[OPQ][0-9][A-Z0-9]{3}[0-9]|[A-NR-Z][0-9]([A-Z][A-Z0-9]{2}[0-9]){1,2}\", \"i\");\n const match = uniprotAccRegex.exec(id);\n return match[0];\n}\n\nfunction getUniProtFeatures(prot, callback) {\n const url = \"https://www.ebi.ac.uk/proteins/api/proteins/\" + extractUniprotAccession(prot.id);\n d3__WEBPACK_IMPORTED_MODULE_0__[\"json\"](url, function (json) {\n let annotations = prot.annotationSets.get(\"UNIPROTKB\");\n if (typeof annotations === \"undefined\") {\n annotations = [];\n prot.annotationSets.set(\"UNIPROTKB\", annotations);\n }\n var uniprotJsonFeatures = json.features.filter(function (ft) {\n return ft.type === \"DOMAIN\";\n });\n for (let feature of uniprotJsonFeatures) {\n feature.seqDatum = new _viz_sequence_datum__WEBPACK_IMPORTED_MODULE_3__[\"SequenceDatum\"](null, feature.begin + \"-\" + feature.end);\n annotations.push(feature);\n }\n // prot.showHighlight(true);\n callback();\n });\n}\n\nfunction getSuperFamFeatures(prot, callback) {\n const url = \"https://supfam.mrc-lmb.cam.ac.uk/SUPERFAMILY/cgi-bin/das/up/features?segment=\" + extractUniprotAccession(prot.id);\n d3__WEBPACK_IMPORTED_MODULE_0__[\"xml\"](url, function (xml) {\n let annotations = prot.annotationSets.get(\"SUPERFAMILY\");\n if (typeof annotations === \"undefined\") {\n annotations = [];\n prot.annotationSets.set(\"SUPERFAMILY\", annotations);\n }\n const xmlDoc = new DOMParser().parseFromString(new XMLSerializer().serializeToString(xml), \"text/xml\");\n const xmlFeatures = xmlDoc.getElementsByTagName(\"FEATURE\");\n for (let xmlFeature of xmlFeatures) {\n const type = xmlFeature.getElementsByTagName(\"TYPE\")[0]; //might need to watch for text nodes getting mixed in here\n const category = type.getAttribute(\"category\");\n if (category === \"miscellaneous\") {\n const name = type.getAttribute(\"id\");\n const start = xmlFeature.getElementsByTagName(\"START\")[0].textContent;\n const end = xmlFeature.getElementsByTagName(\"END\")[0].textContent;\n annotations.push(new _viz_interactor_annotation__WEBPACK_IMPORTED_MODULE_2__[\"Annotation\"](name, new _viz_sequence_datum__WEBPACK_IMPORTED_MODULE_3__[\"SequenceDatum\"](null, start + \"-\" + end)));\n }\n }\n //~ console.log(JSON.stringify(features));\n // prot.showHighlight(true);\n callback();\n });\n}\n\nfunction chooseColors(app) {\n const categories = new Set();\n for (let participant of app.participants.values()) {\n for (let [annotationType, annotationSet] of participant.annotationSets) {\n if (app.annotationSetsShown.get(annotationType) === true) {\n for (let annotation of annotationSet.values()) {\n categories.add(annotation.description);\n }\n }\n }\n }\n\n let colorScheme;\n if (categories.size < 11) {\n colorScheme = d3__WEBPACK_IMPORTED_MODULE_0__[\"scale\"].ordinal().range(d3_scale_chromatic__WEBPACK_IMPORTED_MODULE_1__[\"schemeTableau10\"]);//colorbrewer.Dark2[catCount].slice().reverse());\n // } else if (catCount < 13) {\n // // var reversed = colorbrewer.Paired[catCount];//.slice().reverse();\n // colorScheme = d3.scale.ordinal().range(d3_chromatic.schemeSet3);\n } else {\n colorScheme = d3__WEBPACK_IMPORTED_MODULE_0__[\"scale\"].category20();\n }\n\n for (let participant of app.participants.values()) {\n for (let [annotationType, annotations] of participant.annotationSets) {\n if (app.annotationSetsShown.get(annotationType) === true) {\n for (let anno of annotations) {\n\n let color;\n if (anno.description === \"No annotations\") {\n color = \"#eeeeee\";\n } else {\n color = colorScheme(anno.description);\n }\n\n //ToDO - way more of these are being created than needed\n app.createHatchedFill(\"checkers_\" + anno.description, color);\n const checkedFill = \"url('#checkers_\" + anno.description + \"')\";\n if (anno.fuzzyStart) {\n anno.fuzzyStart.setAttribute(\"fill\", checkedFill);\n // anno.fuzzyStart.setAttribute(\"stroke\", color);\n }\n if (anno.certain) {\n anno.certain.setAttribute(\"fill\", color);\n // anno.certain.setAttribute(\"stroke\", color);\n }\n if (anno.fuzzyEnd) {\n anno.fuzzyEnd.setAttribute(\"fill\", checkedFill);\n // anno.fuzzyEnd.setAttribute(\"stroke\", color);\n }\n }\n }\n }\n }\n\n app.featureColors = colorScheme;\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/js/annotations.js.js","sources":["webpack://complexviewer/./src/js/annotations.js?f992"],"sourcesContent":["import * as d3 from \"d3\";\nimport * as d3_chromatic from \"d3-scale-chromatic\";\nimport {Annotation} from \"./viz/interactor/annotation\";\nimport {SequenceDatum} from \"./viz/sequence-datum\";\n\n\n//todo - cache annotations in memory\nexport function fetchAnnotations(annotationChoice, /*App*/ app, callback) {\n    annotationChoice = annotationChoice.toUpperCase();\n    // we only show annotations on proteins\n    const proteins = Array.from(app.participants.values()).filter(function (value) {\n        return value.type === \"protein\";\n    });\n\n    function clearHighlights(){\n        for (let prot of proteins){\n            prot.showHighlight(false);\n        }\n    }\n    let protsAnnotated = 0;\n    const molCount = proteins.length;\n\n    if (annotationChoice === \"INTERACTOR\") {\n        if (app.proteinCount < 21) {\n            for (let prot of proteins) {\n                const annotation = new Annotation(prot.json.label, new SequenceDatum(null, 1 + \"-\" + prot.size));\n                let annotations = prot.annotationSets.get(annotationChoice);\n                if (typeof annotationSet === \"undefined\") {\n                    annotations = [];\n                    prot.annotationSets.set(annotationChoice, annotations);\n                }\n                annotations.push(annotation);\n            }\n            // app.annotationSetsShown.set(\"INTERACTOR\", true);\n        } else {\n            alert(\"Too many (> 20) - can't color by interactor.\");\n        }\n        callback();\n    } else if (annotationChoice.toUpperCase() === \"SUPERFAMILY\") {\n        for (let prot of proteins) {\n            getSuperFamFeatures(prot, function () {\n                protsAnnotated++;\n                if (protsAnnotated === molCount) {\n                    // clearHighlights();\n                    callback();\n                }\n            });\n        }\n    } else if (annotationChoice.toUpperCase() === \"UNIPROTKB\") {\n        for (let prot of proteins) {\n            getUniProtFeatures(prot, function () {\n                protsAnnotated++;\n                if (protsAnnotated === molCount) {\n                    // clearHighlights();\n                    callback();\n                }\n            });\n        }\n    }\n}\n\nfunction extractUniprotAccession(id) {\n    const uniprotAccRegex = new RegExp(\"[OPQ][0-9][A-Z0-9]{3}[0-9]|[A-NR-Z][0-9]([A-Z][A-Z0-9]{2}[0-9]){1,2}\", \"i\");\n    const match = uniprotAccRegex.exec(id);\n    return match[0];\n}\n\nfunction getUniProtFeatures(prot, callback) {\n    const url = \"https://www.ebi.ac.uk/proteins/api/proteins/\" + extractUniprotAccession(prot.id);\n    d3.json(url, function (json) {\n        let annotations = prot.annotationSets.get(\"UNIPROTKB\");\n        if (typeof annotations === \"undefined\") {\n            annotations = [];\n            prot.annotationSets.set(\"UNIPROTKB\", annotations);\n        }\n        var uniprotJsonFeatures = json.features.filter(function (ft) {\n            return ft.type === \"DOMAIN\";\n        });\n        for (let feature of uniprotJsonFeatures) {\n            feature.seqDatum = new SequenceDatum(null, feature.begin + \"-\" + feature.end);\n            annotations.push(feature);\n        }\n        // prot.showHighlight(true);\n        callback();\n    });\n}\n\nfunction getSuperFamFeatures(prot, callback) {\n    const url = \"https://supfam.mrc-lmb.cam.ac.uk/SUPERFAMILY/cgi-bin/das/up/features?segment=\" + extractUniprotAccession(prot.id);\n    d3.xml(url, function (xml) {\n        let annotations = prot.annotationSets.get(\"SUPERFAMILY\");\n        if (typeof annotations === \"undefined\") {\n            annotations = [];\n            prot.annotationSets.set(\"SUPERFAMILY\", annotations);\n        }\n        const xmlDoc = new DOMParser().parseFromString(new XMLSerializer().serializeToString(xml), \"text/xml\");\n        const xmlFeatures = xmlDoc.getElementsByTagName(\"FEATURE\");\n        for (let xmlFeature of xmlFeatures) {\n            const type = xmlFeature.getElementsByTagName(\"TYPE\")[0]; //might need to watch for text nodes getting mixed in here\n            const category = type.getAttribute(\"category\");\n            if (category === \"miscellaneous\") {\n                const name = type.getAttribute(\"id\");\n                const start = xmlFeature.getElementsByTagName(\"START\")[0].textContent;\n                const end = xmlFeature.getElementsByTagName(\"END\")[0].textContent;\n                annotations.push(new Annotation(name, new SequenceDatum(null, start + \"-\" + end)));\n            }\n        }\n        //~ console.log(JSON.stringify(features));\n        // prot.showHighlight(true);\n        callback();\n    });\n}\n\nexport function chooseColors(app) {\n    const categories = new Set();\n    for (let participant of app.participants.values()) {\n        for (let [annotationType, annotationSet] of participant.annotationSets) {\n            if (app.annotationSetsShown.get(annotationType) === true) {\n                for (let annotation of annotationSet.values()) {\n                    categories.add(annotation.description);\n                }\n            }\n        }\n    }\n\n    let colorScheme;\n    if (categories.size < 11) {\n        colorScheme = d3.scale.ordinal().range(d3_chromatic.schemeTableau10);//colorbrewer.Dark2[catCount].slice().reverse());\n        // } else if (catCount < 13) {\n        //     // var reversed = colorbrewer.Paired[catCount];//.slice().reverse();\n        //     colorScheme = d3.scale.ordinal().range(d3_chromatic.schemeSet3);\n    } else {\n        colorScheme = d3.scale.category20();\n    }\n\n    for (let participant of app.participants.values()) {\n        for (let [annotationType, annotations] of participant.annotationSets) {\n            if (app.annotationSetsShown.get(annotationType) === true) {\n                for (let anno of annotations) {\n\n                    let color;\n                    if (anno.description === \"No annotations\") {\n                        color = \"#eeeeee\";\n                    } else {\n                        color = colorScheme(anno.description);\n                    }\n\n                    //ToDO - way more of these are being created than needed\n                    app.createHatchedFill(\"checkers_\" + anno.description, color);\n                    const checkedFill = \"url('#checkers_\" + anno.description + \"')\";\n                    if (anno.fuzzyStart) {\n                        anno.fuzzyStart.setAttribute(\"fill\", checkedFill);\n                        // anno.fuzzyStart.setAttribute(\"stroke\", color);\n                    }\n                    if (anno.certain) {\n                        anno.certain.setAttribute(\"fill\", color);\n                        // anno.certain.setAttribute(\"stroke\", color);\n                    }\n                    if (anno.fuzzyEnd) {\n                        anno.fuzzyEnd.setAttribute(\"fill\", checkedFill);\n                        // anno.fuzzyEnd.setAttribute(\"stroke\", color);\n                    }\n                }\n            }\n        }\n    }\n\n    app.featureColors = colorScheme;\n}"],"mappings":"AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/js/annotations.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"fetchAnnotations\", function() { return fetchAnnotations; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"chooseColors\", function() { return chooseColors; });\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! d3 */ \"./node_modules/d3/d3.js\");\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(d3__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var d3_scale_chromatic__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! d3-scale-chromatic */ \"./node_modules/d3-scale-chromatic/src/index.js\");\n/* harmony import */ var _viz_interactor_annotation__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./viz/interactor/annotation */ \"./src/js/viz/interactor/annotation.js\");\n/* harmony import */ var _viz_sequence_datum__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./viz/sequence-datum */ \"./src/js/viz/sequence-datum.js\");\n\n\n\n\n\n\n//todo - cache annotations in memory\nfunction fetchAnnotations(annotationChoice, /*App*/ app, callback) {\n annotationChoice = annotationChoice.toUpperCase();\n // we only show annotations on proteins\n const proteins = Array.from(app.participants.values()).filter(function (value) {\n return value.type === \"protein\";\n });\n\n function clearHighlights(){\n for (let prot of proteins){\n prot.showHighlight(false);\n }\n }\n let protsAnnotated = 0;\n const molCount = proteins.length;\n\n if (annotationChoice === \"INTERACTOR\") { //todo - move this out of here\n if (app.proteinCount < 21) {\n for (let prot of proteins) {\n const annotation = new _viz_interactor_annotation__WEBPACK_IMPORTED_MODULE_2__[\"Annotation\"](prot.json.label, new _viz_sequence_datum__WEBPACK_IMPORTED_MODULE_3__[\"SequenceDatum\"](null, 1 + \"-\" + prot.size));\n let annotations = prot.annotationSets.get(annotationChoice);\n if (typeof annotationSet === \"undefined\") {\n annotations = [];\n prot.annotationSets.set(annotationChoice, annotations);\n }\n annotations.push(annotation);\n }\n // app.annotationSetsShown.set(\"INTERACTOR\", true);\n } else {\n // alert(\"Too many (> 20) - can't color by interactor.\"); // people are gong to complain about why arent my interactor colours showing up\n }\n callback();\n } else if (annotationChoice.toUpperCase() === \"SUPERFAMILY\") {\n for (let prot of proteins) {\n getSuperFamFeatures(prot, function () {\n protsAnnotated++;\n if (protsAnnotated === molCount) {\n // clearHighlights();\n callback();\n }\n });\n }\n } else if (annotationChoice.toUpperCase() === \"UNIPROTKB\") {\n for (let prot of proteins) {\n getUniProtFeatures(prot, function () {\n protsAnnotated++;\n if (protsAnnotated === molCount) {\n // clearHighlights();\n callback();\n }\n });\n }\n }\n}\n\nfunction extractUniprotAccession(id) {\n const uniprotAccRegex = new RegExp(\"[OPQ][0-9][A-Z0-9]{3}[0-9]|[A-NR-Z][0-9]([A-Z][A-Z0-9]{2}[0-9]){1,2}\", \"i\");\n const match = uniprotAccRegex.exec(id);\n return match[0];\n}\n\nfunction getUniProtFeatures(prot, callback) {\n const url = \"https://www.ebi.ac.uk/proteins/api/proteins/\" + extractUniprotAccession(prot.id);\n d3__WEBPACK_IMPORTED_MODULE_0__[\"json\"](url, function (json) {\n let annotations = prot.annotationSets.get(\"UNIPROTKB\");\n if (typeof annotations === \"undefined\") {\n annotations = [];\n prot.annotationSets.set(\"UNIPROTKB\", annotations);\n }\n var uniprotJsonFeatures = json.features.filter(function (ft) {\n return ft.type === \"DOMAIN\";\n });\n for (let feature of uniprotJsonFeatures) {\n feature.seqDatum = new _viz_sequence_datum__WEBPACK_IMPORTED_MODULE_3__[\"SequenceDatum\"](null, feature.begin + \"-\" + feature.end);\n annotations.push(feature);\n }\n // prot.showHighlight(true);\n callback();\n });\n}\n\nfunction getSuperFamFeatures(prot, callback) {\n const url = \"https://supfam.mrc-lmb.cam.ac.uk/SUPERFAMILY/cgi-bin/das/up/features?segment=\" + extractUniprotAccession(prot.id);\n d3__WEBPACK_IMPORTED_MODULE_0__[\"xml\"](url, function (xml) {\n let annotations = prot.annotationSets.get(\"SUPERFAMILY\");\n if (typeof annotations === \"undefined\") {\n annotations = [];\n prot.annotationSets.set(\"SUPERFAMILY\", annotations);\n }\n const xmlDoc = new DOMParser().parseFromString(new XMLSerializer().serializeToString(xml), \"text/xml\");\n const xmlFeatures = xmlDoc.getElementsByTagName(\"FEATURE\");\n for (let xmlFeature of xmlFeatures) {\n const type = xmlFeature.getElementsByTagName(\"TYPE\")[0]; //might need to watch for text nodes getting mixed in here\n const category = type.getAttribute(\"category\");\n if (category === \"miscellaneous\") {\n const name = type.getAttribute(\"id\");\n const start = xmlFeature.getElementsByTagName(\"START\")[0].textContent;\n const end = xmlFeature.getElementsByTagName(\"END\")[0].textContent;\n annotations.push(new _viz_interactor_annotation__WEBPACK_IMPORTED_MODULE_2__[\"Annotation\"](name, new _viz_sequence_datum__WEBPACK_IMPORTED_MODULE_3__[\"SequenceDatum\"](null, start + \"-\" + end)));\n }\n }\n //~ console.log(JSON.stringify(features));\n // prot.showHighlight(true);\n callback();\n });\n}\n\nfunction chooseColors(app) {\n const categories = new Set();\n for (let participant of app.participants.values()) {\n for (let [annotationType, annotationSet] of participant.annotationSets) {\n if (app.annotationSetsShown.get(annotationType) === true) {\n for (let annotation of annotationSet.values()) {\n categories.add(annotation.description);\n }\n }\n }\n }\n\n let colorScheme;\n if (categories.size < 11) {\n colorScheme = d3__WEBPACK_IMPORTED_MODULE_0__[\"scale\"].ordinal().range(d3_scale_chromatic__WEBPACK_IMPORTED_MODULE_1__[\"schemeTableau10\"]);//colorbrewer.Dark2[catCount].slice().reverse());\n // } else if (catCount < 13) {\n // // var reversed = colorbrewer.Paired[catCount];//.slice().reverse();\n // colorScheme = d3.scale.ordinal().range(d3_chromatic.schemeSet3);\n } else {\n colorScheme = d3__WEBPACK_IMPORTED_MODULE_0__[\"scale\"].category20();\n }\n\n for (let participant of app.participants.values()) {\n for (let [annotationType, annotations] of participant.annotationSets) {\n if (app.annotationSetsShown.get(annotationType) === true) {\n for (let anno of annotations) {\n\n let color;\n if (anno.description === \"No annotations\") {\n color = \"#eeeeee\";\n } else {\n color = colorScheme(anno.description);\n }\n\n //ToDO - way more of these are being created than needed\n app.createHatchedFill(\"checkers_\" + anno.description, color);\n const checkedFill = \"url('#checkers_\" + anno.description + \"')\";\n if (anno.fuzzyStart) {\n anno.fuzzyStart.setAttribute(\"fill\", checkedFill);\n // anno.fuzzyStart.setAttribute(\"stroke\", color);\n }\n if (anno.certain) {\n anno.certain.setAttribute(\"fill\", color);\n // anno.certain.setAttribute(\"stroke\", color);\n }\n if (anno.fuzzyEnd) {\n anno.fuzzyEnd.setAttribute(\"fill\", checkedFill);\n // anno.fuzzyEnd.setAttribute(\"stroke\", color);\n }\n }\n }\n }\n }\n\n app.featureColors = colorScheme;\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/js/annotations.js.js","sources":["webpack://complexviewer/./src/js/annotations.js?f992"],"sourcesContent":["import * as d3 from \"d3\";\nimport * as d3_chromatic from \"d3-scale-chromatic\";\nimport {Annotation} from \"./viz/interactor/annotation\";\nimport {SequenceDatum} from \"./viz/sequence-datum\";\n\n\n//todo - cache annotations in memory\nexport function fetchAnnotations(annotationChoice, /*App*/ app, callback) {\n    annotationChoice = annotationChoice.toUpperCase();\n    // we only show annotations on proteins\n    const proteins = Array.from(app.participants.values()).filter(function (value) {\n        return value.type === \"protein\";\n    });\n\n    function clearHighlights(){\n        for (let prot of proteins){\n            prot.showHighlight(false);\n        }\n    }\n    let protsAnnotated = 0;\n    const molCount = proteins.length;\n\n    if (annotationChoice === \"INTERACTOR\") { //todo - move this out of here\n        if (app.proteinCount < 21) {\n            for (let prot of proteins) {\n                const annotation = new Annotation(prot.json.label, new SequenceDatum(null, 1 + \"-\" + prot.size));\n                let annotations = prot.annotationSets.get(annotationChoice);\n                if (typeof annotationSet === \"undefined\") {\n                    annotations = [];\n                    prot.annotationSets.set(annotationChoice, annotations);\n                }\n                annotations.push(annotation);\n            }\n            // app.annotationSetsShown.set(\"INTERACTOR\", true);\n        } else {\n            // alert(\"Too many (> 20) - can't color by interactor.\"); // people are gong to complain about why arent my interactor colours showing up\n        }\n        callback();\n    } else if (annotationChoice.toUpperCase() === \"SUPERFAMILY\") {\n        for (let prot of proteins) {\n            getSuperFamFeatures(prot, function () {\n                protsAnnotated++;\n                if (protsAnnotated === molCount) {\n                    // clearHighlights();\n                    callback();\n                }\n            });\n        }\n    } else if (annotationChoice.toUpperCase() === \"UNIPROTKB\") {\n        for (let prot of proteins) {\n            getUniProtFeatures(prot, function () {\n                protsAnnotated++;\n                if (protsAnnotated === molCount) {\n                    // clearHighlights();\n                    callback();\n                }\n            });\n        }\n    }\n}\n\nfunction extractUniprotAccession(id) {\n    const uniprotAccRegex = new RegExp(\"[OPQ][0-9][A-Z0-9]{3}[0-9]|[A-NR-Z][0-9]([A-Z][A-Z0-9]{2}[0-9]){1,2}\", \"i\");\n    const match = uniprotAccRegex.exec(id);\n    return match[0];\n}\n\nfunction getUniProtFeatures(prot, callback) {\n    const url = \"https://www.ebi.ac.uk/proteins/api/proteins/\" + extractUniprotAccession(prot.id);\n    d3.json(url, function (json) {\n        let annotations = prot.annotationSets.get(\"UNIPROTKB\");\n        if (typeof annotations === \"undefined\") {\n            annotations = [];\n            prot.annotationSets.set(\"UNIPROTKB\", annotations);\n        }\n        var uniprotJsonFeatures = json.features.filter(function (ft) {\n            return ft.type === \"DOMAIN\";\n        });\n        for (let feature of uniprotJsonFeatures) {\n            feature.seqDatum = new SequenceDatum(null, feature.begin + \"-\" + feature.end);\n            annotations.push(feature);\n        }\n        // prot.showHighlight(true);\n        callback();\n    });\n}\n\nfunction getSuperFamFeatures(prot, callback) {\n    const url = \"https://supfam.mrc-lmb.cam.ac.uk/SUPERFAMILY/cgi-bin/das/up/features?segment=\" + extractUniprotAccession(prot.id);\n    d3.xml(url, function (xml) {\n        let annotations = prot.annotationSets.get(\"SUPERFAMILY\");\n        if (typeof annotations === \"undefined\") {\n            annotations = [];\n            prot.annotationSets.set(\"SUPERFAMILY\", annotations);\n        }\n        const xmlDoc = new DOMParser().parseFromString(new XMLSerializer().serializeToString(xml), \"text/xml\");\n        const xmlFeatures = xmlDoc.getElementsByTagName(\"FEATURE\");\n        for (let xmlFeature of xmlFeatures) {\n            const type = xmlFeature.getElementsByTagName(\"TYPE\")[0]; //might need to watch for text nodes getting mixed in here\n            const category = type.getAttribute(\"category\");\n            if (category === \"miscellaneous\") {\n                const name = type.getAttribute(\"id\");\n                const start = xmlFeature.getElementsByTagName(\"START\")[0].textContent;\n                const end = xmlFeature.getElementsByTagName(\"END\")[0].textContent;\n                annotations.push(new Annotation(name, new SequenceDatum(null, start + \"-\" + end)));\n            }\n        }\n        //~ console.log(JSON.stringify(features));\n        // prot.showHighlight(true);\n        callback();\n    });\n}\n\nexport function chooseColors(app) {\n    const categories = new Set();\n    for (let participant of app.participants.values()) {\n        for (let [annotationType, annotationSet] of participant.annotationSets) {\n            if (app.annotationSetsShown.get(annotationType) === true) {\n                for (let annotation of annotationSet.values()) {\n                    categories.add(annotation.description);\n                }\n            }\n        }\n    }\n\n    let colorScheme;\n    if (categories.size < 11) {\n        colorScheme = d3.scale.ordinal().range(d3_chromatic.schemeTableau10);//colorbrewer.Dark2[catCount].slice().reverse());\n        // } else if (catCount < 13) {\n        //     // var reversed = colorbrewer.Paired[catCount];//.slice().reverse();\n        //     colorScheme = d3.scale.ordinal().range(d3_chromatic.schemeSet3);\n    } else {\n        colorScheme = d3.scale.category20();\n    }\n\n    for (let participant of app.participants.values()) {\n        for (let [annotationType, annotations] of participant.annotationSets) {\n            if (app.annotationSetsShown.get(annotationType) === true) {\n                for (let anno of annotations) {\n\n                    let color;\n                    if (anno.description === \"No annotations\") {\n                        color = \"#eeeeee\";\n                    } else {\n                        color = colorScheme(anno.description);\n                    }\n\n                    //ToDO - way more of these are being created than needed\n                    app.createHatchedFill(\"checkers_\" + anno.description, color);\n                    const checkedFill = \"url('#checkers_\" + anno.description + \"')\";\n                    if (anno.fuzzyStart) {\n                        anno.fuzzyStart.setAttribute(\"fill\", checkedFill);\n                        // anno.fuzzyStart.setAttribute(\"stroke\", color);\n                    }\n                    if (anno.certain) {\n                        anno.certain.setAttribute(\"fill\", color);\n                        // anno.certain.setAttribute(\"stroke\", color);\n                    }\n                    if (anno.fuzzyEnd) {\n                        anno.fuzzyEnd.setAttribute(\"fill\", checkedFill);\n                        // anno.fuzzyEnd.setAttribute(\"stroke\", color);\n                    }\n                }\n            }\n        }\n    }\n\n    app.featureColors = colorScheme;\n}"],"mappings":"AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/js/annotations.js\n"); /***/ }), @@ -1142,7 +1142,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"App\", function() { return App; });\n/* harmony import */ var _css_xinet_css__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../css/xinet.css */ \"./src/css/xinet.css\");\n/* harmony import */ var _css_xinet_css__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_css_xinet_css__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _package_json__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../package.json */ \"./package.json\");\nvar _package_json__WEBPACK_IMPORTED_MODULE_1___namespace = /*#__PURE__*/__webpack_require__.t(/*! ../../package.json */ \"./package.json\", 1);\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! d3 */ \"./node_modules/d3/d3.js\");\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(d3__WEBPACK_IMPORTED_MODULE_2__);\n/* harmony import */ var d3_scale_chromatic__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! d3-scale-chromatic */ \"./node_modules/d3-scale-chromatic/src/index.js\");\n/* harmony import */ var _cola__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./cola */ \"./src/js/cola.js\");\n/* harmony import */ var _cola__WEBPACK_IMPORTED_MODULE_4___default = /*#__PURE__*/__webpack_require__.n(_cola__WEBPACK_IMPORTED_MODULE_4__);\n/* harmony import */ var _read_mijson__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./read-mijson */ \"./src/js/read-mijson.js\");\n/* harmony import */ var _annotations__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./annotations */ \"./src/js/annotations.js\");\n/* harmony import */ var _color_scheme_key__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ./color-scheme-key */ \"./src/js/color-scheme-key.js\");\n/* harmony import */ var _viz_link_nary_link__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ./viz/link/nary-link */ \"./src/js/viz/link/nary-link.js\");\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! ./config */ \"./src/js/config.js\");\n// eslint-disable-next-line no-unused-vars\n\n\n\n\n\n\n\n\n// import {SymbolKey} from \"./symbol-key\";\n\n\n\n\n// could refactor everything to use ES6 class syntax\n// but https://benmccormick.org/2015/04/07/es6-classes-and-backbone-js\n// \"ES6 classes don’t support adding properties directly to the class instance, only functions/methods\"\n// so backbone doesn't work\n// so continuing to use prototypical inheritance in things for time being\n\nfunction App(/*HTMLDivElement*/networkDiv) {\n // this.debug = true; // things aren't exactly lined up in the bounding boxes cola is using, to do so breaks symetery of some things\n this.el = networkDiv;\n\n this.STATES = {};\n this.STATES.MOUSE_UP = 0; //start state, also set when mouse up on svgElement\n this.STATES.PANNING = 1; //set by mouse down on svgElement - left button, no shift or util\n this.STATES.DRAGGING = 2; //set by mouse down on Protein or Link\n this.STATES.ROTATING = 3; //set by mouse down on Rotator, drag?\n this.STATES.SELECTING = 4; //set by mouse down on svgElement- right button or left button shift or util, drag\n\n //avoids prob with 'save - web page complete'\n this.el.textContent = \"\"; //https://stackoverflow.com/questions/3955229/remove-all-child-elements-of-a-dom-node-in-javascript\n\n this.d3cola = _cola__WEBPACK_IMPORTED_MODULE_4__[\"d3adaptor\"]();\n\n const customMenuSel = d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](this.el)\n .append(\"div\").classed(\"custom-menu-margin\", true)\n .append(\"div\").classed(\"custom-menu\", true)\n .append(\"ul\");\n\n const self = this;\n const collapse = customMenuSel.append(\"li\").classed(\"collapse\", true); //.append(\"button\");\n collapse.text(\"Collapse\");\n collapse[0][0].onclick = function (evt) {\n self.collapseProtein(evt);\n };\n const scaleButtonsListItemSel = customMenuSel.append(\"li\").text(\"Scale: \");\n\n this.barScales = [0.01, 0.2, 1, 2, 4, 8];\n const scaleButtons = scaleButtonsListItemSel.selectAll(\"ul.custom-menu\")\n .data(this.barScales)\n .enter()\n .append(\"div\")\n .attr(\"class\", \"barScale\")\n .append(\"label\");\n scaleButtons.append(\"span\")\n .text(function (d) {\n if (d === 8) return \"AA\";\n else return d;\n });\n scaleButtons.append(\"input\")\n // .attr (\"id\", function(d) { return d*100; })\n .attr(\"class\", function (d) {\n return \"scaleButton scaleButton_\" + (d * 100);\n })\n .attr(\"name\", \"scaleButtons\")\n .attr(\"type\", \"radio\")\n .on(\"change\", function (d) {\n self.preventDefaultsAndStopPropagation(d);\n self.contextMenuProt.setStickScale(d, self.contextMenuPoint);\n });\n\n const contextMenu = d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](\".custom-menu-margin\").node();\n contextMenu.onmouseout = function (evt) {\n let e = evt.relatedTarget;\n do {\n if (e === this) return;\n e = e.parentNode;\n } while (e);\n self.contextMenuProt = null;\n d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](this).style(\"display\", \"none\");\n };\n\n\n //create SVG element\n this.svgElement = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"svg\");\n this.svgElement.setAttribute(\"id\", \"complexViewerSVG\");\n\n //add listeners\n this.svgElement.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.svgElement.onmousemove = function (evt) {\n self.mouseMove(evt);\n };\n this.svgElement.onmouseup = function (evt) {\n self.mouseUp(evt);\n };\n this.svgElement.onmouseout = function (evt) {\n self.hideTooltip(evt);\n };\n this.lastMouseUp = new Date().getTime();\n /*this.svgElement.ontouchstart = function(evt) {\n self.touchStart(evt);\n };\n this.svgElement.ontouchmove = function(evt) {\n self.touchMove(evt);\n };\n this.svgElement.ontouchend = function(evt) {\n self.touchEnd(evt);\n };\n */\n\n this.el.oncontextmenu = function (evt) {\n if (evt.preventDefault) { // necessary for addEventListener, works with traditional\n evt.preventDefault();\n }\n evt.returnValue = false; // necessary for attachEvent, works with traditional\n return false; // works with traditional, not with attachEvent or addEventListener\n };\n\n //legend changed callbacks\n this.colorSchemeKeyDivs = new Set();\n\n this.el.appendChild(this.svgElement);\n\n // various groups needed\n this.container = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"g\");\n this.container.setAttribute(\"id\", \"container\");\n\n const svg = d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](this.svgElement);\n this.defs = svg.append(\"defs\");\n this.createHatchedFill(\"checkers_uncertain\", \"black\");\n\n //markers\n const data = [{\n id: 1,\n name: \"diamond\",\n path: \"M 0,-7.0710768 L 0,7.0710589 L 7.0710462,0 z\",\n viewbox: \"-15 -15 25 25\",\n transform: \"scale(1.5) translate(-5,0)\",\n color: \"black\"\n }];\n\n this.defs.selectAll(\"marker\")\n .data(data)\n .enter()\n .append(\"svg:marker\")\n .attr(\"id\", function (d) {\n return \"marker_\" + d.name;\n })\n .attr(\"markerHeight\", 15)\n .attr(\"markerWidth\", 15)\n .attr(\"markerUnits\", \"userSpaceOnUse\")\n .attr(\"orient\", \"auto\")\n .attr(\"refX\", 0)\n .attr(\"refY\", 0)\n .attr(\"viewBox\", function (d) {\n return d.viewbox;\n })\n .append(\"svg:path\")\n .attr(\"d\", function (d) {\n return d.path;\n })\n .attr(\"fill\", function (d) {\n return d.color;\n })\n .attr(\"transform\", function (d) {\n return d.transform;\n });\n\n this.acknowledgement = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"g\");\n const ackText = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"text\");\n ackText.innerHTML = \"ComplexViewer \"\n + _package_json__WEBPACK_IMPORTED_MODULE_1__[\"version\"] + \"by Rappsilber Laboratory\";\n\n this.acknowledgement.appendChild(ackText);\n ackText.classList.add(\"xlv_text\");\n ackText.setAttribute(\"font-size\", \"8pt\");\n this.svgElement.appendChild(this.acknowledgement);\n\n this.naryLinks = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"g\");\n this.naryLinks.setAttribute(\"id\", \"naryLinks\");\n this.container.appendChild(this.naryLinks);\n\n this.p_pLinksWide = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"g\");\n this.p_pLinksWide.setAttribute(\"id\", \"p_pLinksWide\");\n this.container.appendChild(this.p_pLinksWide);\n\n this.highlights = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"g\");\n this.highlights.setAttribute(\"class\", \"highlights\"); //interactors also contain highlight groups\n this.container.appendChild(this.highlights);\n\n this.res_resLinks = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"g\");\n this.res_resLinks.setAttribute(\"id\", \"res_resLinks\");\n this.container.appendChild(this.res_resLinks);\n\n this.p_pLinks = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"g\");\n this.p_pLinks.setAttribute(\"id\", \"p_pLinks\");\n this.container.appendChild(this.p_pLinks);\n\n this.proteinUpper = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"g\");\n this.proteinUpper.setAttribute(\"id\", \"proteinUpper\");\n this.container.appendChild(this.proteinUpper);\n\n this.selfRes_resLinks = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"g\");\n this.selfRes_resLinks.setAttribute(\"id\", \"res_resLinks\");\n this.container.appendChild(this.selfRes_resLinks);\n\n this.svgElement.appendChild(this.container);\n\n //showing title as tooltips is NOT part of svg spec (even though some browsers do this)\n //also more responsive / more control if we do out own\n this.tooltip = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"text\");\n this.tooltip.setAttribute(\"x\", \"0\");\n this.tooltip.setAttribute(\"y\", \"0\");\n this.tooltip.setAttribute(\"class\", \"xlv_text\");\n const tooltipTextNode = document.createTextNode(\"tooltip\");\n\n this.tooltip.appendChild(tooltipTextNode);\n\n this.tooltip_bg = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"rect\");\n this.tooltip_bg.setAttribute(\"class\", \"tooltip_bg\");\n\n this.tooltip_bg.setAttribute(\"fill-opacity\", \"0.75\");\n this.tooltip_bg.setAttribute(\"stroke-opacity\", \"1\");\n this.tooltip_bg.setAttribute(\"stroke-width\", \"1\");\n\n this.tooltip_subBg = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_9__[\"svgns\"], \"rect\");\n this.tooltip_subBg.setAttribute(\"fill\", \"white\");\n this.tooltip_subBg.setAttribute(\"stroke\", \"white\");\n this.tooltip_subBg.setAttribute(\"class\", \"tooltip_bg\");\n this.tooltip_subBg.setAttribute(\"opacity\", \"1\");\n this.tooltip_subBg.setAttribute(\"stroke-width\", \"1\");\n\n this.svgElement.appendChild(this.tooltip_subBg);\n this.svgElement.appendChild(this.tooltip_bg);\n this.svgElement.appendChild(this.tooltip);\n\n this.clear();\n}\n\nApp.prototype.createHatchedFill = function (name, color) {\n const pattern = this.defs.append(\"pattern\")\n .attr(\"id\", name)\n .attr(\"patternUnits\", \"userSpaceOnUse\")\n .attr(\"x\", 0)\n .attr(\"y\", 0)\n .attr(\"width\", 12)\n .attr(\"height\", 12)\n .attr(\"patternTransform\", \"rotate(45)\");\n\n pattern.append(\"rect\")\n .attr(\"x\", 0)\n .attr(\"y\", 2)\n .attr(\"width\", 12)\n .attr(\"height\", 4)\n .attr(\"fill\", color);\n\n pattern.append(\"rect\")\n .attr(\"x\", 0)\n .attr(\"y\", 8)\n .attr(\"width\", 12)\n .attr(\"height\", 4)\n .attr(\"fill\", color);\n};\n\nApp.prototype.clear = function () {\n this.d3cola.stop();\n\n this.annotationSetsShown = new Map();\n // this.annotationSetsShown.set(\"MI FEATURES\", true);\n\n //lighten colors\n const complexColors = [];\n for (let c of d3_scale_chromatic__WEBPACK_IMPORTED_MODULE_3__[\"schemePastel2\"]) {//colorbrewer.Pastel2[8]) {\n const hsl = d3__WEBPACK_IMPORTED_MODULE_2__[\"hsl\"](c);\n hsl.l = 0.9;\n complexColors.push(hsl + \"\");\n }\n\n _viz_link_nary_link__WEBPACK_IMPORTED_MODULE_8__[\"NaryLink\"].naryColors = d3__WEBPACK_IMPORTED_MODULE_2__[\"scale\"].ordinal().range(complexColors);\n this.defs.selectAll(\".feature_checkers\").remove();\n\n this.naryLinks.textContent = \"\";\n this.p_pLinksWide.textContent = \"\";\n this.highlights.textContent = \"\";\n this.p_pLinks.textContent = \"\";\n this.res_resLinks.textContent = \"\";\n this.proteinUpper.textContent = \"\";\n this.selfRes_resLinks.textContent = \"\";\n\n // if we are dragging something at the moment - this will be the element that is dragged\n this.dragElement = null;\n // from where did we start dragging\n this.dragStart = {};\n\n this.participants = new Map();\n this.allNaryLinks = new Map();\n this.allBinaryLinks = new Map();\n this.allUnaryLinks = new Map();\n this.allSequenceLinks = new Map();\n this.complexes = [];\n\n this.proteinCount = 0;\n this.z = 1;\n this.hideTooltip();\n this.state = this.STATES.MOUSE_UP;\n};\n\nApp.prototype.collapseProtein = function () {\n d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](\".custom-menu-margin\").style(\"display\", \"none\");\n this.contextMenuProt.setForm(0, this.contextMenuPoint);\n this.contextMenuProt = null;\n};\n\nApp.prototype.init = function () {\n this.d3cola.stop();\n\n // let i = 0;\n for (let participant of this.participants.values()) {\n if (participant.type != \"complex\") {\n // let pos = rotatePointAboutPoint([0, -500], [0, 0], (360 / this.participants.size * i));\n participant.setPosition(-500, -500);//pos[0], pos[1]);\n // if (participant.type === \"protein\") {\n // participant.setPositionalFeatures();\n // }\n }\n // i++;\n }\n this.updateAnnotations();\n this.checkLinks(); //totally needed, not sure why tbh todo - check this out\n this.setAllLinkCoordinates(); // just to move them off screen at first\n\n let maxSeqLength = 0;\n for (let participant of this.participants.values()) {\n if (participant.size > maxSeqLength) {\n maxSeqLength = participant.size;\n }\n }\n const width = this.svgElement.parentNode.clientWidth;\n const defaultPixPerRes = width * 0.8 / maxSeqLength;\n //console.log(\"defaultPixPerRes:\" + defaultPixPerRes);\n // https://stackoverflow.com/questions/12141150/from-list-of-integers-get-number-closest-to-a-given-value/12141511#12141511\n function takeClosest(myList, myNumber) {\n const bisect = d3__WEBPACK_IMPORTED_MODULE_2__[\"bisector\"](function (d) {\n return d;\n }).left;\n const pos = bisect(myList, myNumber);\n if (pos === 0 || pos === 1) {\n return myList[1]; // don't return smallest scale as default\n }\n if (pos === myList.length) {\n return myList[myList.length - 1];\n }\n return myList[pos - 1];\n }\n\n this.defaultBarScale = takeClosest(this.barScales, defaultPixPerRes);\n //console.log(\"default bar scale: \" + this.defaultBarScale)\n\n for (let participant of this.participants.values()) {\n if (participant.type !== \"complex\") {\n this.proteinUpper.appendChild(participant.upperGroup);\n if (participant.json.type.name === \"protein\") {\n // participant.initSelfLinkSVG(); // todo - may not even do anything, not sure its working\n participant.stickZoom = this.defaultBarScale;\n if (this.participants.size < 4) {\n participant.toStickNoTransition();\n }\n }\n }\n }\n\n this.autoLayout();\n};\n\nApp.prototype.zoomToExtent = function () {\n const width = this.svgElement.parentNode.clientWidth;\n const height = this.svgElement.parentNode.clientHeight;\n const bbox = this.container.getBBox();\n let xr = (width / bbox.width).toFixed(4) - 0;\n let yr = (height / bbox.height).toFixed(4) - 0;\n let scaleFactor;\n if (yr < xr) {\n scaleFactor = yr;\n } else {\n scaleFactor = xr;\n }\n if (scaleFactor < 1) { ///didn't fit in div\n //console.log(\"no fit\", scaleFactor);\n xr = (width - 40) / (bbox.width);\n yr = (height - 40) / (bbox.height);\n let scaleFactor;\n if (yr < xr) {\n scaleFactor = yr;\n } else {\n scaleFactor = xr;\n }\n\n if (scaleFactor > this.z) {\n scaleFactor = this.z;\n }\n\n //bbox.x + x = 0;\n let x = -bbox.x + (20 / scaleFactor);\n //box.y + y = 0\n let y = -bbox.y + (20 / scaleFactor);\n this.container.setAttribute(\"transform\", \"scale(\" + scaleFactor + \") translate(\" + x + \" \" + y + \") \");\n this.z = this.container.getCTM().inverse().a;\n } else {\n //console.log(\"fit\", scaleFactor);\n // this.container.setAttribute(\"transform\", \"scale(\" + 1 + \") translate(\" + -(width/2) + \" \" + -bbox.y + \")\");\n const deltaWidth = width - bbox.width;\n const deltaHeight = height - bbox.height;\n //bbox.x + x = deltaWidth /2;\n let x = (deltaWidth / 2) - bbox.x;\n //box.y + y = deltaHeight / 2\n let y = (deltaHeight / 2) - bbox.y;\n this.container.setAttribute(\"transform\", \"scale(\" + 1 + \") translate(\" + x + \" \" + y + \")\");\n this.z = 1;\n }\n\n //todo - following could be tided up by using acknowledgement bbox or positioning att's of text\n this.acknowledgement.setAttribute(\"transform\", \"translate(\" + (width - 150) + \", \" + (height - 30) + \")\");\n};\n\n//listeners also attached to mouse events by Interactor (and Rotator) and Link, those consume their events\n//mouse down on svgElement must be allowed to propogate (to fire event on Prots/Links)\n\nApp.prototype.mouseDown = function (evt) {\n //prevent default, but allow propogation\n evt.preventDefault();\n this.d3cola.stop();\n const p = this.getEventPoint(evt); // seems to be correct, see below\n this.dragStart = this.mouseToSVG(p.x, p.y);\n return false;\n};\n\n// dragging/rotation/panning/selecting\nApp.prototype.mouseMove = function (evt) {\n const p = this.getEventPoint(evt); // seems to be correct, see below\n const c = this.mouseToSVG(p.x, p.y);\n\n if (this.dragElement != null) { //dragging or rotating\n this.hideTooltip();\n const dx = this.dragStart.x - c.x;\n const dy = this.dragStart.y - c.y;\n\n if (this.state === this.STATES.DRAGGING) {\n // we are currently dragging things around\n let ox, oy, nx, ny;\n if (!this.dragElement.ix) {\n for (let participant of this.dragElement.participants) {\n participant.changePosition(dx, dy);\n }\n this.setAllLinkCoordinates();\n } else {\n ox = this.dragElement.ix;\n oy = this.dragElement.iy;\n nx = ox - dx;\n ny = oy - dy;\n this.dragElement.setPosition(nx, ny);\n this.dragElement.setAllLinkCoordinates();\n }\n this.dragStart = c;\n } else { //not dragging or rotating yet, maybe we should start\n // don't start dragging just on a click - we need to move the mouse a bit first\n if (Math.sqrt(dx * dx + dy * dy) > (5 * this.z)) {\n this.state = this.STATES.DRAGGING;\n\n }\n }\n } else {\n this.showTooltip(p);\n }\n return false;\n};\n\n// this ends all dragging and rotating\nApp.prototype.mouseUp = function (evt) {\n const time = new Date().getTime();\n //console.log(\"Mouse up: \" + evt.srcElement + \" \" + (time - this.lastMouseUp));\n this.preventDefaultsAndStopPropagation(evt);\n //eliminate some spurious mouse up events\n if ((time - this.lastMouseUp) > 150) {\n\n const p = this.getEventPoint(evt); // seems to be correct, see below\n const c = this.mouseToSVG(p.x, p.y);\n\n if (this.dragElement && this.dragElement.type === \"protein\") { /// todo be consistent about how to check if thing is protein\n if (!(this.state === this.STATES.DRAGGING || this.state === this.STATES.ROTATING)) { //not dragging or rotating\n if (this.dragElement.form === 0) {\n this.dragElement.setForm(1);\n } else {\n this.contextMenuProt = this.dragElement;\n this.contextMenuPoint = c;\n const menu = d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](\".custom-menu-margin\");\n menu.style(\"top\", (evt.pageY - 20) + \"px\").style(\"left\", (evt.pageX - 20) + \"px\").style(\"display\", \"block\");\n d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](\".scaleButton_\" + (this.dragElement.stickZoom * 100)).property(\"checked\", true);\n }\n }\n }\n }\n\n this.dragElement = null;\n this.state = this.STATES.MOUSE_UP;\n\n this.lastMouseUp = time;\n return false;\n};\n\n//gets mouse position\nApp.prototype.getEventPoint = function (evt) {\n const p = this.svgElement.createSVGPoint();\n let element = this.svgElement.parentNode;\n let top = 0,\n left = 0;\n do {\n top += element.offsetTop || 0;\n left += element.offsetLeft || 0;\n element = element.offsetParent;\n } while (element);\n p.x = evt.pageX - left;\n p.y = evt.pageY - top;\n return p;\n};\n\n//stop event propogation and defaults; only do what we ask\nApp.prototype.preventDefaultsAndStopPropagation = function (evt) {\n if (evt.stopPropagation)\n evt.stopPropagation();\n if (evt.cancelBubble != null)\n evt.cancelBubble = true;\n if (evt.preventDefault)\n evt.preventDefault();\n};\n\n/**\n * Handle touchstart event.\n\n App.prototype.touchStart = function(evt) {\n //prevent default, but allow propogation\n evt.preventDefault();\n\n //stop force layout\n if (typeof this.d3cola !== 'undefined' && this.d3cola != null) {\n this.d3cola.stop();\n }\n\n var p = this.getTouchEventPoint(evt); // seems to be correct, see below\n this.dragStart = this.mouseToSVG(p.x, p.y);\n};\n\n // dragging/rotation/panning/selecting\n App.prototype.touchMove = function(evt) {\n // if (this.sequenceInitComplete) { // just being cautious\n var p = this.getTouchEventPoint(evt); // seems to be correct, see below\n var c = this.mouseToSVG(p.x, p.y);\n\n if (this.dragElement != null) { //dragging or rotating\n this.hideTooltip();\n var dx = this.dragStart.x - c.x;\n var dy = this.dragStart.y - c.y;\n\n if (this.state === this.STATES.DRAGGING) {\n // we are currently dragging things around\n var ox, oy, nx, ny;\n if (typeof this.dragElement.ix=== 'undefined') { // if not an Interactor\n var nodes = this.dragElement.interactors;\n var nodeCount = nodes.length;\n for (var i = 0; i < nodeCount; i++) {\n var protein = nodes[i];\n ox = protein.cx;\n oy = protein.cy;\n nx = ox - dx;\n ny = oy - dy;\n protein.setPosition(nx, ny);\n protein.setAllLinkCoordinates();\n }\n for (i = 0; i < nodeCount; i++) {\n nodes[i].setAllLinkCoordinates();\n }\n } else {\n ox = this.dragElement.cx;\n oy = this.dragElement.cy;\n nx = ox - dx;\n ny = oy - dy;\n this.dragElement.setPosition(nx, ny);\n this.dragElement.setAllLinkCoordinates();\n }\n this.dragStart = c;\n } else { //not dragging or rotating yet, maybe we should start\n // don't start dragging just on a click - we need to move the mouse a bit first\n if (Math.sqrt(dx * dx + dy * dy) > (5 * this.z)) {\n this.state = this.STATES.DRAGGING;\n\n }\n }\n } else {\n this.showTooltip(p);\n }\n return false;\n};\n\n// this ends all dragging and rotating\nApp.prototype.touchEnd = function(evt) {\n var time = new Date().getTime();\n //console.log(\"Mouse up: \" + evt.srcElement + \" \" + (time - this.lastMouseUp));\n this.preventDefaultsAndStopPropagation(evt);\n //eliminate some spurious mouse up events\n if ((time - this.lastMouseUp) > 150) {\n\n var p = this.getTouchEventPoint(evt); // seems to be correct, see below\n var c = this.mouseToSVG(p.x, p.y);\n\n if (this.dragElement != null) {\n if (!(this.state === this.STATES.DRAGGING || this.state === this.STATES.ROTATING)) { //not dragging or rotating\n if (this.dragElement.form === 0) {\n this.dragElement.setForm(1);\n } else {\n this.contextMenuProt = this.dragElement;\n this.contextMenuPoint = c;\n var menu = d3.select(\".custom-menu-margin\")\n menu.style(\"top\", (evt.pageY - 20) + \"px\").style(\"left\", (evt.pageX - 20) + \"px\").style(\"display\", \"block\");\n d3.select(\".scaleButton_\" + (this.dragElement.stickZoom * 100)).property(\"checked\", true)\n }\n }\n }\n }\n\n this.dragElement = null;\n this.whichRotator = -1;\n this.state = this.STATES.MOUSE_UP;\n\n this.lastMouseUp = time;\n return false;\n};\n\n//gets mouse position\nApp.prototype.getTouchEventPoint = function(evt) {\n var p = this.svgElement.createSVGPoint();\n var element = this.svgElement.parentNode;\n var top = 0,\n left = 0;\n do {\n top += element.offsetTop || 0;\n left += element.offsetLeft || 0;\n element = element.offsetParent;\n } while (element);\n p.x = evt.touches[0].pageX - left;\n p.y = evt.touches[0].pageY - top;\n return p;\n};\n */\nApp.prototype.autoLayout = function () {\n this.d3cola.stop();\n const self = this;\n\n // needed to ensure consistent results\n for (let p of self.participants.values()) {\n delete p.x;\n delete p.y;\n delete p.px;\n delete p.py;\n delete p.bounds;\n p.fixed = 0;\n }\n\n //// prune leaves from network then layout, then add back leaves and layout again (fixes haemoglobin)\n const pruned = [];\n for (let participant of self.participants.values()) {\n if (participant.binaryLinks.size > 2 && participant.type !== \"complex\") {\n pruned.push(participant);\n }\n }\n const allNodesExceptComplexes = Array.from(self.participants.values()).filter(function (value) {\n return value.type !== \"complex\";\n });\n\n if (pruned.length < allNodesExceptComplexes.length\n && pruned.length > 3 && self.participants.size < 9) {\n // <9 include hemoglobin, possibly some other small cases, but is catious, tends to mess other things up\n // console.log(prunedIn);\n doLayout(pruned, true);\n } else {\n doLayout(allNodesExceptComplexes, self.complexes.length > 0);\n }\n\n function doLayout(nodes, preRun) {\n const layoutObj = {}; // todo get rid\n layoutObj.nodes = nodes;\n layoutObj.links = [];\n\n const molLookUp = {};\n let mi = 0;\n for (let mol of nodes) {\n molLookUp[mol.id] = mi;\n mi++;\n }\n\n for (let binaryLink of self.allBinaryLinks.values()) {\n const fromMol = binaryLink.participants[0];\n const toMol = binaryLink.participants[1];\n // if (preRun || (fromMol.binaryLinks.size === 4 || toMol.binaryLinks.size == 4)) {\n const source = fromMol; //molLookUp[fromMol.id];\n const target = toMol; //molLookUp[toMol.id];\n\n if (source !== target && nodes.indexOf(source) !== -1 && nodes.indexOf(target) !== -1) { // todo - check what this is doing\n const linkObj = {};\n linkObj.source = molLookUp[fromMol.id];\n linkObj.target = molLookUp[toMol.id];\n linkObj.id = binaryLink.id;\n layoutObj.links.push(linkObj);\n }\n // }\n }\n\n const groups = [];\n if (!preRun && self.complexes) {\n for (let g of self.complexes) {\n g.leaves = [];\n g.groups = [];\n for (let interactor of g.naryLink.participants) {\n if (interactor.type !== \"complex\") {\n g.leaves.push(layoutObj.nodes.indexOf(interactor));\n }\n }\n groups.push(g);\n }\n for (let g of self.complexes) {\n for (let interactor of g.naryLink.participants) {\n if (interactor.type === \"complex\") {\n //console.log(groups.indexOf(interactor));\n g.groups.push(groups.indexOf(interactor));\n }\n }\n }\n }\n\n // if (preRun) {\n // layoutObj.nodes = layoutObj.nodes.concat(prunedOut);\n // }\n\n // self.d3cola.convergenceThreshold = 0.01;\n //console.log(\"groups\", groups);\n delete self.d3cola._lastStress;\n delete self.d3cola._alpha;\n delete self.d3cola._descent;\n delete self.d3cola._rootGroup;\n\n let linkLength = (nodes.length < 30) ? 30 : 20;\n const width = self.svgElement.parentNode.clientWidth;\n const height = self.svgElement.parentNode.clientHeight;\n //console.log(\"**\", layoutObj);\n self.d3cola.size([height - 40, width - 40])\n .nodes(layoutObj.nodes).groups(groups).links(layoutObj.links).avoidOverlaps(true);\n let groupDebugSel, participantDebugSel;\n if (self.debug) {\n groupDebugSel = d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](self.container).selectAll(\".group\")\n .data(groups);\n\n groupDebugSel.enter().append(\"rect\")\n .classed(\"group\", true)\n .attr({\n rx: 5,\n ry: 5\n })\n .style(\"stroke\", \"blue\")\n .style(\"fill\", \"none\");\n\n participantDebugSel = d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](self.container).selectAll(\".node\")\n .data(layoutObj.nodes);\n\n participantDebugSel.enter().append(\"rect\")\n .classed(\"node\", true)\n .attr({\n rx: 5,\n ry: 5\n })\n .style(\"stroke\", \"red\")\n .style(\"fill\", \"none\");\n\n groupDebugSel.exit().remove();\n participantDebugSel.exit().remove();\n }\n\n const startTime = Date.now();\n self.d3cola.symmetricDiffLinkLengths(linkLength)\n .on(\"tick\", function () {\n if (Date.now() - startTime > 400) {//!preRun) {\n const nodes = self.d3cola.nodes();\n for (let node of nodes) {\n node.setPosition(node.x, node.y);\n }\n self.setAllLinkCoordinates();\n self.zoomToExtent();\n if (self.debug) {\n groupDebugSel.attr({\n x: function (d) {\n return d.bounds.x;// + (width / 2);\n },\n y: function (d) {\n return d.bounds.y;// + (height / 2);\n },\n width: function (d) {\n return d.bounds.width();\n },\n height: function (d) {\n return d.bounds.height();\n }\n });\n\n participantDebugSel.attr({\n x: function (d) {\n return d.bounds.x;// + (width / 2);\n },\n y: function (d) {\n return d.bounds.y;// + (height / 2);\n },\n width: function (d) {\n return d.bounds.width();\n },\n height: function (d) {\n return d.bounds.height();\n }\n });\n }\n }\n })\n .on(\"end\", function () {\n if (preRun) {\n // alert(\"initial run complete\");\n // // for (let p of layoutObj.nodes) {\n // // p.fixed = 1;\n // // }\n // // let nodesExceptComplexes = Array.from(self.participants.values());\n // // allNodesExceptComplexes = allNodesExceptComplexes.filter(function (value) {\n // // return value.type !== \"complex\";\n // // });\n doLayout(allNodesExceptComplexes, false);\n } else {\n for (let node of nodes) {\n node.setPosition(node.x, node.y);\n }\n self.setAllLinkCoordinates();\n self.zoomToExtent();\n }\n });\n if (preRun) {\n self.d3cola.start(23, 23, 0, 0, true);//, false, false);\n } else {\n self.d3cola.start(0, 23, 23, 0, true);//, false, false);\n }\n }\n};\n\nApp.prototype.getSVG = function () { //todo - update after styling of svg is moved to css\n let svgXml = this.svgElement.outerHTML.replace(//i, \"\"); //take out white background fill\n const viewBox = \"viewBox=\\\"0 0 \" + this.svgElement.parentNode.clientWidth + \" \" + this.svgElement.parentNode.clientHeight + \"\\\" \";\n svgXml = svgXml.replace(\"\" +\n \"\" +\n svgXml;\n};\n\n// transform the mouse-position into a position on the svg\nApp.prototype.mouseToSVG = function (x, y) {\n const p = this.svgElement.createSVGPoint();\n p.x = x;\n p.y = y;\n return p.matrixTransform(this.container.getCTM().inverse());\n};\n\n// reads MI JSON format\nApp.prototype.readMIJSON = function (miJson, expand = true) {\n Object(_read_mijson__WEBPACK_IMPORTED_MODULE_5__[\"readMijson\"])(miJson, this, expand);\n};\n\nApp.prototype.checkLinks = function () {\n function checkAll(linkMap) {\n for (let link of linkMap.values()) {\n link.check();\n }\n }\n\n checkAll(this.allNaryLinks);\n checkAll(this.allBinaryLinks);\n checkAll(this.allUnaryLinks);\n checkAll(this.allSequenceLinks);\n};\n\nApp.prototype.setAllLinkCoordinates = function () {\n function setAll(linkMap) {\n for (let link of linkMap.values()) {\n link.setLinkCoordinates(true); // true means don't propogate changes from naryLink up to complex, everythings getting refreshed anyway\n }\n }\n\n setAll(this.allNaryLinks);\n setAll(this.allBinaryLinks);\n setAll(this.allUnaryLinks);\n setAll(this.allSequenceLinks);\n};\n\nApp.prototype.showTooltip = function (p) {\n let ttX, ttY;\n const length = this.tooltip.getComputedTextLength() + 16;\n const width = this.svgElement.parentNode.clientWidth;\n const height = this.svgElement.parentNode.clientHeight;\n if (p.x + 20 + length < width) {\n ttX = p.x;\n } else {\n ttX = width - length - 20;\n }\n\n if (p.y + 60 < height) {\n ttY = p.y;\n } else {\n ttY = height - 60;\n }\n this.tooltip.setAttribute(\"x\", ttX + 22);\n this.tooltip.setAttribute(\"y\", ttY + 47);\n this.tooltip_bg.setAttribute(\"x\", ttX + 16);\n this.tooltip_bg.setAttribute(\"y\", ttY + 28);\n this.tooltip_subBg.setAttribute(\"x\", ttX + 16);\n this.tooltip_subBg.setAttribute(\"y\", ttY + 28);\n};\n\nApp.prototype.setTooltip = function (text, color) {\n if (text) {\n this.tooltip.firstChild.data = text.toString().replace(/&(quot);/g, \"\\\"\");\n this.tooltip.setAttribute(\"display\", \"block\");\n const length = this.tooltip.getComputedTextLength();\n this.tooltip_bg.setAttribute(\"width\", length + 16);\n this.tooltip_subBg.setAttribute(\"width\", length + 16);\n if (typeof color !== \"undefined\" && color != null) {\n this.tooltip_bg.setAttribute(\"fill\", color);\n this.tooltip_bg.setAttribute(\"stroke\", color);\n this.tooltip_bg.setAttribute(\"fill-opacity\", \"0.5\");\n } else {\n this.tooltip_bg.setAttribute(\"fill\", \"white\");\n this.tooltip_bg.setAttribute(\"stroke\", \"grey\");\n }\n this.tooltip_bg.setAttribute(\"height\", \"28\");\n this.tooltip_subBg.setAttribute(\"height\", \"28\");\n this.tooltip_bg.setAttribute(\"display\", \"block\");\n this.tooltip_subBg.setAttribute(\"display\", \"block\");\n } else {\n this.hideTooltip();\n }\n};\n\nApp.prototype.hideTooltip = function () {\n this.tooltip.setAttribute(\"display\", \"none\");\n this.tooltip_bg.setAttribute(\"display\", \"none\");\n this.tooltip_subBg.setAttribute(\"display\", \"none\");\n};\n\nApp.prototype.addColorSchemeKey = function (/*HTMLDivElement*/ div) {\n this.colorSchemeKeyDivs.add(div);\n _color_scheme_key__WEBPACK_IMPORTED_MODULE_7__[\"update\"](div, this);\n};\n\nApp.prototype.removeColorSchemeKey = function (/*HTMLDivElement*/ colorSchemeKeyDiv) {\n this.colorSchemeKeyDivs.remove(colorSchemeKeyDiv);\n colorSchemeKeyDiv.textContent = \"\";\n};\n\n//for backwards compatibility (noe?), tbh i might have made a bit of a mess here\nApp.prototype.setAnnotations = function (annoChoice) {\n annoChoice = annoChoice.toUpperCase();\n for (let annoType of this.annotationSetsShown.keys()) {\n this.showAnnotations(annoType, annoChoice === annoType);\n }\n this.showAnnotations(annoChoice, true);\n};\n\nApp.prototype.showAnnotations = function (annoChoice, show) {\n annoChoice = annoChoice.toUpperCase();\n const self = this;\n let setShown = this.annotationSetsShown.get(annoChoice);\n if (typeof setShown === \"undefined\" && annoChoice !== \"MIFEATURES\") {\n Object(_annotations__WEBPACK_IMPORTED_MODULE_6__[\"fetchAnnotations\"])(annoChoice, this, function () {\n self.annotationSetsShown.set(annoChoice, show);\n self.updateAnnotations();\n });\n } else {\n this.annotationSetsShown.set(annoChoice, show);\n this.updateAnnotations();\n }\n};\n\nApp.prototype.updateAnnotations = function () {\n // //clear all annot's\n for (let mol of this.participants.values()) {\n if (mol.id.indexOf(\"uniprotkb_\") === 0) { //LIMIT IT TO PROTEINS\n mol.clearPositionalFeatures();\n }\n }\n Object(_annotations__WEBPACK_IMPORTED_MODULE_6__[\"chooseColors\"])(this);\n this.colorSchemeChanged();\n\n for (let mol of this.participants.values()) {\n if (mol.id.indexOf(\"uniprotkb_\") === 0) { //LIMIT IT TO PROTEINS\n mol.setPositionalFeatures();\n }\n }\n Object(_annotations__WEBPACK_IMPORTED_MODULE_6__[\"chooseColors\"])(this);\n this.colorSchemeChanged();\n};\n\nApp.prototype.colorSchemeChanged = function () {\n for (let div of this.colorSchemeKeyDivs) {\n _color_scheme_key__WEBPACK_IMPORTED_MODULE_7__[\"update\"](div, this);\n }\n};\n\nApp.prototype.getComplexColors = function () {\n return _viz_link_nary_link__WEBPACK_IMPORTED_MODULE_8__[\"NaryLink\"].naryColors;\n};\n\nApp.prototype.getFeatureColors = function () {\n return this.featureColors;\n};\n\nApp.prototype.collapseAll = function () {\n for (let participant of this.participants.values()) {\n if (participant.form === 1) {\n participant.setForm(0);\n }\n }\n};\n\nApp.prototype.expandAll = function () {\n for (let participant of this.participants.values()) {\n if (participant.form === 0) {\n participant.setForm(1);\n }\n }\n};\n\n//from noe\nApp.prototype.expandAndCollapseSelection = function (moleculesSelected) {\n const molecules = this.molecules.values();\n for (var m = 0; m < molecules.length; m++) {\n var molecule = molecules[m];\n var molecule_id = molecule.json.identifier.id;\n if (moleculesSelected.includes(molecule_id)) {\n if (molecule.form === 0) {\n molecule.setForm(1);\n }\n } else if (molecule.form === 1) {\n molecule.setForm(0);\n }\n }\n};\n\n\n// export function makeSymbolKey(targetDiv){\n// new SymbolKey(targetDiv);\n// }\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/js/app.js.js","sources":["webpack://complexviewer/./src/js/app.js?90e9"],"sourcesContent":["// eslint-disable-next-line no-unused-vars\nimport * as css from \"../css/xinet.css\";\nimport {version} from \"../../package.json\";\nimport * as d3 from \"d3\";\nimport * as d3_chromatic from \"d3-scale-chromatic\";\nimport * as cola from \"./cola\";\nimport {readMijson} from \"./read-mijson\";\nimport {chooseColors, fetchAnnotations} from \"./annotations\";\n\n// import {SymbolKey} from \"./symbol-key\";\nimport * as ColorSchemeKey from \"./color-scheme-key\";\nimport {NaryLink} from \"./viz/link/nary-link\";\nimport {svgns} from \"./config\";\n\n// could refactor everything to use ES6 class syntax\n// but https://benmccormick.org/2015/04/07/es6-classes-and-backbone-js\n// \"ES6 classes don’t support adding properties directly to the class instance, only functions/methods\"\n// so backbone doesn't work\n// so continuing to use prototypical inheritance in things for time being\n\nexport function App(/*HTMLDivElement*/networkDiv) {\n    // this.debug = true; // things aren't exactly lined up in the bounding boxes cola is using, to do so breaks symetery of some things\n    this.el = networkDiv;\n\n    this.STATES = {};\n    this.STATES.MOUSE_UP = 0; //start state, also set when mouse up on svgElement\n    this.STATES.PANNING = 1; //set by mouse down on svgElement - left button, no shift or util\n    this.STATES.DRAGGING = 2; //set by mouse down on Protein or Link\n    this.STATES.ROTATING = 3; //set by mouse down on Rotator, drag?\n    this.STATES.SELECTING = 4; //set by mouse down on svgElement- right button or left button shift or util, drag\n\n    //avoids prob with 'save - web page complete'\n    this.el.textContent = \"\"; //https://stackoverflow.com/questions/3955229/remove-all-child-elements-of-a-dom-node-in-javascript\n\n    this.d3cola = cola.d3adaptor();\n\n    const customMenuSel = d3.select(this.el)\n        .append(\"div\").classed(\"custom-menu-margin\", true)\n        .append(\"div\").classed(\"custom-menu\", true)\n        .append(\"ul\");\n\n    const self = this;\n    const collapse = customMenuSel.append(\"li\").classed(\"collapse\", true); //.append(\"button\");\n    collapse.text(\"Collapse\");\n    collapse[0][0].onclick = function (evt) {\n        self.collapseProtein(evt);\n    };\n    const scaleButtonsListItemSel = customMenuSel.append(\"li\").text(\"Scale: \");\n\n    this.barScales = [0.01, 0.2, 1, 2, 4, 8];\n    const scaleButtons = scaleButtonsListItemSel.selectAll(\"ul.custom-menu\")\n        .data(this.barScales)\n        .enter()\n        .append(\"div\")\n        .attr(\"class\", \"barScale\")\n        .append(\"label\");\n    scaleButtons.append(\"span\")\n        .text(function (d) {\n            if (d === 8) return \"AA\";\n            else return d;\n        });\n    scaleButtons.append(\"input\")\n        // .attr (\"id\", function(d) { return d*100; })\n        .attr(\"class\", function (d) {\n            return \"scaleButton scaleButton_\" + (d * 100);\n        })\n        .attr(\"name\", \"scaleButtons\")\n        .attr(\"type\", \"radio\")\n        .on(\"change\", function (d) {\n            self.preventDefaultsAndStopPropagation(d);\n            self.contextMenuProt.setStickScale(d, self.contextMenuPoint);\n        });\n\n    const contextMenu = d3.select(\".custom-menu-margin\").node();\n    contextMenu.onmouseout = function (evt) {\n        let e = evt.relatedTarget;\n        do {\n            if (e === this) return;\n            e = e.parentNode;\n        } while (e);\n        self.contextMenuProt = null;\n        d3.select(this).style(\"display\", \"none\");\n    };\n\n\n    //create SVG element\n    this.svgElement = document.createElementNS(svgns, \"svg\");\n    this.svgElement.setAttribute(\"id\", \"complexViewerSVG\");\n\n    //add listeners\n    this.svgElement.onmousedown = function (evt) {\n        self.mouseDown(evt);\n    };\n    this.svgElement.onmousemove = function (evt) {\n        self.mouseMove(evt);\n    };\n    this.svgElement.onmouseup = function (evt) {\n        self.mouseUp(evt);\n    };\n    this.svgElement.onmouseout = function (evt) {\n        self.hideTooltip(evt);\n    };\n    this.lastMouseUp = new Date().getTime();\n    /*this.svgElement.ontouchstart = function(evt) {\n        self.touchStart(evt);\n    };\n    this.svgElement.ontouchmove = function(evt) {\n        self.touchMove(evt);\n    };\n    this.svgElement.ontouchend = function(evt) {\n        self.touchEnd(evt);\n    };\n    */\n\n    this.el.oncontextmenu = function (evt) {\n        if (evt.preventDefault) { // necessary for addEventListener, works with traditional\n            evt.preventDefault();\n        }\n        evt.returnValue = false; // necessary for attachEvent, works with traditional\n        return false; // works with traditional, not with attachEvent or addEventListener\n    };\n\n    //legend changed callbacks\n    this.colorSchemeKeyDivs = new Set();\n\n    this.el.appendChild(this.svgElement);\n\n    // various groups needed\n    this.container = document.createElementNS(svgns, \"g\");\n    this.container.setAttribute(\"id\", \"container\");\n\n    const svg = d3.select(this.svgElement);\n    this.defs = svg.append(\"defs\");\n    this.createHatchedFill(\"checkers_uncertain\", \"black\");\n\n    //markers\n    const data = [{\n        id: 1,\n        name: \"diamond\",\n        path: \"M 0,-7.0710768 L  0,7.0710589 L 7.0710462,0  z\",\n        viewbox: \"-15 -15 25 25\",\n        transform: \"scale(1.5) translate(-5,0)\",\n        color: \"black\"\n    }];\n\n    this.defs.selectAll(\"marker\")\n        .data(data)\n        .enter()\n        .append(\"svg:marker\")\n        .attr(\"id\", function (d) {\n            return \"marker_\" + d.name;\n        })\n        .attr(\"markerHeight\", 15)\n        .attr(\"markerWidth\", 15)\n        .attr(\"markerUnits\", \"userSpaceOnUse\")\n        .attr(\"orient\", \"auto\")\n        .attr(\"refX\", 0)\n        .attr(\"refY\", 0)\n        .attr(\"viewBox\", function (d) {\n            return d.viewbox;\n        })\n        .append(\"svg:path\")\n        .attr(\"d\", function (d) {\n            return d.path;\n        })\n        .attr(\"fill\", function (d) {\n            return d.color;\n        })\n        .attr(\"transform\", function (d) {\n            return d.transform;\n        });\n\n    this.acknowledgement = document.createElementNS(svgns, \"g\");\n    const ackText = document.createElementNS(svgns, \"text\");\n    ackText.innerHTML = \"<a href='https://academic.oup.com/bioinformatics/article/33/22/3673/4061280' target='_blank'><tspan x='0' dy='1.2em' style='text-decoration: underline'>ComplexViewer \"\n        + version + \"</tspan></a><tspan x='0' dy='1.2em'>by <a href='http://rappsilberlab.org/' target='_blank'>Rappsilber Laboratory</a></tspan>\";\n\n    this.acknowledgement.appendChild(ackText);\n    ackText.classList.add(\"xlv_text\");\n    ackText.setAttribute(\"font-size\", \"8pt\");\n    this.svgElement.appendChild(this.acknowledgement);\n\n    this.naryLinks = document.createElementNS(svgns, \"g\");\n    this.naryLinks.setAttribute(\"id\", \"naryLinks\");\n    this.container.appendChild(this.naryLinks);\n\n    this.p_pLinksWide = document.createElementNS(svgns, \"g\");\n    this.p_pLinksWide.setAttribute(\"id\", \"p_pLinksWide\");\n    this.container.appendChild(this.p_pLinksWide);\n\n    this.highlights = document.createElementNS(svgns, \"g\");\n    this.highlights.setAttribute(\"class\", \"highlights\"); //interactors also contain highlight groups\n    this.container.appendChild(this.highlights);\n\n    this.res_resLinks = document.createElementNS(svgns, \"g\");\n    this.res_resLinks.setAttribute(\"id\", \"res_resLinks\");\n    this.container.appendChild(this.res_resLinks);\n\n    this.p_pLinks = document.createElementNS(svgns, \"g\");\n    this.p_pLinks.setAttribute(\"id\", \"p_pLinks\");\n    this.container.appendChild(this.p_pLinks);\n\n    this.proteinUpper = document.createElementNS(svgns, \"g\");\n    this.proteinUpper.setAttribute(\"id\", \"proteinUpper\");\n    this.container.appendChild(this.proteinUpper);\n\n    this.selfRes_resLinks = document.createElementNS(svgns, \"g\");\n    this.selfRes_resLinks.setAttribute(\"id\", \"res_resLinks\");\n    this.container.appendChild(this.selfRes_resLinks);\n\n    this.svgElement.appendChild(this.container);\n\n    //showing title as tooltips is NOT part of svg spec (even though some browsers do this)\n    //also more responsive / more control if we do out own\n    this.tooltip = document.createElementNS(svgns, \"text\");\n    this.tooltip.setAttribute(\"x\", \"0\");\n    this.tooltip.setAttribute(\"y\", \"0\");\n    this.tooltip.setAttribute(\"class\", \"xlv_text\");\n    const tooltipTextNode = document.createTextNode(\"tooltip\");\n\n    this.tooltip.appendChild(tooltipTextNode);\n\n    this.tooltip_bg = document.createElementNS(svgns, \"rect\");\n    this.tooltip_bg.setAttribute(\"class\", \"tooltip_bg\");\n\n    this.tooltip_bg.setAttribute(\"fill-opacity\", \"0.75\");\n    this.tooltip_bg.setAttribute(\"stroke-opacity\", \"1\");\n    this.tooltip_bg.setAttribute(\"stroke-width\", \"1\");\n\n    this.tooltip_subBg = document.createElementNS(svgns, \"rect\");\n    this.tooltip_subBg.setAttribute(\"fill\", \"white\");\n    this.tooltip_subBg.setAttribute(\"stroke\", \"white\");\n    this.tooltip_subBg.setAttribute(\"class\", \"tooltip_bg\");\n    this.tooltip_subBg.setAttribute(\"opacity\", \"1\");\n    this.tooltip_subBg.setAttribute(\"stroke-width\", \"1\");\n\n    this.svgElement.appendChild(this.tooltip_subBg);\n    this.svgElement.appendChild(this.tooltip_bg);\n    this.svgElement.appendChild(this.tooltip);\n\n    this.clear();\n}\n\nApp.prototype.createHatchedFill = function (name, color) {\n    const pattern = this.defs.append(\"pattern\")\n        .attr(\"id\", name)\n        .attr(\"patternUnits\", \"userSpaceOnUse\")\n        .attr(\"x\", 0)\n        .attr(\"y\", 0)\n        .attr(\"width\", 12)\n        .attr(\"height\", 12)\n        .attr(\"patternTransform\", \"rotate(45)\");\n\n    pattern.append(\"rect\")\n        .attr(\"x\", 0)\n        .attr(\"y\", 2)\n        .attr(\"width\", 12)\n        .attr(\"height\", 4)\n        .attr(\"fill\", color);\n\n    pattern.append(\"rect\")\n        .attr(\"x\", 0)\n        .attr(\"y\", 8)\n        .attr(\"width\", 12)\n        .attr(\"height\", 4)\n        .attr(\"fill\", color);\n};\n\nApp.prototype.clear = function () {\n    this.d3cola.stop();\n\n    this.annotationSetsShown = new Map();\n    // this.annotationSetsShown.set(\"MI FEATURES\", true);\n\n    //lighten colors\n    const complexColors = [];\n    for (let c of d3_chromatic.schemePastel2) {//colorbrewer.Pastel2[8]) {\n        const hsl = d3.hsl(c);\n        hsl.l = 0.9;\n        complexColors.push(hsl + \"\");\n    }\n\n    NaryLink.naryColors = d3.scale.ordinal().range(complexColors);\n    this.defs.selectAll(\".feature_checkers\").remove();\n\n    this.naryLinks.textContent = \"\";\n    this.p_pLinksWide.textContent = \"\";\n    this.highlights.textContent = \"\";\n    this.p_pLinks.textContent = \"\";\n    this.res_resLinks.textContent = \"\";\n    this.proteinUpper.textContent = \"\";\n    this.selfRes_resLinks.textContent = \"\";\n\n    // if we are dragging something at the moment - this will be the element that is dragged\n    this.dragElement = null;\n    // from where did we start dragging\n    this.dragStart = {};\n\n    this.participants = new Map();\n    this.allNaryLinks = new Map();\n    this.allBinaryLinks = new Map();\n    this.allUnaryLinks = new Map();\n    this.allSequenceLinks = new Map();\n    this.complexes = [];\n\n    this.proteinCount = 0;\n    this.z = 1;\n    this.hideTooltip();\n    this.state = this.STATES.MOUSE_UP;\n};\n\nApp.prototype.collapseProtein = function () {\n    d3.select(\".custom-menu-margin\").style(\"display\", \"none\");\n    this.contextMenuProt.setForm(0, this.contextMenuPoint);\n    this.contextMenuProt = null;\n};\n\nApp.prototype.init = function () {\n    this.d3cola.stop();\n\n    // let i = 0;\n    for (let participant of this.participants.values()) {\n        if (participant.type != \"complex\") {\n            // let pos = rotatePointAboutPoint([0, -500], [0, 0], (360 / this.participants.size * i));\n            participant.setPosition(-500, -500);//pos[0], pos[1]);\n            // if (participant.type === \"protein\") {\n            //     participant.setPositionalFeatures();\n            // }\n        }\n        // i++;\n    }\n    this.updateAnnotations();\n    this.checkLinks(); //totally needed, not sure why tbh todo - check this out\n    this.setAllLinkCoordinates(); // just to move them off screen at first\n\n    let maxSeqLength = 0;\n    for (let participant of this.participants.values()) {\n        if (participant.size > maxSeqLength) {\n            maxSeqLength = participant.size;\n        }\n    }\n    const width = this.svgElement.parentNode.clientWidth;\n    const defaultPixPerRes = width * 0.8 / maxSeqLength;\n    //console.log(\"defaultPixPerRes:\" + defaultPixPerRes);\n    // https://stackoverflow.com/questions/12141150/from-list-of-integers-get-number-closest-to-a-given-value/12141511#12141511\n    function takeClosest(myList, myNumber) {\n        const bisect = d3.bisector(function (d) {\n            return d;\n        }).left;\n        const pos = bisect(myList, myNumber);\n        if (pos === 0 || pos === 1) {\n            return myList[1]; // don't return smallest scale as default\n        }\n        if (pos === myList.length) {\n            return myList[myList.length - 1];\n        }\n        return myList[pos - 1];\n    }\n\n    this.defaultBarScale = takeClosest(this.barScales, defaultPixPerRes);\n    //console.log(\"default bar scale: \" + this.defaultBarScale)\n\n    for (let participant of this.participants.values()) {\n        if (participant.type !== \"complex\") {\n            this.proteinUpper.appendChild(participant.upperGroup);\n            if (participant.json.type.name === \"protein\") {\n                // participant.initSelfLinkSVG(); // todo - may not even do anything, not sure its working\n                participant.stickZoom = this.defaultBarScale;\n                if (this.participants.size < 4) {\n                    participant.toStickNoTransition();\n                }\n            }\n        }\n    }\n\n    this.autoLayout();\n};\n\nApp.prototype.zoomToExtent = function () {\n    const width = this.svgElement.parentNode.clientWidth;\n    const height = this.svgElement.parentNode.clientHeight;\n    const bbox = this.container.getBBox();\n    let xr = (width / bbox.width).toFixed(4) - 0;\n    let yr = (height / bbox.height).toFixed(4) - 0;\n    let scaleFactor;\n    if (yr < xr) {\n        scaleFactor = yr;\n    } else {\n        scaleFactor = xr;\n    }\n    if (scaleFactor < 1) { ///didn't fit in div\n        //console.log(\"no fit\", scaleFactor);\n        xr = (width - 40) / (bbox.width);\n        yr = (height - 40) / (bbox.height);\n        let scaleFactor;\n        if (yr < xr) {\n            scaleFactor = yr;\n        } else {\n            scaleFactor = xr;\n        }\n\n        if (scaleFactor > this.z) {\n            scaleFactor = this.z;\n        }\n\n        //bbox.x + x = 0;\n        let x = -bbox.x + (20 / scaleFactor);\n        //box.y + y = 0\n        let y = -bbox.y + (20 / scaleFactor);\n        this.container.setAttribute(\"transform\", \"scale(\" + scaleFactor + \") translate(\" + x + \" \" + y + \") \");\n        this.z = this.container.getCTM().inverse().a;\n    } else {\n        //console.log(\"fit\", scaleFactor);\n        // this.container.setAttribute(\"transform\", \"scale(\" + 1 + \") translate(\" + -(width/2) + \" \" + -bbox.y + \")\");\n        const deltaWidth = width - bbox.width;\n        const deltaHeight = height - bbox.height;\n        //bbox.x + x = deltaWidth /2;\n        let x = (deltaWidth / 2) - bbox.x;\n        //box.y + y = deltaHeight / 2\n        let y = (deltaHeight / 2) - bbox.y;\n        this.container.setAttribute(\"transform\", \"scale(\" + 1 + \") translate(\" + x + \" \" + y + \")\");\n        this.z = 1;\n    }\n\n    //todo - following could be tided up by using acknowledgement bbox or positioning att's of text\n    this.acknowledgement.setAttribute(\"transform\", \"translate(\" + (width - 150) + \", \" + (height - 30) + \")\");\n};\n\n//listeners also attached to mouse events by Interactor (and Rotator) and Link, those consume their events\n//mouse down on svgElement must be allowed to propogate (to fire event on Prots/Links)\n\nApp.prototype.mouseDown = function (evt) {\n    //prevent default, but allow propogation\n    evt.preventDefault();\n    this.d3cola.stop();\n    const p = this.getEventPoint(evt); // seems to be correct, see below\n    this.dragStart = this.mouseToSVG(p.x, p.y);\n    return false;\n};\n\n// dragging/rotation/panning/selecting\nApp.prototype.mouseMove = function (evt) {\n    const p = this.getEventPoint(evt); // seems to be correct, see below\n    const c = this.mouseToSVG(p.x, p.y);\n\n    if (this.dragElement != null) { //dragging or rotating\n        this.hideTooltip();\n        const dx = this.dragStart.x - c.x;\n        const dy = this.dragStart.y - c.y;\n\n        if (this.state === this.STATES.DRAGGING) {\n            // we are currently dragging things around\n            let ox, oy, nx, ny;\n            if (!this.dragElement.ix) {\n                for (let participant of this.dragElement.participants) {\n                    participant.changePosition(dx, dy);\n                }\n                this.setAllLinkCoordinates();\n            } else {\n                ox = this.dragElement.ix;\n                oy = this.dragElement.iy;\n                nx = ox - dx;\n                ny = oy - dy;\n                this.dragElement.setPosition(nx, ny);\n                this.dragElement.setAllLinkCoordinates();\n            }\n            this.dragStart = c;\n        } else { //not dragging or rotating yet, maybe we should start\n            // don't start dragging just on a click - we need to move the mouse a bit first\n            if (Math.sqrt(dx * dx + dy * dy) > (5 * this.z)) {\n                this.state = this.STATES.DRAGGING;\n\n            }\n        }\n    } else {\n        this.showTooltip(p);\n    }\n    return false;\n};\n\n// this ends all dragging and rotating\nApp.prototype.mouseUp = function (evt) {\n    const time = new Date().getTime();\n    //console.log(\"Mouse up: \" + evt.srcElement + \" \" + (time - this.lastMouseUp));\n    this.preventDefaultsAndStopPropagation(evt);\n    //eliminate some spurious mouse up events\n    if ((time - this.lastMouseUp) > 150) {\n\n        const p = this.getEventPoint(evt); // seems to be correct, see below\n        const c = this.mouseToSVG(p.x, p.y);\n\n        if (this.dragElement && this.dragElement.type === \"protein\") { /// todo be consistent about how to check if thing is protein\n            if (!(this.state === this.STATES.DRAGGING || this.state === this.STATES.ROTATING)) { //not dragging or rotating\n                if (this.dragElement.form === 0) {\n                    this.dragElement.setForm(1);\n                } else {\n                    this.contextMenuProt = this.dragElement;\n                    this.contextMenuPoint = c;\n                    const menu = d3.select(\".custom-menu-margin\");\n                    menu.style(\"top\", (evt.pageY - 20) + \"px\").style(\"left\", (evt.pageX - 20) + \"px\").style(\"display\", \"block\");\n                    d3.select(\".scaleButton_\" + (this.dragElement.stickZoom * 100)).property(\"checked\", true);\n                }\n            }\n        }\n    }\n\n    this.dragElement = null;\n    this.state = this.STATES.MOUSE_UP;\n\n    this.lastMouseUp = time;\n    return false;\n};\n\n//gets mouse position\nApp.prototype.getEventPoint = function (evt) {\n    const p = this.svgElement.createSVGPoint();\n    let element = this.svgElement.parentNode;\n    let top = 0,\n        left = 0;\n    do {\n        top += element.offsetTop || 0;\n        left += element.offsetLeft || 0;\n        element = element.offsetParent;\n    } while (element);\n    p.x = evt.pageX - left;\n    p.y = evt.pageY - top;\n    return p;\n};\n\n//stop event propogation and defaults; only do what we ask\nApp.prototype.preventDefaultsAndStopPropagation = function (evt) {\n    if (evt.stopPropagation)\n        evt.stopPropagation();\n    if (evt.cancelBubble != null)\n        evt.cancelBubble = true;\n    if (evt.preventDefault)\n        evt.preventDefault();\n};\n\n/**\n * Handle touchstart event.\n\n App.prototype.touchStart = function(evt) {\n    //prevent default, but allow propogation\n    evt.preventDefault();\n\n    //stop force layout\n    if (typeof this.d3cola !== 'undefined' && this.d3cola != null) {\n        this.d3cola.stop();\n    }\n\n    var p = this.getTouchEventPoint(evt); // seems to be correct, see below\n    this.dragStart = this.mouseToSVG(p.x, p.y);\n};\n\n // dragging/rotation/panning/selecting\n App.prototype.touchMove = function(evt) {\n    // if (this.sequenceInitComplete) { // just being cautious\n    var p = this.getTouchEventPoint(evt); // seems to be correct, see below\n    var c = this.mouseToSVG(p.x, p.y);\n\n    if (this.dragElement != null) { //dragging or rotating\n        this.hideTooltip();\n        var dx = this.dragStart.x - c.x;\n        var dy = this.dragStart.y - c.y;\n\n        if (this.state === this.STATES.DRAGGING) {\n            // we are currently dragging things around\n            var ox, oy, nx, ny;\n            if (typeof this.dragElement.ix=== 'undefined') { // if not an Interactor\n                var nodes = this.dragElement.interactors;\n                var nodeCount = nodes.length;\n                for (var i = 0; i < nodeCount; i++) {\n                    var protein = nodes[i];\n                    ox = protein.cx;\n                    oy = protein.cy;\n                    nx = ox - dx;\n                    ny = oy - dy;\n                    protein.setPosition(nx, ny);\n                    protein.setAllLinkCoordinates();\n                }\n                for (i = 0; i < nodeCount; i++) {\n                    nodes[i].setAllLinkCoordinates();\n                }\n            } else {\n                ox = this.dragElement.cx;\n                oy = this.dragElement.cy;\n                nx = ox - dx;\n                ny = oy - dy;\n                this.dragElement.setPosition(nx, ny);\n                this.dragElement.setAllLinkCoordinates();\n            }\n            this.dragStart = c;\n        } else { //not dragging or rotating yet, maybe we should start\n            // don't start dragging just on a click - we need to move the mouse a bit first\n            if (Math.sqrt(dx * dx + dy * dy) > (5 * this.z)) {\n                this.state = this.STATES.DRAGGING;\n\n            }\n        }\n    } else {\n        this.showTooltip(p);\n    }\n    return false;\n};\n\n// this ends all dragging and rotating\nApp.prototype.touchEnd = function(evt) {\n    var time = new Date().getTime();\n    //console.log(\"Mouse up: \" + evt.srcElement + \" \" + (time - this.lastMouseUp));\n    this.preventDefaultsAndStopPropagation(evt);\n    //eliminate some spurious mouse up events\n    if ((time - this.lastMouseUp) > 150) {\n\n        var p = this.getTouchEventPoint(evt); // seems to be correct, see below\n        var c = this.mouseToSVG(p.x, p.y);\n\n        if (this.dragElement != null) {\n            if (!(this.state === this.STATES.DRAGGING || this.state === this.STATES.ROTATING)) { //not dragging or rotating\n                if (this.dragElement.form === 0) {\n                    this.dragElement.setForm(1);\n                } else {\n                    this.contextMenuProt = this.dragElement;\n                    this.contextMenuPoint = c;\n                    var menu = d3.select(\".custom-menu-margin\")\n                    menu.style(\"top\", (evt.pageY - 20) + \"px\").style(\"left\", (evt.pageX - 20) + \"px\").style(\"display\", \"block\");\n                    d3.select(\".scaleButton_\" + (this.dragElement.stickZoom * 100)).property(\"checked\", true)\n                }\n            }\n        }\n    }\n\n    this.dragElement = null;\n    this.whichRotator = -1;\n    this.state = this.STATES.MOUSE_UP;\n\n    this.lastMouseUp = time;\n    return false;\n};\n\n//gets mouse position\nApp.prototype.getTouchEventPoint = function(evt) {\n    var p = this.svgElement.createSVGPoint();\n    var element = this.svgElement.parentNode;\n    var top = 0,\n        left = 0;\n    do {\n        top += element.offsetTop || 0;\n        left += element.offsetLeft || 0;\n        element = element.offsetParent;\n    } while (element);\n    p.x = evt.touches[0].pageX - left;\n    p.y = evt.touches[0].pageY - top;\n    return p;\n};\n */\nApp.prototype.autoLayout = function () {\n    this.d3cola.stop();\n    const self = this;\n\n    // needed to ensure consistent results\n    for (let p of self.participants.values()) {\n        delete p.x;\n        delete p.y;\n        delete p.px;\n        delete p.py;\n        delete p.bounds;\n        p.fixed = 0;\n    }\n\n    //// prune leaves from network then layout, then add back leaves and layout again (fixes haemoglobin)\n    const pruned = [];\n    for (let participant of self.participants.values()) {\n        if (participant.binaryLinks.size > 2 && participant.type !== \"complex\") {\n            pruned.push(participant);\n        }\n    }\n    const allNodesExceptComplexes = Array.from(self.participants.values()).filter(function (value) {\n        return value.type !== \"complex\";\n    });\n\n    if (pruned.length < allNodesExceptComplexes.length\n        && pruned.length > 3 && self.participants.size < 9) {\n        // <9 include hemoglobin, possibly some other small cases, but is catious, tends to mess other things up\n        // console.log(prunedIn);\n        doLayout(pruned, true);\n    } else {\n        doLayout(allNodesExceptComplexes, self.complexes.length > 0);\n    }\n\n    function doLayout(nodes, preRun) {\n        const layoutObj = {}; // todo get rid\n        layoutObj.nodes = nodes;\n        layoutObj.links = [];\n\n        const molLookUp = {};\n        let mi = 0;\n        for (let mol of nodes) {\n            molLookUp[mol.id] = mi;\n            mi++;\n        }\n\n        for (let binaryLink of self.allBinaryLinks.values()) {\n            const fromMol = binaryLink.participants[0];\n            const toMol = binaryLink.participants[1];\n            // if (preRun || (fromMol.binaryLinks.size === 4 || toMol.binaryLinks.size == 4)) {\n            const source = fromMol; //molLookUp[fromMol.id];\n            const target = toMol; //molLookUp[toMol.id];\n\n            if (source !== target && nodes.indexOf(source) !== -1 && nodes.indexOf(target) !== -1) { // todo - check what this is doing\n                const linkObj = {};\n                linkObj.source = molLookUp[fromMol.id];\n                linkObj.target = molLookUp[toMol.id];\n                linkObj.id = binaryLink.id;\n                layoutObj.links.push(linkObj);\n            }\n            // }\n        }\n\n        const groups = [];\n        if (!preRun && self.complexes) {\n            for (let g of self.complexes) {\n                g.leaves = [];\n                g.groups = [];\n                for (let interactor of g.naryLink.participants) {\n                    if (interactor.type !== \"complex\") {\n                        g.leaves.push(layoutObj.nodes.indexOf(interactor));\n                    }\n                }\n                groups.push(g);\n            }\n            for (let g of self.complexes) {\n                for (let interactor of g.naryLink.participants) {\n                    if (interactor.type === \"complex\") {\n                        //console.log(groups.indexOf(interactor));\n                        g.groups.push(groups.indexOf(interactor));\n                    }\n                }\n            }\n        }\n\n        // if (preRun) {\n        //    layoutObj.nodes = layoutObj.nodes.concat(prunedOut);\n        // }\n\n        // self.d3cola.convergenceThreshold = 0.01;\n        //console.log(\"groups\", groups);\n        delete self.d3cola._lastStress;\n        delete self.d3cola._alpha;\n        delete self.d3cola._descent;\n        delete self.d3cola._rootGroup;\n\n        let linkLength = (nodes.length < 30) ? 30 : 20;\n        const width = self.svgElement.parentNode.clientWidth;\n        const height = self.svgElement.parentNode.clientHeight;\n        //console.log(\"**\", layoutObj);\n        self.d3cola.size([height - 40, width - 40])\n            .nodes(layoutObj.nodes).groups(groups).links(layoutObj.links).avoidOverlaps(true);\n        let groupDebugSel, participantDebugSel;\n        if (self.debug) {\n            groupDebugSel = d3.select(self.container).selectAll(\".group\")\n                .data(groups);\n\n            groupDebugSel.enter().append(\"rect\")\n                .classed(\"group\", true)\n                .attr({\n                    rx: 5,\n                    ry: 5\n                })\n                .style(\"stroke\", \"blue\")\n                .style(\"fill\", \"none\");\n\n            participantDebugSel = d3.select(self.container).selectAll(\".node\")\n                .data(layoutObj.nodes);\n\n            participantDebugSel.enter().append(\"rect\")\n                .classed(\"node\", true)\n                .attr({\n                    rx: 5,\n                    ry: 5\n                })\n                .style(\"stroke\", \"red\")\n                .style(\"fill\", \"none\");\n\n            groupDebugSel.exit().remove();\n            participantDebugSel.exit().remove();\n        }\n\n        const startTime = Date.now();\n        self.d3cola.symmetricDiffLinkLengths(linkLength)\n            .on(\"tick\", function () {\n                if (Date.now() - startTime > 400) {//!preRun) {\n                    const nodes = self.d3cola.nodes();\n                    for (let node of nodes) {\n                        node.setPosition(node.x, node.y);\n                    }\n                    self.setAllLinkCoordinates();\n                    self.zoomToExtent();\n                    if (self.debug) {\n                        groupDebugSel.attr({\n                            x: function (d) {\n                                return d.bounds.x;// + (width / 2);\n                            },\n                            y: function (d) {\n                                return d.bounds.y;// + (height / 2);\n                            },\n                            width: function (d) {\n                                return d.bounds.width();\n                            },\n                            height: function (d) {\n                                return d.bounds.height();\n                            }\n                        });\n\n                        participantDebugSel.attr({\n                            x: function (d) {\n                                return d.bounds.x;// + (width / 2);\n                            },\n                            y: function (d) {\n                                return d.bounds.y;// + (height / 2);\n                            },\n                            width: function (d) {\n                                return d.bounds.width();\n                            },\n                            height: function (d) {\n                                return d.bounds.height();\n                            }\n                        });\n                    }\n                }\n            })\n            .on(\"end\", function () {\n                if (preRun) {\n                    // alert(\"initial run complete\");\n                    //     // for (let p of layoutObj.nodes) {\n                    //     //         p.fixed = 1;\n                    //     // }\n                    //     // let nodesExceptComplexes = Array.from(self.participants.values());\n                    //     // allNodesExceptComplexes = allNodesExceptComplexes.filter(function (value) {\n                    //     //     return value.type !== \"complex\";\n                    //     // });\n                    doLayout(allNodesExceptComplexes, false);\n                } else {\n                    for (let node of nodes) {\n                        node.setPosition(node.x, node.y);\n                    }\n                    self.setAllLinkCoordinates();\n                    self.zoomToExtent();\n                }\n            });\n        if (preRun) {\n            self.d3cola.start(23, 23, 0, 0, true);//, false, false);\n        } else {\n            self.d3cola.start(0, 23, 23, 0, true);//, false, false);\n        }\n    }\n};\n\nApp.prototype.getSVG = function () { //todo - update after styling of svg is moved to css\n    let svgXml = this.svgElement.outerHTML.replace(/<rect .*?\\/rect>/i, \"\"); //take out white background fill\n    const viewBox = \"viewBox=\\\"0 0 \" + this.svgElement.parentNode.clientWidth + \" \" + this.svgElement.parentNode.clientHeight + \"\\\" \";\n    svgXml = svgXml.replace(\"<svg \", \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" xmlns:xlink=\\\"http://www.w3.org/1999/xlink\\\" xmlns:ev=\\\"http://www.w3.org/2001/xml-events\\\" \" + viewBox);\n\n    return \"<?xml version=\\\"1.0\\\" encoding=\\\"UTF-8\\\" standalone=\\\"no\\\"?>\" +\n        \"<!DOCTYPE svg PUBLIC \\\"-//W3C//DTD SVG 1.1//EN\\\" \\\"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd\\\">\" +\n        svgXml;\n};\n\n// transform the mouse-position into a position on the svg\nApp.prototype.mouseToSVG = function (x, y) {\n    const p = this.svgElement.createSVGPoint();\n    p.x = x;\n    p.y = y;\n    return p.matrixTransform(this.container.getCTM().inverse());\n};\n\n// reads MI JSON format\nApp.prototype.readMIJSON = function (miJson, expand = true) {\n    readMijson(miJson, this, expand);\n};\n\nApp.prototype.checkLinks = function () {\n    function checkAll(linkMap) {\n        for (let link of linkMap.values()) {\n            link.check();\n        }\n    }\n\n    checkAll(this.allNaryLinks);\n    checkAll(this.allBinaryLinks);\n    checkAll(this.allUnaryLinks);\n    checkAll(this.allSequenceLinks);\n};\n\nApp.prototype.setAllLinkCoordinates = function () {\n    function setAll(linkMap) {\n        for (let link of linkMap.values()) {\n            link.setLinkCoordinates(true); // true means don't propogate changes from naryLink up to complex, everythings getting refreshed anyway\n        }\n    }\n\n    setAll(this.allNaryLinks);\n    setAll(this.allBinaryLinks);\n    setAll(this.allUnaryLinks);\n    setAll(this.allSequenceLinks);\n};\n\nApp.prototype.showTooltip = function (p) {\n    let ttX, ttY;\n    const length = this.tooltip.getComputedTextLength() + 16;\n    const width = this.svgElement.parentNode.clientWidth;\n    const height = this.svgElement.parentNode.clientHeight;\n    if (p.x + 20 + length < width) {\n        ttX = p.x;\n    } else {\n        ttX = width - length - 20;\n    }\n\n    if (p.y + 60 < height) {\n        ttY = p.y;\n    } else {\n        ttY = height - 60;\n    }\n    this.tooltip.setAttribute(\"x\", ttX + 22);\n    this.tooltip.setAttribute(\"y\", ttY + 47);\n    this.tooltip_bg.setAttribute(\"x\", ttX + 16);\n    this.tooltip_bg.setAttribute(\"y\", ttY + 28);\n    this.tooltip_subBg.setAttribute(\"x\", ttX + 16);\n    this.tooltip_subBg.setAttribute(\"y\", ttY + 28);\n};\n\nApp.prototype.setTooltip = function (text, color) {\n    if (text) {\n        this.tooltip.firstChild.data = text.toString().replace(/&(quot);/g, \"\\\"\");\n        this.tooltip.setAttribute(\"display\", \"block\");\n        const length = this.tooltip.getComputedTextLength();\n        this.tooltip_bg.setAttribute(\"width\", length + 16);\n        this.tooltip_subBg.setAttribute(\"width\", length + 16);\n        if (typeof color !== \"undefined\" && color != null) {\n            this.tooltip_bg.setAttribute(\"fill\", color);\n            this.tooltip_bg.setAttribute(\"stroke\", color);\n            this.tooltip_bg.setAttribute(\"fill-opacity\", \"0.5\");\n        } else {\n            this.tooltip_bg.setAttribute(\"fill\", \"white\");\n            this.tooltip_bg.setAttribute(\"stroke\", \"grey\");\n        }\n        this.tooltip_bg.setAttribute(\"height\", \"28\");\n        this.tooltip_subBg.setAttribute(\"height\", \"28\");\n        this.tooltip_bg.setAttribute(\"display\", \"block\");\n        this.tooltip_subBg.setAttribute(\"display\", \"block\");\n    } else {\n        this.hideTooltip();\n    }\n};\n\nApp.prototype.hideTooltip = function () {\n    this.tooltip.setAttribute(\"display\", \"none\");\n    this.tooltip_bg.setAttribute(\"display\", \"none\");\n    this.tooltip_subBg.setAttribute(\"display\", \"none\");\n};\n\nApp.prototype.addColorSchemeKey = function (/*HTMLDivElement*/ div) {\n    this.colorSchemeKeyDivs.add(div);\n    ColorSchemeKey.update(div, this);\n};\n\nApp.prototype.removeColorSchemeKey = function (/*HTMLDivElement*/ colorSchemeKeyDiv) {\n    this.colorSchemeKeyDivs.remove(colorSchemeKeyDiv);\n    colorSchemeKeyDiv.textContent = \"\";\n};\n\n//for backwards compatibility (noe?), tbh i might have made a bit of a mess here\nApp.prototype.setAnnotations = function (annoChoice) {\n    annoChoice = annoChoice.toUpperCase();\n    for (let annoType of this.annotationSetsShown.keys()) {\n        this.showAnnotations(annoType, annoChoice === annoType);\n    }\n    this.showAnnotations(annoChoice, true);\n};\n\nApp.prototype.showAnnotations = function (annoChoice, show) {\n    annoChoice = annoChoice.toUpperCase();\n    const self = this;\n    let setShown = this.annotationSetsShown.get(annoChoice);\n    if (typeof setShown === \"undefined\" && annoChoice !== \"MIFEATURES\") {\n        fetchAnnotations(annoChoice, this, function () {\n            self.annotationSetsShown.set(annoChoice, show);\n            self.updateAnnotations();\n        });\n    } else {\n        this.annotationSetsShown.set(annoChoice, show);\n        this.updateAnnotations();\n    }\n};\n\nApp.prototype.updateAnnotations = function () {\n    // //clear all annot's\n    for (let mol of this.participants.values()) {\n        if (mol.id.indexOf(\"uniprotkb_\") === 0) { //LIMIT IT TO PROTEINS\n            mol.clearPositionalFeatures();\n        }\n    }\n    chooseColors(this);\n    this.colorSchemeChanged();\n\n    for (let mol of this.participants.values()) {\n        if (mol.id.indexOf(\"uniprotkb_\") === 0) { //LIMIT IT TO PROTEINS\n            mol.setPositionalFeatures();\n        }\n    }\n    chooseColors(this);\n    this.colorSchemeChanged();\n};\n\nApp.prototype.colorSchemeChanged = function () {\n    for (let div of this.colorSchemeKeyDivs) {\n        ColorSchemeKey.update(div, this);\n    }\n};\n\nApp.prototype.getComplexColors = function () {\n    return NaryLink.naryColors;\n};\n\nApp.prototype.getFeatureColors = function () {\n    return this.featureColors;\n};\n\nApp.prototype.collapseAll = function () {\n    for (let participant of this.participants.values()) {\n        if (participant.form === 1) {\n            participant.setForm(0);\n        }\n    }\n};\n\nApp.prototype.expandAll = function () {\n    for (let participant of this.participants.values()) {\n        if (participant.form === 0) {\n            participant.setForm(1);\n        }\n    }\n};\n\n//from noe\nApp.prototype.expandAndCollapseSelection = function (moleculesSelected) {\n    const molecules = this.molecules.values();\n    for (var m = 0; m < molecules.length; m++) {\n        var molecule = molecules[m];\n        var molecule_id = molecule.json.identifier.id;\n        if (moleculesSelected.includes(molecule_id)) {\n            if (molecule.form === 0) {\n                molecule.setForm(1);\n            }\n        } else if (molecule.form === 1) {\n            molecule.setForm(0);\n        }\n    }\n};\n\n\n// export function makeSymbolKey(targetDiv){\n//     new SymbolKey(targetDiv);\n// }\n"],"mappings":"AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/js/app.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"App\", function() { return App; });\n/* harmony import */ var _css_xinet_css__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../css/xinet.css */ \"./src/css/xinet.css\");\n/* harmony import */ var _css_xinet_css__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_css_xinet_css__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _package_json__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../package.json */ \"./package.json\");\nvar _package_json__WEBPACK_IMPORTED_MODULE_1___namespace = /*#__PURE__*/__webpack_require__.t(/*! ../../package.json */ \"./package.json\", 1);\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! d3 */ \"./node_modules/d3/d3.js\");\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(d3__WEBPACK_IMPORTED_MODULE_2__);\n/* harmony import */ var d3_scale_chromatic__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! d3-scale-chromatic */ \"./node_modules/d3-scale-chromatic/src/index.js\");\n/* harmony import */ var _cola__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./cola */ \"./src/js/cola.js\");\n/* harmony import */ var _cola__WEBPACK_IMPORTED_MODULE_4___default = /*#__PURE__*/__webpack_require__.n(_cola__WEBPACK_IMPORTED_MODULE_4__);\n/* harmony import */ var _read_mijson__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./read-mijson */ \"./src/js/read-mijson.js\");\n/* harmony import */ var _annotations__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./annotations */ \"./src/js/annotations.js\");\n/* harmony import */ var _svgexp__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ./svgexp */ \"./src/js/svgexp.js\");\n/* harmony import */ var _color_scheme_key__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ./color-scheme-key */ \"./src/js/color-scheme-key.js\");\n/* harmony import */ var _viz_link_nary_link__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! ./viz/link/nary-link */ \"./src/js/viz/link/nary-link.js\");\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_10__ = __webpack_require__(/*! ./config */ \"./src/js/config.js\");\n// eslint-disable-next-line no-unused-vars\n\n\n\n\n\n\n\n\n\n// import {SymbolKey} from \"./symbol-key\";\n\n\n\n\n// could refactor everything to use ES6 class syntax\n// but https://benmccormick.org/2015/04/07/es6-classes-and-backbone-js\n// \"ES6 classes don’t support adding properties directly to the class instance, only functions/methods\"\n// so backbone doesn't work\n// so continuing to use prototypical inheritance in things for time being\n\nfunction App(/*HTMLDivElement*/networkDiv) {\n // this.debug = true; // things aren't exactly lined up in the bounding boxes cola is using, to do so breaks symetery of some things\n this.el = networkDiv;\n\n this.STATES = {};\n this.STATES.MOUSE_UP = 0; //start state, also set when mouse up on svgElement\n this.STATES.PANNING = 1; //set by mouse down on svgElement - left button, no shift or util\n this.STATES.DRAGGING = 2; //set by mouse down on Protein or Link\n this.STATES.ROTATING = 3; //set by mouse down on Rotator, drag?\n this.STATES.SELECTING = 4; //set by mouse down on svgElement- right button or left button shift or util, drag\n\n //avoids prob with 'save - web page complete'\n this.el.textContent = \"\"; //https://stackoverflow.com/questions/3955229/remove-all-child-elements-of-a-dom-node-in-javascript\n\n this.d3cola = _cola__WEBPACK_IMPORTED_MODULE_4__[\"d3adaptor\"]();\n\n const customMenuSel = d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](this.el)\n .append(\"div\").classed(\"custom-menu-margin\", true)\n .append(\"div\").classed(\"custom-menu\", true)\n .append(\"ul\");\n\n const self = this;\n const collapse = customMenuSel.append(\"li\").classed(\"collapse\", true); //.append(\"button\");\n collapse.text(\"Collapse\");\n collapse[0][0].onclick = function (evt) {\n self.collapseProtein(evt);\n };\n const scaleButtonsListItemSel = customMenuSel.append(\"li\").text(\"Scale: \");\n\n this.barScales = [0.01, 0.2, 1, 2, 4, 8];\n const scaleButtons = scaleButtonsListItemSel.selectAll(\"ul.custom-menu\")\n .data(this.barScales)\n .enter()\n .append(\"div\")\n .attr(\"class\", \"barScale\")\n .append(\"label\");\n scaleButtons.append(\"span\")\n .text(function (d) {\n if (d === 8) return \"AA\";\n else return d;\n });\n scaleButtons.append(\"input\")\n // .attr (\"id\", function(d) { return d*100; })\n .attr(\"class\", function (d) {\n return \"scaleButton scaleButton_\" + (d * 100);\n })\n .attr(\"name\", \"scaleButtons\")\n .attr(\"type\", \"radio\")\n .on(\"change\", function (d) {\n self.preventDefaultsAndStopPropagation(d);\n self.contextMenuProt.setStickScale(d, self.contextMenuPoint);\n });\n\n const contextMenu = d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](\".custom-menu-margin\").node();\n contextMenu.onmouseout = function (evt) {\n let e = evt.relatedTarget;\n do {\n if (e === this) return;\n e = e.parentNode;\n } while (e);\n self.contextMenuProt = null;\n d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](this).style(\"display\", \"none\");\n };\n\n\n //create SVG element\n this.svgElement = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"svg\");\n this.svgElement.setAttribute(\"id\", \"complexViewerSVG\");\n\n //add listeners\n this.svgElement.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.svgElement.onmousemove = function (evt) {\n self.mouseMove(evt);\n };\n this.svgElement.onmouseup = function (evt) {\n self.mouseUp(evt);\n };\n this.svgElement.onmouseout = function (evt) {\n self.hideTooltip(evt);\n };\n this.lastMouseUp = new Date().getTime();\n /*this.svgElement.ontouchstart = function(evt) {\n self.touchStart(evt);\n };\n this.svgElement.ontouchmove = function(evt) {\n self.touchMove(evt);\n };\n this.svgElement.ontouchend = function(evt) {\n self.touchEnd(evt);\n };\n */\n\n this.el.oncontextmenu = function (evt) {\n if (evt.preventDefault) { // necessary for addEventListener, works with traditional\n evt.preventDefault();\n }\n evt.returnValue = false; // necessary for attachEvent, works with traditional\n return false; // works with traditional, not with attachEvent or addEventListener\n };\n\n //legend changed callbacks\n this.colorSchemeKeyDivs = new Set();\n\n this.el.appendChild(this.svgElement);\n\n // various groups needed\n this.container = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"g\");\n this.container.setAttribute(\"id\", \"container\");\n\n const svg = d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](this.svgElement);\n this.defs = svg.append(\"defs\");\n\n this.acknowledgement = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"g\");\n const ackText = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"text\");\n ackText.innerHTML = \"ComplexViewer \"\n + _package_json__WEBPACK_IMPORTED_MODULE_1__[\"version\"] + \"by Rappsilber Laboratory\";\n\n this.acknowledgement.appendChild(ackText);\n ackText.setAttribute(\"font-size\", \"8pt\");\n this.svgElement.appendChild(this.acknowledgement);\n\n this.naryLinks = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"g\");\n this.naryLinks.setAttribute(\"id\", \"naryLinks\");\n this.container.appendChild(this.naryLinks);\n\n this.p_pLinksWide = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"g\");\n this.p_pLinksWide.setAttribute(\"id\", \"p_pLinksWide\");\n this.container.appendChild(this.p_pLinksWide);\n\n this.highlights = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"g\");\n this.highlights.setAttribute(\"class\", \"highlights\"); //interactors also contain highlight groups\n this.container.appendChild(this.highlights);\n\n this.res_resLinks = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"g\");\n this.res_resLinks.setAttribute(\"id\", \"res_resLinks\");\n this.container.appendChild(this.res_resLinks);\n\n this.p_pLinks = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"g\");\n this.p_pLinks.setAttribute(\"id\", \"p_pLinks\");\n this.container.appendChild(this.p_pLinks);\n\n this.proteinUpper = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"g\");\n this.proteinUpper.setAttribute(\"id\", \"proteinUpper\");\n this.container.appendChild(this.proteinUpper);\n\n this.selfRes_resLinks = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"g\");\n this.selfRes_resLinks.setAttribute(\"id\", \"res_resLinks\");\n this.container.appendChild(this.selfRes_resLinks);\n\n this.svgElement.appendChild(this.container);\n\n //showing title as tooltips is NOT part of svg spec (even though some browsers do this)\n //also more responsive / more control if we do out own\n this.tooltip = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"text\");\n this.tooltip.setAttribute(\"x\", \"0\");\n this.tooltip.setAttribute(\"y\", \"0\");\n const tooltipTextNode = document.createTextNode(\"tooltip\");\n this.tooltip.classList.add(\"label\", \"tooltip\");\n\n this.tooltip.appendChild(tooltipTextNode);\n\n this.tooltip_bg = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"rect\");\n this.tooltip_bg.classList.add(\"tooltip-background\");\n\n this.tooltip_subBg = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_10__[\"svgns\"], \"rect\");\n this.tooltip_subBg.classList.add(\"tooltip-sub-background\");\n\n this.svgElement.appendChild(this.tooltip_subBg);\n this.svgElement.appendChild(this.tooltip_bg);\n this.svgElement.appendChild(this.tooltip);\n\n this.clear();\n}\n\nApp.prototype.createHatchedFill = function (name, color) {\n if (!this.checkedHatchNames.has(name)) {\n const pattern = this.defs.append(\"pattern\")\n .attr(\"id\", name)\n .attr(\"patternUnits\", \"userSpaceOnUse\")\n .attr(\"x\", 0)\n .attr(\"y\", 0)\n .attr(\"width\", 12)\n .attr(\"height\", 12)\n .attr(\"patternTransform\", \"rotate(45)\");\n\n pattern.append(\"rect\")\n .attr(\"x\", 0)\n .attr(\"y\", 2)\n .attr(\"width\", 12)\n .attr(\"height\", 4)\n .attr(\"fill\", color);\n\n pattern.append(\"rect\")\n .attr(\"x\", 0)\n .attr(\"y\", 8)\n .attr(\"width\", 12)\n .attr(\"height\", 4)\n .attr(\"fill\", color);\n\n this.checkedHatchNames.add(name);\n }\n};\n\nApp.prototype.clear = function () {\n this.d3cola.stop();\n\n this.annotationSetsShown = new Map();\n // this.annotationSetsShown.set(\"MI FEATURES\", true);\n\n this.checkedHatchNames = new Set();\n\n //lighten colors\n const complexColors = [];\n for (let c of d3_scale_chromatic__WEBPACK_IMPORTED_MODULE_3__[\"schemePastel2\"]) {//colorbrewer.Pastel2[8]) {\n const hsl = d3__WEBPACK_IMPORTED_MODULE_2__[\"hsl\"](c);\n hsl.l = 0.9;\n complexColors.push(hsl + \"\");\n }\n\n _viz_link_nary_link__WEBPACK_IMPORTED_MODULE_9__[\"NaryLink\"].naryColors = d3__WEBPACK_IMPORTED_MODULE_2__[\"scale\"].ordinal().range(complexColors);\n this.defs.selectAll(\".feature_checkers\").remove();\n\n this.naryLinks.textContent = \"\";\n this.p_pLinksWide.textContent = \"\";\n this.highlights.textContent = \"\";\n this.p_pLinks.textContent = \"\";\n this.res_resLinks.textContent = \"\";\n this.proteinUpper.textContent = \"\";\n this.selfRes_resLinks.textContent = \"\";\n\n // if we are dragging something at the moment - this will be the element that is dragged\n this.dragElement = null;\n // from where did we start dragging\n this.dragStart = {};\n\n this.participants = new Map();\n this.allNaryLinks = new Map();\n this.allBinaryLinks = new Map();\n this.allUnaryLinks = new Map();\n this.allSequenceLinks = new Map();\n this.complexes = [];\n\n this.proteinCount = 0;\n this.z = 1;\n this.hideTooltip();\n this.state = this.STATES.MOUSE_UP;\n};\n\nApp.prototype.collapseProtein = function () {\n d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](\".custom-menu-margin\").style(\"display\", \"none\");\n this.contextMenuProt.setForm(0, this.contextMenuPoint);\n this.contextMenuProt = null;\n};\n\nApp.prototype.init = function () {\n this.d3cola.stop();\n\n for (let participant of this.participants.values()) {\n if (participant.type != \"complex\") {\n participant.setPosition(-500, -500);\n }\n }\n this.updateAnnotations();\n this.checkLinks(); //totally needed, not sure why tbh todo - check this out\n this.setAllLinkCoordinates(); // just to move them off screen at first\n\n let maxSeqLength = 0;\n for (let participant of this.participants.values()) {\n if (participant.size > maxSeqLength) {\n maxSeqLength = participant.size;\n }\n }\n const width = this.svgElement.parentNode.clientWidth;\n const defaultPixPerRes = width * 0.8 / maxSeqLength;\n //console.log(\"defaultPixPerRes:\" + defaultPixPerRes);\n // https://stackoverflow.com/questions/12141150/from-list-of-integers-get-number-closest-to-a-given-value/12141511#12141511\n function takeClosest(myList, myNumber) {\n const bisect = d3__WEBPACK_IMPORTED_MODULE_2__[\"bisector\"](function (d) {\n return d;\n }).left;\n const pos = bisect(myList, myNumber);\n if (pos === 0 || pos === 1) {\n return myList[1]; // don't return smallest scale as default\n }\n if (pos === myList.length) {\n return myList[myList.length - 1];\n }\n return myList[pos - 1];\n }\n\n this.defaultBarScale = takeClosest(this.barScales, defaultPixPerRes);\n //console.log(\"default bar scale: \" + this.defaultBarScale)\n\n for (let participant of this.participants.values()) {\n if (participant.type !== \"complex\") {\n this.proteinUpper.appendChild(participant.upperGroup);\n if (participant.json.type.name === \"protein\") {\n // participant.initSelfLinkSVG(); // todo - may not even do anything, not sure its working\n participant.stickZoom = this.defaultBarScale;\n if (this.participants.size < 4) {\n participant.toStickNoTransition();\n }\n }\n }\n }\n\n this.autoLayout();\n};\n\nApp.prototype.zoomToExtent = function () {\n const width = this.svgElement.parentNode.clientWidth;\n const height = this.svgElement.parentNode.clientHeight;\n const bbox = this.container.getBBox();\n let xr = (width / bbox.width).toFixed(4) - 0;\n let yr = (height / bbox.height).toFixed(4) - 0;\n let scaleFactor;\n if (yr < xr) {\n scaleFactor = yr;\n } else {\n scaleFactor = xr;\n }\n if (scaleFactor < 1) { ///didn't fit in div\n //console.log(\"no fit\", scaleFactor);\n xr = (width - 40) / (bbox.width);\n yr = (height - 40) / (bbox.height);\n let scaleFactor;\n if (yr < xr) {\n scaleFactor = yr;\n } else {\n scaleFactor = xr;\n }\n\n if (scaleFactor > this.z) {\n scaleFactor = this.z;\n }\n\n //bbox.x + x = 0;\n let x = -bbox.x + (20 / scaleFactor);\n //box.y + y = 0\n let y = -bbox.y + (20 / scaleFactor);\n this.container.setAttribute(\"transform\", \"scale(\" + scaleFactor + \") translate(\" + x + \" \" + y + \") \");\n this.z = this.container.getCTM().inverse().a;\n } else {\n //console.log(\"fit\", scaleFactor);\n // this.container.setAttribute(\"transform\", \"scale(\" + 1 + \") translate(\" + -(width/2) + \" \" + -bbox.y + \")\");\n const deltaWidth = width - bbox.width;\n const deltaHeight = height - bbox.height;\n //bbox.x + x = deltaWidth /2;\n let x = (deltaWidth / 2) - bbox.x;\n //box.y + y = deltaHeight / 2\n let y = (deltaHeight / 2) - bbox.y;\n this.container.setAttribute(\"transform\", \"scale(\" + 1 + \") translate(\" + x + \" \" + y + \")\");\n this.z = 1;\n }\n\n //todo - following could be tided up by using acknowledgement bbox or positioning att's of text\n this.acknowledgement.setAttribute(\"transform\", \"translate(\" + (width - 150) + \", \" + (height - 30) + \")\");\n};\n\n//listeners also attached to mouse events by Interactor (and Rotator) and Link, those consume their events\n//mouse down on svgElement must be allowed to propogate (to fire event on Prots/Links)\n\nApp.prototype.mouseDown = function (evt) {\n //prevent default, but allow propogation\n evt.preventDefault();\n this.d3cola.stop();\n const p = this.getEventPoint(evt); // seems to be correct, see below\n this.dragStart = this.mouseToSVG(p.x, p.y);\n return false;\n};\n\n// dragging/rotation/panning/selecting\nApp.prototype.mouseMove = function (evt) {\n const p = this.getEventPoint(evt); // seems to be correct, see below\n const c = this.mouseToSVG(p.x, p.y);\n\n if (this.dragElement != null) { //dragging or rotating\n this.hideTooltip();\n const dx = this.dragStart.x - c.x;\n const dy = this.dragStart.y - c.y;\n\n if (this.state === this.STATES.DRAGGING) {\n // we are currently dragging things around\n let ox, oy, nx, ny;\n if (!this.dragElement.ix) {\n for (let participant of this.dragElement.participants) {\n participant.changePosition(dx, dy);\n }\n this.setAllLinkCoordinates();\n } else {\n ox = this.dragElement.ix;\n oy = this.dragElement.iy;\n nx = ox - dx;\n ny = oy - dy;\n this.dragElement.setPosition(nx, ny);\n this.dragElement.setAllLinkCoordinates();\n }\n this.dragStart = c;\n } else { //not dragging or rotating yet, maybe we should start\n // don't start dragging just on a click - we need to move the mouse a bit first\n if (Math.sqrt(dx * dx + dy * dy) > (5 * this.z)) {\n this.state = this.STATES.DRAGGING;\n\n }\n }\n } else {\n this.showTooltip(p);\n }\n return false;\n};\n\n// this ends all dragging and rotating\nApp.prototype.mouseUp = function (evt) {\n const time = new Date().getTime();\n //console.log(\"Mouse up: \" + evt.srcElement + \" \" + (time - this.lastMouseUp));\n this.preventDefaultsAndStopPropagation(evt);\n //eliminate some spurious mouse up events\n if ((time - this.lastMouseUp) > 150) {\n\n const p = this.getEventPoint(evt); // seems to be correct, see below\n const c = this.mouseToSVG(p.x, p.y);\n\n if (this.dragElement && this.dragElement.type === \"protein\") { /// todo be consistent about how to check if thing is protein\n if (!(this.state === this.STATES.DRAGGING || this.state === this.STATES.ROTATING)) { //not dragging or rotating\n if (this.dragElement.form === 0) {\n this.dragElement.setForm(1);\n } else {\n this.contextMenuProt = this.dragElement;\n this.contextMenuPoint = c;\n const menu = d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](\".custom-menu-margin\");\n menu.style(\"top\", (evt.pageY - 20) + \"px\").style(\"left\", (evt.pageX - 20) + \"px\").style(\"display\", \"block\");\n d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](\".scaleButton_\" + (this.dragElement.stickZoom * 100)).property(\"checked\", true);\n }\n }\n }\n }\n\n this.dragElement = null;\n this.state = this.STATES.MOUSE_UP;\n\n this.lastMouseUp = time;\n return false;\n};\n\n//gets mouse position\nApp.prototype.getEventPoint = function (evt) {\n const p = this.svgElement.createSVGPoint();\n let element = this.svgElement.parentNode;\n let top = 0,\n left = 0;\n do {\n top += element.offsetTop || 0;\n left += element.offsetLeft || 0;\n element = element.offsetParent;\n } while (element);\n p.x = evt.pageX - left;\n p.y = evt.pageY - top;\n return p;\n};\n\n//stop event propogation and defaults; only do what we ask\nApp.prototype.preventDefaultsAndStopPropagation = function (evt) {\n if (evt.stopPropagation)\n evt.stopPropagation();\n if (evt.cancelBubble != null)\n evt.cancelBubble = true;\n if (evt.preventDefault)\n evt.preventDefault();\n};\n\n/**\n * Handle touchstart event.\n\n App.prototype.touchStart = function(evt) {\n //prevent default, but allow propogation\n evt.preventDefault();\n\n //stop force layout\n if (typeof this.d3cola !== 'undefined' && this.d3cola != null) {\n this.d3cola.stop();\n }\n\n var p = this.getTouchEventPoint(evt); // seems to be correct, see below\n this.dragStart = this.mouseToSVG(p.x, p.y);\n};\n\n // dragging/rotation/panning/selecting\n App.prototype.touchMove = function(evt) {\n // if (this.sequenceInitComplete) { // just being cautious\n var p = this.getTouchEventPoint(evt); // seems to be correct, see below\n var c = this.mouseToSVG(p.x, p.y);\n\n if (this.dragElement != null) { //dragging or rotating\n this.hideTooltip();\n var dx = this.dragStart.x - c.x;\n var dy = this.dragStart.y - c.y;\n\n if (this.state === this.STATES.DRAGGING) {\n // we are currently dragging things around\n var ox, oy, nx, ny;\n if (typeof this.dragElement.ix=== 'undefined') { // if not an Interactor\n var nodes = this.dragElement.interactors;\n var nodeCount = nodes.length;\n for (var i = 0; i < nodeCount; i++) {\n var protein = nodes[i];\n ox = protein.cx;\n oy = protein.cy;\n nx = ox - dx;\n ny = oy - dy;\n protein.setPosition(nx, ny);\n protein.setAllLinkCoordinates();\n }\n for (i = 0; i < nodeCount; i++) {\n nodes[i].setAllLinkCoordinates();\n }\n } else {\n ox = this.dragElement.cx;\n oy = this.dragElement.cy;\n nx = ox - dx;\n ny = oy - dy;\n this.dragElement.setPosition(nx, ny);\n this.dragElement.setAllLinkCoordinates();\n }\n this.dragStart = c;\n } else { //not dragging or rotating yet, maybe we should start\n // don't start dragging just on a click - we need to move the mouse a bit first\n if (Math.sqrt(dx * dx + dy * dy) > (5 * this.z)) {\n this.state = this.STATES.DRAGGING;\n\n }\n }\n } else {\n this.showTooltip(p);\n }\n return false;\n};\n\n// this ends all dragging and rotating\nApp.prototype.touchEnd = function(evt) {\n var time = new Date().getTime();\n //console.log(\"Mouse up: \" + evt.srcElement + \" \" + (time - this.lastMouseUp));\n this.preventDefaultsAndStopPropagation(evt);\n //eliminate some spurious mouse up events\n if ((time - this.lastMouseUp) > 150) {\n\n var p = this.getTouchEventPoint(evt); // seems to be correct, see below\n var c = this.mouseToSVG(p.x, p.y);\n\n if (this.dragElement != null) {\n if (!(this.state === this.STATES.DRAGGING || this.state === this.STATES.ROTATING)) { //not dragging or rotating\n if (this.dragElement.form === 0) {\n this.dragElement.setForm(1);\n } else {\n this.contextMenuProt = this.dragElement;\n this.contextMenuPoint = c;\n var menu = d3.select(\".custom-menu-margin\")\n menu.style(\"top\", (evt.pageY - 20) + \"px\").style(\"left\", (evt.pageX - 20) + \"px\").style(\"display\", \"block\");\n d3.select(\".scaleButton_\" + (this.dragElement.stickZoom * 100)).property(\"checked\", true)\n }\n }\n }\n }\n\n this.dragElement = null;\n this.whichRotator = -1;\n this.state = this.STATES.MOUSE_UP;\n\n this.lastMouseUp = time;\n return false;\n};\n\n//gets mouse position\nApp.prototype.getTouchEventPoint = function(evt) {\n var p = this.svgElement.createSVGPoint();\n var element = this.svgElement.parentNode;\n var top = 0,\n left = 0;\n do {\n top += element.offsetTop || 0;\n left += element.offsetLeft || 0;\n element = element.offsetParent;\n } while (element);\n p.x = evt.touches[0].pageX - left;\n p.y = evt.touches[0].pageY - top;\n return p;\n};\n */\nApp.prototype.autoLayout = function () {\n this.d3cola.stop();\n const self = this;\n\n // needed to ensure consistent results\n for (let p of self.participants.values()) {\n delete p.x;\n delete p.y;\n delete p.px;\n delete p.py;\n delete p.bounds;\n p.fixed = 0;\n }\n\n //// prune leaves from network then layout, then add back leaves and layout again (fixes haemoglobin)\n const pruned = [];\n for (let participant of self.participants.values()) {\n if (participant.binaryLinks.size > 2 && participant.type !== \"complex\") {\n pruned.push(participant);\n }\n }\n const allNodesExceptComplexes = Array.from(self.participants.values()).filter(function (value) {\n return value.type !== \"complex\";\n });\n\n if (pruned.length < allNodesExceptComplexes.length\n && pruned.length > 3 && self.participants.size < 9) {\n // <9 include hemoglobin, possibly some other small cases, but is catious, tends to mess other things up\n // console.log(prunedIn);\n doLayout(pruned, true);\n } else {\n doLayout(allNodesExceptComplexes, self.complexes.length > 0);\n }\n\n function doLayout(nodes, preRun) {\n const layoutObj = {}; // todo get rid\n layoutObj.nodes = nodes;\n layoutObj.links = [];\n\n const molLookUp = {};\n let mi = 0;\n for (let mol of nodes) {\n molLookUp[mol.id] = mi;\n mi++;\n }\n\n for (let binaryLink of self.allBinaryLinks.values()) {\n const fromMol = binaryLink.participants[0];\n const toMol = binaryLink.participants[1];\n // if (preRun || (fromMol.binaryLinks.size === 4 || toMol.binaryLinks.size == 4)) {\n const source = fromMol; //molLookUp[fromMol.id];\n const target = toMol; //molLookUp[toMol.id];\n\n if (source !== target && nodes.indexOf(source) !== -1 && nodes.indexOf(target) !== -1) { // todo - check what this is doing\n const linkObj = {};\n linkObj.source = molLookUp[fromMol.id];\n linkObj.target = molLookUp[toMol.id];\n linkObj.id = binaryLink.id;\n layoutObj.links.push(linkObj);\n }\n // }\n }\n\n const groups = [];\n if (!preRun && self.complexes) {\n for (let g of self.complexes) {\n g.leaves = [];\n g.groups = [];\n for (let interactor of g.naryLink.participants) {\n if (interactor.type !== \"complex\") {\n g.leaves.push(layoutObj.nodes.indexOf(interactor));\n }\n }\n groups.push(g);\n }\n for (let g of self.complexes) {\n for (let interactor of g.naryLink.participants) {\n if (interactor.type === \"complex\") {\n //console.log(groups.indexOf(interactor));\n g.groups.push(groups.indexOf(interactor));\n }\n }\n }\n }\n\n //console.log(\"groups\", groups);\n delete self.d3cola._lastStress;\n delete self.d3cola._alpha;\n delete self.d3cola._descent;\n delete self.d3cola._rootGroup;\n\n let linkLength = (nodes.length < 30) ? 30 : 20;\n const width = self.svgElement.parentNode.clientWidth;\n const height = self.svgElement.parentNode.clientHeight;\n //console.log(\"**\", layoutObj);\n self.d3cola.size([height - 40, width - 40])\n .nodes(layoutObj.nodes).groups(groups).links(layoutObj.links).avoidOverlaps(true);\n let groupDebugSel, participantDebugSel;\n if (self.debug) {\n groupDebugSel = d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](self.container).selectAll(\".group\")\n .data(groups);\n\n groupDebugSel.enter().append(\"rect\")\n .classed(\"group\", true)\n .attr({\n rx: 5,\n ry: 5\n })\n .style(\"stroke\", \"blue\")\n .style(\"fill\", \"none\");\n\n participantDebugSel = d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](self.container).selectAll(\".node\")\n .data(layoutObj.nodes);\n\n participantDebugSel.enter().append(\"rect\")\n .classed(\"node\", true)\n .attr({\n rx: 5,\n ry: 5\n })\n .style(\"stroke\", \"red\")\n .style(\"fill\", \"none\");\n\n groupDebugSel.exit().remove();\n participantDebugSel.exit().remove();\n }\n\n const startTime = Date.now();\n self.d3cola.symmetricDiffLinkLengths(linkLength)\n .on(\"tick\", function () {\n if (Date.now() - startTime > 750) {//!preRun) {\n const nodes = self.d3cola.nodes();\n for (let node of nodes) {\n node.setPosition(node.x, node.y);\n }\n self.setAllLinkCoordinates();\n self.zoomToExtent();\n if (self.debug) {\n groupDebugSel.attr({\n x: function (d) {\n return d.bounds.x;// + (width / 2);\n },\n y: function (d) {\n return d.bounds.y;// + (height / 2);\n },\n width: function (d) {\n return d.bounds.width();\n },\n height: function (d) {\n return d.bounds.height();\n }\n });\n\n participantDebugSel.attr({\n x: function (d) {\n return d.bounds.x;// + (width / 2);\n },\n y: function (d) {\n return d.bounds.y;// + (height / 2);\n },\n width: function (d) {\n return d.bounds.width();\n },\n height: function (d) {\n return d.bounds.height();\n }\n });\n }\n }\n })\n .on(\"end\", function () {\n if (preRun) {\n // alert(\"initial run complete\");\n // // for (let p of layoutObj.nodes) {\n // // p.fixed = 1;\n // // }\n doLayout(allNodesExceptComplexes, false);\n } else {\n for (let node of nodes) {\n node.setPosition(node.x, node.y);\n }\n self.setAllLinkCoordinates();\n self.zoomToExtent();\n }\n });\n if (preRun) {\n self.d3cola.start(23, 23, 0, 0, true);//, false, false);\n } else {\n self.d3cola.start(0, 23, 23, 0, true);//, false, false);\n }\n }\n};\n\nApp.prototype.getSVG = function () { //todo - somewhat broken, annotations missing\n var svgSel = d3__WEBPACK_IMPORTED_MODULE_2__[\"select\"](this.el).selectAll(\"svg\");\n var svgArr = [svgSel.node()];\n var svgStrings = _svgexp__WEBPACK_IMPORTED_MODULE_7__[\"svgUtils\"].capture(svgArr);\n var svgXML = _svgexp__WEBPACK_IMPORTED_MODULE_7__[\"svgUtils\"].makeXMLStr(new XMLSerializer(), svgStrings[0]);\n\n return svgXML;\n\n // var fileName = this.filenameStateString().substring(0, 240);\n // download(svgXML, 'application/svg', fileName + \".svg\");\n\n};\n\n// App.prototype.getSVG = function () {\n// let svgXml = this.svgElement.outerHTML.replace(//i, \"\"); //take out white background fill\n// const viewBox = \"viewBox=\\\"0 0 \" + this.svgElement.parentNode.clientWidth + \" \" + this.svgElement.parentNode.clientHeight + \"\\\" \";\n// svgXml = svgXml.replace(\"\" +\n// \"\" +\n// svgXml;\n// };\n\n// transform the mouse-position into a position on the svg\nApp.prototype.mouseToSVG = function (x, y) {\n const p = this.svgElement.createSVGPoint();\n p.x = x;\n p.y = y;\n return p.matrixTransform(this.container.getCTM().inverse());\n};\n\n// reads MI JSON format\nApp.prototype.readMIJSON = function (miJson, expand = true) {\n Object(_read_mijson__WEBPACK_IMPORTED_MODULE_5__[\"readMijson\"])(miJson, this, expand);\n};\n\nApp.prototype.checkLinks = function () {\n for (let link of this.allNaryLinks.values()) {\n link.check();\n }\n for (let link of this.allBinaryLinks.values()) {\n link.check();\n }\n for (let link of this.allUnaryLinks.values()) {\n link.check();\n }\n for (let link of this.allSequenceLinks.values()) {\n link.check();\n }\n};\n\nApp.prototype.setAllLinkCoordinates = function () {\n for (let link of this.allNaryLinks.values()) {\n link.setLinkCoordinates();\n }\n for (let link of this.allBinaryLinks.values()) {\n link.setLinkCoordinates();\n }\n for (let link of this.allUnaryLinks.values()) {\n link.setLinkCoordinates();\n }\n for (let link of this.allSequenceLinks.values()) {\n link.setLinkCoordinates();\n }\n};\n\nApp.prototype.showTooltip = function (p) {\n let ttX, ttY;\n const length = this.tooltip.getComputedTextLength() + 16;\n const width = this.svgElement.parentNode.clientWidth;\n const height = this.svgElement.parentNode.clientHeight;\n if (p.x + 20 + length < width) {\n ttX = p.x;\n } else {\n ttX = width - length - 20;\n }\n\n if (p.y + 60 < height) {\n ttY = p.y;\n } else {\n ttY = height - 60;\n }\n this.tooltip.setAttribute(\"x\", ttX + 22);\n this.tooltip.setAttribute(\"y\", ttY + 47);\n this.tooltip_bg.setAttribute(\"x\", ttX + 16);\n this.tooltip_bg.setAttribute(\"y\", ttY + 28);\n this.tooltip_subBg.setAttribute(\"x\", ttX + 16);\n this.tooltip_subBg.setAttribute(\"y\", ttY + 28);\n};\n\nApp.prototype.setTooltip = function (text, color) {\n if (text) {\n this.tooltip.firstChild.data = text.toString().replace(/&(quot);/g, \"\\\"\");\n this.tooltip.setAttribute(\"display\", \"block\");\n const length = this.tooltip.getComputedTextLength();\n this.tooltip_bg.setAttribute(\"width\", length + 16);\n this.tooltip_subBg.setAttribute(\"width\", length + 16);\n if (typeof color !== \"undefined\" && color != null) {\n this.tooltip_bg.setAttribute(\"fill\", color);\n this.tooltip_bg.setAttribute(\"stroke\", color);\n this.tooltip_bg.setAttribute(\"fill-opacity\", \"0.5\");\n } else {\n this.tooltip_bg.setAttribute(\"fill\", \"white\");\n this.tooltip_bg.setAttribute(\"stroke\", \"grey\");\n }\n // todo - whats this height for?\n this.tooltip_bg.setAttribute(\"height\", \"28\");\n this.tooltip_subBg.setAttribute(\"height\", \"28\");\n this.tooltip_bg.setAttribute(\"display\", \"block\");\n this.tooltip_subBg.setAttribute(\"display\", \"block\");\n } else {\n this.hideTooltip();\n }\n};\n\nApp.prototype.hideTooltip = function () {\n this.tooltip.setAttribute(\"display\", \"none\");\n this.tooltip_bg.setAttribute(\"display\", \"none\");\n this.tooltip_subBg.setAttribute(\"display\", \"none\");\n};\n\nApp.prototype.addColorSchemeKey = function (/*HTMLDivElement*/ div) {\n this.colorSchemeKeyDivs.add(div);\n _color_scheme_key__WEBPACK_IMPORTED_MODULE_8__[\"update\"](div, this);\n};\n\nApp.prototype.removeColorSchemeKey = function (/*HTMLDivElement*/ colorSchemeKeyDiv) {\n this.colorSchemeKeyDivs.remove(colorSchemeKeyDiv);\n colorSchemeKeyDiv.textContent = \"\";\n};\n\n//for backwards compatibility (noe?), tbh i might have made a bit of a mess here\nApp.prototype.setAnnotations = function (annoChoice) {\n annoChoice = annoChoice.toUpperCase();\n for (let annoType of this.annotationSetsShown.keys()) {\n this.showAnnotations(annoType, annoChoice === annoType);\n }\n this.showAnnotations(annoChoice, true);\n};\n\nApp.prototype.showAnnotations = function (annoChoice, show) {\n annoChoice = annoChoice.toUpperCase();\n const self = this;\n let setShown = this.annotationSetsShown.get(annoChoice);\n if (typeof setShown === \"undefined\" && annoChoice !== \"MIFEATURES\") {\n Object(_annotations__WEBPACK_IMPORTED_MODULE_6__[\"fetchAnnotations\"])(annoChoice, this, function () {\n self.annotationSetsShown.set(annoChoice, show);\n self.updateAnnotations();\n });\n } else {\n this.annotationSetsShown.set(annoChoice, show);\n this.updateAnnotations();\n }\n};\n\nApp.prototype.updateAnnotations = function () {\n // //clear all annot's\n for (let mol of this.participants.values()) {\n if (mol.id.indexOf(\"uniprotkb_\") === 0) { //LIMIT IT TO PROTEINS\n mol.clearPositionalFeatures();\n }\n }\n Object(_annotations__WEBPACK_IMPORTED_MODULE_6__[\"chooseColors\"])(this);\n this.colorSchemeChanged();\n\n for (let mol of this.participants.values()) {\n if (mol.id.indexOf(\"uniprotkb_\") === 0) { //LIMIT IT TO PROTEINS\n mol.setPositionalFeatures();\n }\n }\n Object(_annotations__WEBPACK_IMPORTED_MODULE_6__[\"chooseColors\"])(this);\n this.colorSchemeChanged();\n};\n\nApp.prototype.colorSchemeChanged = function () {\n for (let div of this.colorSchemeKeyDivs) {\n _color_scheme_key__WEBPACK_IMPORTED_MODULE_8__[\"update\"](div, this);\n }\n};\n\nApp.prototype.getComplexColors = function () {\n return _viz_link_nary_link__WEBPACK_IMPORTED_MODULE_9__[\"NaryLink\"].naryColors;\n};\n\nApp.prototype.getFeatureColors = function () {\n return this.featureColors;\n};\n\nApp.prototype.collapseAll = function () {\n for (let participant of this.participants.values()) {\n if (participant.form === 1) {\n participant.setForm(0);\n }\n }\n};\n\nApp.prototype.expandAll = function () {\n for (let participant of this.participants.values()) {\n if (participant.form === 0) {\n participant.setForm(1);\n }\n }\n};\n\n//from noe\nApp.prototype.expandAndCollapseSelection = function (moleculesSelected) {\n for (let participant of this.participants.values()) {\n const molecule_id = participant.json.identifier.id;\n if (moleculesSelected.includes(molecule_id)) {\n if (participant.form === 0) {\n participant.setForm(1);\n }\n } else if (participant.form === 1) {\n participant.setForm(0);\n }\n }\n};\n\n// export function makeSymbolKey(targetDiv){\n// new SymbolKey(targetDiv);\n// }//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/js/app.js.js","sources":["webpack://complexviewer/./src/js/app.js?90e9"],"sourcesContent":["// eslint-disable-next-line no-unused-vars\nimport * as css from \"../css/xinet.css\";\nimport {version} from \"../../package.json\";\nimport * as d3 from \"d3\";\nimport * as d3_chromatic from \"d3-scale-chromatic\";\nimport * as cola from \"./cola\";\nimport {readMijson} from \"./read-mijson\";\nimport {chooseColors, fetchAnnotations} from \"./annotations\";\nimport {svgUtils} from \"./svgexp\";\n\n// import {SymbolKey} from \"./symbol-key\";\nimport * as ColorSchemeKey from \"./color-scheme-key\";\nimport {NaryLink} from \"./viz/link/nary-link\";\nimport {svgns} from \"./config\";\n\n// could refactor everything to use ES6 class syntax\n// but https://benmccormick.org/2015/04/07/es6-classes-and-backbone-js\n// \"ES6 classes don’t support adding properties directly to the class instance, only functions/methods\"\n// so backbone doesn't work\n// so continuing to use prototypical inheritance in things for time being\n\nexport function App(/*HTMLDivElement*/networkDiv) {\n    // this.debug = true; // things aren't exactly lined up in the bounding boxes cola is using, to do so breaks symetery of some things\n    this.el = networkDiv;\n\n    this.STATES = {};\n    this.STATES.MOUSE_UP = 0; //start state, also set when mouse up on svgElement\n    this.STATES.PANNING = 1; //set by mouse down on svgElement - left button, no shift or util\n    this.STATES.DRAGGING = 2; //set by mouse down on Protein or Link\n    this.STATES.ROTATING = 3; //set by mouse down on Rotator, drag?\n    this.STATES.SELECTING = 4; //set by mouse down on svgElement- right button or left button shift or util, drag\n\n    //avoids prob with 'save - web page complete'\n    this.el.textContent = \"\"; //https://stackoverflow.com/questions/3955229/remove-all-child-elements-of-a-dom-node-in-javascript\n\n    this.d3cola = cola.d3adaptor();\n\n    const customMenuSel = d3.select(this.el)\n        .append(\"div\").classed(\"custom-menu-margin\", true)\n        .append(\"div\").classed(\"custom-menu\", true)\n        .append(\"ul\");\n\n    const self = this;\n    const collapse = customMenuSel.append(\"li\").classed(\"collapse\", true); //.append(\"button\");\n    collapse.text(\"Collapse\");\n    collapse[0][0].onclick = function (evt) {\n        self.collapseProtein(evt);\n    };\n    const scaleButtonsListItemSel = customMenuSel.append(\"li\").text(\"Scale: \");\n\n    this.barScales = [0.01, 0.2, 1, 2, 4, 8];\n    const scaleButtons = scaleButtonsListItemSel.selectAll(\"ul.custom-menu\")\n        .data(this.barScales)\n        .enter()\n        .append(\"div\")\n        .attr(\"class\", \"barScale\")\n        .append(\"label\");\n    scaleButtons.append(\"span\")\n        .text(function (d) {\n            if (d === 8) return \"AA\";\n            else return d;\n        });\n    scaleButtons.append(\"input\")\n        // .attr (\"id\", function(d) { return d*100; })\n        .attr(\"class\", function (d) {\n            return \"scaleButton scaleButton_\" + (d * 100);\n        })\n        .attr(\"name\", \"scaleButtons\")\n        .attr(\"type\", \"radio\")\n        .on(\"change\", function (d) {\n            self.preventDefaultsAndStopPropagation(d);\n            self.contextMenuProt.setStickScale(d, self.contextMenuPoint);\n        });\n\n    const contextMenu = d3.select(\".custom-menu-margin\").node();\n    contextMenu.onmouseout = function (evt) {\n        let e = evt.relatedTarget;\n        do {\n            if (e === this) return;\n            e = e.parentNode;\n        } while (e);\n        self.contextMenuProt = null;\n        d3.select(this).style(\"display\", \"none\");\n    };\n\n\n    //create SVG element\n    this.svgElement = document.createElementNS(svgns, \"svg\");\n    this.svgElement.setAttribute(\"id\", \"complexViewerSVG\");\n\n    //add listeners\n    this.svgElement.onmousedown = function (evt) {\n        self.mouseDown(evt);\n    };\n    this.svgElement.onmousemove = function (evt) {\n        self.mouseMove(evt);\n    };\n    this.svgElement.onmouseup = function (evt) {\n        self.mouseUp(evt);\n    };\n    this.svgElement.onmouseout = function (evt) {\n        self.hideTooltip(evt);\n    };\n    this.lastMouseUp = new Date().getTime();\n    /*this.svgElement.ontouchstart = function(evt) {\n        self.touchStart(evt);\n    };\n    this.svgElement.ontouchmove = function(evt) {\n        self.touchMove(evt);\n    };\n    this.svgElement.ontouchend = function(evt) {\n        self.touchEnd(evt);\n    };\n    */\n\n    this.el.oncontextmenu = function (evt) {\n        if (evt.preventDefault) { // necessary for addEventListener, works with traditional\n            evt.preventDefault();\n        }\n        evt.returnValue = false; // necessary for attachEvent, works with traditional\n        return false; // works with traditional, not with attachEvent or addEventListener\n    };\n\n    //legend changed callbacks\n    this.colorSchemeKeyDivs = new Set();\n\n    this.el.appendChild(this.svgElement);\n\n    // various groups needed\n    this.container = document.createElementNS(svgns, \"g\");\n    this.container.setAttribute(\"id\", \"container\");\n\n    const svg = d3.select(this.svgElement);\n    this.defs = svg.append(\"defs\");\n\n    this.acknowledgement = document.createElementNS(svgns, \"g\");\n    const ackText = document.createElementNS(svgns, \"text\");\n    ackText.innerHTML = \"<a href='https://academic.oup.com/bioinformatics/article/33/22/3673/4061280' target='_blank'><tspan x='0' dy='1.2em' style='text-decoration: underline'>ComplexViewer \"\n        + version + \"</tspan></a><tspan x='0' dy='1.2em'>by <a href='http://rappsilberlab.org/' target='_blank'>Rappsilber Laboratory</a></tspan>\";\n\n    this.acknowledgement.appendChild(ackText);\n    ackText.setAttribute(\"font-size\", \"8pt\");\n    this.svgElement.appendChild(this.acknowledgement);\n\n    this.naryLinks = document.createElementNS(svgns, \"g\");\n    this.naryLinks.setAttribute(\"id\", \"naryLinks\");\n    this.container.appendChild(this.naryLinks);\n\n    this.p_pLinksWide = document.createElementNS(svgns, \"g\");\n    this.p_pLinksWide.setAttribute(\"id\", \"p_pLinksWide\");\n    this.container.appendChild(this.p_pLinksWide);\n\n    this.highlights = document.createElementNS(svgns, \"g\");\n    this.highlights.setAttribute(\"class\", \"highlights\"); //interactors also contain highlight groups\n    this.container.appendChild(this.highlights);\n\n    this.res_resLinks = document.createElementNS(svgns, \"g\");\n    this.res_resLinks.setAttribute(\"id\", \"res_resLinks\");\n    this.container.appendChild(this.res_resLinks);\n\n    this.p_pLinks = document.createElementNS(svgns, \"g\");\n    this.p_pLinks.setAttribute(\"id\", \"p_pLinks\");\n    this.container.appendChild(this.p_pLinks);\n\n    this.proteinUpper = document.createElementNS(svgns, \"g\");\n    this.proteinUpper.setAttribute(\"id\", \"proteinUpper\");\n    this.container.appendChild(this.proteinUpper);\n\n    this.selfRes_resLinks = document.createElementNS(svgns, \"g\");\n    this.selfRes_resLinks.setAttribute(\"id\", \"res_resLinks\");\n    this.container.appendChild(this.selfRes_resLinks);\n\n    this.svgElement.appendChild(this.container);\n\n    //showing title as tooltips is NOT part of svg spec (even though some browsers do this)\n    //also more responsive / more control if we do out own\n    this.tooltip = document.createElementNS(svgns, \"text\");\n    this.tooltip.setAttribute(\"x\", \"0\");\n    this.tooltip.setAttribute(\"y\", \"0\");\n    const tooltipTextNode = document.createTextNode(\"tooltip\");\n    this.tooltip.classList.add(\"label\", \"tooltip\");\n\n    this.tooltip.appendChild(tooltipTextNode);\n\n    this.tooltip_bg = document.createElementNS(svgns, \"rect\");\n    this.tooltip_bg.classList.add(\"tooltip-background\");\n\n    this.tooltip_subBg = document.createElementNS(svgns, \"rect\");\n    this.tooltip_subBg.classList.add(\"tooltip-sub-background\");\n\n    this.svgElement.appendChild(this.tooltip_subBg);\n    this.svgElement.appendChild(this.tooltip_bg);\n    this.svgElement.appendChild(this.tooltip);\n\n    this.clear();\n}\n\nApp.prototype.createHatchedFill = function (name, color) {\n    if (!this.checkedHatchNames.has(name)) {\n        const pattern = this.defs.append(\"pattern\")\n            .attr(\"id\", name)\n            .attr(\"patternUnits\", \"userSpaceOnUse\")\n            .attr(\"x\", 0)\n            .attr(\"y\", 0)\n            .attr(\"width\", 12)\n            .attr(\"height\", 12)\n            .attr(\"patternTransform\", \"rotate(45)\");\n\n        pattern.append(\"rect\")\n            .attr(\"x\", 0)\n            .attr(\"y\", 2)\n            .attr(\"width\", 12)\n            .attr(\"height\", 4)\n            .attr(\"fill\", color);\n\n        pattern.append(\"rect\")\n            .attr(\"x\", 0)\n            .attr(\"y\", 8)\n            .attr(\"width\", 12)\n            .attr(\"height\", 4)\n            .attr(\"fill\", color);\n\n        this.checkedHatchNames.add(name);\n    }\n};\n\nApp.prototype.clear = function () {\n    this.d3cola.stop();\n\n    this.annotationSetsShown = new Map();\n    // this.annotationSetsShown.set(\"MI FEATURES\", true);\n\n    this.checkedHatchNames = new Set();\n\n    //lighten colors\n    const complexColors = [];\n    for (let c of d3_chromatic.schemePastel2) {//colorbrewer.Pastel2[8]) {\n        const hsl = d3.hsl(c);\n        hsl.l = 0.9;\n        complexColors.push(hsl + \"\");\n    }\n\n    NaryLink.naryColors = d3.scale.ordinal().range(complexColors);\n    this.defs.selectAll(\".feature_checkers\").remove();\n\n    this.naryLinks.textContent = \"\";\n    this.p_pLinksWide.textContent = \"\";\n    this.highlights.textContent = \"\";\n    this.p_pLinks.textContent = \"\";\n    this.res_resLinks.textContent = \"\";\n    this.proteinUpper.textContent = \"\";\n    this.selfRes_resLinks.textContent = \"\";\n\n    // if we are dragging something at the moment - this will be the element that is dragged\n    this.dragElement = null;\n    // from where did we start dragging\n    this.dragStart = {};\n\n    this.participants = new Map();\n    this.allNaryLinks = new Map();\n    this.allBinaryLinks = new Map();\n    this.allUnaryLinks = new Map();\n    this.allSequenceLinks = new Map();\n    this.complexes = [];\n\n    this.proteinCount = 0;\n    this.z = 1;\n    this.hideTooltip();\n    this.state = this.STATES.MOUSE_UP;\n};\n\nApp.prototype.collapseProtein = function () {\n    d3.select(\".custom-menu-margin\").style(\"display\", \"none\");\n    this.contextMenuProt.setForm(0, this.contextMenuPoint);\n    this.contextMenuProt = null;\n};\n\nApp.prototype.init = function () {\n    this.d3cola.stop();\n\n    for (let participant of this.participants.values()) {\n        if (participant.type != \"complex\") {\n            participant.setPosition(-500, -500);\n        }\n    }\n    this.updateAnnotations();\n    this.checkLinks(); //totally needed, not sure why tbh todo - check this out\n    this.setAllLinkCoordinates(); // just to move them off screen at first\n\n    let maxSeqLength = 0;\n    for (let participant of this.participants.values()) {\n        if (participant.size > maxSeqLength) {\n            maxSeqLength = participant.size;\n        }\n    }\n    const width = this.svgElement.parentNode.clientWidth;\n    const defaultPixPerRes = width * 0.8 / maxSeqLength;\n    //console.log(\"defaultPixPerRes:\" + defaultPixPerRes);\n    // https://stackoverflow.com/questions/12141150/from-list-of-integers-get-number-closest-to-a-given-value/12141511#12141511\n    function takeClosest(myList, myNumber) {\n        const bisect = d3.bisector(function (d) {\n            return d;\n        }).left;\n        const pos = bisect(myList, myNumber);\n        if (pos === 0 || pos === 1) {\n            return myList[1]; // don't return smallest scale as default\n        }\n        if (pos === myList.length) {\n            return myList[myList.length - 1];\n        }\n        return myList[pos - 1];\n    }\n\n    this.defaultBarScale = takeClosest(this.barScales, defaultPixPerRes);\n    //console.log(\"default bar scale: \" + this.defaultBarScale)\n\n    for (let participant of this.participants.values()) {\n        if (participant.type !== \"complex\") {\n            this.proteinUpper.appendChild(participant.upperGroup);\n            if (participant.json.type.name === \"protein\") {\n                // participant.initSelfLinkSVG(); // todo - may not even do anything, not sure its working\n                participant.stickZoom = this.defaultBarScale;\n                if (this.participants.size < 4) {\n                    participant.toStickNoTransition();\n                }\n            }\n        }\n    }\n\n    this.autoLayout();\n};\n\nApp.prototype.zoomToExtent = function () {\n    const width = this.svgElement.parentNode.clientWidth;\n    const height = this.svgElement.parentNode.clientHeight;\n    const bbox = this.container.getBBox();\n    let xr = (width / bbox.width).toFixed(4) - 0;\n    let yr = (height / bbox.height).toFixed(4) - 0;\n    let scaleFactor;\n    if (yr < xr) {\n        scaleFactor = yr;\n    } else {\n        scaleFactor = xr;\n    }\n    if (scaleFactor < 1) { ///didn't fit in div\n        //console.log(\"no fit\", scaleFactor);\n        xr = (width - 40) / (bbox.width);\n        yr = (height - 40) / (bbox.height);\n        let scaleFactor;\n        if (yr < xr) {\n            scaleFactor = yr;\n        } else {\n            scaleFactor = xr;\n        }\n\n        if (scaleFactor > this.z) {\n            scaleFactor = this.z;\n        }\n\n        //bbox.x + x = 0;\n        let x = -bbox.x + (20 / scaleFactor);\n        //box.y + y = 0\n        let y = -bbox.y + (20 / scaleFactor);\n        this.container.setAttribute(\"transform\", \"scale(\" + scaleFactor + \") translate(\" + x + \" \" + y + \") \");\n        this.z = this.container.getCTM().inverse().a;\n    } else {\n        //console.log(\"fit\", scaleFactor);\n        // this.container.setAttribute(\"transform\", \"scale(\" + 1 + \") translate(\" + -(width/2) + \" \" + -bbox.y + \")\");\n        const deltaWidth = width - bbox.width;\n        const deltaHeight = height - bbox.height;\n        //bbox.x + x = deltaWidth /2;\n        let x = (deltaWidth / 2) - bbox.x;\n        //box.y + y = deltaHeight / 2\n        let y = (deltaHeight / 2) - bbox.y;\n        this.container.setAttribute(\"transform\", \"scale(\" + 1 + \") translate(\" + x + \" \" + y + \")\");\n        this.z = 1;\n    }\n\n    //todo - following could be tided up by using acknowledgement bbox or positioning att's of text\n    this.acknowledgement.setAttribute(\"transform\", \"translate(\" + (width - 150) + \", \" + (height - 30) + \")\");\n};\n\n//listeners also attached to mouse events by Interactor (and Rotator) and Link, those consume their events\n//mouse down on svgElement must be allowed to propogate (to fire event on Prots/Links)\n\nApp.prototype.mouseDown = function (evt) {\n    //prevent default, but allow propogation\n    evt.preventDefault();\n    this.d3cola.stop();\n    const p = this.getEventPoint(evt); // seems to be correct, see below\n    this.dragStart = this.mouseToSVG(p.x, p.y);\n    return false;\n};\n\n// dragging/rotation/panning/selecting\nApp.prototype.mouseMove = function (evt) {\n    const p = this.getEventPoint(evt); // seems to be correct, see below\n    const c = this.mouseToSVG(p.x, p.y);\n\n    if (this.dragElement != null) { //dragging or rotating\n        this.hideTooltip();\n        const dx = this.dragStart.x - c.x;\n        const dy = this.dragStart.y - c.y;\n\n        if (this.state === this.STATES.DRAGGING) {\n            // we are currently dragging things around\n            let ox, oy, nx, ny;\n            if (!this.dragElement.ix) {\n                for (let participant of this.dragElement.participants) {\n                    participant.changePosition(dx, dy);\n                }\n                this.setAllLinkCoordinates();\n            } else {\n                ox = this.dragElement.ix;\n                oy = this.dragElement.iy;\n                nx = ox - dx;\n                ny = oy - dy;\n                this.dragElement.setPosition(nx, ny);\n                this.dragElement.setAllLinkCoordinates();\n            }\n            this.dragStart = c;\n        } else { //not dragging or rotating yet, maybe we should start\n            // don't start dragging just on a click - we need to move the mouse a bit first\n            if (Math.sqrt(dx * dx + dy * dy) > (5 * this.z)) {\n                this.state = this.STATES.DRAGGING;\n\n            }\n        }\n    } else {\n        this.showTooltip(p);\n    }\n    return false;\n};\n\n// this ends all dragging and rotating\nApp.prototype.mouseUp = function (evt) {\n    const time = new Date().getTime();\n    //console.log(\"Mouse up: \" + evt.srcElement + \" \" + (time - this.lastMouseUp));\n    this.preventDefaultsAndStopPropagation(evt);\n    //eliminate some spurious mouse up events\n    if ((time - this.lastMouseUp) > 150) {\n\n        const p = this.getEventPoint(evt); // seems to be correct, see below\n        const c = this.mouseToSVG(p.x, p.y);\n\n        if (this.dragElement && this.dragElement.type === \"protein\") { /// todo be consistent about how to check if thing is protein\n            if (!(this.state === this.STATES.DRAGGING || this.state === this.STATES.ROTATING)) { //not dragging or rotating\n                if (this.dragElement.form === 0) {\n                    this.dragElement.setForm(1);\n                } else {\n                    this.contextMenuProt = this.dragElement;\n                    this.contextMenuPoint = c;\n                    const menu = d3.select(\".custom-menu-margin\");\n                    menu.style(\"top\", (evt.pageY - 20) + \"px\").style(\"left\", (evt.pageX - 20) + \"px\").style(\"display\", \"block\");\n                    d3.select(\".scaleButton_\" + (this.dragElement.stickZoom * 100)).property(\"checked\", true);\n                }\n            }\n        }\n    }\n\n    this.dragElement = null;\n    this.state = this.STATES.MOUSE_UP;\n\n    this.lastMouseUp = time;\n    return false;\n};\n\n//gets mouse position\nApp.prototype.getEventPoint = function (evt) {\n    const p = this.svgElement.createSVGPoint();\n    let element = this.svgElement.parentNode;\n    let top = 0,\n        left = 0;\n    do {\n        top += element.offsetTop || 0;\n        left += element.offsetLeft || 0;\n        element = element.offsetParent;\n    } while (element);\n    p.x = evt.pageX - left;\n    p.y = evt.pageY - top;\n    return p;\n};\n\n//stop event propogation and defaults; only do what we ask\nApp.prototype.preventDefaultsAndStopPropagation = function (evt) {\n    if (evt.stopPropagation)\n        evt.stopPropagation();\n    if (evt.cancelBubble != null)\n        evt.cancelBubble = true;\n    if (evt.preventDefault)\n        evt.preventDefault();\n};\n\n/**\n * Handle touchstart event.\n\n App.prototype.touchStart = function(evt) {\n    //prevent default, but allow propogation\n    evt.preventDefault();\n\n    //stop force layout\n    if (typeof this.d3cola !== 'undefined' && this.d3cola != null) {\n        this.d3cola.stop();\n    }\n\n    var p = this.getTouchEventPoint(evt); // seems to be correct, see below\n    this.dragStart = this.mouseToSVG(p.x, p.y);\n};\n\n // dragging/rotation/panning/selecting\n App.prototype.touchMove = function(evt) {\n    // if (this.sequenceInitComplete) { // just being cautious\n    var p = this.getTouchEventPoint(evt); // seems to be correct, see below\n    var c = this.mouseToSVG(p.x, p.y);\n\n    if (this.dragElement != null) { //dragging or rotating\n        this.hideTooltip();\n        var dx = this.dragStart.x - c.x;\n        var dy = this.dragStart.y - c.y;\n\n        if (this.state === this.STATES.DRAGGING) {\n            // we are currently dragging things around\n            var ox, oy, nx, ny;\n            if (typeof this.dragElement.ix=== 'undefined') { // if not an Interactor\n                var nodes = this.dragElement.interactors;\n                var nodeCount = nodes.length;\n                for (var i = 0; i < nodeCount; i++) {\n                    var protein = nodes[i];\n                    ox = protein.cx;\n                    oy = protein.cy;\n                    nx = ox - dx;\n                    ny = oy - dy;\n                    protein.setPosition(nx, ny);\n                    protein.setAllLinkCoordinates();\n                }\n                for (i = 0; i < nodeCount; i++) {\n                    nodes[i].setAllLinkCoordinates();\n                }\n            } else {\n                ox = this.dragElement.cx;\n                oy = this.dragElement.cy;\n                nx = ox - dx;\n                ny = oy - dy;\n                this.dragElement.setPosition(nx, ny);\n                this.dragElement.setAllLinkCoordinates();\n            }\n            this.dragStart = c;\n        } else { //not dragging or rotating yet, maybe we should start\n            // don't start dragging just on a click - we need to move the mouse a bit first\n            if (Math.sqrt(dx * dx + dy * dy) > (5 * this.z)) {\n                this.state = this.STATES.DRAGGING;\n\n            }\n        }\n    } else {\n        this.showTooltip(p);\n    }\n    return false;\n};\n\n// this ends all dragging and rotating\nApp.prototype.touchEnd = function(evt) {\n    var time = new Date().getTime();\n    //console.log(\"Mouse up: \" + evt.srcElement + \" \" + (time - this.lastMouseUp));\n    this.preventDefaultsAndStopPropagation(evt);\n    //eliminate some spurious mouse up events\n    if ((time - this.lastMouseUp) > 150) {\n\n        var p = this.getTouchEventPoint(evt); // seems to be correct, see below\n        var c = this.mouseToSVG(p.x, p.y);\n\n        if (this.dragElement != null) {\n            if (!(this.state === this.STATES.DRAGGING || this.state === this.STATES.ROTATING)) { //not dragging or rotating\n                if (this.dragElement.form === 0) {\n                    this.dragElement.setForm(1);\n                } else {\n                    this.contextMenuProt = this.dragElement;\n                    this.contextMenuPoint = c;\n                    var menu = d3.select(\".custom-menu-margin\")\n                    menu.style(\"top\", (evt.pageY - 20) + \"px\").style(\"left\", (evt.pageX - 20) + \"px\").style(\"display\", \"block\");\n                    d3.select(\".scaleButton_\" + (this.dragElement.stickZoom * 100)).property(\"checked\", true)\n                }\n            }\n        }\n    }\n\n    this.dragElement = null;\n    this.whichRotator = -1;\n    this.state = this.STATES.MOUSE_UP;\n\n    this.lastMouseUp = time;\n    return false;\n};\n\n//gets mouse position\nApp.prototype.getTouchEventPoint = function(evt) {\n    var p = this.svgElement.createSVGPoint();\n    var element = this.svgElement.parentNode;\n    var top = 0,\n        left = 0;\n    do {\n        top += element.offsetTop || 0;\n        left += element.offsetLeft || 0;\n        element = element.offsetParent;\n    } while (element);\n    p.x = evt.touches[0].pageX - left;\n    p.y = evt.touches[0].pageY - top;\n    return p;\n};\n */\nApp.prototype.autoLayout = function () {\n    this.d3cola.stop();\n    const self = this;\n\n    // needed to ensure consistent results\n    for (let p of self.participants.values()) {\n        delete p.x;\n        delete p.y;\n        delete p.px;\n        delete p.py;\n        delete p.bounds;\n        p.fixed = 0;\n    }\n\n    //// prune leaves from network then layout, then add back leaves and layout again (fixes haemoglobin)\n    const pruned = [];\n    for (let participant of self.participants.values()) {\n        if (participant.binaryLinks.size > 2 && participant.type !== \"complex\") {\n            pruned.push(participant);\n        }\n    }\n    const allNodesExceptComplexes = Array.from(self.participants.values()).filter(function (value) {\n        return value.type !== \"complex\";\n    });\n\n    if (pruned.length < allNodesExceptComplexes.length\n        && pruned.length > 3 && self.participants.size < 9) {\n        // <9 include hemoglobin, possibly some other small cases, but is catious, tends to mess other things up\n        // console.log(prunedIn);\n        doLayout(pruned, true);\n    } else {\n        doLayout(allNodesExceptComplexes, self.complexes.length > 0);\n    }\n\n    function doLayout(nodes, preRun) {\n        const layoutObj = {}; // todo get rid\n        layoutObj.nodes = nodes;\n        layoutObj.links = [];\n\n        const molLookUp = {};\n        let mi = 0;\n        for (let mol of nodes) {\n            molLookUp[mol.id] = mi;\n            mi++;\n        }\n\n        for (let binaryLink of self.allBinaryLinks.values()) {\n            const fromMol = binaryLink.participants[0];\n            const toMol = binaryLink.participants[1];\n            // if (preRun || (fromMol.binaryLinks.size === 4 || toMol.binaryLinks.size == 4)) {\n            const source = fromMol; //molLookUp[fromMol.id];\n            const target = toMol; //molLookUp[toMol.id];\n\n            if (source !== target && nodes.indexOf(source) !== -1 && nodes.indexOf(target) !== -1) { // todo - check what this is doing\n                const linkObj = {};\n                linkObj.source = molLookUp[fromMol.id];\n                linkObj.target = molLookUp[toMol.id];\n                linkObj.id = binaryLink.id;\n                layoutObj.links.push(linkObj);\n            }\n            // }\n        }\n\n        const groups = [];\n        if (!preRun && self.complexes) {\n            for (let g of self.complexes) {\n                g.leaves = [];\n                g.groups = [];\n                for (let interactor of g.naryLink.participants) {\n                    if (interactor.type !== \"complex\") {\n                        g.leaves.push(layoutObj.nodes.indexOf(interactor));\n                    }\n                }\n                groups.push(g);\n            }\n            for (let g of self.complexes) {\n                for (let interactor of g.naryLink.participants) {\n                    if (interactor.type === \"complex\") {\n                        //console.log(groups.indexOf(interactor));\n                        g.groups.push(groups.indexOf(interactor));\n                    }\n                }\n            }\n        }\n\n        //console.log(\"groups\", groups);\n        delete self.d3cola._lastStress;\n        delete self.d3cola._alpha;\n        delete self.d3cola._descent;\n        delete self.d3cola._rootGroup;\n\n        let linkLength = (nodes.length < 30) ? 30 : 20;\n        const width = self.svgElement.parentNode.clientWidth;\n        const height = self.svgElement.parentNode.clientHeight;\n        //console.log(\"**\", layoutObj);\n        self.d3cola.size([height - 40, width - 40])\n            .nodes(layoutObj.nodes).groups(groups).links(layoutObj.links).avoidOverlaps(true);\n        let groupDebugSel, participantDebugSel;\n        if (self.debug) {\n            groupDebugSel = d3.select(self.container).selectAll(\".group\")\n                .data(groups);\n\n            groupDebugSel.enter().append(\"rect\")\n                .classed(\"group\", true)\n                .attr({\n                    rx: 5,\n                    ry: 5\n                })\n                .style(\"stroke\", \"blue\")\n                .style(\"fill\", \"none\");\n\n            participantDebugSel = d3.select(self.container).selectAll(\".node\")\n                .data(layoutObj.nodes);\n\n            participantDebugSel.enter().append(\"rect\")\n                .classed(\"node\", true)\n                .attr({\n                    rx: 5,\n                    ry: 5\n                })\n                .style(\"stroke\", \"red\")\n                .style(\"fill\", \"none\");\n\n            groupDebugSel.exit().remove();\n            participantDebugSel.exit().remove();\n        }\n\n        const startTime = Date.now();\n        self.d3cola.symmetricDiffLinkLengths(linkLength)\n            .on(\"tick\", function () {\n                if (Date.now() - startTime > 750) {//!preRun) {\n                    const nodes = self.d3cola.nodes();\n                    for (let node of nodes) {\n                        node.setPosition(node.x, node.y);\n                    }\n                    self.setAllLinkCoordinates();\n                    self.zoomToExtent();\n                    if (self.debug) {\n                        groupDebugSel.attr({\n                            x: function (d) {\n                                return d.bounds.x;// + (width / 2);\n                            },\n                            y: function (d) {\n                                return d.bounds.y;// + (height / 2);\n                            },\n                            width: function (d) {\n                                return d.bounds.width();\n                            },\n                            height: function (d) {\n                                return d.bounds.height();\n                            }\n                        });\n\n                        participantDebugSel.attr({\n                            x: function (d) {\n                                return d.bounds.x;// + (width / 2);\n                            },\n                            y: function (d) {\n                                return d.bounds.y;// + (height / 2);\n                            },\n                            width: function (d) {\n                                return d.bounds.width();\n                            },\n                            height: function (d) {\n                                return d.bounds.height();\n                            }\n                        });\n                    }\n                }\n            })\n            .on(\"end\", function () {\n                if (preRun) {\n                    // alert(\"initial run complete\");\n                    //     // for (let p of layoutObj.nodes) {\n                    //     //         p.fixed = 1;\n                    //     // }\n                    doLayout(allNodesExceptComplexes, false);\n                } else {\n                    for (let node of nodes) {\n                        node.setPosition(node.x, node.y);\n                    }\n                    self.setAllLinkCoordinates();\n                    self.zoomToExtent();\n                }\n            });\n        if (preRun) {\n            self.d3cola.start(23, 23, 0, 0, true);//, false, false);\n        } else {\n            self.d3cola.start(0, 23, 23, 0, true);//, false, false);\n        }\n    }\n};\n\nApp.prototype.getSVG = function () { //todo - somewhat broken, annotations missing\n    var svgSel = d3.select(this.el).selectAll(\"svg\");\n    var svgArr = [svgSel.node()];\n    var svgStrings = svgUtils.capture(svgArr);\n    var svgXML = svgUtils.makeXMLStr(new XMLSerializer(), svgStrings[0]);\n\n    return svgXML;\n\n    // var fileName = this.filenameStateString().substring(0, 240);\n    // download(svgXML, 'application/svg', fileName + \".svg\");\n\n};\n\n// App.prototype.getSVG = function () {\n//     let svgXml = this.svgElement.outerHTML.replace(/<rect .*?\\/rect>/i, \"\"); //take out white background fill\n//     const viewBox = \"viewBox=\\\"0 0 \" + this.svgElement.parentNode.clientWidth + \" \" + this.svgElement.parentNode.clientHeight + \"\\\" \";\n//     svgXml = svgXml.replace(\"<svg \", \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" xmlns:xlink=\\\"http://www.w3.org/1999/xlink\\\" xmlns:ev=\\\"http://www.w3.org/2001/xml-events\\\" \" + viewBox);\n//\n//     return \"<?xml version=\\\"1.0\\\" encoding=\\\"UTF-8\\\" standalone=\\\"no\\\"?>\" +\n//         \"<!DOCTYPE svg PUBLIC \\\"-//W3C//DTD SVG 1.1//EN\\\" \\\"http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd\\\">\" +\n//         svgXml;\n// };\n\n// transform the mouse-position into a position on the svg\nApp.prototype.mouseToSVG = function (x, y) {\n    const p = this.svgElement.createSVGPoint();\n    p.x = x;\n    p.y = y;\n    return p.matrixTransform(this.container.getCTM().inverse());\n};\n\n// reads MI JSON format\nApp.prototype.readMIJSON = function (miJson, expand = true) {\n    readMijson(miJson, this, expand);\n};\n\nApp.prototype.checkLinks = function () {\n    for (let link of this.allNaryLinks.values()) {\n        link.check();\n    }\n    for (let link of this.allBinaryLinks.values()) {\n        link.check();\n    }\n    for (let link of this.allUnaryLinks.values()) {\n        link.check();\n    }\n    for (let link of this.allSequenceLinks.values()) {\n        link.check();\n    }\n};\n\nApp.prototype.setAllLinkCoordinates = function () {\n    for (let link of this.allNaryLinks.values()) {\n        link.setLinkCoordinates();\n    }\n    for (let link of this.allBinaryLinks.values()) {\n        link.setLinkCoordinates();\n    }\n    for (let link of this.allUnaryLinks.values()) {\n        link.setLinkCoordinates();\n    }\n    for (let link of this.allSequenceLinks.values()) {\n        link.setLinkCoordinates();\n    }\n};\n\nApp.prototype.showTooltip = function (p) {\n    let ttX, ttY;\n    const length = this.tooltip.getComputedTextLength() + 16;\n    const width = this.svgElement.parentNode.clientWidth;\n    const height = this.svgElement.parentNode.clientHeight;\n    if (p.x + 20 + length < width) {\n        ttX = p.x;\n    } else {\n        ttX = width - length - 20;\n    }\n\n    if (p.y + 60 < height) {\n        ttY = p.y;\n    } else {\n        ttY = height - 60;\n    }\n    this.tooltip.setAttribute(\"x\", ttX + 22);\n    this.tooltip.setAttribute(\"y\", ttY + 47);\n    this.tooltip_bg.setAttribute(\"x\", ttX + 16);\n    this.tooltip_bg.setAttribute(\"y\", ttY + 28);\n    this.tooltip_subBg.setAttribute(\"x\", ttX + 16);\n    this.tooltip_subBg.setAttribute(\"y\", ttY + 28);\n};\n\nApp.prototype.setTooltip = function (text, color) {\n    if (text) {\n        this.tooltip.firstChild.data = text.toString().replace(/&(quot);/g, \"\\\"\");\n        this.tooltip.setAttribute(\"display\", \"block\");\n        const length = this.tooltip.getComputedTextLength();\n        this.tooltip_bg.setAttribute(\"width\", length + 16);\n        this.tooltip_subBg.setAttribute(\"width\", length + 16);\n        if (typeof color !== \"undefined\" && color != null) {\n            this.tooltip_bg.setAttribute(\"fill\", color);\n            this.tooltip_bg.setAttribute(\"stroke\", color);\n            this.tooltip_bg.setAttribute(\"fill-opacity\", \"0.5\");\n        } else {\n            this.tooltip_bg.setAttribute(\"fill\", \"white\");\n            this.tooltip_bg.setAttribute(\"stroke\", \"grey\");\n        }\n        // todo - whats this height for?\n        this.tooltip_bg.setAttribute(\"height\", \"28\");\n        this.tooltip_subBg.setAttribute(\"height\", \"28\");\n        this.tooltip_bg.setAttribute(\"display\", \"block\");\n        this.tooltip_subBg.setAttribute(\"display\", \"block\");\n    } else {\n        this.hideTooltip();\n    }\n};\n\nApp.prototype.hideTooltip = function () {\n    this.tooltip.setAttribute(\"display\", \"none\");\n    this.tooltip_bg.setAttribute(\"display\", \"none\");\n    this.tooltip_subBg.setAttribute(\"display\", \"none\");\n};\n\nApp.prototype.addColorSchemeKey = function (/*HTMLDivElement*/ div) {\n    this.colorSchemeKeyDivs.add(div);\n    ColorSchemeKey.update(div, this);\n};\n\nApp.prototype.removeColorSchemeKey = function (/*HTMLDivElement*/ colorSchemeKeyDiv) {\n    this.colorSchemeKeyDivs.remove(colorSchemeKeyDiv);\n    colorSchemeKeyDiv.textContent = \"\";\n};\n\n//for backwards compatibility (noe?), tbh i might have made a bit of a mess here\nApp.prototype.setAnnotations = function (annoChoice) {\n    annoChoice = annoChoice.toUpperCase();\n    for (let annoType of this.annotationSetsShown.keys()) {\n        this.showAnnotations(annoType, annoChoice === annoType);\n    }\n    this.showAnnotations(annoChoice, true);\n};\n\nApp.prototype.showAnnotations = function (annoChoice, show) {\n    annoChoice = annoChoice.toUpperCase();\n    const self = this;\n    let setShown = this.annotationSetsShown.get(annoChoice);\n    if (typeof setShown === \"undefined\" && annoChoice !== \"MIFEATURES\") {\n        fetchAnnotations(annoChoice, this, function () {\n            self.annotationSetsShown.set(annoChoice, show);\n            self.updateAnnotations();\n        });\n    } else {\n        this.annotationSetsShown.set(annoChoice, show);\n        this.updateAnnotations();\n    }\n};\n\nApp.prototype.updateAnnotations = function () {\n    // //clear all annot's\n    for (let mol of this.participants.values()) {\n        if (mol.id.indexOf(\"uniprotkb_\") === 0) { //LIMIT IT TO PROTEINS\n            mol.clearPositionalFeatures();\n        }\n    }\n    chooseColors(this);\n    this.colorSchemeChanged();\n\n    for (let mol of this.participants.values()) {\n        if (mol.id.indexOf(\"uniprotkb_\") === 0) { //LIMIT IT TO PROTEINS\n            mol.setPositionalFeatures();\n        }\n    }\n    chooseColors(this);\n    this.colorSchemeChanged();\n};\n\nApp.prototype.colorSchemeChanged = function () {\n    for (let div of this.colorSchemeKeyDivs) {\n        ColorSchemeKey.update(div, this);\n    }\n};\n\nApp.prototype.getComplexColors = function () {\n    return NaryLink.naryColors;\n};\n\nApp.prototype.getFeatureColors = function () {\n    return this.featureColors;\n};\n\nApp.prototype.collapseAll = function () {\n    for (let participant of this.participants.values()) {\n        if (participant.form === 1) {\n            participant.setForm(0);\n        }\n    }\n};\n\nApp.prototype.expandAll = function () {\n    for (let participant of this.participants.values()) {\n        if (participant.form === 0) {\n            participant.setForm(1);\n        }\n    }\n};\n\n//from noe\nApp.prototype.expandAndCollapseSelection = function (moleculesSelected) {\n    for (let participant of this.participants.values()) {\n        const molecule_id = participant.json.identifier.id;\n        if (moleculesSelected.includes(molecule_id)) {\n            if (participant.form === 0) {\n                participant.setForm(1);\n            }\n        } else if (participant.form === 1) {\n            participant.setForm(0);\n        }\n    }\n};\n\n// export function makeSymbolKey(targetDiv){\n//     new SymbolKey(targetDiv);\n// }"],"mappings":"AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/js/app.js\n"); /***/ }), @@ -1165,7 +1165,7 @@ eval("var require;var require;(function(f){if(true){module.exports=f()}else { va /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"update\", function() { return update; });\n//import * as RGBColor from \"rgbcolor\";\n\nfunction update(/*HTMLDivElement*/ div, /*App*/app) {\n div.textContent = \"\";\n\n const complexColorScheme = app.getComplexColors();\n const complexColorTable = document.createElement(\"table\");\n complexColorTable.classList.add(\"color_key\", \"complex_colors\");\n const th = complexColorTable.createTHead();\n th.textContent = \"Complexes\";\n const ccDomain = complexColorScheme.domain();\n const ccRange = complexColorScheme.range();\n for (let i = 0; i < ccDomain.length; i++) {\n const tr = complexColorTable.insertRow();\n const tc1 = tr.insertCell();\n tc1.style.backgroundColor = ccRange[i % 6];\n const tc2 = tr.insertCell();\n tc2.textContent = ccDomain[i];\n console.log(i + \" \" + ccDomain[i] + \" \" + ccRange[i]);\n }\n div.appendChild(complexColorTable);\n\n const featureColorScheme = app.getFeatureColors();\n if (featureColorScheme) {\n const featureColorTable = document.createElement(\"table\");\n featureColorTable.classList.add(\"color_key\", \"feature_colors\");\n const th2 = featureColorTable.createTHead();\n th2.textContent = \"Features\";\n const domain = featureColorScheme.domain();\n const range = featureColorScheme.range();\n for (let i = 0; i < domain.length; i++) {\n const tr = featureColorTable.insertRow();\n const tc1 = tr.insertCell();\n // make transparent version of color\n //const temp = new RGBColor(range[i % 20]).;\n tc1.style.backgroundColor = range[i % 20];//\"rgba(\" + temp.r + \",\" + temp.g + \",\" + temp.b + \", 0.6)\";\n const tc2 = tr.insertCell();\n tc2.textContent = domain[i];\n }\n div.appendChild(featureColorTable);\n }\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvY29sb3Itc2NoZW1lLWtleS5qcy5qcyIsInNvdXJjZXMiOlsid2VicGFjazovL2NvbXBsZXh2aWV3ZXIvLi9zcmMvanMvY29sb3Itc2NoZW1lLWtleS5qcz8wYTllIl0sInNvdXJjZXNDb250ZW50IjpbIi8vaW1wb3J0ICogYXMgUkdCQ29sb3IgZnJvbSBcInJnYmNvbG9yXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiB1cGRhdGUoLypIVE1MRGl2RWxlbWVudCovIGRpdiwgLypBcHAqL2FwcCkge1xuICAgIGRpdi50ZXh0Q29udGVudCA9IFwiXCI7XG5cbiAgICBjb25zdCBjb21wbGV4Q29sb3JTY2hlbWUgPSBhcHAuZ2V0Q29tcGxleENvbG9ycygpO1xuICAgIGNvbnN0IGNvbXBsZXhDb2xvclRhYmxlID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcInRhYmxlXCIpO1xuICAgIGNvbXBsZXhDb2xvclRhYmxlLmNsYXNzTGlzdC5hZGQoXCJjb2xvcl9rZXlcIiwgXCJjb21wbGV4X2NvbG9yc1wiKTtcbiAgICBjb25zdCB0aCA9IGNvbXBsZXhDb2xvclRhYmxlLmNyZWF0ZVRIZWFkKCk7XG4gICAgdGgudGV4dENvbnRlbnQgPSBcIkNvbXBsZXhlc1wiO1xuICAgIGNvbnN0IGNjRG9tYWluID0gY29tcGxleENvbG9yU2NoZW1lLmRvbWFpbigpO1xuICAgIGNvbnN0IGNjUmFuZ2UgPSBjb21wbGV4Q29sb3JTY2hlbWUucmFuZ2UoKTtcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IGNjRG9tYWluLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgIGNvbnN0IHRyID0gY29tcGxleENvbG9yVGFibGUuaW5zZXJ0Um93KCk7XG4gICAgICAgIGNvbnN0IHRjMSA9IHRyLmluc2VydENlbGwoKTtcbiAgICAgICAgdGMxLnN0eWxlLmJhY2tncm91bmRDb2xvciA9IGNjUmFuZ2VbaSAlIDZdO1xuICAgICAgICBjb25zdCB0YzIgPSB0ci5pbnNlcnRDZWxsKCk7XG4gICAgICAgIHRjMi50ZXh0Q29udGVudCA9IGNjRG9tYWluW2ldO1xuICAgICAgICBjb25zb2xlLmxvZyhpICsgXCIgXCIgKyBjY0RvbWFpbltpXSArIFwiIFwiICsgY2NSYW5nZVtpXSk7XG4gICAgfVxuICAgIGRpdi5hcHBlbmRDaGlsZChjb21wbGV4Q29sb3JUYWJsZSk7XG5cbiAgICBjb25zdCBmZWF0dXJlQ29sb3JTY2hlbWUgPSBhcHAuZ2V0RmVhdHVyZUNvbG9ycygpO1xuICAgIGlmIChmZWF0dXJlQ29sb3JTY2hlbWUpIHtcbiAgICAgICAgY29uc3QgZmVhdHVyZUNvbG9yVGFibGUgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwidGFibGVcIik7XG4gICAgICAgIGZlYXR1cmVDb2xvclRhYmxlLmNsYXNzTGlzdC5hZGQoXCJjb2xvcl9rZXlcIiwgXCJmZWF0dXJlX2NvbG9yc1wiKTtcbiAgICAgICAgY29uc3QgdGgyID0gZmVhdHVyZUNvbG9yVGFibGUuY3JlYXRlVEhlYWQoKTtcbiAgICAgICAgdGgyLnRleHRDb250ZW50ID0gXCJGZWF0dXJlc1wiO1xuICAgICAgICBjb25zdCBkb21haW4gPSBmZWF0dXJlQ29sb3JTY2hlbWUuZG9tYWluKCk7XG4gICAgICAgIGNvbnN0IHJhbmdlID0gZmVhdHVyZUNvbG9yU2NoZW1lLnJhbmdlKCk7XG4gICAgICAgIGZvciAobGV0IGkgPSAwOyBpIDwgZG9tYWluLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgICAgICBjb25zdCB0ciA9IGZlYXR1cmVDb2xvclRhYmxlLmluc2VydFJvdygpO1xuICAgICAgICAgICAgY29uc3QgdGMxID0gdHIuaW5zZXJ0Q2VsbCgpO1xuICAgICAgICAgICAgLy8gbWFrZSB0cmFuc3BhcmVudCB2ZXJzaW9uIG9mIGNvbG9yXG4gICAgICAgICAgICAvL2NvbnN0IHRlbXAgPSBuZXcgUkdCQ29sb3IocmFuZ2VbaSAlIDIwXSkuO1xuICAgICAgICAgICAgdGMxLnN0eWxlLmJhY2tncm91bmRDb2xvciA9IHJhbmdlW2kgJSAyMF07Ly9cInJnYmEoXCIgKyB0ZW1wLnIgKyBcIixcIiArIHRlbXAuZyArIFwiLFwiICsgdGVtcC5iICsgXCIsIDAuNilcIjtcbiAgICAgICAgICAgIGNvbnN0IHRjMiA9IHRyLmluc2VydENlbGwoKTtcbiAgICAgICAgICAgIHRjMi50ZXh0Q29udGVudCA9IGRvbWFpbltpXTtcbiAgICAgICAgfVxuICAgICAgICBkaXYuYXBwZW5kQ2hpbGQoZmVhdHVyZUNvbG9yVGFibGUpO1xuICAgIH1cbn0iXSwibWFwcGluZ3MiOiJBQUFBO0FBQUE7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./src/js/color-scheme-key.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"update\", function() { return update; });\n//import * as RGBColor from \"rgbcolor\";\n\nfunction update(/*HTMLDivElement*/ div, /*App*/app) {\n div.textContent = \"\";\n\n const complexColorScheme = app.getComplexColors();\n const complexColorTable = document.createElement(\"table\");\n complexColorTable.classList.add(\"color_key\", \"complex_colors\");\n const th = complexColorTable.createTHead();\n th.textContent = \"Complexes\";\n const ccDomain = complexColorScheme.domain();\n const ccRange = complexColorScheme.range();\n for (let i = 0; i < ccDomain.length; i++) {\n const tr = complexColorTable.insertRow();\n const tc1 = tr.insertCell();\n tc1.style.backgroundColor = ccRange[i % 6];\n const tc2 = tr.insertCell();\n tc2.textContent = ccDomain[i];\n //console.log(i + \" \" + ccDomain[i] + \" \" + ccRange[i]);\n }\n div.appendChild(complexColorTable);\n\n const featureColorScheme = app.getFeatureColors();\n if (featureColorScheme) {\n const featureColorTable = document.createElement(\"table\");\n featureColorTable.classList.add(\"color_key\", \"feature_colors\");\n const th2 = featureColorTable.createTHead();\n th2.textContent = \"Features\";\n const domain = featureColorScheme.domain();\n const range = featureColorScheme.range();\n for (let i = 0; i < domain.length; i++) {\n const tr = featureColorTable.insertRow();\n const tc1 = tr.insertCell();\n // make transparent version of color\n //const temp = new RGBColor(range[i % 20]).;\n tc1.style.backgroundColor = range[i % 20];//\"rgba(\" + temp.r + \",\" + temp.g + \",\" + temp.b + \", 0.6)\";\n const tc2 = tr.insertCell();\n tc2.textContent = domain[i];\n }\n div.appendChild(featureColorTable);\n }\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvY29sb3Itc2NoZW1lLWtleS5qcy5qcyIsInNvdXJjZXMiOlsid2VicGFjazovL2NvbXBsZXh2aWV3ZXIvLi9zcmMvanMvY29sb3Itc2NoZW1lLWtleS5qcz8wYTllIl0sInNvdXJjZXNDb250ZW50IjpbIi8vaW1wb3J0ICogYXMgUkdCQ29sb3IgZnJvbSBcInJnYmNvbG9yXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiB1cGRhdGUoLypIVE1MRGl2RWxlbWVudCovIGRpdiwgLypBcHAqL2FwcCkge1xuICAgIGRpdi50ZXh0Q29udGVudCA9IFwiXCI7XG5cbiAgICBjb25zdCBjb21wbGV4Q29sb3JTY2hlbWUgPSBhcHAuZ2V0Q29tcGxleENvbG9ycygpO1xuICAgIGNvbnN0IGNvbXBsZXhDb2xvclRhYmxlID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudChcInRhYmxlXCIpO1xuICAgIGNvbXBsZXhDb2xvclRhYmxlLmNsYXNzTGlzdC5hZGQoXCJjb2xvcl9rZXlcIiwgXCJjb21wbGV4X2NvbG9yc1wiKTtcbiAgICBjb25zdCB0aCA9IGNvbXBsZXhDb2xvclRhYmxlLmNyZWF0ZVRIZWFkKCk7XG4gICAgdGgudGV4dENvbnRlbnQgPSBcIkNvbXBsZXhlc1wiO1xuICAgIGNvbnN0IGNjRG9tYWluID0gY29tcGxleENvbG9yU2NoZW1lLmRvbWFpbigpO1xuICAgIGNvbnN0IGNjUmFuZ2UgPSBjb21wbGV4Q29sb3JTY2hlbWUucmFuZ2UoKTtcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IGNjRG9tYWluLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgIGNvbnN0IHRyID0gY29tcGxleENvbG9yVGFibGUuaW5zZXJ0Um93KCk7XG4gICAgICAgIGNvbnN0IHRjMSA9IHRyLmluc2VydENlbGwoKTtcbiAgICAgICAgdGMxLnN0eWxlLmJhY2tncm91bmRDb2xvciA9IGNjUmFuZ2VbaSAlIDZdO1xuICAgICAgICBjb25zdCB0YzIgPSB0ci5pbnNlcnRDZWxsKCk7XG4gICAgICAgIHRjMi50ZXh0Q29udGVudCA9IGNjRG9tYWluW2ldO1xuICAgICAgICAvL2NvbnNvbGUubG9nKGkgKyBcIiBcIiArIGNjRG9tYWluW2ldICsgXCIgXCIgKyBjY1JhbmdlW2ldKTtcbiAgICB9XG4gICAgZGl2LmFwcGVuZENoaWxkKGNvbXBsZXhDb2xvclRhYmxlKTtcblxuICAgIGNvbnN0IGZlYXR1cmVDb2xvclNjaGVtZSA9IGFwcC5nZXRGZWF0dXJlQ29sb3JzKCk7XG4gICAgaWYgKGZlYXR1cmVDb2xvclNjaGVtZSkge1xuICAgICAgICBjb25zdCBmZWF0dXJlQ29sb3JUYWJsZSA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJ0YWJsZVwiKTtcbiAgICAgICAgZmVhdHVyZUNvbG9yVGFibGUuY2xhc3NMaXN0LmFkZChcImNvbG9yX2tleVwiLCBcImZlYXR1cmVfY29sb3JzXCIpO1xuICAgICAgICBjb25zdCB0aDIgPSBmZWF0dXJlQ29sb3JUYWJsZS5jcmVhdGVUSGVhZCgpO1xuICAgICAgICB0aDIudGV4dENvbnRlbnQgPSBcIkZlYXR1cmVzXCI7XG4gICAgICAgIGNvbnN0IGRvbWFpbiA9IGZlYXR1cmVDb2xvclNjaGVtZS5kb21haW4oKTtcbiAgICAgICAgY29uc3QgcmFuZ2UgPSBmZWF0dXJlQ29sb3JTY2hlbWUucmFuZ2UoKTtcbiAgICAgICAgZm9yIChsZXQgaSA9IDA7IGkgPCBkb21haW4ubGVuZ3RoOyBpKyspIHtcbiAgICAgICAgICAgIGNvbnN0IHRyID0gZmVhdHVyZUNvbG9yVGFibGUuaW5zZXJ0Um93KCk7XG4gICAgICAgICAgICBjb25zdCB0YzEgPSB0ci5pbnNlcnRDZWxsKCk7XG4gICAgICAgICAgICAvLyBtYWtlIHRyYW5zcGFyZW50IHZlcnNpb24gb2YgY29sb3JcbiAgICAgICAgICAgIC8vY29uc3QgdGVtcCA9IG5ldyBSR0JDb2xvcihyYW5nZVtpICUgMjBdKS47XG4gICAgICAgICAgICB0YzEuc3R5bGUuYmFja2dyb3VuZENvbG9yID0gcmFuZ2VbaSAlIDIwXTsvL1wicmdiYShcIiArIHRlbXAuciArIFwiLFwiICsgdGVtcC5nICsgXCIsXCIgKyB0ZW1wLmIgKyBcIiwgMC42KVwiO1xuICAgICAgICAgICAgY29uc3QgdGMyID0gdHIuaW5zZXJ0Q2VsbCgpO1xuICAgICAgICAgICAgdGMyLnRleHRDb250ZW50ID0gZG9tYWluW2ldO1xuICAgICAgICB9XG4gICAgICAgIGRpdi5hcHBlbmRDaGlsZChmZWF0dXJlQ29sb3JUYWJsZSk7XG4gICAgfVxufSJdLCJtYXBwaW5ncyI6IkFBQUE7QUFBQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSIsInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./src/js/color-scheme-key.js\n"); /***/ }), @@ -1173,11 +1173,11 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /*!**************************!*\ !*** ./src/js/config.js ***! \**************************/ -/*! exports provided: svgns, highlightColour, LABEL_Y, rotatePointAboutPoint */ +/*! exports provided: svgns, LABEL_Y, rotatePointAboutPoint */ /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"svgns\", function() { return svgns; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"highlightColour\", function() { return highlightColour; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"LABEL_Y\", function() { return LABEL_Y; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"rotatePointAboutPoint\", function() { return rotatePointAboutPoint; });\nconst svgns = \"http://www.w3.org/2000/svg\";//, // namespace for svg elements\n// xlinkNS: 'http://www.w3.org/1999/xlink', // namespace for xlink, for use/defs elements\n\nconst highlightColour = \"#ffff99\"; //, //\"#fdc086\"); // todo use css\n\nconst LABEL_Y = -5; // todo this isn't needed\n// selectedColour: '#ffff99',\n//\n// Polymer: {\n// STICKHEIGHT: 20,\n// MAXSIZE: 20,\n// transitionTime: 650\n// }\n// };\nfunction rotatePointAboutPoint (p, o, theta) {\n theta = (theta / 360) * Math.PI * 2; //TODO: change theta arg to radians not degrees\n const rx = Math.cos(theta) * (p[0] - o[0]) - Math.sin(theta) * (p[1] - o[1]) + o[0];\n const ry = Math.sin(theta) * (p[0] - o[0]) + Math.cos(theta) * (p[1] - o[1]) + o[1];\n return [rx, ry];\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvY29uZmlnLmpzLmpzIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vY29tcGxleHZpZXdlci8uL3NyYy9qcy9jb25maWcuanM/MTRhYyJdLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3Qgc3ZnbnMgPSBcImh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnXCI7Ly8sIC8vIG5hbWVzcGFjZSBmb3Igc3ZnIGVsZW1lbnRzXG4vLyAgICAgeGxpbmtOUzogJ2h0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsnLCAvLyBuYW1lc3BhY2UgZm9yIHhsaW5rLCBmb3IgdXNlL2RlZnMgZWxlbWVudHNcblxuZXhwb3J0IGNvbnN0IGhpZ2hsaWdodENvbG91ciA9IFwiI2ZmZmY5OVwiOyAvLywgLy9cIiNmZGMwODZcIik7IC8vIHRvZG8gdXNlIGNzc1xuXG5leHBvcnQgY29uc3QgTEFCRUxfWSA9IC01OyAvLyB0b2RvIHRoaXMgaXNuJ3QgbmVlZGVkXG4vLyAgICAgc2VsZWN0ZWRDb2xvdXI6ICcjZmZmZjk5Jyxcbi8vXG4vLyAgICAgUG9seW1lcjoge1xuLy8gICAgICAgICBTVElDS0hFSUdIVDogMjAsXG4vLyAgICAgICAgIE1BWFNJWkU6IDIwLFxuLy8gICAgICAgICB0cmFuc2l0aW9uVGltZTogNjUwXG4vLyAgICAgfVxuLy8gfTtcbmV4cG9ydCBmdW5jdGlvbiByb3RhdGVQb2ludEFib3V0UG9pbnQgKHAsIG8sIHRoZXRhKSB7XG4gICAgdGhldGEgPSAodGhldGEgLyAzNjApICogTWF0aC5QSSAqIDI7IC8vVE9ETzogY2hhbmdlIHRoZXRhIGFyZyB0byByYWRpYW5zIG5vdCBkZWdyZWVzXG4gICAgY29uc3QgcnggPSBNYXRoLmNvcyh0aGV0YSkgKiAocFswXSAtIG9bMF0pIC0gTWF0aC5zaW4odGhldGEpICogKHBbMV0gLSBvWzFdKSArIG9bMF07XG4gICAgY29uc3QgcnkgPSBNYXRoLnNpbih0aGV0YSkgKiAocFswXSAtIG9bMF0pICsgTWF0aC5jb3ModGhldGEpICogKHBbMV0gLSBvWzFdKSArIG9bMV07XG4gICAgcmV0dXJuIFtyeCwgcnldO1xufSJdLCJtYXBwaW5ncyI6IkFBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./src/js/config.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"svgns\", function() { return svgns; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"LABEL_Y\", function() { return LABEL_Y; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"rotatePointAboutPoint\", function() { return rotatePointAboutPoint; });\nconst svgns = \"http://www.w3.org/2000/svg\";//, // namespace for svg elements\n// xlinkNS: 'http://www.w3.org/1999/xlink', // namespace for xlink, for use/defs elements\n\nconst LABEL_Y = -5; // todo this isn't needed\n// selectedColour: '#ffff99',\n//\n// Polymer: {\n// STICKHEIGHT: 20,\n// MAXSIZE: 20,\n// transitionTime: 650\n// }\n// };\nfunction rotatePointAboutPoint (p, o, theta) {\n theta = (theta / 360) * Math.PI * 2; //TODO: change theta arg to radians not degrees\n const rx = Math.cos(theta) * (p[0] - o[0]) - Math.sin(theta) * (p[1] - o[1]) + o[0];\n const ry = Math.sin(theta) * (p[0] - o[0]) + Math.cos(theta) * (p[1] - o[1]) + o[1];\n return [rx, ry];\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvY29uZmlnLmpzLmpzIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vY29tcGxleHZpZXdlci8uL3NyYy9qcy9jb25maWcuanM/MTRhYyJdLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3Qgc3ZnbnMgPSBcImh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnXCI7Ly8sIC8vIG5hbWVzcGFjZSBmb3Igc3ZnIGVsZW1lbnRzXG4vLyAgICAgeGxpbmtOUzogJ2h0dHA6Ly93d3cudzMub3JnLzE5OTkveGxpbmsnLCAvLyBuYW1lc3BhY2UgZm9yIHhsaW5rLCBmb3IgdXNlL2RlZnMgZWxlbWVudHNcblxuZXhwb3J0IGNvbnN0IExBQkVMX1kgPSAtNTsgLy8gdG9kbyB0aGlzIGlzbid0IG5lZWRlZFxuLy8gICAgIHNlbGVjdGVkQ29sb3VyOiAnI2ZmZmY5OScsXG4vL1xuLy8gICAgIFBvbHltZXI6IHtcbi8vICAgICAgICAgU1RJQ0tIRUlHSFQ6IDIwLFxuLy8gICAgICAgICBNQVhTSVpFOiAyMCxcbi8vICAgICAgICAgdHJhbnNpdGlvblRpbWU6IDY1MFxuLy8gICAgIH1cbi8vIH07XG5leHBvcnQgZnVuY3Rpb24gcm90YXRlUG9pbnRBYm91dFBvaW50IChwLCBvLCB0aGV0YSkge1xuICAgIHRoZXRhID0gKHRoZXRhIC8gMzYwKSAqIE1hdGguUEkgKiAyOyAvL1RPRE86IGNoYW5nZSB0aGV0YSBhcmcgdG8gcmFkaWFucyBub3QgZGVncmVlc1xuICAgIGNvbnN0IHJ4ID0gTWF0aC5jb3ModGhldGEpICogKHBbMF0gLSBvWzBdKSAtIE1hdGguc2luKHRoZXRhKSAqIChwWzFdIC0gb1sxXSkgKyBvWzBdO1xuICAgIGNvbnN0IHJ5ID0gTWF0aC5zaW4odGhldGEpICogKHBbMF0gLSBvWzBdKSArIE1hdGguY29zKHRoZXRhKSAqIChwWzFdIC0gb1sxXSkgKyBvWzFdO1xuICAgIHJldHVybiBbcngsIHJ5XTtcbn0iXSwibWFwcGluZ3MiOiJBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./src/js/config.js\n"); /***/ }), @@ -1205,6 +1205,18 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /***/ }), +/***/ "./src/js/svgexp.js": +/*!**************************!*\ + !*** ./src/js/svgexp.js ***! + \**************************/ +/*! exports provided: svgUtils */ +/***/ (function(module, __webpack_exports__, __webpack_require__) { + +"use strict"; +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"svgUtils\", function() { return svgUtils; });\n/**\n * Created by cs22 on 04/12/14.\n */\n\nconst svgUtils = {\n \n capture: function (svgElems) {\n return svgElems.map (function(svg) { return svgUtils.makeSVGDoc (svg); });\n },\n\n getAllSVGElements: function () {\n // search through all document objects, including those in iframes\n var allIFrames = [].slice.apply (document.getElementsByTagName('iframe'));\n var docs = [document];\n allIFrames.forEach (function (iframe) {\n try {\n docs.push (iframe.contentDocument || iframe.contentWindow.document);\n }\n catch (e) {\n console.log (\"Protected cross-domain IFrame\", iframe);\n }\n });\n\n var allSvgs = [];\n docs.forEach (function(doc) {\n var allDocSvgs = [].slice.apply (doc.getElementsByTagName('svg'));\n allSvgs.push.apply (allSvgs, allDocSvgs);\n });\n return allSvgs;\n },\n\n\n makeSVGDoc: function (svgElem) {\n // clone node\n var cloneSVG = svgElem.cloneNode (true);\n var ownerDoc = cloneSVG.ownerDocument || document;\n svgUtils.pruneInvisibleSubtrees (cloneSVG, svgElem);\n\n // find all styles inherited/referenced at or below this node\n var styles = svgUtils.usedStyles (svgElem, true, true);\n\n // collect relevant info on parent chain of svg node\n var predecessorInfo = svgUtils.parentChain (svgElem, styles);\n \n var addDummy = function (dummySVGElem, cloneSVG, origSVG, transferAttr) {\n dummySVGElem.appendChild (cloneSVG);\n Object.keys(transferAttr).forEach (function (attr) {\n var val = cloneSVG.getAttribute (attr) || cloneSVG.style [attr] || svgUtils.getComputedStyleCssText (origSVG, attr);\n if (val != null) {\n dummySVGElem.setAttribute (attr, val);\n var attrVal = transferAttr[attr];\n if (attrVal.replace) {\n cloneSVG.setAttribute (attr, attrVal.replace);\n } else if (attrVal.delete) {\n cloneSVG.removeAttribute (attr);\n }\n }\n });\n };\n\n // make a chain of dummy svg nodes to include classes / ids of parent chain of our original svg\n // this means any styles referenced within the svg that depend on the presence of these classes/ids are fired\n var transferAttr = {width: {replace: \"100%\"}, height: {replace: \"100%\"}, xmlns: {delete: true}};\n var parentAdded = false;\n for (var p = 0; p < predecessorInfo.length; p++) {\n var pinf = predecessorInfo [p];\n //var dummySVGElem = ownerDoc.createElement (\"svg\");\n var dummySVGElem = ownerDoc.createElementNS (\"http://www.w3.org/2000/svg\", \"svg\");\n var empty = true;\n Object.keys(pinf).forEach (function (key) {\n if (pinf[key]) {\n dummySVGElem.setAttribute (key, pinf[key]);\n empty = false;\n }\n });\n // If the dummy svg has no relevant id, classes or computed style then ignore it, otherwise make it the new root\n if (!empty) {\n addDummy (dummySVGElem, cloneSVG, svgElem, transferAttr);\n cloneSVG = dummySVGElem;\n parentAdded = true;\n }\n }\n\n // if no dummy parent added in previous section, but our svg isn't root then add one as placeholder\n if (svgElem.parentNode != null && !parentAdded) {\n var dummySVGElem = ownerDoc.createElementNS (\"http://www.w3.org/2000/svg\", \"svg\");\n addDummy (dummySVGElem, cloneSVG, svgElem, transferAttr);\n cloneSVG = dummySVGElem;\n parentAdded = true;\n }\n\n // Copy svg's computed style (it's style context) if a dummy parent node has been introduced\n if (parentAdded) {\n cloneSVG.setAttribute (\"style\", svgUtils.getComputedStyleCssText (svgElem));\n }\n\n cloneSVG.setAttribute (\"version\", \"1.1\");\n //cloneSVG.setAttribute (\"xmlns\", \"http://www.w3.org/2000/svg\"); // XMLSerializer does this\n //cloneSVG.setAttribute (\"xmlns:xlink\", \"http://www.w3.org/1999/xlink\"); // when I used setAttributeNS it ballsed up\n\t\t// however using these attributeNS calls work, and stops errors in IE11. Win.\n\t\tcloneSVG.setAttributeNS (\"http://www.w3.org/2000/xmlns/\", \"xmlns\", \"http://www.w3.org/2000/svg\"); // XMLSerializer does this\n cloneSVG.setAttributeNS (\"http://www.w3.org/2000/xmlns/\", \"xmlns:xlink\", \"http://www.w3.org/1999/xlink\"); // when I used setAttributeNS it ballsed up\n\n\n var styleElem = ownerDoc.createElement (\"style\");\n styleElem.setAttribute (\"type\", \"text/css\");\n var styleText = ownerDoc.createTextNode (styles.join(\"\\n\"));\n styleElem.appendChild (styleText);\n cloneSVG.insertBefore (styleElem, cloneSVG.firstChild);\n\n return cloneSVG;\n },\n \n // Because firefox returns cssText as empty\n // https://bugzilla.mozilla.org/show_bug.cgi?id=137687\n getComputedStyleCssText: function (element, field) {\n var style = window.getComputedStyle(element);\n if (field) {\n return style[field];\n }\n\n if (style.cssText != \"\") {\n return style.cssText;\n }\n\n var cssText = \"\";\n for (var i = 0; i < style.length; i++) {\n var styleName = style[i];\n var propVal = style.getPropertyValue(styleName);\n cssText += styleName + \": \" + propVal + \"; \";\n }\n\n return cssText;\n },\n \n doPruneInvisible: true,\n \n pruneConditionSets: [{\"display\": \"none\"}, {\"visibility\": \"hidden\"}, {\"opacity\": \"0\"}, {\"fill-opacity\": \"0\", \"stroke-opacity\": \"0\"}, {\"fill-opacity\": \"0\", \"stroke\": \"none\"}, {\"fill\": \"none\", \"stroke-opacity\": \"0\"}],\n \n pruneInvisibleSubtrees: function (clonedElement, matchingOriginalElement) {\n if (svgUtils.doPruneInvisible) {\n var style = window.getComputedStyle (matchingOriginalElement); // cloned (unattached) nodes in chrome at least don't have computed styles\n var prune = false;\n \n svgUtils.pruneConditionSets.forEach (function (conditionSet) {\n if (!prune) {\n var allConditionsMet = true;\n Object.keys(conditionSet).forEach (function (condition) {\n var condVal = conditionSet[condition];\n var eStyle = style[condition];\n var eAttr = matchingOriginalElement.getAttribute(condition);\n if (!(eStyle === condVal || (!eStyle && eAttr === condVal))) {\n allConditionsMet = false; \n }\n });\n prune = allConditionsMet;\n }\n });\n if (prune && clonedElement.parentNode) {\n clonedElement.parentNode.removeChild (clonedElement);\n //console.log (\"removed\", clonedElement);\n } else {\n var clonedChildren = clonedElement.children;\n var matchingOriginalChildren = matchingOriginalElement.children;\n //console.log (\"kept\", clonedElement, style.display, style.visibility, style.opacity, style[\"stroke-opacity\"], style[\"fill-opacity\"], style);\n //console.log (element, \"children\", children);\n if (clonedChildren && clonedChildren.length) {\n // count backwards because removing a child will break the 'i' counter if we go forwards\n // e.g. if children=[A,B,C,D] and i=2, if we delete[C] then children becomes [A,B,D],\n // and when i then increments to 3, expecting D, instead we find the end of loop, and don't test D\n // PS. And if we fixed that we'd then need a separate counter for the original child elements anyways so backwards it is\n for (var i = clonedChildren.length; --i >= 0;) {\n svgUtils.pruneInvisibleSubtrees (clonedChildren[i], matchingOriginalChildren[i]);\n }\n }\n }\n }\n },\n\n parentChain: function (elem, styles) {\n // Capture id / classes of svg's parent chain.\n var ownerDoc = elem.ownerDocument || document;\n var elemArr = [];\n while (elem.parentNode !== ownerDoc && elem.parentNode !== null) {\n elem = elem.parentNode;\n elemArr.push ({id: elem.id, class: elem.getAttribute(\"class\") || \"\"});\n }\n\n // see if id or element class are referenced in any styles collected below the svg node\n // if not, null the id / class as they're not going to be relevant\n elemArr.forEach (function (elemData) {\n var presences = {id: false, class: false};\n var classes = elemData.class.split(\" \").filter(function(a) { return a.length > 0; }); // v1.13: may be multiple classes in a containing class attribute\n styles.forEach (function (style) {\n for (var c = 0; c < classes.length; c++) {\n if (style.indexOf (\".\"+classes[c]) >= 0) {\n presences.class = true;\n break; // no need to keep looking through rest of classtypes if one is needed\n }\n }\n if (elemData.id && style.indexOf (\"#\"+elemData.id) >= 0) {\n presences.id = true;\n }\n });\n Object.keys(presences).forEach (function (presence) {\n if (!presences[presence]) { elemData[presence] = undefined; }\n });\n });\n\n return elemArr;\n },\n\n // code adapted from user adardesign's answer in http://stackoverflow.com/questions/13204785/is-it-possible-to-read-the-styles-of-css-classes-not-being-used-in-the-dom-using\n usedStyles: function (elem, subtree, both) {\n var needed = [], rule;\n var ownerDoc = elem.ownerDocument || document;\n var CSSSheets = ownerDoc.styleSheets;\n\n for(var j=0; j < CSSSheets.length; j++){\n\t\t\t// stop accessing empty style sheets (1.15), catch security exceptions (1.20)\n\t\t\ttry{\n\t\t\t\tif (CSSSheets[j].cssRules == null) {\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\t\t\t} catch (err) {\n\t\t\t\tcontinue;\n\t\t\t}\n\t\t\t\n for(var i=0; i < CSSSheets[j].cssRules.length; i++){\n rule = CSSSheets[j].cssRules[i];\n var match = false;\n // Issue reported, css rule '[ng:cloak], [ng-cloak], [data-ng-cloak], [x-ng-cloak], .ng-cloak, .x-ng-cloak, .ng-hide:not(.ng-hide-animate)' gives error\n // It's the [ng:cloak] bit that does the damage\n // Fix found from https://github.com/exupero/saveSvgAsPng/issues/11 - but the css rule isn't applied\n try {\n if (subtree) {\n match = elem.querySelectorAll(rule.selectorText).length > 0;\n }\n if (!subtree || both) {\n match |= elem.matches(rule.selectorText);\n }\n }\n catch (err) {\n console.warn (\"CSS selector error: \"+rule.selectorText+\". Often angular issue.\", err);\n }\n if (match) { needed.push (rule.cssText); }\n }\n }\n\n return needed;\n },\n \n makeXMLStr: function (xmls, svgDoc) {\n var xmlStr = xmls.serializeToString(svgDoc);\n // serializing adds an xmlns attribute to the style element ('cos it thinks we want xhtml), which knackers it for inkscape, here we chop it out\n xmlStr = xmlStr.split(\"xmlns=\\\"http://www.w3.org/1999/xhtml\\\"\").join(\"\");\n return xmlStr;\n },\n\n // saveSVGDocs: function (svgDocs) {\n // var xmls = new XMLSerializer();\n // svgDocs.forEach (function (svgDoc, i) {\n // var xmlStr = svgUtils.makeXMLStr (xmls, svgDoc);\n // var blob = new Blob([xmlStr], {type: \"image/svg+xml\"});\n // saveAs(blob, \"saved\"+i+\".svg\");\n // });\n // },\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/js/svgexp.js.js","sources":["webpack://complexviewer/./src/js/svgexp.js?ad9d"],"sourcesContent":["/**\n * Created by cs22 on 04/12/14.\n */\n\nexport const svgUtils = {\n    \n    capture: function (svgElems) {\n        return svgElems.map (function(svg) { return svgUtils.makeSVGDoc (svg); });\n    },\n\n    getAllSVGElements: function () {\n        // search through all document objects, including those in iframes\n        var allIFrames = [].slice.apply (document.getElementsByTagName('iframe'));\n        var docs = [document];\n        allIFrames.forEach (function (iframe) {\n            try {\n                docs.push (iframe.contentDocument || iframe.contentWindow.document);\n            }\n            catch (e) {\n                console.log (\"Protected cross-domain IFrame\", iframe);\n            }\n        });\n\n        var allSvgs = [];\n        docs.forEach (function(doc) {\n            var allDocSvgs = [].slice.apply (doc.getElementsByTagName('svg'));\n            allSvgs.push.apply (allSvgs, allDocSvgs);\n        });\n        return allSvgs;\n    },\n\n\n    makeSVGDoc: function (svgElem) {\n        // clone node\n        var cloneSVG = svgElem.cloneNode (true);\n        var ownerDoc = cloneSVG.ownerDocument || document;\n        svgUtils.pruneInvisibleSubtrees (cloneSVG, svgElem);\n\n        // find all styles inherited/referenced at or below this node\n        var styles = svgUtils.usedStyles (svgElem, true, true);\n\n        // collect relevant info on parent chain of svg node\n        var predecessorInfo = svgUtils.parentChain (svgElem, styles);\n        \n        var addDummy = function (dummySVGElem, cloneSVG, origSVG, transferAttr) {\n            dummySVGElem.appendChild (cloneSVG);\n            Object.keys(transferAttr).forEach (function (attr) {\n                var val = cloneSVG.getAttribute (attr) || cloneSVG.style [attr] || svgUtils.getComputedStyleCssText (origSVG, attr);\n                if (val != null) {\n                    dummySVGElem.setAttribute (attr, val);\n                    var attrVal = transferAttr[attr];\n                    if (attrVal.replace) {\n                        cloneSVG.setAttribute (attr, attrVal.replace);\n                    } else if (attrVal.delete) {\n                        cloneSVG.removeAttribute (attr);\n                    }\n                }\n            });\n        };\n\n        // make a chain of dummy svg nodes to include classes / ids of parent chain of our original svg\n        // this means any styles referenced within the svg that depend on the presence of these classes/ids are fired\n        var transferAttr = {width: {replace: \"100%\"}, height: {replace: \"100%\"}, xmlns: {delete: true}};\n        var parentAdded = false;\n        for (var p = 0; p < predecessorInfo.length; p++) {\n            var pinf = predecessorInfo [p];\n            //var dummySVGElem = ownerDoc.createElement (\"svg\");\n            var dummySVGElem = ownerDoc.createElementNS (\"http://www.w3.org/2000/svg\", \"svg\");\n            var empty = true;\n            Object.keys(pinf).forEach (function (key) {\n                if (pinf[key]) {\n                    dummySVGElem.setAttribute (key, pinf[key]);\n                    empty = false;\n                }\n            });\n            // If the dummy svg has no relevant id, classes or computed style then ignore it, otherwise make it the new root\n            if (!empty) {\n                addDummy (dummySVGElem, cloneSVG, svgElem, transferAttr);\n                cloneSVG = dummySVGElem;\n                parentAdded = true;\n            }\n        }\n\n        // if no dummy parent added in previous section, but our svg isn't root then add one as placeholder\n        if (svgElem.parentNode != null && !parentAdded) {\n            var dummySVGElem = ownerDoc.createElementNS (\"http://www.w3.org/2000/svg\", \"svg\");\n            addDummy (dummySVGElem, cloneSVG, svgElem, transferAttr);\n            cloneSVG = dummySVGElem;\n            parentAdded = true;\n        }\n\n        // Copy svg's computed style (it's style context) if a dummy parent node has been introduced\n        if (parentAdded) {\n            cloneSVG.setAttribute (\"style\", svgUtils.getComputedStyleCssText (svgElem));\n        }\n\n        cloneSVG.setAttribute (\"version\", \"1.1\");\n        //cloneSVG.setAttribute (\"xmlns\", \"http://www.w3.org/2000/svg\");    // XMLSerializer does this\n        //cloneSVG.setAttribute (\"xmlns:xlink\", \"http://www.w3.org/1999/xlink\");  // when I used setAttributeNS it ballsed up\n\t\t// however using these attributeNS calls work, and stops errors in IE11. Win.\n\t\tcloneSVG.setAttributeNS (\"http://www.w3.org/2000/xmlns/\", \"xmlns\", \"http://www.w3.org/2000/svg\");    // XMLSerializer does this\n        cloneSVG.setAttributeNS (\"http://www.w3.org/2000/xmlns/\", \"xmlns:xlink\", \"http://www.w3.org/1999/xlink\");  // when I used setAttributeNS it ballsed up\n\n\n        var styleElem = ownerDoc.createElement (\"style\");\n        styleElem.setAttribute (\"type\", \"text/css\");\n        var styleText = ownerDoc.createTextNode (styles.join(\"\\n\"));\n        styleElem.appendChild (styleText);\n        cloneSVG.insertBefore (styleElem, cloneSVG.firstChild);\n\n        return cloneSVG;\n    },\n    \n    // Because firefox returns cssText as empty\n    // https://bugzilla.mozilla.org/show_bug.cgi?id=137687\n    getComputedStyleCssText: function (element, field) {\n        var style = window.getComputedStyle(element);\n        if (field) {\n            return style[field];\n        }\n\n        if (style.cssText != \"\") {\n            return style.cssText;\n        }\n\n        var cssText = \"\";\n        for (var i = 0; i < style.length; i++) {\n            var styleName = style[i];\n            var propVal = style.getPropertyValue(styleName);\n            cssText += styleName + \": \" + propVal + \"; \";\n        }\n\n        return cssText;\n    },\n    \n    doPruneInvisible: true,\n    \n    pruneConditionSets: [{\"display\": \"none\"}, {\"visibility\": \"hidden\"}, {\"opacity\": \"0\"}, {\"fill-opacity\": \"0\", \"stroke-opacity\": \"0\"}, {\"fill-opacity\": \"0\", \"stroke\": \"none\"}, {\"fill\": \"none\", \"stroke-opacity\": \"0\"}],\n    \n    pruneInvisibleSubtrees: function (clonedElement, matchingOriginalElement) {\n        if (svgUtils.doPruneInvisible) {\n            var style = window.getComputedStyle (matchingOriginalElement);  // cloned (unattached) nodes in chrome at least don't have computed styles\n            var prune = false;\n            \n            svgUtils.pruneConditionSets.forEach (function (conditionSet) {\n                if (!prune) {\n                    var allConditionsMet = true;\n                    Object.keys(conditionSet).forEach (function (condition) {\n                        var condVal = conditionSet[condition];\n                        var eStyle = style[condition];\n                        var eAttr = matchingOriginalElement.getAttribute(condition);\n                        if (!(eStyle === condVal || (!eStyle && eAttr === condVal))) {\n                            allConditionsMet = false; \n                        }\n                    });\n                    prune = allConditionsMet;\n                }\n            });\n            if (prune && clonedElement.parentNode) {\n                clonedElement.parentNode.removeChild (clonedElement);\n                //console.log (\"removed\", clonedElement);\n            } else {\n                var clonedChildren = clonedElement.children;\n                var matchingOriginalChildren = matchingOriginalElement.children;\n                //console.log (\"kept\", clonedElement, style.display, style.visibility, style.opacity, style[\"stroke-opacity\"], style[\"fill-opacity\"], style);\n                //console.log (element, \"children\", children);\n                if (clonedChildren && clonedChildren.length) {\n                    // count backwards because removing a child will break the 'i' counter if we go forwards\n                    // e.g. if children=[A,B,C,D] and i=2, if we delete[C] then children becomes [A,B,D],\n                    // and when i then increments to 3, expecting D, instead we find the end of loop, and don't test D\n                    // PS. And if we fixed that we'd then need a separate counter for the original child elements anyways so backwards it is\n                    for (var i = clonedChildren.length; --i >= 0;) {\n                        svgUtils.pruneInvisibleSubtrees (clonedChildren[i], matchingOriginalChildren[i]);\n                    }\n                }\n            }\n        }\n    },\n\n    parentChain: function (elem, styles) {\n        // Capture id / classes of svg's parent chain.\n        var ownerDoc = elem.ownerDocument || document;\n        var elemArr = [];\n        while (elem.parentNode !== ownerDoc && elem.parentNode !== null) {\n            elem = elem.parentNode;\n            elemArr.push ({id: elem.id, class: elem.getAttribute(\"class\") || \"\"});\n        }\n\n        // see if id or element class are referenced in any styles collected below the svg node\n        // if not, null the id / class as they're not going to be relevant\n        elemArr.forEach (function (elemData) {\n            var presences = {id: false, class: false};\n            var classes = elemData.class.split(\" \").filter(function(a) { return a.length > 0; });   // v1.13: may be multiple classes in a containing class attribute\n            styles.forEach (function (style) {\n                for (var c = 0; c < classes.length; c++) {\n                    if (style.indexOf (\".\"+classes[c]) >= 0) {\n                        presences.class = true;\n                        break;  // no need to keep looking through rest of classtypes if one is needed\n                    }\n                }\n                if (elemData.id && style.indexOf (\"#\"+elemData.id) >= 0) {\n                    presences.id = true;\n                }\n            });\n            Object.keys(presences).forEach (function (presence) {\n                if (!presences[presence]) { elemData[presence] = undefined; }\n            });\n        });\n\n        return elemArr;\n    },\n\n    // code adapted from user adardesign's answer in http://stackoverflow.com/questions/13204785/is-it-possible-to-read-the-styles-of-css-classes-not-being-used-in-the-dom-using\n    usedStyles: function (elem, subtree, both) {\n        var needed = [], rule;\n        var ownerDoc = elem.ownerDocument || document;\n        var CSSSheets = ownerDoc.styleSheets;\n\n        for(var j=0; j < CSSSheets.length; j++){\n\t\t\t// stop accessing empty style sheets (1.15), catch security exceptions (1.20)\n\t\t\ttry{\n\t\t\t\tif (CSSSheets[j].cssRules == null) {\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\t\t\t} catch (err) {\n\t\t\t\tcontinue;\n\t\t\t}\n\t\t\t\n            for(var i=0; i < CSSSheets[j].cssRules.length; i++){\n                rule = CSSSheets[j].cssRules[i];\n                var match = false;\n                // Issue reported, css rule '[ng:cloak], [ng-cloak], [data-ng-cloak], [x-ng-cloak], .ng-cloak, .x-ng-cloak, .ng-hide:not(.ng-hide-animate)' gives error\n                // It's the [ng:cloak] bit that does the damage\n                // Fix found from https://github.com/exupero/saveSvgAsPng/issues/11 - but the css rule isn't applied\n                try {\n                    if (subtree) {\n                        match = elem.querySelectorAll(rule.selectorText).length > 0;\n                    }\n                    if (!subtree || both) {\n                        match |= elem.matches(rule.selectorText);\n                    }\n                }\n                catch (err) {\n                    console.warn (\"CSS selector error: \"+rule.selectorText+\". Often angular issue.\", err);\n                }\n                if (match) { needed.push (rule.cssText); }\n            }\n        }\n\n        return needed;\n    },\n    \n    makeXMLStr: function (xmls, svgDoc) {\n        var xmlStr = xmls.serializeToString(svgDoc);\n        // serializing adds an xmlns attribute to the style element ('cos it thinks we want xhtml), which knackers it for inkscape, here we chop it out\n        xmlStr = xmlStr.split(\"xmlns=\\\"http://www.w3.org/1999/xhtml\\\"\").join(\"\");\n        return xmlStr;\n    },\n\n    // saveSVGDocs: function (svgDocs) {\n    //     var xmls = new XMLSerializer();\n    //     svgDocs.forEach (function (svgDoc, i) {\n    //         var xmlStr = svgUtils.makeXMLStr (xmls, svgDoc);\n    //         var blob = new Blob([xmlStr], {type: \"image/svg+xml\"});\n    //         saveAs(blob, \"saved\"+i+\".svg\");\n    //     });\n    // },\n};"],"mappings":"AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/js/svgexp.js\n"); + +/***/ }), + /***/ "./src/js/viz/interactor/annotation.js": /*!*********************************************!*\ !*** ./src/js/viz/interactor/annotation.js ***! @@ -1213,7 +1225,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"Annotation\", function() { return Annotation; });\n//constructor for annotations\nfunction Annotation(annotationName, seqDatum) {\n console.log(\"**\", annotationName, seqDatum);\n this.description = annotationName.trim();\n this.seqDatum = seqDatum;\n}\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvdml6L2ludGVyYWN0b3IvYW5ub3RhdGlvbi5qcy5qcyIsInNvdXJjZXMiOlsid2VicGFjazovL2NvbXBsZXh2aWV3ZXIvLi9zcmMvanMvdml6L2ludGVyYWN0b3IvYW5ub3RhdGlvbi5qcz8yYTZkIl0sInNvdXJjZXNDb250ZW50IjpbIi8vY29uc3RydWN0b3IgZm9yIGFubm90YXRpb25zXG5leHBvcnQgZnVuY3Rpb24gQW5ub3RhdGlvbihhbm5vdGF0aW9uTmFtZSwgc2VxRGF0dW0pIHtcbiAgICBjb25zb2xlLmxvZyhcIioqXCIsIGFubm90YXRpb25OYW1lLCBzZXFEYXR1bSk7XG4gICAgdGhpcy5kZXNjcmlwdGlvbiA9IGFubm90YXRpb25OYW1lLnRyaW0oKTtcbiAgICB0aGlzLnNlcURhdHVtID0gc2VxRGF0dW07XG59XG4iXSwibWFwcGluZ3MiOiJBQUFBO0FBQUE7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7Iiwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./src/js/viz/interactor/annotation.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"Annotation\", function() { return Annotation; });\n//constructor for annotations\nfunction Annotation(annotationName, seqDatum) {\n // console.log(\"**\", annotationName, seqDatum);\n this.description = annotationName.trim();\n this.seqDatum = seqDatum;\n}\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvdml6L2ludGVyYWN0b3IvYW5ub3RhdGlvbi5qcy5qcyIsInNvdXJjZXMiOlsid2VicGFjazovL2NvbXBsZXh2aWV3ZXIvLi9zcmMvanMvdml6L2ludGVyYWN0b3IvYW5ub3RhdGlvbi5qcz8yYTZkIl0sInNvdXJjZXNDb250ZW50IjpbIi8vY29uc3RydWN0b3IgZm9yIGFubm90YXRpb25zXG5leHBvcnQgZnVuY3Rpb24gQW5ub3RhdGlvbihhbm5vdGF0aW9uTmFtZSwgc2VxRGF0dW0pIHtcbiAgICAvLyBjb25zb2xlLmxvZyhcIioqXCIsIGFubm90YXRpb25OYW1lLCBzZXFEYXR1bSk7XG4gICAgdGhpcy5kZXNjcmlwdGlvbiA9IGFubm90YXRpb25OYW1lLnRyaW0oKTtcbiAgICB0aGlzLnNlcURhdHVtID0gc2VxRGF0dW07XG59XG4iXSwibWFwcGluZ3MiOiJBQUFBO0FBQUE7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7Iiwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./src/js/viz/interactor/annotation.js\n"); /***/ }), @@ -1249,7 +1261,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"Complex\", function() { return Complex; });\n/* harmony import */ var _interactor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./interactor */ \"./src/js/viz/interactor/interactor.js\");\n/* harmony import */ var point2d__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! point2d */ \"./node_modules/point2d/index.js\");\n/* harmony import */ var point2d__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(point2d__WEBPACK_IMPORTED_MODULE_1__);\n/* harmony import */ var intersectionjs__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! intersectionjs */ \"./node_modules/intersectionjs/intersection.js\");\n/* harmony import */ var intersectionjs__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(intersectionjs__WEBPACK_IMPORTED_MODULE_2__);\n\n\n\n\nfunction Complex(id, app) {\n this.init(id, app);\n this.type = \"complex\";\n this.padding = 20;\n\n // const self = this;\n // // its bad if you end up with these getting called\n // Object.defineProperty(this, \"width\", {\n // get: function height() {\n // return self.naryLink.path.getBBox().width;\n // //return 160;\n // }\n // });\n // Object.defineProperty(this, \"height\", {\n // get: function height() {\n // return self.naryLink.path.getBBox().height;\n // //return 160;\n // }\n // });\n}\n\nComplex.prototype = new _interactor__WEBPACK_IMPORTED_MODULE_0__[\"Interactor\"]();\n\n// Complex.prototype = {\n// get width() {\n// return this.naryLink.path.getBBox().width;\n// },\n// get height() {\n// return this.naryLink.path.getBBox().height;\n// },\n// };\n\nComplex.prototype.initLink = function (naryLink) {\n this.naryLink = naryLink;\n naryLink.path.setAttribute(\"stroke\", \"white\");\n naryLink.path.setAttribute(\"stroke-linejoin\", \"round\");\n naryLink.path.setAttribute(\"stroke-width\", 4);\n};\n\nComplex.prototype.getPosition = function (originPoint) {\n const mapped = this.naryLink.mapped;//getMappedCoordinates();\n const mc = mapped.length;\n let xSum = 0,\n ySum = 0;\n for (let m = 0; m < mc; m++) {\n xSum += mapped[m][0];\n ySum += mapped[m][1];\n }\n let center = [xSum / mc, ySum / mc];\n if (originPoint) {\n // if (participant.type === \"complex\"){\n // startPoint = participant.getPosition();\n let naryPath = this.naryLink.hull;\n let iPath = [];\n for (let p of naryPath) {\n iPath.push(new point2d__WEBPACK_IMPORTED_MODULE_1__(p[0], p[1]));\n }\n let a1 = new point2d__WEBPACK_IMPORTED_MODULE_1__(center[0], center[1]);\n let a2 = new point2d__WEBPACK_IMPORTED_MODULE_1__(originPoint[0], originPoint[1]);\n let intersect = intersectionjs__WEBPACK_IMPORTED_MODULE_2__[\"intersectLinePolygon\"](a1, a2, iPath);\n if (intersect.points[0]) {\n return [intersect.points[0].x, intersect.points[0].y];\n }\n }\n return center;\n};\n\nComplex.prototype.setPosition = function () {\n console.error(\"bad - called setPosition on \", this);\n};\n\nComplex.prototype.changePosition = function (dx, dy) {\n for (let participant of this.naryLink.participants){\n participant.changePosition(dx, dy);\n }\n};\n\nComplex.prototype.getResidueCoordinates = function () {\n return this.getPosition();\n};\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvdml6L2ludGVyYWN0b3IvY29tcGxleC5qcy5qcyIsInNvdXJjZXMiOlsid2VicGFjazovL2NvbXBsZXh2aWV3ZXIvLi9zcmMvanMvdml6L2ludGVyYWN0b3IvY29tcGxleC5qcz8zOTQ4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7SW50ZXJhY3Rvcn0gZnJvbSBcIi4vaW50ZXJhY3RvclwiO1xuaW1wb3J0ICogYXMgUG9pbnQyRCBmcm9tIFwicG9pbnQyZFwiO1xuaW1wb3J0ICogYXMgSW50ZXJzZWN0aW9uIGZyb20gXCJpbnRlcnNlY3Rpb25qc1wiO1xuXG5leHBvcnQgZnVuY3Rpb24gQ29tcGxleChpZCwgYXBwKSB7XG4gICAgdGhpcy5pbml0KGlkLCBhcHApO1xuICAgIHRoaXMudHlwZSA9IFwiY29tcGxleFwiO1xuICAgIHRoaXMucGFkZGluZyA9IDIwO1xuXG4gICAgLy8gY29uc3Qgc2VsZiA9IHRoaXM7XG4gICAgLy8gLy8gaXRzIGJhZCBpZiB5b3UgZW5kIHVwIHdpdGggdGhlc2UgZ2V0dGluZyBjYWxsZWRcbiAgICAvLyBPYmplY3QuZGVmaW5lUHJvcGVydHkodGhpcywgXCJ3aWR0aFwiLCB7XG4gICAgLy8gICAgIGdldDogZnVuY3Rpb24gaGVpZ2h0KCkge1xuICAgIC8vICAgICAgICAgcmV0dXJuIHNlbGYubmFyeUxpbmsucGF0aC5nZXRCQm94KCkud2lkdGg7XG4gICAgLy8gICAgICAgICAvL3JldHVybiAxNjA7XG4gICAgLy8gICAgIH1cbiAgICAvLyB9KTtcbiAgICAvLyBPYmplY3QuZGVmaW5lUHJvcGVydHkodGhpcywgXCJoZWlnaHRcIiwge1xuICAgIC8vICAgICBnZXQ6IGZ1bmN0aW9uIGhlaWdodCgpIHtcbiAgICAvLyAgICAgICAgIHJldHVybiBzZWxmLm5hcnlMaW5rLnBhdGguZ2V0QkJveCgpLmhlaWdodDtcbiAgICAvLyAgICAgICAgIC8vcmV0dXJuIDE2MDtcbiAgICAvLyAgICAgfVxuICAgIC8vIH0pO1xufVxuXG5Db21wbGV4LnByb3RvdHlwZSA9IG5ldyBJbnRlcmFjdG9yKCk7XG5cbi8vIENvbXBsZXgucHJvdG90eXBlID0ge1xuLy8gICAgIGdldCB3aWR0aCgpIHtcbi8vICAgICAgICAgcmV0dXJuIHRoaXMubmFyeUxpbmsucGF0aC5nZXRCQm94KCkud2lkdGg7XG4vLyAgICAgfSxcbi8vICAgICBnZXQgaGVpZ2h0KCkge1xuLy8gICAgICAgICByZXR1cm4gdGhpcy5uYXJ5TGluay5wYXRoLmdldEJCb3goKS5oZWlnaHQ7XG4vLyAgICAgfSxcbi8vIH07XG5cbkNvbXBsZXgucHJvdG90eXBlLmluaXRMaW5rID0gZnVuY3Rpb24gKG5hcnlMaW5rKSB7XG4gICAgdGhpcy5uYXJ5TGluayA9IG5hcnlMaW5rO1xuICAgIG5hcnlMaW5rLnBhdGguc2V0QXR0cmlidXRlKFwic3Ryb2tlXCIsIFwid2hpdGVcIik7XG4gICAgbmFyeUxpbmsucGF0aC5zZXRBdHRyaWJ1dGUoXCJzdHJva2UtbGluZWpvaW5cIiwgXCJyb3VuZFwiKTtcbiAgICBuYXJ5TGluay5wYXRoLnNldEF0dHJpYnV0ZShcInN0cm9rZS13aWR0aFwiLCA0KTtcbn07XG5cbkNvbXBsZXgucHJvdG90eXBlLmdldFBvc2l0aW9uID0gZnVuY3Rpb24gKG9yaWdpblBvaW50KSB7XG4gICAgY29uc3QgbWFwcGVkID0gdGhpcy5uYXJ5TGluay5tYXBwZWQ7Ly9nZXRNYXBwZWRDb29yZGluYXRlcygpO1xuICAgIGNvbnN0IG1jID0gbWFwcGVkLmxlbmd0aDtcbiAgICBsZXQgeFN1bSA9IDAsXG4gICAgICAgIHlTdW0gPSAwO1xuICAgIGZvciAobGV0IG0gPSAwOyBtIDwgbWM7IG0rKykge1xuICAgICAgICB4U3VtICs9IG1hcHBlZFttXVswXTtcbiAgICAgICAgeVN1bSArPSBtYXBwZWRbbV1bMV07XG4gICAgfVxuICAgIGxldCBjZW50ZXIgPSBbeFN1bSAvIG1jLCB5U3VtIC8gbWNdO1xuICAgIGlmIChvcmlnaW5Qb2ludCkge1xuICAgIC8vIGlmIChwYXJ0aWNpcGFudC50eXBlID09PSBcImNvbXBsZXhcIil7XG4gICAgLy8gICAgIHN0YXJ0UG9pbnQgPSBwYXJ0aWNpcGFudC5nZXRQb3NpdGlvbigpO1xuICAgICAgICBsZXQgbmFyeVBhdGggPSB0aGlzLm5hcnlMaW5rLmh1bGw7XG4gICAgICAgIGxldCBpUGF0aCA9IFtdO1xuICAgICAgICBmb3IgKGxldCBwIG9mIG5hcnlQYXRoKSB7XG4gICAgICAgICAgICBpUGF0aC5wdXNoKG5ldyBQb2ludDJEKHBbMF0sIHBbMV0pKTtcbiAgICAgICAgfVxuICAgICAgICBsZXQgYTEgPSBuZXcgUG9pbnQyRChjZW50ZXJbMF0sIGNlbnRlclsxXSk7XG4gICAgICAgIGxldCBhMiA9IG5ldyBQb2ludDJEKG9yaWdpblBvaW50WzBdLCBvcmlnaW5Qb2ludFsxXSk7XG4gICAgICAgIGxldCBpbnRlcnNlY3QgPSBJbnRlcnNlY3Rpb24uaW50ZXJzZWN0TGluZVBvbHlnb24oYTEsIGEyLCBpUGF0aCk7XG4gICAgICAgIGlmIChpbnRlcnNlY3QucG9pbnRzWzBdKSB7XG4gICAgICAgICAgICByZXR1cm4gW2ludGVyc2VjdC5wb2ludHNbMF0ueCwgaW50ZXJzZWN0LnBvaW50c1swXS55XTtcbiAgICAgICAgfVxuICAgIH1cbiAgICByZXR1cm4gY2VudGVyO1xufTtcblxuQ29tcGxleC5wcm90b3R5cGUuc2V0UG9zaXRpb24gPSBmdW5jdGlvbiAoKSB7XG4gICAgY29uc29sZS5lcnJvcihcImJhZCAtIGNhbGxlZCBzZXRQb3NpdGlvbiBvbiBcIiwgdGhpcyk7XG59O1xuXG5Db21wbGV4LnByb3RvdHlwZS5jaGFuZ2VQb3NpdGlvbiA9IGZ1bmN0aW9uIChkeCwgZHkpIHtcbiAgICBmb3IgKGxldCBwYXJ0aWNpcGFudCBvZiB0aGlzLm5hcnlMaW5rLnBhcnRpY2lwYW50cyl7XG4gICAgICAgIHBhcnRpY2lwYW50LmNoYW5nZVBvc2l0aW9uKGR4LCBkeSk7XG4gICAgfVxufTtcblxuQ29tcGxleC5wcm90b3R5cGUuZ2V0UmVzaWR1ZUNvb3JkaW5hdGVzID0gZnVuY3Rpb24gKCkge1xuICAgIHJldHVybiB0aGlzLmdldFBvc2l0aW9uKCk7XG59O1xuXG4iXSwibWFwcGluZ3MiOiJBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7Iiwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./src/js/viz/interactor/complex.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"Complex\", function() { return Complex; });\n/* harmony import */ var _interactor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./interactor */ \"./src/js/viz/interactor/interactor.js\");\n/* harmony import */ var point2d__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! point2d */ \"./node_modules/point2d/index.js\");\n/* harmony import */ var point2d__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(point2d__WEBPACK_IMPORTED_MODULE_1__);\n/* harmony import */ var intersectionjs__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! intersectionjs */ \"./node_modules/intersectionjs/intersection.js\");\n/* harmony import */ var intersectionjs__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(intersectionjs__WEBPACK_IMPORTED_MODULE_2__);\n\n\n\n\nfunction Complex(id, app) {\n this.init(id, app);\n this.type = \"complex\";\n this.padding = 22;\n\n // const self = this;\n // // its bad if you end up with these getting called\n // Object.defineProperty(this, \"width\", {\n // get: function height() {\n // return self.naryLink.path.getBBox().width;\n // //return 160;\n // }\n // });\n // Object.defineProperty(this, \"height\", {\n // get: function height() {\n // return self.naryLink.path.getBBox().height;\n // //return 160;\n // }\n // });\n}\n\nComplex.prototype = new _interactor__WEBPACK_IMPORTED_MODULE_0__[\"Interactor\"]();\n\nComplex.prototype.initLink = function (naryLink) {\n this.naryLink = naryLink;\n this.naryLink.path.classList.add(\"complex-outline\");\n};\n\nComplex.prototype.setLinked = function () {\n\n this.naryLink.path2.classList.add(\"linked-complex\");\n};\n\n\nComplex.prototype.getPosition = function (originPoint) {\n const mapped = this.naryLink.mapped;//getMappedCoordinates();\n const mc = mapped.length;\n let xSum = 0,\n ySum = 0;\n for (let m = 0; m < mc; m++) {\n xSum += mapped[m][0];\n ySum += mapped[m][1];\n }\n let center = [xSum / mc, ySum / mc];\n if (originPoint) {\n // if (participant.type === \"complex\"){\n // startPoint = participant.getPosition();\n let naryPath = this.naryLink.hull;\n let iPath = [];\n for (let p of naryPath) {\n iPath.push(new point2d__WEBPACK_IMPORTED_MODULE_1__(p[0], p[1]));\n }\n let a1 = new point2d__WEBPACK_IMPORTED_MODULE_1__(center[0], center[1]);\n let a2 = new point2d__WEBPACK_IMPORTED_MODULE_1__(originPoint[0], originPoint[1]);\n let intersect = intersectionjs__WEBPACK_IMPORTED_MODULE_2__[\"intersectLinePolygon\"](a1, a2, iPath);\n if (intersect.points[0]) {\n return [intersect.points[0].x, intersect.points[0].y];\n }\n this.setLinked();\n }\n return center;\n};\n\nComplex.prototype.setPosition = function () {\n console.error(\"bad - called setPosition on \", this);\n};\n\nComplex.prototype.changePosition = function (dx, dy) {\n for (let participant of this.naryLink.participants){\n participant.changePosition(dx, dy);\n }\n};\n\nComplex.prototype.getResidueCoordinates = function () {\n return this.getPosition();\n};\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvdml6L2ludGVyYWN0b3IvY29tcGxleC5qcy5qcyIsInNvdXJjZXMiOlsid2VicGFjazovL2NvbXBsZXh2aWV3ZXIvLi9zcmMvanMvdml6L2ludGVyYWN0b3IvY29tcGxleC5qcz8zOTQ4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7SW50ZXJhY3Rvcn0gZnJvbSBcIi4vaW50ZXJhY3RvclwiO1xuaW1wb3J0ICogYXMgUG9pbnQyRCBmcm9tIFwicG9pbnQyZFwiO1xuaW1wb3J0ICogYXMgSW50ZXJzZWN0aW9uIGZyb20gXCJpbnRlcnNlY3Rpb25qc1wiO1xuXG5leHBvcnQgZnVuY3Rpb24gQ29tcGxleChpZCwgYXBwKSB7XG4gICAgdGhpcy5pbml0KGlkLCBhcHApO1xuICAgIHRoaXMudHlwZSA9IFwiY29tcGxleFwiO1xuICAgIHRoaXMucGFkZGluZyA9IDIyO1xuXG4gICAgLy8gY29uc3Qgc2VsZiA9IHRoaXM7XG4gICAgLy8gLy8gaXRzIGJhZCBpZiB5b3UgZW5kIHVwIHdpdGggdGhlc2UgZ2V0dGluZyBjYWxsZWRcbiAgICAvLyBPYmplY3QuZGVmaW5lUHJvcGVydHkodGhpcywgXCJ3aWR0aFwiLCB7XG4gICAgLy8gICAgIGdldDogZnVuY3Rpb24gaGVpZ2h0KCkge1xuICAgIC8vICAgICAgICAgcmV0dXJuIHNlbGYubmFyeUxpbmsucGF0aC5nZXRCQm94KCkud2lkdGg7XG4gICAgLy8gICAgICAgICAvL3JldHVybiAxNjA7XG4gICAgLy8gICAgIH1cbiAgICAvLyB9KTtcbiAgICAvLyBPYmplY3QuZGVmaW5lUHJvcGVydHkodGhpcywgXCJoZWlnaHRcIiwge1xuICAgIC8vICAgICBnZXQ6IGZ1bmN0aW9uIGhlaWdodCgpIHtcbiAgICAvLyAgICAgICAgIHJldHVybiBzZWxmLm5hcnlMaW5rLnBhdGguZ2V0QkJveCgpLmhlaWdodDtcbiAgICAvLyAgICAgICAgIC8vcmV0dXJuIDE2MDtcbiAgICAvLyAgICAgfVxuICAgIC8vIH0pO1xufVxuXG5Db21wbGV4LnByb3RvdHlwZSA9IG5ldyBJbnRlcmFjdG9yKCk7XG5cbkNvbXBsZXgucHJvdG90eXBlLmluaXRMaW5rID0gZnVuY3Rpb24gKG5hcnlMaW5rKSB7XG4gICAgdGhpcy5uYXJ5TGluayA9IG5hcnlMaW5rO1xuICAgIHRoaXMubmFyeUxpbmsucGF0aC5jbGFzc0xpc3QuYWRkKFwiY29tcGxleC1vdXRsaW5lXCIpO1xufTtcblxuQ29tcGxleC5wcm90b3R5cGUuc2V0TGlua2VkID0gZnVuY3Rpb24gKCkge1xuXG4gICAgdGhpcy5uYXJ5TGluay5wYXRoMi5jbGFzc0xpc3QuYWRkKFwibGlua2VkLWNvbXBsZXhcIik7XG59O1xuXG5cbkNvbXBsZXgucHJvdG90eXBlLmdldFBvc2l0aW9uID0gZnVuY3Rpb24gKG9yaWdpblBvaW50KSB7XG4gICAgY29uc3QgbWFwcGVkID0gdGhpcy5uYXJ5TGluay5tYXBwZWQ7Ly9nZXRNYXBwZWRDb29yZGluYXRlcygpO1xuICAgIGNvbnN0IG1jID0gbWFwcGVkLmxlbmd0aDtcbiAgICBsZXQgeFN1bSA9IDAsXG4gICAgICAgIHlTdW0gPSAwO1xuICAgIGZvciAobGV0IG0gPSAwOyBtIDwgbWM7IG0rKykge1xuICAgICAgICB4U3VtICs9IG1hcHBlZFttXVswXTtcbiAgICAgICAgeVN1bSArPSBtYXBwZWRbbV1bMV07XG4gICAgfVxuICAgIGxldCBjZW50ZXIgPSBbeFN1bSAvIG1jLCB5U3VtIC8gbWNdO1xuICAgIGlmIChvcmlnaW5Qb2ludCkge1xuICAgIC8vIGlmIChwYXJ0aWNpcGFudC50eXBlID09PSBcImNvbXBsZXhcIil7XG4gICAgLy8gICAgIHN0YXJ0UG9pbnQgPSBwYXJ0aWNpcGFudC5nZXRQb3NpdGlvbigpO1xuICAgICAgICBsZXQgbmFyeVBhdGggPSB0aGlzLm5hcnlMaW5rLmh1bGw7XG4gICAgICAgIGxldCBpUGF0aCA9IFtdO1xuICAgICAgICBmb3IgKGxldCBwIG9mIG5hcnlQYXRoKSB7XG4gICAgICAgICAgICBpUGF0aC5wdXNoKG5ldyBQb2ludDJEKHBbMF0sIHBbMV0pKTtcbiAgICAgICAgfVxuICAgICAgICBsZXQgYTEgPSBuZXcgUG9pbnQyRChjZW50ZXJbMF0sIGNlbnRlclsxXSk7XG4gICAgICAgIGxldCBhMiA9IG5ldyBQb2ludDJEKG9yaWdpblBvaW50WzBdLCBvcmlnaW5Qb2ludFsxXSk7XG4gICAgICAgIGxldCBpbnRlcnNlY3QgPSBJbnRlcnNlY3Rpb24uaW50ZXJzZWN0TGluZVBvbHlnb24oYTEsIGEyLCBpUGF0aCk7XG4gICAgICAgIGlmIChpbnRlcnNlY3QucG9pbnRzWzBdKSB7XG4gICAgICAgICAgICByZXR1cm4gW2ludGVyc2VjdC5wb2ludHNbMF0ueCwgaW50ZXJzZWN0LnBvaW50c1swXS55XTtcbiAgICAgICAgfVxuICAgICAgICB0aGlzLnNldExpbmtlZCgpO1xuICAgIH1cbiAgICByZXR1cm4gY2VudGVyO1xufTtcblxuQ29tcGxleC5wcm90b3R5cGUuc2V0UG9zaXRpb24gPSBmdW5jdGlvbiAoKSB7XG4gICAgY29uc29sZS5lcnJvcihcImJhZCAtIGNhbGxlZCBzZXRQb3NpdGlvbiBvbiBcIiwgdGhpcyk7XG59O1xuXG5Db21wbGV4LnByb3RvdHlwZS5jaGFuZ2VQb3NpdGlvbiA9IGZ1bmN0aW9uIChkeCwgZHkpIHtcbiAgICBmb3IgKGxldCBwYXJ0aWNpcGFudCBvZiB0aGlzLm5hcnlMaW5rLnBhcnRpY2lwYW50cyl7XG4gICAgICAgIHBhcnRpY2lwYW50LmNoYW5nZVBvc2l0aW9uKGR4LCBkeSk7XG4gICAgfVxufTtcblxuQ29tcGxleC5wcm90b3R5cGUuZ2V0UmVzaWR1ZUNvb3JkaW5hdGVzID0gZnVuY3Rpb24gKCkge1xuICAgIHJldHVybiB0aGlzLmdldFBvc2l0aW9uKCk7XG59O1xuXG4iXSwibWFwcGluZ3MiOiJBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOyIsInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./src/js/viz/interactor/complex.js\n"); /***/ }), @@ -1285,7 +1297,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"Interactor\", function() { return Interactor; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"trig\", function() { return trig; });\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n\n\nfunction Interactor() {\n}\n\nInteractor.prototype = {\n get width() {\n // console.log(this.upperGroup.getBBox().width);\n return this.upperGroup.getBBox().width;\n },\n get height() {\n return 40;//this.upperGroup.getBBox().height;\n },\n};\n\nInteractor.prototype.init = function (id, app, json, name){\n this.id = id;\n this.app = app;\n this.json = json;\n this.name = name;\n\n //annotations indexed by annotation set name (\"MI FEATURES\", \"SUPERFAMILY\", etc)\n this.annotationSets = new Map();\n\n //links\n this.naryLinks = new Map();\n this.binaryLinks = new Map();\n this.sequenceLinks = new Map();\n};\n\nInteractor.prototype.initLabel = function (){\n\n this.labelSVG = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_0__[\"svgns\"], \"text\");\n this.labelSVG.setAttribute(\"text-anchor\", \"end\");\n this.labelSVG.setAttribute(\"fill\", \"black\");\n this.labelSVG.setAttribute(\"x\", \"0\");\n this.labelSVG.setAttribute(\"y\", \"10\");\n this.labelSVG.setAttribute(\"class\", \"xlv_text proteinLabel\");\n this.labelSVG.setAttribute(\"font-family\", \"Arial\");\n this.labelSVG.setAttribute(\"font-size\", \"16\");\n\n //choose label text\n if (this.name) {\n this.labelText = this.name;\n } else {\n this.labelText = this.id;\n }\n if (this.labelText.length > 25) {\n this.labelText = this.labelText.substr(0, 16) + \"...\";\n }\n\n this.labelText = this.name;\n this.labelTextNode = document.createTextNode(this.labelText);\n this.labelSVG.appendChild(this.labelTextNode);\n this.labelSVG.setAttribute(\"transform\",\n \"translate( -\" + this.getSymbolRadius() + \" \" + _config__WEBPACK_IMPORTED_MODULE_0__[\"LABEL_Y\"] + \")\");\n this.upperGroup.appendChild(this.labelSVG);\n};\n\nInteractor.prototype.initOutline = function (){\n this.outline.setAttribute(\"stroke\", \"black\");\n this.outline.setAttribute(\"stroke-width\", \"1\");\n this.outline.setAttribute(\"stroke-opacity\", \"1\");\n this.outline.setAttribute(\"fill-opacity\", \"1\");\n this.outline.setAttribute(\"fill\", \"#ffffff\");\n //append outline\n this.upperGroup.appendChild(this.outline);\n};\n\nInteractor.prototype.initListeners = function (){\n // events\n const self = this;\n // this.upperGroup.setAttribute('pointer-events','all');\n this.upperGroup.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.upperGroup.onmouseover = function (evt) {\n self.mouseOver(evt);\n };\n this.upperGroup.onmouseout = function (evt) {\n self.mouseOut(evt);\n };\n // this.upperGroup.ontouchstart = function (evt) {\n // self.touchStart(evt);\n // };\n\n //~ this.upperGroup.ontouchmove = function(evt) {};\n //~ this.upperGroup.ontouchend = function(evt) {\n //~ self.ctrl.message(\"protein touch end\");\n //~ self.mouseOut(evt);\n //~ };\n //~ this.upperGroup.ontouchenter = function(evt) {\n //~ self.message(\"protein touch enter\");\n //~ self.touchStart(evt);\n //~ };\n //~ this.upperGroup.ontouchleave = function(evt) {\n //~ self.message(\"protein touch leave\");\n //~ self.mouseOut(evt);\n //~ };\n //~ this.upperGroup.ontouchcancel = function(evt) {\n //~ self.message(\"protein touch cancel\");\n //~ self.mouseOut(evt);\n //~ };\n};\n\nInteractor.prototype.addStoichiometryLabel = function (stoichiometry) {\n if (this.labelSVG) { //complexes don't have labels (yet?)\n // noinspection JSUndefinedPropertyAssignment\n this.labelSVG.childNodes[0].data = this.labelSVG.childNodes[0].data + \" [\" + stoichiometry + \"]\";\n }\n};\n\nInteractor.prototype.mouseDown = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt); //see MouseEvents.js\n this.app.d3cola.stop();\n this.app.dragElement = this;\n const p = this.app.getEventPoint(evt);\n this.app.dragStart = this.app.mouseToSVG(p.x, p.y);\n return false;\n};\n\n//// TODO: test on touch screen\n// Interactor.prototype.touchStart = function(evt) {\n// this.util.preventDefaultsAndStopPropagation(evt); //see MouseEvents.js\n// if (this.util.d3cola !== undefined) {\n// this.util.d3cola.stop();\n// }\n// this.util.dragElement = this;\n// //store start location\n// var p = this.util.getTouchEventPoint(evt);\n// this.util.dragStart = this.util.mouseToSVG(p.x, p.y);\n// return false;\n// };\n\nInteractor.prototype.mouseOver = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt);\n this.showHighlight(true);\n //~ this.util.setTooltip(this.id);\n return false;\n};\n\nInteractor.prototype.mouseOut = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt);\n this.showHighlight(false);\n this.app.hideTooltip();\n return false;\n};\n\nInteractor.prototype.getSymbolRadius = function () {\n return 15;\n};\n\n\nInteractor.prototype.showHighlight = function () {\n};\n\nInteractor.prototype.getPosition = function () {\n return [this.ix, this.iy]; // todo - type of return is kind of inconsistent\n};\n\nInteractor.prototype.setPosition = function (x, y) {\n this.px = this.ix;\n this.py = this.iy;\n this.ix = x;\n this.iy = y;\n this.upperGroup.setAttribute(\"transform\", \"translate(\" + this.ix + \" \" + this.iy + \")\");\n};\n\nInteractor.prototype.changePosition = function (x, y) {\n this.px = this.ix;\n this.py = this.iy;\n this.ix -= x;\n this.iy -= y;\n this.upperGroup.setAttribute(\"transform\", \"translate(\" + this.ix + \" \" + this.iy + \")\");\n // this.setAllLinkCoordinates(); // todo - look at calls\n};\n\nInteractor.prototype.getAggregateSelfLinkPath = function () {\n const intraR = this.getSymbolRadius() + 7;\n const sectorSize = 45;\n const arcStart = trig(intraR, 25 + sectorSize);\n const arcEnd = trig(intraR, -25 + sectorSize);\n const cp1 = trig(intraR, 40 + sectorSize);\n const cp2 = trig(intraR, -40 + sectorSize);\n return \"M 0,0 \" +\n \"Q \" + cp1.x + \",\" + -cp1.y + \" \" + arcStart.x + \",\" + -arcStart.y +\n \" A \" + intraR + \" \" + intraR + \" 0 0 1 \" + arcEnd.x + \",\" + -arcEnd.y +\n \" Q \" + cp2.x + \",\" + -cp2.y + \" 0,0\";\n};\n\nInteractor.prototype.checkLinks = function () {\n function checkAll(linkMap) {\n for (let link of linkMap.values()) {\n link.check();\n }\n }\n\n // checkAll(this.naryLinks); // hacked out to fix ordering of nLinks\n checkAll(this.binaryLinks);\n checkAll(this.sequenceLinks);\n if (this.selfLink) {\n this.selfLink.check();\n }\n};\n\n// update all lines (e.g after a move)\nInteractor.prototype.setAllLinkCoordinates = function () {\n for (let link of this.naryLinks.values()) {\n link.setLinkCoordinates();\n }\n for (let link of this.binaryLinks.values()) {\n link.setLinkCoordinates();\n }\n if (this.selfLink) {\n this.selfLink.setLinkCoordinates();\n }\n for (let link of this.sequenceLinks.values()) {\n link.setLinkCoordinates();\n }\n};\n\nfunction trig (radius, angleDegrees) {\n //x = rx + radius * cos(theta) and y = ry + radius * sin(theta)\n const radians = (angleDegrees / 360) * Math.PI * 2;\n return {\n x: (radius * Math.cos(radians)),\n y: (radius * Math.sin(radians))\n };\n}\n//\n// Interactor.prototype.setForm = function () {\n// };\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/js/viz/interactor/interactor.js.js","sources":["webpack://complexviewer/./src/js/viz/interactor/interactor.js?e808"],"sourcesContent":["import {LABEL_Y, svgns} from \"../../config\";\n\nexport function Interactor() {\n}\n\nInteractor.prototype = {\n    get width() {\n        // console.log(this.upperGroup.getBBox().width);\n        return this.upperGroup.getBBox().width;\n    },\n    get height() {\n        return 40;//this.upperGroup.getBBox().height;\n    },\n};\n\nInteractor.prototype.init = function (id, app, json, name){\n    this.id = id;\n    this.app = app;\n    this.json = json;\n    this.name = name;\n\n    //annotations indexed by annotation set name (\"MI FEATURES\", \"SUPERFAMILY\", etc)\n    this.annotationSets = new Map();\n\n    //links\n    this.naryLinks = new Map();\n    this.binaryLinks = new Map();\n    this.sequenceLinks = new Map();\n};\n\nInteractor.prototype.initLabel = function (){\n\n    this.labelSVG = document.createElementNS(svgns, \"text\");\n    this.labelSVG.setAttribute(\"text-anchor\", \"end\");\n    this.labelSVG.setAttribute(\"fill\", \"black\");\n    this.labelSVG.setAttribute(\"x\", \"0\");\n    this.labelSVG.setAttribute(\"y\", \"10\");\n    this.labelSVG.setAttribute(\"class\", \"xlv_text proteinLabel\");\n    this.labelSVG.setAttribute(\"font-family\", \"Arial\");\n    this.labelSVG.setAttribute(\"font-size\", \"16\");\n\n    //choose label text\n    if (this.name) {\n        this.labelText = this.name;\n    } else {\n        this.labelText = this.id;\n    }\n    if (this.labelText.length > 25) {\n        this.labelText = this.labelText.substr(0, 16) + \"...\";\n    }\n\n    this.labelText = this.name;\n    this.labelTextNode = document.createTextNode(this.labelText);\n    this.labelSVG.appendChild(this.labelTextNode);\n    this.labelSVG.setAttribute(\"transform\",\n        \"translate( -\" + this.getSymbolRadius() + \" \" + LABEL_Y + \")\");\n    this.upperGroup.appendChild(this.labelSVG);\n};\n\nInteractor.prototype.initOutline = function (){\n    this.outline.setAttribute(\"stroke\", \"black\");\n    this.outline.setAttribute(\"stroke-width\", \"1\");\n    this.outline.setAttribute(\"stroke-opacity\", \"1\");\n    this.outline.setAttribute(\"fill-opacity\", \"1\");\n    this.outline.setAttribute(\"fill\", \"#ffffff\");\n    //append outline\n    this.upperGroup.appendChild(this.outline);\n};\n\nInteractor.prototype.initListeners = function (){\n    // events\n    const self = this;\n    //    this.upperGroup.setAttribute('pointer-events','all');\n    this.upperGroup.onmousedown = function (evt) {\n        self.mouseDown(evt);\n    };\n    this.upperGroup.onmouseover = function (evt) {\n        self.mouseOver(evt);\n    };\n    this.upperGroup.onmouseout = function (evt) {\n        self.mouseOut(evt);\n    };\n    // this.upperGroup.ontouchstart = function (evt) {\n    //     self.touchStart(evt);\n    // };\n\n    //~ this.upperGroup.ontouchmove = function(evt) {};\n    //~ this.upperGroup.ontouchend = function(evt) {\n    //~ self.ctrl.message(\"protein touch end\");\n    //~ self.mouseOut(evt);\n    //~ };\n    //~ this.upperGroup.ontouchenter = function(evt) {\n    //~ self.message(\"protein touch enter\");\n    //~ self.touchStart(evt);\n    //~ };\n    //~ this.upperGroup.ontouchleave = function(evt) {\n    //~ self.message(\"protein touch leave\");\n    //~ self.mouseOut(evt);\n    //~ };\n    //~ this.upperGroup.ontouchcancel = function(evt) {\n    //~ self.message(\"protein touch cancel\");\n    //~ self.mouseOut(evt);\n    //~ };\n};\n\nInteractor.prototype.addStoichiometryLabel = function (stoichiometry) {\n    if (this.labelSVG) { //complexes don't have labels (yet?)\n        // noinspection JSUndefinedPropertyAssignment\n        this.labelSVG.childNodes[0].data = this.labelSVG.childNodes[0].data + \" [\" + stoichiometry + \"]\";\n    }\n};\n\nInteractor.prototype.mouseDown = function (evt) {\n    this.app.preventDefaultsAndStopPropagation(evt); //see MouseEvents.js\n    this.app.d3cola.stop();\n    this.app.dragElement = this;\n    const p = this.app.getEventPoint(evt);\n    this.app.dragStart = this.app.mouseToSVG(p.x, p.y);\n    return false;\n};\n\n//// TODO: test on touch screen\n// Interactor.prototype.touchStart = function(evt) {\n//     this.util.preventDefaultsAndStopPropagation(evt); //see MouseEvents.js\n//     if (this.util.d3cola !== undefined) {\n//         this.util.d3cola.stop();\n//     }\n//     this.util.dragElement = this;\n//     //store start location\n//     var p = this.util.getTouchEventPoint(evt);\n//     this.util.dragStart = this.util.mouseToSVG(p.x, p.y);\n//     return false;\n// };\n\nInteractor.prototype.mouseOver = function (evt) {\n    this.app.preventDefaultsAndStopPropagation(evt);\n    this.showHighlight(true);\n    //~ this.util.setTooltip(this.id);\n    return false;\n};\n\nInteractor.prototype.mouseOut = function (evt) {\n    this.app.preventDefaultsAndStopPropagation(evt);\n    this.showHighlight(false);\n    this.app.hideTooltip();\n    return false;\n};\n\nInteractor.prototype.getSymbolRadius = function () {\n    return 15;\n};\n\n\nInteractor.prototype.showHighlight = function () {\n};\n\nInteractor.prototype.getPosition = function () {\n    return [this.ix, this.iy]; // todo - type of return is kind of inconsistent\n};\n\nInteractor.prototype.setPosition = function (x, y) {\n    this.px = this.ix;\n    this.py = this.iy;\n    this.ix = x;\n    this.iy = y;\n    this.upperGroup.setAttribute(\"transform\", \"translate(\" + this.ix + \" \" + this.iy + \")\");\n};\n\nInteractor.prototype.changePosition = function (x, y) {\n    this.px = this.ix;\n    this.py = this.iy;\n    this.ix -= x;\n    this.iy -= y;\n    this.upperGroup.setAttribute(\"transform\", \"translate(\" + this.ix + \" \" + this.iy + \")\");\n    // this.setAllLinkCoordinates(); // todo - look at calls\n};\n\nInteractor.prototype.getAggregateSelfLinkPath = function () {\n    const intraR = this.getSymbolRadius() + 7;\n    const sectorSize = 45;\n    const arcStart = trig(intraR, 25 + sectorSize);\n    const arcEnd = trig(intraR, -25 + sectorSize);\n    const cp1 = trig(intraR, 40 + sectorSize);\n    const cp2 = trig(intraR, -40 + sectorSize);\n    return \"M 0,0 \" +\n        \"Q \" + cp1.x + \",\" + -cp1.y + \" \" + arcStart.x + \",\" + -arcStart.y +\n        \" A \" + intraR + \" \" + intraR + \" 0 0 1 \" + arcEnd.x + \",\" + -arcEnd.y +\n        \" Q \" + cp2.x + \",\" + -cp2.y + \" 0,0\";\n};\n\nInteractor.prototype.checkLinks = function () {\n    function checkAll(linkMap) {\n        for (let link of linkMap.values()) {\n            link.check();\n        }\n    }\n\n    // checkAll(this.naryLinks); // hacked out to fix ordering of nLinks\n    checkAll(this.binaryLinks);\n    checkAll(this.sequenceLinks);\n    if (this.selfLink) {\n        this.selfLink.check();\n    }\n};\n\n// update all lines (e.g after a move)\nInteractor.prototype.setAllLinkCoordinates = function () {\n    for (let link of this.naryLinks.values()) {\n        link.setLinkCoordinates();\n    }\n    for (let link of this.binaryLinks.values()) {\n        link.setLinkCoordinates();\n    }\n    if (this.selfLink) {\n        this.selfLink.setLinkCoordinates();\n    }\n    for (let link of this.sequenceLinks.values()) {\n            link.setLinkCoordinates();\n    }\n};\n\nexport function trig (radius, angleDegrees) {\n    //x = rx + radius * cos(theta) and y = ry + radius * sin(theta)\n    const radians = (angleDegrees / 360) * Math.PI * 2;\n    return {\n        x: (radius * Math.cos(radians)),\n        y: (radius * Math.sin(radians))\n    };\n}\n//\n// Interactor.prototype.setForm = function () {\n// };\n"],"mappings":"AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/js/viz/interactor/interactor.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"Interactor\", function() { return Interactor; });\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"trig\", function() { return trig; });\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n\n\nfunction Interactor() {\n}\n\nInteractor.prototype = {\n get width() {\n // console.log(this.upperGroup.getBBox().width);\n return this.upperGroup.getBBox().width;\n },\n get height() {\n return 40;//this.upperGroup.getBBox().height;\n },\n};\n\nInteractor.prototype.init = function (id, app, json, name){\n this.id = id;\n this.app = app;\n this.json = json;\n this.name = name;\n\n //todo - think 'type' should be a property here (except for complex, can just return json.type.name)\n\n //annotations indexed by annotation set name (\"MIFEATURES\", \"SUPERFAMILY\", etc)\n this.annotationSets = new Map();\n //links\n this.naryLinks = new Map();\n this.binaryLinks = new Map();\n this.sequenceLinks = new Map();\n};\n\nInteractor.prototype.initLabel = function (){\n this.labelSVG = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_0__[\"svgns\"], \"text\");\n this.labelSVG.setAttribute(\"x\", \"0\"); // css?\n this.labelSVG.setAttribute(\"y\", \"10\");\n this.labelSVG.classList.add(\"label\");\n //choose label text\n if (this.name) {\n this.labelText = this.name;\n } else {\n this.labelText = this.id;\n }\n if (this.labelText.length > 25) {\n this.labelText = this.labelText.substr(0, 16) + \"...\";\n }\n this.labelText = this.name;\n this.labelTextNode = document.createTextNode(this.labelText);\n this.labelSVG.appendChild(this.labelTextNode);\n this.labelSVG.setAttribute(\"transform\",\n \"translate( -\" + this.getSymbolRadius() + \" \" + _config__WEBPACK_IMPORTED_MODULE_0__[\"LABEL_Y\"] + \")\");\n this.upperGroup.appendChild(this.labelSVG);\n};\n\nInteractor.prototype.initOutline = function (){\n this.outline.classList.add(\"outline\");\n this.upperGroup.appendChild(this.outline);\n};\n\nInteractor.prototype.initListeners = function (){\n // events\n const self = this;\n // this.upperGroup.setAttribute('pointer-events','all');\n this.upperGroup.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.upperGroup.onmouseover = function (evt) {\n self.mouseOver(evt);\n };\n this.upperGroup.onmouseout = function (evt) {\n self.mouseOut(evt);\n };\n // this.upperGroup.ontouchstart = function (evt) {\n // self.touchStart(evt);\n // };\n\n //~ this.upperGroup.ontouchmove = function(evt) {};\n //~ this.upperGroup.ontouchend = function(evt) {\n //~ self.ctrl.message(\"protein touch end\");\n //~ self.mouseOut(evt);\n //~ };\n //~ this.upperGroup.ontouchenter = function(evt) {\n //~ self.message(\"protein touch enter\");\n //~ self.touchStart(evt);\n //~ };\n //~ this.upperGroup.ontouchleave = function(evt) {\n //~ self.message(\"protein touch leave\");\n //~ self.mouseOut(evt);\n //~ };\n //~ this.upperGroup.ontouchcancel = function(evt) {\n //~ self.message(\"protein touch cancel\");\n //~ self.mouseOut(evt);\n //~ };\n};\n\nInteractor.prototype.addStoichiometryLabel = function (stoichiometry) {\n if (this.labelSVG) { //complexes don't have labels (yet?)\n // noinspection JSUndefinedPropertyAssignment\n this.labelSVG.childNodes[0].data = this.labelSVG.childNodes[0].data + \" [\" + stoichiometry + \"]\";\n }\n};\n\nInteractor.prototype.mouseDown = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt); //see MouseEvents.js\n this.app.d3cola.stop();\n this.app.dragElement = this;\n const p = this.app.getEventPoint(evt);\n this.app.dragStart = this.app.mouseToSVG(p.x, p.y);\n return false;\n};\n\n//// TODO: test on touch screen\n// Interactor.prototype.touchStart = function(evt) {\n// this.util.preventDefaultsAndStopPropagation(evt); //see MouseEvents.js\n// if (this.util.d3cola !== undefined) {\n// this.util.d3cola.stop();\n// }\n// this.util.dragElement = this;\n// //store start location\n// var p = this.util.getTouchEventPoint(evt);\n// this.util.dragStart = this.util.mouseToSVG(p.x, p.y);\n// return false;\n// };\n\nInteractor.prototype.mouseOver = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt);\n this.showHighlight(true);\n //~ this.util.setTooltip(this.id);\n return false;\n};\n\nInteractor.prototype.mouseOut = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt);\n this.showHighlight(false);\n this.app.hideTooltip();\n return false;\n};\n\nInteractor.prototype.getSymbolRadius = function () {\n return 15;\n};\n\n\nInteractor.prototype.showHighlight = function () {\n};\n\nInteractor.prototype.getPosition = function () {\n return [this.ix, this.iy]; // todo - type of return is kind of inconsistent\n};\n\nInteractor.prototype.setPosition = function (x, y) {\n this.px = this.ix;\n this.py = this.iy;\n this.ix = x;\n this.iy = y;\n this.upperGroup.setAttribute(\"transform\", \"translate(\" + this.ix + \" \" + this.iy + \")\");\n};\n\nInteractor.prototype.changePosition = function (x, y) {\n this.px = this.ix;\n this.py = this.iy;\n this.ix -= x;\n this.iy -= y;\n this.upperGroup.setAttribute(\"transform\", \"translate(\" + this.ix + \" \" + this.iy + \")\");\n // this.setAllLinkCoordinates(); // todo - look at calls\n};\n\nInteractor.prototype.getAggregateSelfLinkPath = function () {\n const intraR = this.getSymbolRadius() + 7;\n const sectorSize = 45;\n const arcStart = trig(intraR, 25 + sectorSize);\n const arcEnd = trig(intraR, -25 + sectorSize);\n const cp1 = trig(intraR, 40 + sectorSize);\n const cp2 = trig(intraR, -40 + sectorSize);\n return \"M 0,0 \" +\n \"Q \" + cp1.x + \",\" + -cp1.y + \" \" + arcStart.x + \",\" + -arcStart.y +\n \" A \" + intraR + \" \" + intraR + \" 0 0 1 \" + arcEnd.x + \",\" + -arcEnd.y +\n \" Q \" + cp2.x + \",\" + -cp2.y + \" 0,0\";\n};\n\nInteractor.prototype.checkLinks = function () {\n function checkAll(linkMap) {\n for (let link of linkMap.values()) {\n link.check();\n }\n }\n\n // checkAll(this.naryLinks); // hacked out to fix ordering of nLinks\n checkAll(this.binaryLinks);\n checkAll(this.sequenceLinks);\n if (this.selfLink) {\n this.selfLink.check();\n }\n};\n\n// update all lines (e.g after a move)\nInteractor.prototype.setAllLinkCoordinates = function () {\n for (let link of this.naryLinks.values()) {\n link.setLinkCoordinates();\n }\n for (let link of this.binaryLinks.values()) {\n link.setLinkCoordinates();\n }\n if (this.selfLink) {\n this.selfLink.setLinkCoordinates();\n }\n for (let link of this.sequenceLinks.values()) {\n link.setLinkCoordinates();\n }\n};\n\nfunction trig (radius, angleDegrees) {\n //x = rx + radius * cos(theta) and y = ry + radius * sin(theta)\n const radians = (angleDegrees / 360) * Math.PI * 2;\n return {\n x: (radius * Math.cos(radians)),\n y: (radius * Math.sin(radians))\n };\n}\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/js/viz/interactor/interactor.js.js","sources":["webpack://complexviewer/./src/js/viz/interactor/interactor.js?e808"],"sourcesContent":["import {LABEL_Y, svgns} from \"../../config\";\n\nexport function Interactor() {\n}\n\nInteractor.prototype = {\n    get width() {\n        // console.log(this.upperGroup.getBBox().width);\n        return this.upperGroup.getBBox().width;\n    },\n    get height() {\n        return 40;//this.upperGroup.getBBox().height;\n    },\n};\n\nInteractor.prototype.init = function (id, app, json, name){\n    this.id = id;\n    this.app = app;\n    this.json = json;\n    this.name = name;\n\n    //todo - think 'type' should be  a property here (except for complex, can just return json.type.name)\n\n    //annotations indexed by annotation set name (\"MIFEATURES\", \"SUPERFAMILY\", etc)\n    this.annotationSets = new Map();\n    //links\n    this.naryLinks = new Map();\n    this.binaryLinks = new Map();\n    this.sequenceLinks = new Map();\n};\n\nInteractor.prototype.initLabel = function (){\n    this.labelSVG = document.createElementNS(svgns, \"text\");\n    this.labelSVG.setAttribute(\"x\", \"0\"); // css?\n    this.labelSVG.setAttribute(\"y\", \"10\");\n    this.labelSVG.classList.add(\"label\");\n    //choose label text\n    if (this.name) {\n        this.labelText = this.name;\n    } else {\n        this.labelText = this.id;\n    }\n    if (this.labelText.length > 25) {\n        this.labelText = this.labelText.substr(0, 16) + \"...\";\n    }\n    this.labelText = this.name;\n    this.labelTextNode = document.createTextNode(this.labelText);\n    this.labelSVG.appendChild(this.labelTextNode);\n    this.labelSVG.setAttribute(\"transform\",\n        \"translate( -\" + this.getSymbolRadius() + \" \" + LABEL_Y + \")\");\n    this.upperGroup.appendChild(this.labelSVG);\n};\n\nInteractor.prototype.initOutline = function (){\n    this.outline.classList.add(\"outline\");\n    this.upperGroup.appendChild(this.outline);\n};\n\nInteractor.prototype.initListeners = function (){\n    // events\n    const self = this;\n    //    this.upperGroup.setAttribute('pointer-events','all');\n    this.upperGroup.onmousedown = function (evt) {\n        self.mouseDown(evt);\n    };\n    this.upperGroup.onmouseover = function (evt) {\n        self.mouseOver(evt);\n    };\n    this.upperGroup.onmouseout = function (evt) {\n        self.mouseOut(evt);\n    };\n    // this.upperGroup.ontouchstart = function (evt) {\n    //     self.touchStart(evt);\n    // };\n\n    //~ this.upperGroup.ontouchmove = function(evt) {};\n    //~ this.upperGroup.ontouchend = function(evt) {\n    //~ self.ctrl.message(\"protein touch end\");\n    //~ self.mouseOut(evt);\n    //~ };\n    //~ this.upperGroup.ontouchenter = function(evt) {\n    //~ self.message(\"protein touch enter\");\n    //~ self.touchStart(evt);\n    //~ };\n    //~ this.upperGroup.ontouchleave = function(evt) {\n    //~ self.message(\"protein touch leave\");\n    //~ self.mouseOut(evt);\n    //~ };\n    //~ this.upperGroup.ontouchcancel = function(evt) {\n    //~ self.message(\"protein touch cancel\");\n    //~ self.mouseOut(evt);\n    //~ };\n};\n\nInteractor.prototype.addStoichiometryLabel = function (stoichiometry) {\n    if (this.labelSVG) { //complexes don't have labels (yet?)\n        // noinspection JSUndefinedPropertyAssignment\n        this.labelSVG.childNodes[0].data = this.labelSVG.childNodes[0].data + \" [\" + stoichiometry + \"]\";\n    }\n};\n\nInteractor.prototype.mouseDown = function (evt) {\n    this.app.preventDefaultsAndStopPropagation(evt); //see MouseEvents.js\n    this.app.d3cola.stop();\n    this.app.dragElement = this;\n    const p = this.app.getEventPoint(evt);\n    this.app.dragStart = this.app.mouseToSVG(p.x, p.y);\n    return false;\n};\n\n//// TODO: test on touch screen\n// Interactor.prototype.touchStart = function(evt) {\n//     this.util.preventDefaultsAndStopPropagation(evt); //see MouseEvents.js\n//     if (this.util.d3cola !== undefined) {\n//         this.util.d3cola.stop();\n//     }\n//     this.util.dragElement = this;\n//     //store start location\n//     var p = this.util.getTouchEventPoint(evt);\n//     this.util.dragStart = this.util.mouseToSVG(p.x, p.y);\n//     return false;\n// };\n\nInteractor.prototype.mouseOver = function (evt) {\n    this.app.preventDefaultsAndStopPropagation(evt);\n    this.showHighlight(true);\n    //~ this.util.setTooltip(this.id);\n    return false;\n};\n\nInteractor.prototype.mouseOut = function (evt) {\n    this.app.preventDefaultsAndStopPropagation(evt);\n    this.showHighlight(false);\n    this.app.hideTooltip();\n    return false;\n};\n\nInteractor.prototype.getSymbolRadius = function () {\n    return 15;\n};\n\n\nInteractor.prototype.showHighlight = function () {\n};\n\nInteractor.prototype.getPosition = function () {\n    return [this.ix, this.iy]; // todo - type of return is kind of inconsistent\n};\n\nInteractor.prototype.setPosition = function (x, y) {\n    this.px = this.ix;\n    this.py = this.iy;\n    this.ix = x;\n    this.iy = y;\n    this.upperGroup.setAttribute(\"transform\", \"translate(\" + this.ix + \" \" + this.iy + \")\");\n};\n\nInteractor.prototype.changePosition = function (x, y) {\n    this.px = this.ix;\n    this.py = this.iy;\n    this.ix -= x;\n    this.iy -= y;\n    this.upperGroup.setAttribute(\"transform\", \"translate(\" + this.ix + \" \" + this.iy + \")\");\n    // this.setAllLinkCoordinates(); // todo - look at calls\n};\n\nInteractor.prototype.getAggregateSelfLinkPath = function () {\n    const intraR = this.getSymbolRadius() + 7;\n    const sectorSize = 45;\n    const arcStart = trig(intraR, 25 + sectorSize);\n    const arcEnd = trig(intraR, -25 + sectorSize);\n    const cp1 = trig(intraR, 40 + sectorSize);\n    const cp2 = trig(intraR, -40 + sectorSize);\n    return \"M 0,0 \" +\n        \"Q \" + cp1.x + \",\" + -cp1.y + \" \" + arcStart.x + \",\" + -arcStart.y +\n        \" A \" + intraR + \" \" + intraR + \" 0 0 1 \" + arcEnd.x + \",\" + -arcEnd.y +\n        \" Q \" + cp2.x + \",\" + -cp2.y + \" 0,0\";\n};\n\nInteractor.prototype.checkLinks = function () {\n    function checkAll(linkMap) {\n        for (let link of linkMap.values()) {\n            link.check();\n        }\n    }\n\n    // checkAll(this.naryLinks); // hacked out to fix ordering of nLinks\n    checkAll(this.binaryLinks);\n    checkAll(this.sequenceLinks);\n    if (this.selfLink) {\n        this.selfLink.check();\n    }\n};\n\n// update all lines (e.g after a move)\nInteractor.prototype.setAllLinkCoordinates = function () {\n    for (let link of this.naryLinks.values()) {\n        link.setLinkCoordinates();\n    }\n    for (let link of this.binaryLinks.values()) {\n        link.setLinkCoordinates();\n    }\n    if (this.selfLink) {\n        this.selfLink.setLinkCoordinates();\n    }\n    for (let link of this.sequenceLinks.values()) {\n            link.setLinkCoordinates();\n    }\n};\n\nexport function trig (radius, angleDegrees) {\n    //x = rx + radius * cos(theta) and y = ry + radius * sin(theta)\n    const radians = (angleDegrees / 360) * Math.PI * 2;\n    return {\n        x: (radius * Math.cos(radians)),\n        y: (radius * Math.sin(radians))\n    };\n}\n"],"mappings":"AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/js/viz/interactor/interactor.js\n"); /***/ }), @@ -1297,7 +1309,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"MoleculeSet\", function() { return MoleculeSet; });\n/* harmony import */ var _interactor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./interactor */ \"./src/js/viz/interactor/interactor.js\");\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n\n\n\nfunction MoleculeSet(id, app, json, name) {\n this.init(id, app, json, name);\n this.upperGroup = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"g\");\n this.initLabel();\n this.outline = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"rect\");\n this.outline.setAttribute(\"x\", \"-20\");\n this.outline.setAttribute(\"y\", \"-10\");\n this.outline.setAttribute(\"width\", \"40\");\n this.outline.setAttribute(\"height\", \"20\");\n this.outline.setAttribute(\"rx\", \"5\");\n this.outline.setAttribute(\"ry\", \"5\");\n this.outline.setAttribute(\"stroke\", \"black\");\n this.outline.setAttribute(\"stroke-width\", \"4\");\n this.outline.setAttribute(\"stroke-opacity\", \"1\");\n this.outline.setAttribute(\"fill-opacity\", \"1\");\n this.outline.setAttribute(\"fill\", \"#ffffff\");\n //append outline\n this.upperGroup.appendChild(this.outline);\n this.upperLine = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"rect\");\n this.upperLine.setAttribute(\"x\", \"-20\");\n this.upperLine.setAttribute(\"y\", \"-10\");\n this.upperLine.setAttribute(\"width\", \"40\");\n this.upperLine.setAttribute(\"height\", \"20\");\n this.upperLine.setAttribute(\"rx\", \"5\");\n this.upperLine.setAttribute(\"ry\", \"5\");\n this.upperLine.setAttribute(\"stroke\", \"white\");\n this.upperLine.setAttribute(\"stroke-width\", \"2\");\n this.upperLine.setAttribute(\"stroke-opacity\", \"1\");\n this.upperLine.setAttribute(\"fill\", \"none\");\n this.upperGroup.appendChild(this.upperLine);\n this.initListeners();\n}\n\nMoleculeSet.prototype = new _interactor__WEBPACK_IMPORTED_MODULE_0__[\"Interactor\"]();\n\nMoleculeSet.prototype.getSymbolRadius = function () {\n return 25;\n};\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvdml6L2ludGVyYWN0b3IvbW9sZWN1bGUtc2V0LmpzLmpzIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vY29tcGxleHZpZXdlci8uL3NyYy9qcy92aXovaW50ZXJhY3Rvci9tb2xlY3VsZS1zZXQuanM/YjZhMiJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0ludGVyYWN0b3J9IGZyb20gXCIuL2ludGVyYWN0b3JcIjtcbmltcG9ydCB7c3ZnbnN9IGZyb20gXCIuLi8uLi9jb25maWdcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIE1vbGVjdWxlU2V0KGlkLCBhcHAsIGpzb24sIG5hbWUpIHtcbiAgICB0aGlzLmluaXQoaWQsIGFwcCwganNvbiwgbmFtZSk7XG4gICAgdGhpcy51cHBlckdyb3VwID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudE5TKHN2Z25zLCBcImdcIik7XG4gICAgdGhpcy5pbml0TGFiZWwoKTtcbiAgICB0aGlzLm91dGxpbmUgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50TlMoc3ZnbnMsIFwicmVjdFwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwieFwiLCBcIi0yMFwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwieVwiLCBcIi0xMFwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwid2lkdGhcIiwgXCI0MFwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwiaGVpZ2h0XCIsIFwiMjBcIik7XG4gICAgdGhpcy5vdXRsaW5lLnNldEF0dHJpYnV0ZShcInJ4XCIsIFwiNVwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwicnlcIiwgXCI1XCIpO1xuICAgIHRoaXMub3V0bGluZS5zZXRBdHRyaWJ1dGUoXCJzdHJva2VcIiwgXCJibGFja1wiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwic3Ryb2tlLXdpZHRoXCIsIFwiNFwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwic3Ryb2tlLW9wYWNpdHlcIiwgXCIxXCIpO1xuICAgIHRoaXMub3V0bGluZS5zZXRBdHRyaWJ1dGUoXCJmaWxsLW9wYWNpdHlcIiwgXCIxXCIpO1xuICAgIHRoaXMub3V0bGluZS5zZXRBdHRyaWJ1dGUoXCJmaWxsXCIsIFwiI2ZmZmZmZlwiKTtcbiAgICAvL2FwcGVuZCBvdXRsaW5lXG4gICAgdGhpcy51cHBlckdyb3VwLmFwcGVuZENoaWxkKHRoaXMub3V0bGluZSk7XG4gICAgdGhpcy51cHBlckxpbmUgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50TlMoc3ZnbnMsIFwicmVjdFwiKTtcbiAgICB0aGlzLnVwcGVyTGluZS5zZXRBdHRyaWJ1dGUoXCJ4XCIsIFwiLTIwXCIpO1xuICAgIHRoaXMudXBwZXJMaW5lLnNldEF0dHJpYnV0ZShcInlcIiwgXCItMTBcIik7XG4gICAgdGhpcy51cHBlckxpbmUuc2V0QXR0cmlidXRlKFwid2lkdGhcIiwgXCI0MFwiKTtcbiAgICB0aGlzLnVwcGVyTGluZS5zZXRBdHRyaWJ1dGUoXCJoZWlnaHRcIiwgXCIyMFwiKTtcbiAgICB0aGlzLnVwcGVyTGluZS5zZXRBdHRyaWJ1dGUoXCJyeFwiLCBcIjVcIik7XG4gICAgdGhpcy51cHBlckxpbmUuc2V0QXR0cmlidXRlKFwicnlcIiwgXCI1XCIpO1xuICAgIHRoaXMudXBwZXJMaW5lLnNldEF0dHJpYnV0ZShcInN0cm9rZVwiLCBcIndoaXRlXCIpO1xuICAgIHRoaXMudXBwZXJMaW5lLnNldEF0dHJpYnV0ZShcInN0cm9rZS13aWR0aFwiLCBcIjJcIik7XG4gICAgdGhpcy51cHBlckxpbmUuc2V0QXR0cmlidXRlKFwic3Ryb2tlLW9wYWNpdHlcIiwgXCIxXCIpO1xuICAgIHRoaXMudXBwZXJMaW5lLnNldEF0dHJpYnV0ZShcImZpbGxcIiwgXCJub25lXCIpO1xuICAgIHRoaXMudXBwZXJHcm91cC5hcHBlbmRDaGlsZCh0aGlzLnVwcGVyTGluZSk7XG4gICAgdGhpcy5pbml0TGlzdGVuZXJzKCk7XG59XG5cbk1vbGVjdWxlU2V0LnByb3RvdHlwZSA9IG5ldyBJbnRlcmFjdG9yKCk7XG5cbk1vbGVjdWxlU2V0LnByb3RvdHlwZS5nZXRTeW1ib2xSYWRpdXMgPSBmdW5jdGlvbiAoKSB7XG4gICAgcmV0dXJuIDI1O1xufTtcbiJdLCJtYXBwaW5ncyI6IkFBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOyIsInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./src/js/viz/interactor/molecule-set.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"MoleculeSet\", function() { return MoleculeSet; });\n/* harmony import */ var _interactor__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./interactor */ \"./src/js/viz/interactor/interactor.js\");\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n\n\n\nfunction MoleculeSet(id, app, json, name) {\n this.init(id, app, json, name);\n this.upperGroup = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"g\");\n this.initLabel();\n this.outline = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"rect\");\n this.outline.setAttribute(\"x\", \"-20\");\n this.outline.setAttribute(\"y\", \"-10\");\n this.outline.setAttribute(\"width\", \"40\");\n this.outline.setAttribute(\"height\", \"20\");\n this.outline.setAttribute(\"rx\", \"5\");\n this.outline.setAttribute(\"ry\", \"5\");\n //todo - css... (initOutline hasn't been called so it doesn't have outlin in its classList)\n this.outline.setAttribute(\"stroke\", \"black\");\n this.outline.setAttribute(\"stroke-width\", \"4\");\n this.outline.setAttribute(\"stroke-opacity\", \"1\");\n this.outline.setAttribute(\"fill-opacity\", \"1\");\n this.outline.setAttribute(\"fill\", \"#ffffff\");\n //append outline\n this.upperGroup.appendChild(this.outline);\n this.upperLine = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"rect\");\n this.upperLine.setAttribute(\"x\", \"-20\");\n this.upperLine.setAttribute(\"y\", \"-10\");\n this.upperLine.setAttribute(\"width\", \"40\");\n this.upperLine.setAttribute(\"height\", \"20\");\n this.upperLine.setAttribute(\"rx\", \"5\");\n this.upperLine.setAttribute(\"ry\", \"5\");\n this.upperLine.setAttribute(\"stroke\", \"white\");\n this.upperLine.setAttribute(\"stroke-width\", \"2\");\n this.upperLine.setAttribute(\"stroke-opacity\", \"1\");\n this.upperLine.setAttribute(\"fill\", \"none\");\n this.upperGroup.appendChild(this.upperLine);\n this.initListeners();\n}\n\nMoleculeSet.prototype = new _interactor__WEBPACK_IMPORTED_MODULE_0__[\"Interactor\"]();\n\nMoleculeSet.prototype.getSymbolRadius = function () {\n return 25;\n};\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvdml6L2ludGVyYWN0b3IvbW9sZWN1bGUtc2V0LmpzLmpzIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vY29tcGxleHZpZXdlci8uL3NyYy9qcy92aXovaW50ZXJhY3Rvci9tb2xlY3VsZS1zZXQuanM/YjZhMiJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0ludGVyYWN0b3J9IGZyb20gXCIuL2ludGVyYWN0b3JcIjtcbmltcG9ydCB7c3ZnbnN9IGZyb20gXCIuLi8uLi9jb25maWdcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIE1vbGVjdWxlU2V0KGlkLCBhcHAsIGpzb24sIG5hbWUpIHtcbiAgICB0aGlzLmluaXQoaWQsIGFwcCwganNvbiwgbmFtZSk7XG4gICAgdGhpcy51cHBlckdyb3VwID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudE5TKHN2Z25zLCBcImdcIik7XG4gICAgdGhpcy5pbml0TGFiZWwoKTtcbiAgICB0aGlzLm91dGxpbmUgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50TlMoc3ZnbnMsIFwicmVjdFwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwieFwiLCBcIi0yMFwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwieVwiLCBcIi0xMFwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwid2lkdGhcIiwgXCI0MFwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwiaGVpZ2h0XCIsIFwiMjBcIik7XG4gICAgdGhpcy5vdXRsaW5lLnNldEF0dHJpYnV0ZShcInJ4XCIsIFwiNVwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwicnlcIiwgXCI1XCIpO1xuICAgIC8vdG9kbyAtIGNzcy4uLiAoaW5pdE91dGxpbmUgaGFzbid0IGJlZW4gY2FsbGVkIHNvIGl0IGRvZXNuJ3QgaGF2ZSBvdXRsaW4gaW4gaXRzIGNsYXNzTGlzdClcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwic3Ryb2tlXCIsIFwiYmxhY2tcIik7XG4gICAgdGhpcy5vdXRsaW5lLnNldEF0dHJpYnV0ZShcInN0cm9rZS13aWR0aFwiLCBcIjRcIik7XG4gICAgdGhpcy5vdXRsaW5lLnNldEF0dHJpYnV0ZShcInN0cm9rZS1vcGFjaXR5XCIsIFwiMVwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwiZmlsbC1vcGFjaXR5XCIsIFwiMVwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwiZmlsbFwiLCBcIiNmZmZmZmZcIik7XG4gICAgLy9hcHBlbmQgb3V0bGluZVxuICAgIHRoaXMudXBwZXJHcm91cC5hcHBlbmRDaGlsZCh0aGlzLm91dGxpbmUpO1xuICAgIHRoaXMudXBwZXJMaW5lID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudE5TKHN2Z25zLCBcInJlY3RcIik7XG4gICAgdGhpcy51cHBlckxpbmUuc2V0QXR0cmlidXRlKFwieFwiLCBcIi0yMFwiKTtcbiAgICB0aGlzLnVwcGVyTGluZS5zZXRBdHRyaWJ1dGUoXCJ5XCIsIFwiLTEwXCIpO1xuICAgIHRoaXMudXBwZXJMaW5lLnNldEF0dHJpYnV0ZShcIndpZHRoXCIsIFwiNDBcIik7XG4gICAgdGhpcy51cHBlckxpbmUuc2V0QXR0cmlidXRlKFwiaGVpZ2h0XCIsIFwiMjBcIik7XG4gICAgdGhpcy51cHBlckxpbmUuc2V0QXR0cmlidXRlKFwicnhcIiwgXCI1XCIpO1xuICAgIHRoaXMudXBwZXJMaW5lLnNldEF0dHJpYnV0ZShcInJ5XCIsIFwiNVwiKTtcbiAgICB0aGlzLnVwcGVyTGluZS5zZXRBdHRyaWJ1dGUoXCJzdHJva2VcIiwgXCJ3aGl0ZVwiKTtcbiAgICB0aGlzLnVwcGVyTGluZS5zZXRBdHRyaWJ1dGUoXCJzdHJva2Utd2lkdGhcIiwgXCIyXCIpO1xuICAgIHRoaXMudXBwZXJMaW5lLnNldEF0dHJpYnV0ZShcInN0cm9rZS1vcGFjaXR5XCIsIFwiMVwiKTtcbiAgICB0aGlzLnVwcGVyTGluZS5zZXRBdHRyaWJ1dGUoXCJmaWxsXCIsIFwibm9uZVwiKTtcbiAgICB0aGlzLnVwcGVyR3JvdXAuYXBwZW5kQ2hpbGQodGhpcy51cHBlckxpbmUpO1xuICAgIHRoaXMuaW5pdExpc3RlbmVycygpO1xufVxuXG5Nb2xlY3VsZVNldC5wcm90b3R5cGUgPSBuZXcgSW50ZXJhY3RvcigpO1xuXG5Nb2xlY3VsZVNldC5wcm90b3R5cGUuZ2V0U3ltYm9sUmFkaXVzID0gZnVuY3Rpb24gKCkge1xuICAgIHJldHVybiAyNTtcbn07XG4iXSwibWFwcGluZ3MiOiJBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOyIsInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./src/js/viz/interactor/molecule-set.js\n"); /***/ }), @@ -1309,7 +1321,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"Polymer\", function() { return Polymer; });\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! d3 */ \"./node_modules/d3/d3.js\");\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(d3__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _interactor__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./interactor */ \"./src/js/viz/interactor/interactor.js\");\n/* harmony import */ var _annotation__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./annotation */ \"./src/js/viz/interactor/annotation.js\");\n/* harmony import */ var _sequence_datum__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../sequence-datum */ \"./src/js/viz/sequence-datum.js\");\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n // transitions and other stuff\n\n\n\n\n\nPolymer.STICKHEIGHT = 20; //height of stick in pixels\nPolymer.MAXSIZE = 0; // residue count of longest sequence\nPolymer.transitionTime = 650;\n\nfunction Polymer() {\n}\n\nPolymer.prototype = new _interactor__WEBPACK_IMPORTED_MODULE_1__[\"Interactor\"]();\n\n//sequence = amino acids in UPPERCASE, digits or lowercase can be used for modification info\nPolymer.prototype.setSequence = function (sequence) {\n //remove modification site info from sequence\n this.sequence = sequence.replace(/[^A-Z]/g, \"\");\n this.size = this.sequence.length;\n};\n\nPolymer.prototype.getSymbolRadius = function () {\n return 15;\n};\n\nPolymer.prototype.showHighlight = function (show) {\n if (show === true) {\n this.highlight.setAttribute(\"stroke-opacity\", \"1\");\n } else {\n this.highlight.setAttribute(\"stroke-opacity\", \"0\");\n }\n};\n\nPolymer.minXDist = 30;\nPolymer.prototype.setStickScale = function (scale, svgP) {\n const oldScale = this.stickZoom;\n\n //dist from centre\n const dx = (this.ix - svgP.x);\n const dy = (this.iy - svgP.y);\n\n // new dist from centre\n const nx = dx * scale / oldScale;\n const ny = dy * scale / oldScale;\n\n //required change\n const rx = nx - dx;\n let ry = ny - dy;\n\n if (this.rotation === 0 || this.rotation === 180) {\n ry = 0;\n }\n\n //new pos\n const x = this.ix + rx;\n const y = this.iy + ry;\n\n this.stickZoom = scale;\n this.scale();\n this.setPosition(x, y);\n this.setAllLinkCoordinates();\n};\n\nPolymer.prototype.scale = function () {\n const protLength = (this.size) * this.stickZoom;\n if (this.form === 1) {\n const labelTransform = d3__WEBPACK_IMPORTED_MODULE_0__[\"transform\"](this.labelSVG.getAttribute(\"transform\"));\n const k = this.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate)\n .translate((-(((this.size / 2) * this.stickZoom) + (this.nTerminusFeature ? 25 : 10))), _config__WEBPACK_IMPORTED_MODULE_4__[\"LABEL_Y\"]); //.scale(z).translate(-c.x, -c.y);\n this.labelSVG.transform.baseVal.initialize(this.app.svgElement.createSVGTransformFromMatrix(k));\n\n for (let [annotationType, annotations] of this.annotationSets) {\n if (this.app.annotationSetsShown.get(annotationType) === true) {\n for (let anno of annotations) {\n if (anno.fuzzyStart) {\n anno.fuzzyStart.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n }\n if (anno.certain) {\n let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n let tempEnd = anno.seqDatum.end;\n if (anno.seqDatum.uncertainBegin) {\n tempBegin += 1;\n }\n if (anno.seqDatum.uncertainEnd) {\n tempEnd -= 1;\n }\n anno.certain.setAttribute(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno));\n }\n if (anno.fuzzyEnd) {\n anno.fuzzyEnd.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n }\n }\n }\n }\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.background)\n .attr(\"width\", protLength)\n .attr(\"x\", this.getResXwithStickZoom(0.5));\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.outline)\n .attr(\"width\", protLength)\n .attr(\"x\", this.getResXwithStickZoom(0.5));\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.highlight)\n .attr(\"width\", protLength + 5)\n .attr(\"x\", this.getResXwithStickZoom(0.5) - 2.5);\n\n\n this.setScaleGroup();\n }\n};\n\nPolymer.prototype.setScaleGroup = function () {\n this.upperGroup.appendChild(this.ticks); //will do nothing if this.ticks already appended to this.uppergroup\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.ticks).selectAll(\"*\").remove();\n\n this.scaleLabels = [];\n const ScaleTicksPerLabel = 2; // varies with scale?\n let tick = -1;\n const lastTickX = this.getResXwithStickZoom(this.size);\n\n for (let res = 1; res <= this.size; res++) {\n if (res === 1 ||\n ((res % 100 === 0) && (200 * this.stickZoom > Polymer.minXDist)) ||\n ((res % 10 === 0) && (20 * this.stickZoom > Polymer.minXDist))\n ) {\n const tx = this.getResXwithStickZoom(res);\n if (this.stickZoom >= 8 || res !== 1) {\n tickAt(this, tx);\n }\n tick = (tick + 1) % ScaleTicksPerLabel;\n // does this one get a label?\n if (tick === 0) { // && tx > 20) {\n if ((tx + Polymer.minXDist) < lastTickX) {\n scaleLabelAt(this, res, tx);\n }\n }\n }\n if (this.stickZoom >= 8) {\n const seqLabelGroup = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"g\");\n seqLabelGroup.setAttribute(\"transform\", \"translate(\" + this.getResXwithStickZoom(res) + \" \" + 0 + \")\");\n const seqLabel = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"text\");\n seqLabel.setAttribute(\"font-family\", \"'Courier New', monospace\");\n seqLabel.setAttribute(\"font-size\", \"10px\");\n seqLabel.setAttribute(\"text-anchor\", \"middle\");\n seqLabel.setAttribute(\"x\", \"0\");\n seqLabel.setAttribute(\"y\", \"3\");\n seqLabel.appendChild(document.createTextNode(this.sequence[res - 1]));\n seqLabelGroup.appendChild(seqLabel);\n this.scaleLabels.push(seqLabel);\n this.ticks.appendChild(seqLabelGroup);\n }\n }\n scaleLabelAt(this, this.size, lastTickX);\n if (this.stickZoom >= 8) {\n tickAt(this, lastTickX);\n }\n\n function scaleLabelAt(self, text, tickX) {\n const scaleLabelGroup = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"g\");\n scaleLabelGroup.setAttribute(\"transform\", \"translate(\" + tickX + \" \" + 0 + \")\");\n const scaleLabel = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"text\");\n scaleLabel.setAttribute(\"class\", \"xlv_text\");\n // scaleLabel.setAttribute(\"font-family\", \"'Courier New', monospace\");\n scaleLabel.setAttribute(\"font-size\", \"8pt\"); // todo css...\n scaleLabel.setAttribute(\"text-anchor\", \"middle\");\n scaleLabel.setAttribute(\"x\", \"0\");\n scaleLabel.setAttribute(\"y\", Polymer.STICKHEIGHT + 4);\n scaleLabel.appendChild(document.createTextNode(text));\n scaleLabelGroup.appendChild(scaleLabel);\n self.scaleLabels.push(scaleLabel);\n self.ticks.appendChild(scaleLabelGroup);\n }\n\n function tickAt(self, tickX) {\n const tick = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"line\");\n tick.setAttribute(\"x1\", tickX);\n tick.setAttribute(\"y1\", \"5\");\n tick.setAttribute(\"x2\", tickX);\n tick.setAttribute(\"y2\", \"10\");\n tick.setAttribute(\"stroke\", \"black\");\n self.ticks.appendChild(tick);\n }\n};\n\nPolymer.prototype.setForm = function (form, svgP) {\n if (this.busy !== true) {\n if (form === 1) {\n if (this.form !== 1) {\n this.toStick();\n }\n } else {\n // if (this.form !== 0) {\n this.toCircle(svgP);\n // var r = this.getSymbolRadius();\n\n }\n // }\n }\n};\n\nPolymer.prototype.toCircle = function (svgP) {\n //svgP = null;// temp hack - you can uncomment this is you experience things 'flying off screen'\n this.busy = true;\n\n const r = this.getSymbolRadius();\n //\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.background).transition()\n .attr(\"x\", -r).attr(\"y\", -r)\n .attr(\"width\", r * 2).attr(\"height\", r * 2)\n .attr(\"rx\", r).attr(\"ry\", r)\n .duration(Polymer.transitionTime);\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.outline).transition()\n .attr(\"x\", -r).attr(\"y\", -r)\n .attr(\"width\", r * 2).attr(\"height\", r * 2)\n .attr(\"rx\", r).attr(\"ry\", r)\n .duration(Polymer.transitionTime);\n // d3.select(this.annotationsSvgGroup).transition()\n // .attr(\"transform\", \"scale(1, 1)\")\n // .duration(Polymer.transitionTime);\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.highlight).transition()\n .attr(\"x\", -r).attr(\"y\", -r)\n .attr(\"width\", r * 2).attr(\"height\", r * 2)\n .attr(\"rx\", r).attr(\"ry\", r)\n .duration(Polymer.transitionTime);\n\n const stickZoomInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](this.stickZoom, 0);\n // var rotationInterpol = d3.interpolate((this.rotation > 180) ? this.rotation - 360 : this.rotation, 0);\n const labelTransform = d3__WEBPACK_IMPORTED_MODULE_0__[\"transform\"](this.labelSVG.getAttribute(\"transform\"));\n const labelStartPoint = labelTransform.translate[0];\n const labelTranslateInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](labelStartPoint, -(r + 5));\n\n let xInterpol = null,\n yInterpol = null;\n if (typeof svgP !== \"undefined\" && svgP !== null) {\n xInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](this.ix, svgP.x);\n yInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](this.iy, svgP.y);\n }\n\n const self = this;\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.ticks).transition().attr(\"opacity\", 0).duration(Polymer.transitionTime / 4)\n .each(\"end\",\n function () {\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this).selectAll(\"*\").remove();\n }\n );\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.highlight).transition()\n .attr(\"width\", (r * 2) + 5).attr(\"height\", (r * 2) + 5)\n .attr(\"x\", -r - 2.5).attr(\"y\", -r - 2.5)\n .attr(\"rx\", r + 2.5).attr(\"ry\", r + 2.5)\n .duration(Polymer.transitionTime);\n\n function changeFuzzyStartToArcPath(anno) {\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](anno.fuzzyStart).attr(\"d\", self.getAnnotationPieSliceArcPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n }\n\n function changeCertainToArcPath(anno) {\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](anno.certain).attr(\"d\", self.getAnnotationPieSliceArcPath(anno.seqDatum.begin, anno.seqDatum.end, anno));\n }\n\n function changeFuzzyEndToArcPath(anno) {\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](anno.fuzzyEnd).attr(\"d\", self.getAnnotationPieSliceArcPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n }\n\n for (let [annotationType, annotations] of this.annotationSets) {\n if (this.app.annotationSetsShown.get(annotationType) === true) {\n for (let anno of annotations) {\n if (anno.fuzzyStart) {\n const fuzzyStart = anno.fuzzyStart;\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](fuzzyStart).transition().attr(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno))\n .duration(Polymer.transitionTime).each(\"end\",\n function () {\n changeFuzzyStartToArcPath(anno);\n }\n );\n }\n\n if (anno.certain) {\n const certain = anno.certain;\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](certain).transition().attr(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.begin, anno.seqDatum.end, anno))\n .duration(Polymer.transitionTime).each(\"end\",\n function () {\n changeCertainToArcPath(anno);\n }\n );\n }\n\n if (anno.fuzzyEnd) {\n const fuzzyEnd = anno.fuzzyEnd;\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](fuzzyEnd).transition().attr(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno))\n .duration(Polymer.transitionTime).each(\"end\",\n function () {\n changeFuzzyEndToArcPath(anno);\n }\n );\n }\n }\n }\n }\n\n const originalStickZoom = this.stickZoom;\n const originalRotation = this.rotation;\n const cubicInOut = d3__WEBPACK_IMPORTED_MODULE_0__[\"ease\"](\"cubic-in-out\");\n d3__WEBPACK_IMPORTED_MODULE_0__[\"timer\"](function (elapsed) {\n return update(elapsed / Polymer.transitionTime);\n });\n\n function update(interp) {\n const labelTransform = d3__WEBPACK_IMPORTED_MODULE_0__[\"transform\"](self.labelSVG.getAttribute(\"transform\"));\n const k = self.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate).translate(labelTranslateInterpol(cubicInOut(interp)), _config__WEBPACK_IMPORTED_MODULE_4__[\"LABEL_Y\"]); //.scale(z).translate(-c.x, -c.y);\n self.labelSVG.transform.baseVal.initialize(self.app.svgElement.createSVGTransformFromMatrix(k));\n //~\n if (xInterpol !== null) {\n self.setPosition(xInterpol(cubicInOut(interp)), yInterpol(cubicInOut(interp)));\n }\n\n self.stickZoom = stickZoomInterpol(cubicInOut(interp));\n self.setAllLinkCoordinates();\n\n if (interp === 1) { // finished - tidy up\n self.form = 0;\n self.checkLinks();\n self.stickZoom = originalStickZoom;\n self.rotation = originalRotation;\n self.busy = false;\n return true;\n } else if (interp > 1) {\n return update(1);\n } else {\n return false;\n }\n }\n};\n\nPolymer.prototype.toStick = function () {\n this.busy = true;\n this.form = 1;\n\n //remove prot-prot links - would it be better if checkLinks did this? - think not\n const c = this.binaryLinks.values().length;\n for (let l = 0; l < c; l++) {\n const link = this.binaryLinks.values()[l];\n //out with the old\n if (link.shown) {\n link.hide();\n }\n }\n\n const protLength = this.size * this.stickZoom;\n const r = this.getSymbolRadius();\n\n const lengthInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"]((2 * r), protLength);\n const stickZoomInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](0, this.stickZoom);\n const labelTranslateInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](-(r + 5), -(((this.size / 2) * this.stickZoom) + (this.nTerminusFeature ? 25 : 10)));\n\n const origStickZoom = this.stickZoom;\n this.stickZoom = 0;\n this.checkLinks(this.binaryLinks);\n this.checkLinks(this.selfLink);\n this.checkLinks(this.sequenceLinks);\n this.stickZoom = origStickZoom;\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.background).transition() //.attr(\"stroke-opacity\", 1)\n .attr(\"height\", Polymer.STICKHEIGHT)\n .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n .attr(\"rx\", 0).attr(\"ry\", 0)\n .duration(Polymer.transitionTime);\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.outline).transition() //.attr(\"stroke-opacity\", 1)\n .attr(\"height\", Polymer.STICKHEIGHT)\n .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n .attr(\"rx\", 0).attr(\"ry\", 0)\n .duration(Polymer.transitionTime);\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.highlight).transition()\n .attr(\"width\", protLength + 5).attr(\"height\", Polymer.STICKHEIGHT + 5)\n .attr(\"x\", this.getResXwithStickZoom(0.5) - 2.5).attr(\"y\", (-Polymer.STICKHEIGHT / 2) - 2.5)\n .attr(\"rx\", 0).attr(\"ry\", 0)\n .duration(Polymer.transitionTime);\n\n for (let [annotationType, annotations] of this.annotationSets) {\n if (this.app.annotationSetsShown.get(annotationType) === true) {\n for (let anno of annotations) {\n if (anno.fuzzyStart) {\n const fuzzyStart = anno.fuzzyStart;\n fuzzyStart.setAttribute(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](fuzzyStart).transition().attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno))\n .duration(Polymer.transitionTime);\n }\n if (anno.certain) {\n const certain = anno.certain;\n let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n let tempEnd = anno.seqDatum.end;\n if (anno.seqDatum.uncertainBegin) {\n tempBegin += 1;\n }\n if (anno.seqDatum.uncertainEnd) {\n tempEnd -= 1;\n }\n\n certain.setAttribute(\"d\", this.getAnnotationPieSliceApproximatePath(tempBegin, tempEnd, anno));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](certain).transition().attr(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno))\n .duration(Polymer.transitionTime);\n }\n if (anno.fuzzyEnd) {\n const fuzzyEnd = anno.fuzzyEnd;\n fuzzyEnd.setAttribute(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](fuzzyEnd).transition().attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno))\n .duration(Polymer.transitionTime);\n }\n }\n }\n }\n\n const self = this;\n const cubicInOut = d3__WEBPACK_IMPORTED_MODULE_0__[\"ease\"](\"cubic-in-out\");\n d3__WEBPACK_IMPORTED_MODULE_0__[\"timer\"](function (elapsed) {\n return update(elapsed / Polymer.transitionTime);\n });\n\n function update(interp) {\n const labelTransform = d3__WEBPACK_IMPORTED_MODULE_0__[\"transform\"](self.labelSVG.getAttribute(\"transform\"));\n const k = self.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate).translate(labelTranslateInterpol(cubicInOut(interp)), _config__WEBPACK_IMPORTED_MODULE_4__[\"LABEL_Y\"]); //.scale(z).translate(-c.x, -c.y);\n self.labelSVG.transform.baseVal.initialize(self.app.svgElement.createSVGTransformFromMatrix(k));\n\n const currentLength = lengthInterpol(cubicInOut(interp));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](self.highlight).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](self.outline).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](self.background).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n self.stickZoom = stickZoomInterpol(cubicInOut(interp));\n self.setAllLinkCoordinates();\n\n if (interp === 1) { // finished - tidy up\n self.busy = false;\n return true;\n } else if (interp > 1) {\n return update(1);\n } else {\n return false;\n }\n }\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.ticks).attr(\"opacity\", 0);\n this.setScaleGroup();\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.ticks).transition().attr(\"opacity\", 1)\n .delay(Polymer.transitionTime * 0.8).duration(Polymer.transitionTime / 2);\n};\n\n\nPolymer.prototype.toStickNoTransition = function () { //todo tidy\n this.busy = true;\n this.form = 1;\n\n //remove prot-prot links - would it be better if checkLinks did this? - think not\n const c = this.binaryLinks.values().length;\n for (let l = 0; l < c; l++) {\n const link = this.binaryLinks.values()[l];\n //out with the old\n if (link.shown) {\n link.hide();\n }\n }\n\n const protLength = this.size * this.stickZoom;\n const r = this.getSymbolRadius();\n\n const lengthInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"]((2 * r), protLength);\n const labelTranslateInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](-(r + 5), -(((this.size / 2) * this.stickZoom) + (this.nTerminusFeature ? 25 : 10)));\n\n this.checkLinks(this.binaryLinks);\n this.checkLinks(this.selfLink);\n this.checkLinks(this.sequenceLinks);\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.background)\n .attr(\"height\", Polymer.STICKHEIGHT)\n .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n .attr(\"rx\", 0).attr(\"ry\", 0);\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.outline)\n .attr(\"height\", Polymer.STICKHEIGHT)\n .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n .attr(\"rx\", 0).attr(\"ry\", 0);\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.highlight)\n .attr(\"width\", protLength + 5).attr(\"height\", Polymer.STICKHEIGHT + 5)\n .attr(\"x\", this.getResXwithStickZoom(0.5) - 2.5).attr(\"y\", (-Polymer.STICKHEIGHT / 2) - 2.5)\n .attr(\"rx\", 0).attr(\"ry\", 0);\n\n for (let [annotationType, annotations] of this.annotationSets) {\n if (this.app.annotationSetsShown.get(annotationType) === true) {\n for (let anno of annotations) {\n if (anno.fuzzyStart) {\n const fuzzyStart = anno.fuzzyStart;\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](fuzzyStart).attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n }\n if (anno.certain) {\n let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n let tempEnd = anno.seqDatum.end;\n if (anno.seqDatum.uncertainBegin) {\n tempBegin += 1;\n }\n if (anno.seqDatum.uncertainEnd) {\n tempEnd -= 1;\n }\n anno.certain.setAttribute(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno));\n }\n if (anno.fuzzyEnd) {\n const fuzzyEnd = anno.fuzzyEnd;\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](fuzzyEnd) /*.transition()*/ .attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n }\n }\n }\n }\n\n const self = this;\n\n const labelTransform = d3__WEBPACK_IMPORTED_MODULE_0__[\"transform\"](self.labelSVG.getAttribute(\"transform\"));\n const k = self.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate).translate(labelTranslateInterpol(1), _config__WEBPACK_IMPORTED_MODULE_4__[\"LABEL_Y\"]); //.scale(z).translate(-c.x, -c.y);\n self.labelSVG.transform.baseVal.initialize(self.app.svgElement.createSVGTransformFromMatrix(k));\n\n const currentLength = lengthInterpol(1);\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](self.highlight).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](self.outline).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](self.background).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n self.setAllLinkCoordinates();\n\n this.setScaleGroup();\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.ticks).attr(\"opacity\", 1);\n\n self.busy = false;\n};\n\nPolymer.prototype.getResXwithStickZoom = function (r) {\n // if (isNaN(r)) {\n // console.error(\"NOT NUMBER\");\n // }\n if (r === \"n-n\") {\n return (-this.size / 2 * this.stickZoom) - 20;\n } else if (r === \"c-c\") {\n return (this.size / 2 * this.stickZoom) + 20;\n } else {\n return (r - (this.size / 2)) * this.stickZoom;\n }\n};\n\n//calculate the coordinates of a residue (relative to this.util.container)\nPolymer.prototype.getResidueCoordinates = function (r, yOff) {\n if (typeof r === \"undefined\") {\n console.error(\"ERROR: residue number is undefined\");\n }\n let x = this.getResXwithStickZoom(r * 1);// * this.app.z;\n // console.log(\"***\", this.app.z);\n // coz prots don't scale, don't multiple by app.z\n let y;\n if (x !== 0) {\n const l = Math.abs(x);\n const a = Math.acos(x / l);\n const rotRad = (this.rotation / 360) * Math.PI * 2;\n x = l * Math.cos(rotRad + a);\n y = l * Math.sin(rotRad + a);\n if (typeof yOff !== \"undefined\") {\n x += yOff /** this.app.z*/ * Math.cos(rotRad + (Math.PI / 2));\n y += yOff /** this.app.z*/ * Math.sin(rotRad + (Math.PI / 2));\n }\n } else {\n y = yOff;\n }\n x += this.ix;\n y += this.iy;\n return [x, y];\n};\n\nPolymer.prototype.clearPositionalFeatures = function () {\n this.annotations = [];\n this.annotationTypes = [];\n if (this.annotationsSvgGroup) d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.annotationsSvgGroup).selectAll(\"*\").remove();\n};\n\nPolymer.prototype.setPositionalFeatures = function () {\n const self = this;\n\n const toolTipFunc = function (evt) {\n const el = (evt.target.correspondingUseElement) ? evt.target.correspondingUseElement : evt.target;\n self.app.preventDefaultsAndStopPropagation(evt);\n self.app.setTooltip(el.name, el.getAttribute(\"fill\"));\n self.showHighlight(true);\n };\n\n const annotationTypesSet = new Set();\n\n for (let [annotationType, annotationSet] of this.annotationSets) {\n if (this.app.annotationSetsShown.get(annotationType) === true) {\n for (let annotation of annotationSet.values()) {\n\n if (annotation.seqDatum.sequenceDatumString !== \"n-n\" && annotation.seqDatum.sequenceDatumString !== \"c-c\") {\n annotationTypesSet.add(annotation.description);\n }\n }\n }\n }\n this.annotationTypes = Array.from(annotationTypesSet.values()).sort();\n\n for (let [annotationType, annotationSet] of this.annotationSets) {\n if (this.app.annotationSetsShown.get(annotationType) === true) {\n for (let anno of annotationSet.values()) {\n let text = anno.description + \" [\" + (anno.seqDatum ? anno.seqDatum.toString() : anno.seqDatum.begin + \" - \" + anno.seqDatum.end) + \"]\";\n if (anno.description === \"No annotations\") {\n text = \"No annotations\";\n }\n if (anno.seqDatum.uncertainBegin) {\n anno.fuzzyStart = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"path\");\n if (this.form === 0) {\n anno.fuzzyStart.setAttribute(\"d\", this.getAnnotationPieSliceArcPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n } else {\n anno.fuzzyStart.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n }\n anno.fuzzyStart.setAttribute(\"stroke\", \"none\");//-width\", \"1\"); // todo - should be css\n // anno.fuzzyStart.setAttribute(\"fill-opacity\", \"0.6\");\n anno.fuzzyStart.name = text;\n anno.fuzzyStart.onmouseover = toolTipFunc;\n this.annotationsSvgGroup.appendChild(anno.fuzzyStart);\n }\n\n if (anno.seqDatum.begin && anno.seqDatum.end) {\n anno.certain = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"path\");\n let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n let tempEnd = anno.seqDatum.end;\n if (anno.seqDatum.uncertainBegin) {\n tempBegin += 1;\n }\n if (anno.seqDatum.uncertainEnd) {\n tempEnd -= 1;\n }\n if (this.form === 0) {\n anno.certain.setAttribute(\"d\", this.getAnnotationPieSliceArcPath(tempBegin, tempEnd, anno));\n } else {\n anno.certain.setAttribute(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno));\n }\n anno.certain.setAttribute(\"stroke\", \"none\");//-width\", \"1\");\n // anno.certain.setAttribute(\"fill-opacity\", \"0.6\");\n anno.certain.name = text;\n anno.certain.onmouseover = toolTipFunc;\n this.annotationsSvgGroup.appendChild(anno.certain);\n }\n if (anno.seqDatum.uncertainEnd) {\n anno.fuzzyEnd = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"path\");\n if (this.form === 0) {\n anno.fuzzyEnd.setAttribute(\"d\", this.getAnnotationPieSliceArcPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n } else {\n anno.fuzzyEnd.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n }\n anno.fuzzyEnd.setAttribute(\"stroke\", \"none\"); //-width\", \"1\");\n // anno.fuzzyEnd.setAttribute(\"fill-opacity\", \"0.6\");\n anno.fuzzyEnd.name = text;\n anno.fuzzyEnd.onmouseover = toolTipFunc;\n this.annotationsSvgGroup.appendChild(anno.fuzzyEnd);\n }\n }\n }\n }\n};\n\nPolymer.stepsInArc = 5;\n\nPolymer.prototype.getAnnotationPieSliceArcPath = function (startRes, endRes, annotation) {\n const radius = this.getSymbolRadius();// - 2;\n\n let top, bottom, rungHeight;\n const rung = this.annotationTypes.indexOf(annotation.description);\n // console.log(\"rung\", rung, this.annotationTypes);\n if (rung === -1) {\n bottom = 0;\n top = radius;\n } else {\n //Math.sqrt(this.participant.size / Math.PI) * 0.6\n rungHeight = radius / this.annotationTypes.length;\n bottom = (rung * rungHeight);\n top = bottom + rungHeight;\n //\n // bottom = Math.sqrt(rung / this.annotationTypes.length) * radius;\n // top = Math.sqrt(rung + 1 / this.annotationTypes.length) * radius;\n }\n\n // var startAngle = ((startRes - 1) / this.size) * 360;\n // var endAngle = ((endRes - 1) / this.size) * 360;\n let startAngle, endAngle;\n if (startRes === \"n-n\") {\n startAngle = -20; //((startRes - 1) / this.size) * 360;\n endAngle = 0;//((endRes - 1) / this.size) * 360;\n } else if (endRes === \"c-c\") {\n startAngle = 0;//((startRes - 1) / this.size) * 360;\n endAngle = +20; //((endRes) / this.size) * 360;\n } else {\n startAngle = ((startRes - 1) / this.size) * 360;\n endAngle = ((endRes - 1) / this.size) * 360;\n }\n // const arcStart = trig(radius, startAngle - 90);\n // const arcEnd = trig(radius, endAngle - 90);\n let largeArch = 0;\n if ((endAngle - startAngle) > 180 || (endAngle === startAngle)) {\n largeArch = 1;\n }\n\n const p1 = Object(_config__WEBPACK_IMPORTED_MODULE_4__[\"rotatePointAboutPoint\"])([0, bottom], [0, 0], startAngle - 180);\n const p2 = Object(_config__WEBPACK_IMPORTED_MODULE_4__[\"rotatePointAboutPoint\"])([0, top], [0, 0], startAngle - 180);\n const p3 = Object(_config__WEBPACK_IMPORTED_MODULE_4__[\"rotatePointAboutPoint\"])([0, top], [0, 0], endAngle - 180);\n const p4 = Object(_config__WEBPACK_IMPORTED_MODULE_4__[\"rotatePointAboutPoint\"])([0, bottom], [0, 0], endAngle - 180);\n\n //const r = (bottom + top) / 2;\n const path = \"M\" + p1[0] + \",\" + p1[1] + \" L\" + p2[0] + \",\" + p2[1]\n + \" A\" + top + \",\" + top + \" 0 \" + largeArch + \" 1 \" + p3[0] + \",\" + p3[1] + \" L\" + p4[0] + \",\" + p4[1]\n + \" A\" + bottom + \",\" + bottom + \" 0 \" + largeArch + \" 0 \" + p1[0] + \",\" + p1[1] + \" Z\";\n console.log(\"**\", path);\n return path;\n // return \"M0,0 L\" + arcStart.x + \",\" + arcStart.y + \" A\" + radius + \",\" +\n // radius + \" 0 \" + largeArch + \" 1 \" + arcEnd.x + \",\" + arcEnd.y + \" Z\";\n};\n\nPolymer.prototype.getAnnotationPieSliceApproximatePath = function (startRes, endRes, annotation) {\n\n // let top, bottom, rungHeight;\n // const rung = this.annotationTypes.indexOf(annotation.description);\n // // console.log(\"rung\", rung, this.annotationTypes);\n // if (rung === -1) {\n // bottom = Polymer.STICKHEIGHT / 2;\n // top = -Polymer.STICKHEIGHT / 2;\n // } else {\n // rungHeight = Polymer.STICKHEIGHT / this.annotationTypes.length;\n // top = (-Polymer.STICKHEIGHT / 2) + (rung * rungHeight);\n // bottom = top + rungHeight;\n // }\n\n //approximate pie slice\n let startAngle, endAngle;\n if (startRes === \"n-n\") {\n startAngle = -20; //((startRes - 1) / this.size) * 360;\n endAngle = 0;//((endRes) / this.size) * 360;\n } else if (endRes === \"c-c\") {\n startAngle = 0;//((startRes - 1) / this.size) * 360;\n endAngle = +20; //((endRes) / this.size) * 360;\n } else {\n startAngle = ((startRes - 1) / this.size) * 360;\n endAngle = ((endRes) / this.size) * 360;\n }\n const pieRadius = this.getSymbolRadius() - 2;\n // var arcStart = Interactor.trig(pieRadius, startAngle - 90);\n // var arcEnd = Interactor.trig(pieRadius, endAngle - 90);\n let approximatePiePath = \"M 0,0\";\n const stepsInArc = 5;\n for (let sia = 0; sia <= Polymer.stepsInArc; sia++) {\n const angle = startAngle + ((endAngle - startAngle) * (sia / stepsInArc));\n const siaCoord = Object(_interactor__WEBPACK_IMPORTED_MODULE_1__[\"trig\"])(pieRadius, angle - 90);\n approximatePiePath += \" L \" + siaCoord.x + \",\" + siaCoord.y;\n }\n approximatePiePath += \" L \" + 0 + \",\" + 0;\n approximatePiePath += \" Z\";\n return approximatePiePath;\n};\n\nPolymer.prototype.getAnnotationRectPath = function (startRes, endRes, annotation) {\n //domain as rectangle path\n let top, bottom, rungHeight;\n const rung = this.annotationTypes.indexOf(annotation.description);\n // console.log(\"rung\", rung, this.annotationTypes);\n if (rung === -1) {\n bottom = Polymer.STICKHEIGHT / 2;\n top = -Polymer.STICKHEIGHT / 2;\n } else {\n rungHeight = Polymer.STICKHEIGHT / this.annotationTypes.length;\n top = (-Polymer.STICKHEIGHT / 2) + (rung * rungHeight);\n bottom = top + rungHeight;\n }\n\n let annotX, annotSize, annotLength;\n if (startRes === \"n-n\") {\n annotX = this.getResXwithStickZoom(0.5) - 20;\n // var annotSize = (1 + (endRes - startRes));\n annotLength = 20;//annotSize * this.stickZoom;\n } else if (endRes === \"c-c\") {\n annotX = this.getResXwithStickZoom(this.size + 0.5);\n // var annotSize = (1 + (endRes - startRes));\n annotLength = 20;//annotSize * this.stickZoom;\n } else {\n annotX = this.getResXwithStickZoom(startRes - 0.5);\n annotSize = (1 + (endRes - startRes));\n annotLength = annotSize * this.stickZoom;\n }\n let rectPath = \"M \" + annotX + \",\" + bottom;\n for (let sia = 0; sia <= Polymer.stepsInArc; sia++) {\n const step = annotX + (annotLength * (sia / Polymer.stepsInArc));\n rectPath += \" L \" + step + \",\" + top;\n }\n rectPath += \" L \" + (annotX + annotLength) + \",\" + bottom +\n \" Z\";\n return rectPath;\n};\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/js/viz/interactor/polymer.js.js","sources":["webpack://complexviewer/./src/js/viz/interactor/polymer.js?04f6"],"sourcesContent":["import * as d3 from \"d3\"; // transitions and other stuff\nimport {Interactor, trig} from \"./interactor\";\nimport {Annotation} from \"./annotation\";\nimport {SequenceDatum} from \"../sequence-datum\";\nimport {svgns, LABEL_Y, rotatePointAboutPoint} from \"../../config\";\n\nPolymer.STICKHEIGHT = 20; //height of stick in pixels\nPolymer.MAXSIZE = 0; // residue count of longest sequence\nPolymer.transitionTime = 650;\n\nexport function Polymer() {\n}\n\nPolymer.prototype = new Interactor();\n\n//sequence = amino acids in UPPERCASE, digits or lowercase can be used for modification info\nPolymer.prototype.setSequence = function (sequence) {\n    //remove modification site info from sequence\n    this.sequence = sequence.replace(/[^A-Z]/g, \"\");\n    this.size = this.sequence.length;\n};\n\nPolymer.prototype.getSymbolRadius = function () {\n    return 15;\n};\n\nPolymer.prototype.showHighlight = function (show) {\n    if (show === true) {\n        this.highlight.setAttribute(\"stroke-opacity\", \"1\");\n    } else {\n        this.highlight.setAttribute(\"stroke-opacity\", \"0\");\n    }\n};\n\nPolymer.minXDist = 30;\nPolymer.prototype.setStickScale = function (scale, svgP) {\n    const oldScale = this.stickZoom;\n\n    //dist from centre\n    const dx = (this.ix - svgP.x);\n    const dy = (this.iy - svgP.y);\n\n    // new dist from centre\n    const nx = dx * scale / oldScale;\n    const ny = dy * scale / oldScale;\n\n    //required change\n    const rx = nx - dx;\n    let ry = ny - dy;\n\n    if (this.rotation === 0 || this.rotation === 180) {\n        ry = 0;\n    }\n\n    //new pos\n    const x = this.ix + rx;\n    const y = this.iy + ry;\n\n    this.stickZoom = scale;\n    this.scale();\n    this.setPosition(x, y);\n    this.setAllLinkCoordinates();\n};\n\nPolymer.prototype.scale = function () {\n    const protLength = (this.size) * this.stickZoom;\n    if (this.form === 1) {\n        const labelTransform = d3.transform(this.labelSVG.getAttribute(\"transform\"));\n        const k = this.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate)\n            .translate((-(((this.size / 2) * this.stickZoom) + (this.nTerminusFeature ? 25 : 10))), LABEL_Y); //.scale(z).translate(-c.x, -c.y);\n        this.labelSVG.transform.baseVal.initialize(this.app.svgElement.createSVGTransformFromMatrix(k));\n\n        for (let [annotationType, annotations] of this.annotationSets) {\n            if (this.app.annotationSetsShown.get(annotationType) === true) {\n                for (let anno of annotations) {\n                    if (anno.fuzzyStart) {\n                        anno.fuzzyStart.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n                    }\n                    if (anno.certain) {\n                        let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n                        let tempEnd = anno.seqDatum.end;\n                        if (anno.seqDatum.uncertainBegin) {\n                            tempBegin += 1;\n                        }\n                        if (anno.seqDatum.uncertainEnd) {\n                            tempEnd -= 1;\n                        }\n                        anno.certain.setAttribute(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno));\n                    }\n                    if (anno.fuzzyEnd) {\n                        anno.fuzzyEnd.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n                    }\n                }\n            }\n        }\n\n        d3.select(this.background)\n            .attr(\"width\", protLength)\n            .attr(\"x\", this.getResXwithStickZoom(0.5));\n\n        d3.select(this.outline)\n            .attr(\"width\", protLength)\n            .attr(\"x\", this.getResXwithStickZoom(0.5));\n\n        d3.select(this.highlight)\n            .attr(\"width\", protLength + 5)\n            .attr(\"x\", this.getResXwithStickZoom(0.5) - 2.5);\n\n\n        this.setScaleGroup();\n    }\n};\n\nPolymer.prototype.setScaleGroup = function () {\n    this.upperGroup.appendChild(this.ticks); //will do nothing if this.ticks already appended to this.uppergroup\n    d3.select(this.ticks).selectAll(\"*\").remove();\n\n    this.scaleLabels = [];\n    const ScaleTicksPerLabel = 2; // varies with scale?\n    let tick = -1;\n    const lastTickX = this.getResXwithStickZoom(this.size);\n\n    for (let res = 1; res <= this.size; res++) {\n        if (res === 1 ||\n            ((res % 100 === 0) && (200 * this.stickZoom > Polymer.minXDist)) ||\n            ((res % 10 === 0) && (20 * this.stickZoom > Polymer.minXDist))\n        ) {\n            const tx = this.getResXwithStickZoom(res);\n            if (this.stickZoom >= 8 || res !== 1) {\n                tickAt(this, tx);\n            }\n            tick = (tick + 1) % ScaleTicksPerLabel;\n            // does this one get a label?\n            if (tick === 0) { // && tx > 20) {\n                if ((tx + Polymer.minXDist) < lastTickX) {\n                    scaleLabelAt(this, res, tx);\n                }\n            }\n        }\n        if (this.stickZoom >= 8) {\n            const seqLabelGroup = document.createElementNS(svgns, \"g\");\n            seqLabelGroup.setAttribute(\"transform\", \"translate(\" + this.getResXwithStickZoom(res) + \" \" + 0 + \")\");\n            const seqLabel = document.createElementNS(svgns, \"text\");\n            seqLabel.setAttribute(\"font-family\", \"'Courier New', monospace\");\n            seqLabel.setAttribute(\"font-size\", \"10px\");\n            seqLabel.setAttribute(\"text-anchor\", \"middle\");\n            seqLabel.setAttribute(\"x\", \"0\");\n            seqLabel.setAttribute(\"y\", \"3\");\n            seqLabel.appendChild(document.createTextNode(this.sequence[res - 1]));\n            seqLabelGroup.appendChild(seqLabel);\n            this.scaleLabels.push(seqLabel);\n            this.ticks.appendChild(seqLabelGroup);\n        }\n    }\n    scaleLabelAt(this, this.size, lastTickX);\n    if (this.stickZoom >= 8) {\n        tickAt(this, lastTickX);\n    }\n\n    function scaleLabelAt(self, text, tickX) {\n        const scaleLabelGroup = document.createElementNS(svgns, \"g\");\n        scaleLabelGroup.setAttribute(\"transform\", \"translate(\" + tickX + \" \" + 0 + \")\");\n        const scaleLabel = document.createElementNS(svgns, \"text\");\n        scaleLabel.setAttribute(\"class\", \"xlv_text\");\n        // scaleLabel.setAttribute(\"font-family\", \"'Courier New', monospace\");\n        scaleLabel.setAttribute(\"font-size\", \"8pt\"); // todo css...\n        scaleLabel.setAttribute(\"text-anchor\", \"middle\");\n        scaleLabel.setAttribute(\"x\", \"0\");\n        scaleLabel.setAttribute(\"y\", Polymer.STICKHEIGHT + 4);\n        scaleLabel.appendChild(document.createTextNode(text));\n        scaleLabelGroup.appendChild(scaleLabel);\n        self.scaleLabels.push(scaleLabel);\n        self.ticks.appendChild(scaleLabelGroup);\n    }\n\n    function tickAt(self, tickX) {\n        const tick = document.createElementNS(svgns, \"line\");\n        tick.setAttribute(\"x1\", tickX);\n        tick.setAttribute(\"y1\", \"5\");\n        tick.setAttribute(\"x2\", tickX);\n        tick.setAttribute(\"y2\", \"10\");\n        tick.setAttribute(\"stroke\", \"black\");\n        self.ticks.appendChild(tick);\n    }\n};\n\nPolymer.prototype.setForm = function (form, svgP) {\n    if (this.busy !== true) {\n        if (form === 1) {\n            if (this.form !== 1) {\n                this.toStick();\n            }\n        } else {\n            // if (this.form !== 0) {\n            this.toCircle(svgP);\n            // var r = this.getSymbolRadius();\n\n        }\n        // }\n    }\n};\n\nPolymer.prototype.toCircle = function (svgP) {\n    //svgP = null;// temp hack - you can uncomment this is you experience things 'flying off screen'\n    this.busy = true;\n\n    const r = this.getSymbolRadius();\n    //\n    d3.select(this.background).transition()\n        .attr(\"x\", -r).attr(\"y\", -r)\n        .attr(\"width\", r * 2).attr(\"height\", r * 2)\n        .attr(\"rx\", r).attr(\"ry\", r)\n        .duration(Polymer.transitionTime);\n    d3.select(this.outline).transition()\n        .attr(\"x\", -r).attr(\"y\", -r)\n        .attr(\"width\", r * 2).attr(\"height\", r * 2)\n        .attr(\"rx\", r).attr(\"ry\", r)\n        .duration(Polymer.transitionTime);\n    // d3.select(this.annotationsSvgGroup).transition()\n    //     .attr(\"transform\", \"scale(1, 1)\")\n    //     .duration(Polymer.transitionTime);\n    d3.select(this.highlight).transition()\n        .attr(\"x\", -r).attr(\"y\", -r)\n        .attr(\"width\", r * 2).attr(\"height\", r * 2)\n        .attr(\"rx\", r).attr(\"ry\", r)\n        .duration(Polymer.transitionTime);\n\n    const stickZoomInterpol = d3.interpolate(this.stickZoom, 0);\n    // var rotationInterpol = d3.interpolate((this.rotation > 180) ? this.rotation - 360 : this.rotation, 0);\n    const labelTransform = d3.transform(this.labelSVG.getAttribute(\"transform\"));\n    const labelStartPoint = labelTransform.translate[0];\n    const labelTranslateInterpol = d3.interpolate(labelStartPoint, -(r + 5));\n\n    let xInterpol = null,\n        yInterpol = null;\n    if (typeof svgP !== \"undefined\" && svgP !== null) {\n        xInterpol = d3.interpolate(this.ix, svgP.x);\n        yInterpol = d3.interpolate(this.iy, svgP.y);\n    }\n\n    const self = this;\n    d3.select(this.ticks).transition().attr(\"opacity\", 0).duration(Polymer.transitionTime / 4)\n        .each(\"end\",\n            function () {\n                d3.select(this).selectAll(\"*\").remove();\n            }\n        );\n\n    d3.select(this.highlight).transition()\n        .attr(\"width\", (r * 2) + 5).attr(\"height\", (r * 2) + 5)\n        .attr(\"x\", -r - 2.5).attr(\"y\", -r - 2.5)\n        .attr(\"rx\", r + 2.5).attr(\"ry\", r + 2.5)\n        .duration(Polymer.transitionTime);\n\n    function changeFuzzyStartToArcPath(anno) {\n        d3.select(anno.fuzzyStart).attr(\"d\", self.getAnnotationPieSliceArcPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n    }\n\n    function changeCertainToArcPath(anno) {\n        d3.select(anno.certain).attr(\"d\", self.getAnnotationPieSliceArcPath(anno.seqDatum.begin, anno.seqDatum.end, anno));\n    }\n\n    function changeFuzzyEndToArcPath(anno) {\n        d3.select(anno.fuzzyEnd).attr(\"d\", self.getAnnotationPieSliceArcPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n    }\n\n    for (let [annotationType, annotations] of this.annotationSets) {\n        if (this.app.annotationSetsShown.get(annotationType) === true) {\n            for (let anno of annotations) {\n                if (anno.fuzzyStart) {\n                    const fuzzyStart = anno.fuzzyStart;\n                    d3.select(fuzzyStart).transition().attr(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno))\n                        .duration(Polymer.transitionTime).each(\"end\",\n                        function () {\n                            changeFuzzyStartToArcPath(anno);\n                        }\n                    );\n                }\n\n                if (anno.certain) {\n                    const certain = anno.certain;\n                    d3.select(certain).transition().attr(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.begin, anno.seqDatum.end, anno))\n                        .duration(Polymer.transitionTime).each(\"end\",\n                        function () {\n                            changeCertainToArcPath(anno);\n                        }\n                    );\n                }\n\n                if (anno.fuzzyEnd) {\n                    const fuzzyEnd = anno.fuzzyEnd;\n                    d3.select(fuzzyEnd).transition().attr(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno))\n                        .duration(Polymer.transitionTime).each(\"end\",\n                        function () {\n                            changeFuzzyEndToArcPath(anno);\n                        }\n                    );\n                }\n            }\n        }\n    }\n\n    const originalStickZoom = this.stickZoom;\n    const originalRotation = this.rotation;\n    const cubicInOut = d3.ease(\"cubic-in-out\");\n    d3.timer(function (elapsed) {\n        return update(elapsed / Polymer.transitionTime);\n    });\n\n    function update(interp) {\n        const labelTransform = d3.transform(self.labelSVG.getAttribute(\"transform\"));\n        const k = self.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate).translate(labelTranslateInterpol(cubicInOut(interp)), LABEL_Y); //.scale(z).translate(-c.x, -c.y);\n        self.labelSVG.transform.baseVal.initialize(self.app.svgElement.createSVGTransformFromMatrix(k));\n        //~\n        if (xInterpol !== null) {\n            self.setPosition(xInterpol(cubicInOut(interp)), yInterpol(cubicInOut(interp)));\n        }\n\n        self.stickZoom = stickZoomInterpol(cubicInOut(interp));\n        self.setAllLinkCoordinates();\n\n        if (interp === 1) { // finished - tidy up\n            self.form = 0;\n            self.checkLinks();\n            self.stickZoom = originalStickZoom;\n            self.rotation = originalRotation;\n            self.busy = false;\n            return true;\n        } else if (interp > 1) {\n            return update(1);\n        } else {\n            return false;\n        }\n    }\n};\n\nPolymer.prototype.toStick = function () {\n    this.busy = true;\n    this.form = 1;\n\n    //remove prot-prot links - would it be better if checkLinks did this? - think not\n    const c = this.binaryLinks.values().length;\n    for (let l = 0; l < c; l++) {\n        const link = this.binaryLinks.values()[l];\n        //out with the old\n        if (link.shown) {\n            link.hide();\n        }\n    }\n\n    const protLength = this.size * this.stickZoom;\n    const r = this.getSymbolRadius();\n\n    const lengthInterpol = d3.interpolate((2 * r), protLength);\n    const stickZoomInterpol = d3.interpolate(0, this.stickZoom);\n    const labelTranslateInterpol = d3.interpolate(-(r + 5), -(((this.size / 2) * this.stickZoom) + (this.nTerminusFeature ? 25 : 10)));\n\n    const origStickZoom = this.stickZoom;\n    this.stickZoom = 0;\n    this.checkLinks(this.binaryLinks);\n    this.checkLinks(this.selfLink);\n    this.checkLinks(this.sequenceLinks);\n    this.stickZoom = origStickZoom;\n\n    d3.select(this.background).transition() //.attr(\"stroke-opacity\", 1)\n        .attr(\"height\", Polymer.STICKHEIGHT)\n        .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n        .attr(\"rx\", 0).attr(\"ry\", 0)\n        .duration(Polymer.transitionTime);\n\n    d3.select(this.outline).transition() //.attr(\"stroke-opacity\", 1)\n        .attr(\"height\", Polymer.STICKHEIGHT)\n        .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n        .attr(\"rx\", 0).attr(\"ry\", 0)\n        .duration(Polymer.transitionTime);\n\n    d3.select(this.highlight).transition()\n        .attr(\"width\", protLength + 5).attr(\"height\", Polymer.STICKHEIGHT + 5)\n        .attr(\"x\", this.getResXwithStickZoom(0.5) - 2.5).attr(\"y\", (-Polymer.STICKHEIGHT / 2) - 2.5)\n        .attr(\"rx\", 0).attr(\"ry\", 0)\n        .duration(Polymer.transitionTime);\n\n    for (let [annotationType, annotations] of this.annotationSets) {\n        if (this.app.annotationSetsShown.get(annotationType) === true) {\n            for (let anno of annotations) {\n                if (anno.fuzzyStart) {\n                    const fuzzyStart = anno.fuzzyStart;\n                    fuzzyStart.setAttribute(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n                    d3.select(fuzzyStart).transition().attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno))\n                        .duration(Polymer.transitionTime);\n                }\n                if (anno.certain) {\n                    const certain = anno.certain;\n                    let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n                    let tempEnd = anno.seqDatum.end;\n                    if (anno.seqDatum.uncertainBegin) {\n                        tempBegin += 1;\n                    }\n                    if (anno.seqDatum.uncertainEnd) {\n                        tempEnd -= 1;\n                    }\n\n                    certain.setAttribute(\"d\", this.getAnnotationPieSliceApproximatePath(tempBegin, tempEnd, anno));\n                    d3.select(certain).transition().attr(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno))\n                        .duration(Polymer.transitionTime);\n                }\n                if (anno.fuzzyEnd) {\n                    const fuzzyEnd = anno.fuzzyEnd;\n                    fuzzyEnd.setAttribute(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n                    d3.select(fuzzyEnd).transition().attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno))\n                        .duration(Polymer.transitionTime);\n                }\n            }\n        }\n    }\n\n    const self = this;\n    const cubicInOut = d3.ease(\"cubic-in-out\");\n    d3.timer(function (elapsed) {\n        return update(elapsed / Polymer.transitionTime);\n    });\n\n    function update(interp) {\n        const labelTransform = d3.transform(self.labelSVG.getAttribute(\"transform\"));\n        const k = self.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate).translate(labelTranslateInterpol(cubicInOut(interp)), LABEL_Y); //.scale(z).translate(-c.x, -c.y);\n        self.labelSVG.transform.baseVal.initialize(self.app.svgElement.createSVGTransformFromMatrix(k));\n\n        const currentLength = lengthInterpol(cubicInOut(interp));\n        d3.select(self.highlight).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n        d3.select(self.outline).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n        d3.select(self.background).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n        self.stickZoom = stickZoomInterpol(cubicInOut(interp));\n        self.setAllLinkCoordinates();\n\n        if (interp === 1) { // finished - tidy up\n            self.busy = false;\n            return true;\n        } else if (interp > 1) {\n            return update(1);\n        } else {\n            return false;\n        }\n    }\n\n    d3.select(this.ticks).attr(\"opacity\", 0);\n    this.setScaleGroup();\n    d3.select(this.ticks).transition().attr(\"opacity\", 1)\n        .delay(Polymer.transitionTime * 0.8).duration(Polymer.transitionTime / 2);\n};\n\n\nPolymer.prototype.toStickNoTransition = function () { //todo tidy\n    this.busy = true;\n    this.form = 1;\n\n    //remove prot-prot links - would it be better if checkLinks did this? - think not\n    const c = this.binaryLinks.values().length;\n    for (let l = 0; l < c; l++) {\n        const link = this.binaryLinks.values()[l];\n        //out with the old\n        if (link.shown) {\n            link.hide();\n        }\n    }\n\n    const protLength = this.size * this.stickZoom;\n    const r = this.getSymbolRadius();\n\n    const lengthInterpol = d3.interpolate((2 * r), protLength);\n    const labelTranslateInterpol = d3.interpolate(-(r + 5), -(((this.size / 2) * this.stickZoom) + (this.nTerminusFeature ? 25 : 10)));\n\n    this.checkLinks(this.binaryLinks);\n    this.checkLinks(this.selfLink);\n    this.checkLinks(this.sequenceLinks);\n\n    d3.select(this.background)\n        .attr(\"height\", Polymer.STICKHEIGHT)\n        .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n        .attr(\"rx\", 0).attr(\"ry\", 0);\n\n    d3.select(this.outline)\n        .attr(\"height\", Polymer.STICKHEIGHT)\n        .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n        .attr(\"rx\", 0).attr(\"ry\", 0);\n\n    d3.select(this.highlight)\n        .attr(\"width\", protLength + 5).attr(\"height\", Polymer.STICKHEIGHT + 5)\n        .attr(\"x\", this.getResXwithStickZoom(0.5) - 2.5).attr(\"y\", (-Polymer.STICKHEIGHT / 2) - 2.5)\n        .attr(\"rx\", 0).attr(\"ry\", 0);\n\n    for (let [annotationType, annotations] of this.annotationSets) {\n        if (this.app.annotationSetsShown.get(annotationType) === true) {\n            for (let anno of annotations) {\n                if (anno.fuzzyStart) {\n                    const fuzzyStart = anno.fuzzyStart;\n                    d3.select(fuzzyStart).attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n                }\n                if (anno.certain) {\n                    let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n                    let tempEnd = anno.seqDatum.end;\n                    if (anno.seqDatum.uncertainBegin) {\n                        tempBegin += 1;\n                    }\n                    if (anno.seqDatum.uncertainEnd) {\n                        tempEnd -= 1;\n                    }\n                    anno.certain.setAttribute(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno));\n                }\n                if (anno.fuzzyEnd) {\n                    const fuzzyEnd = anno.fuzzyEnd;\n                    d3.select(fuzzyEnd) /*.transition()*/ .attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n                }\n            }\n        }\n    }\n\n    const self = this;\n\n    const labelTransform = d3.transform(self.labelSVG.getAttribute(\"transform\"));\n    const k = self.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate).translate(labelTranslateInterpol(1), LABEL_Y); //.scale(z).translate(-c.x, -c.y);\n    self.labelSVG.transform.baseVal.initialize(self.app.svgElement.createSVGTransformFromMatrix(k));\n\n    const currentLength = lengthInterpol(1);\n    d3.select(self.highlight).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n    d3.select(self.outline).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n    d3.select(self.background).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n    self.setAllLinkCoordinates();\n\n    this.setScaleGroup();\n    d3.select(this.ticks).attr(\"opacity\", 1);\n\n    self.busy = false;\n};\n\nPolymer.prototype.getResXwithStickZoom = function (r) {\n    // if (isNaN(r)) {\n    //     console.error(\"NOT NUMBER\");\n    // }\n    if (r === \"n-n\") {\n        return (-this.size / 2 * this.stickZoom) - 20;\n    } else if (r === \"c-c\") {\n        return (this.size / 2 * this.stickZoom) + 20;\n    } else {\n        return (r - (this.size / 2)) * this.stickZoom;\n    }\n};\n\n//calculate the  coordinates of a residue (relative to this.util.container)\nPolymer.prototype.getResidueCoordinates = function (r, yOff) {\n    if (typeof r === \"undefined\") {\n        console.error(\"ERROR: residue number is undefined\");\n    }\n    let x = this.getResXwithStickZoom(r * 1);// * this.app.z;\n    // console.log(\"***\", this.app.z);\n    // coz prots don't scale, don't multiple by app.z\n    let y;\n    if (x !== 0) {\n        const l = Math.abs(x);\n        const a = Math.acos(x / l);\n        const rotRad = (this.rotation / 360) * Math.PI * 2;\n        x = l * Math.cos(rotRad + a);\n        y = l * Math.sin(rotRad + a);\n        if (typeof yOff !== \"undefined\") {\n            x += yOff /** this.app.z*/ * Math.cos(rotRad + (Math.PI / 2));\n            y += yOff /** this.app.z*/ * Math.sin(rotRad + (Math.PI / 2));\n        }\n    } else {\n        y = yOff;\n    }\n    x += this.ix;\n    y += this.iy;\n    return [x, y];\n};\n\nPolymer.prototype.clearPositionalFeatures = function () {\n    this.annotations = [];\n    this.annotationTypes = [];\n    if (this.annotationsSvgGroup) d3.select(this.annotationsSvgGroup).selectAll(\"*\").remove();\n};\n\nPolymer.prototype.setPositionalFeatures = function () {\n    const self = this;\n\n    const toolTipFunc = function (evt) {\n        const el = (evt.target.correspondingUseElement) ? evt.target.correspondingUseElement : evt.target;\n        self.app.preventDefaultsAndStopPropagation(evt);\n        self.app.setTooltip(el.name, el.getAttribute(\"fill\"));\n        self.showHighlight(true);\n    };\n\n    const annotationTypesSet = new Set();\n\n    for (let [annotationType, annotationSet] of this.annotationSets) {\n        if (this.app.annotationSetsShown.get(annotationType) === true) {\n            for (let annotation of annotationSet.values()) {\n\n                if (annotation.seqDatum.sequenceDatumString !== \"n-n\" && annotation.seqDatum.sequenceDatumString !== \"c-c\") {\n                    annotationTypesSet.add(annotation.description);\n                }\n            }\n        }\n    }\n    this.annotationTypes = Array.from(annotationTypesSet.values()).sort();\n\n    for (let [annotationType, annotationSet] of this.annotationSets) {\n        if (this.app.annotationSetsShown.get(annotationType) === true) {\n            for (let anno of annotationSet.values()) {\n                let text = anno.description + \" [\" + (anno.seqDatum ? anno.seqDatum.toString() : anno.seqDatum.begin + \" - \" + anno.seqDatum.end) + \"]\";\n                if (anno.description === \"No annotations\") {\n                    text = \"No annotations\";\n                }\n                if (anno.seqDatum.uncertainBegin) {\n                    anno.fuzzyStart = document.createElementNS(svgns, \"path\");\n                    if (this.form === 0) {\n                        anno.fuzzyStart.setAttribute(\"d\", this.getAnnotationPieSliceArcPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n                    } else {\n                        anno.fuzzyStart.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n                    }\n                    anno.fuzzyStart.setAttribute(\"stroke\", \"none\");//-width\", \"1\"); // todo - should be css\n                    // anno.fuzzyStart.setAttribute(\"fill-opacity\", \"0.6\");\n                    anno.fuzzyStart.name = text;\n                    anno.fuzzyStart.onmouseover = toolTipFunc;\n                    this.annotationsSvgGroup.appendChild(anno.fuzzyStart);\n                }\n\n                if (anno.seqDatum.begin && anno.seqDatum.end) {\n                    anno.certain = document.createElementNS(svgns, \"path\");\n                    let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n                    let tempEnd = anno.seqDatum.end;\n                    if (anno.seqDatum.uncertainBegin) {\n                        tempBegin += 1;\n                    }\n                    if (anno.seqDatum.uncertainEnd) {\n                        tempEnd -= 1;\n                    }\n                    if (this.form === 0) {\n                        anno.certain.setAttribute(\"d\", this.getAnnotationPieSliceArcPath(tempBegin, tempEnd, anno));\n                    } else {\n                        anno.certain.setAttribute(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno));\n                    }\n                    anno.certain.setAttribute(\"stroke\", \"none\");//-width\", \"1\");\n                    // anno.certain.setAttribute(\"fill-opacity\", \"0.6\");\n                    anno.certain.name = text;\n                    anno.certain.onmouseover = toolTipFunc;\n                    this.annotationsSvgGroup.appendChild(anno.certain);\n                }\n                if (anno.seqDatum.uncertainEnd) {\n                    anno.fuzzyEnd = document.createElementNS(svgns, \"path\");\n                    if (this.form === 0) {\n                        anno.fuzzyEnd.setAttribute(\"d\", this.getAnnotationPieSliceArcPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n                    } else {\n                        anno.fuzzyEnd.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n                    }\n                    anno.fuzzyEnd.setAttribute(\"stroke\", \"none\"); //-width\", \"1\");\n                    // anno.fuzzyEnd.setAttribute(\"fill-opacity\", \"0.6\");\n                    anno.fuzzyEnd.name = text;\n                    anno.fuzzyEnd.onmouseover = toolTipFunc;\n                    this.annotationsSvgGroup.appendChild(anno.fuzzyEnd);\n                }\n            }\n        }\n    }\n};\n\nPolymer.stepsInArc = 5;\n\nPolymer.prototype.getAnnotationPieSliceArcPath = function (startRes, endRes, annotation) {\n    const radius = this.getSymbolRadius();// - 2;\n\n    let top, bottom, rungHeight;\n    const rung = this.annotationTypes.indexOf(annotation.description);\n    // console.log(\"rung\", rung, this.annotationTypes);\n    if (rung === -1) {\n        bottom = 0;\n        top = radius;\n    } else {\n        //Math.sqrt(this.participant.size / Math.PI) * 0.6\n        rungHeight = radius / this.annotationTypes.length;\n        bottom = (rung * rungHeight);\n        top = bottom + rungHeight;\n        //\n        // bottom = Math.sqrt(rung / this.annotationTypes.length) * radius;\n        // top = Math.sqrt(rung + 1 / this.annotationTypes.length) * radius;\n    }\n\n    // var startAngle = ((startRes - 1) / this.size) * 360;\n    // var endAngle = ((endRes - 1) / this.size) * 360;\n    let startAngle, endAngle;\n    if (startRes === \"n-n\") {\n        startAngle = -20; //((startRes - 1) / this.size) * 360;\n        endAngle = 0;//((endRes - 1) / this.size) * 360;\n    } else if (endRes === \"c-c\") {\n        startAngle = 0;//((startRes - 1) / this.size) * 360;\n        endAngle = +20; //((endRes) / this.size) * 360;\n    } else {\n        startAngle = ((startRes - 1) / this.size) * 360;\n        endAngle = ((endRes - 1) / this.size) * 360;\n    }\n    // const arcStart = trig(radius, startAngle - 90);\n    // const arcEnd = trig(radius, endAngle - 90);\n    let largeArch = 0;\n    if ((endAngle - startAngle) > 180 || (endAngle === startAngle)) {\n        largeArch = 1;\n    }\n\n    const p1 = rotatePointAboutPoint([0, bottom], [0, 0], startAngle - 180);\n    const p2 = rotatePointAboutPoint([0, top], [0, 0], startAngle - 180);\n    const p3 = rotatePointAboutPoint([0, top], [0, 0], endAngle - 180);\n    const p4 = rotatePointAboutPoint([0, bottom], [0, 0], endAngle - 180);\n\n    //const r = (bottom + top) / 2;\n    const path = \"M\" + p1[0] + \",\" + p1[1] + \" L\" + p2[0] + \",\" + p2[1]\n        + \" A\" + top + \",\" + top + \" 0 \" + largeArch + \" 1 \" + p3[0] + \",\" + p3[1] + \" L\" + p4[0] + \",\" + p4[1]\n        + \" A\" + bottom + \",\" + bottom + \" 0 \" + largeArch + \" 0 \" + p1[0] + \",\" + p1[1] + \" Z\";\n    console.log(\"**\", path);\n    return path;\n    // return \"M0,0 L\" + arcStart.x + \",\" + arcStart.y + \" A\" + radius + \",\" +\n    //     radius + \" 0 \" + largeArch + \" 1 \" + arcEnd.x + \",\" + arcEnd.y + \" Z\";\n};\n\nPolymer.prototype.getAnnotationPieSliceApproximatePath = function (startRes, endRes, annotation) {\n\n    // let top, bottom, rungHeight;\n    // const rung = this.annotationTypes.indexOf(annotation.description);\n    // // console.log(\"rung\", rung, this.annotationTypes);\n    // if (rung === -1) {\n    //     bottom = Polymer.STICKHEIGHT / 2;\n    //     top = -Polymer.STICKHEIGHT / 2;\n    // } else {\n    //     rungHeight = Polymer.STICKHEIGHT / this.annotationTypes.length;\n    //     top = (-Polymer.STICKHEIGHT / 2) + (rung * rungHeight);\n    //     bottom = top + rungHeight;\n    // }\n\n    //approximate pie slice\n    let startAngle, endAngle;\n    if (startRes === \"n-n\") {\n        startAngle = -20; //((startRes - 1) / this.size) * 360;\n        endAngle = 0;//((endRes) / this.size) * 360;\n    } else if (endRes === \"c-c\") {\n        startAngle = 0;//((startRes - 1) / this.size) * 360;\n        endAngle = +20; //((endRes) / this.size) * 360;\n    } else {\n        startAngle = ((startRes - 1) / this.size) * 360;\n        endAngle = ((endRes) / this.size) * 360;\n    }\n    const pieRadius = this.getSymbolRadius() - 2;\n    // var arcStart = Interactor.trig(pieRadius, startAngle - 90);\n    // var arcEnd = Interactor.trig(pieRadius, endAngle - 90);\n    let approximatePiePath = \"M 0,0\";\n    const stepsInArc = 5;\n    for (let sia = 0; sia <= Polymer.stepsInArc; sia++) {\n        const angle = startAngle + ((endAngle - startAngle) * (sia / stepsInArc));\n        const siaCoord = trig(pieRadius, angle - 90);\n        approximatePiePath += \" L \" + siaCoord.x + \",\" + siaCoord.y;\n    }\n    approximatePiePath += \" L \" + 0 + \",\" + 0;\n    approximatePiePath += \"  Z\";\n    return approximatePiePath;\n};\n\nPolymer.prototype.getAnnotationRectPath = function (startRes, endRes, annotation) {\n    //domain as rectangle path\n    let top, bottom, rungHeight;\n    const rung = this.annotationTypes.indexOf(annotation.description);\n    // console.log(\"rung\", rung, this.annotationTypes);\n    if (rung === -1) {\n        bottom = Polymer.STICKHEIGHT / 2;\n        top = -Polymer.STICKHEIGHT / 2;\n    } else {\n        rungHeight = Polymer.STICKHEIGHT / this.annotationTypes.length;\n        top = (-Polymer.STICKHEIGHT / 2) + (rung * rungHeight);\n        bottom = top + rungHeight;\n    }\n\n    let annotX, annotSize, annotLength;\n    if (startRes === \"n-n\") {\n        annotX = this.getResXwithStickZoom(0.5) - 20;\n        // var annotSize = (1 + (endRes - startRes));\n        annotLength = 20;//annotSize * this.stickZoom;\n    } else if (endRes === \"c-c\") {\n        annotX = this.getResXwithStickZoom(this.size + 0.5);\n        // var annotSize = (1 + (endRes - startRes));\n        annotLength = 20;//annotSize * this.stickZoom;\n    } else {\n        annotX = this.getResXwithStickZoom(startRes - 0.5);\n        annotSize = (1 + (endRes - startRes));\n        annotLength = annotSize * this.stickZoom;\n    }\n    let rectPath = \"M \" + annotX + \",\" + bottom;\n    for (let sia = 0; sia <= Polymer.stepsInArc; sia++) {\n        const step = annotX + (annotLength * (sia / Polymer.stepsInArc));\n        rectPath += \" L \" + step + \",\" + top;\n    }\n    rectPath += \" L \" + (annotX + annotLength) + \",\" + bottom +\n        \" Z\";\n    return rectPath;\n};\n"],"mappings":"AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/js/viz/interactor/polymer.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"Polymer\", function() { return Polymer; });\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! d3 */ \"./node_modules/d3/d3.js\");\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(d3__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _interactor__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./interactor */ \"./src/js/viz/interactor/interactor.js\");\n/* harmony import */ var _annotation__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./annotation */ \"./src/js/viz/interactor/annotation.js\");\n/* harmony import */ var _sequence_datum__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../sequence-datum */ \"./src/js/viz/sequence-datum.js\");\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n // transitions and other stuff\n\n\n\n\n\nPolymer.STICKHEIGHT = 20; //height of stick in pixels\nPolymer.MAXSIZE = 0; // residue count of longest sequence\nPolymer.transitionTime = 650;\n\nfunction Polymer() {\n}\n\nPolymer.prototype = new _interactor__WEBPACK_IMPORTED_MODULE_1__[\"Interactor\"]();\n\n//sequence = amino acids in UPPERCASE, digits or lowercase can be used for modification info\nPolymer.prototype.setSequence = function (sequence) {\n //remove modification site info from sequence\n this.sequence = sequence.replace(/[^A-Z]/g, \"\");\n this.size = this.sequence.length;\n};\n\nPolymer.prototype.getSymbolRadius = function () {\n return 15;\n};\n\nPolymer.prototype.showHighlight = function (show) {\n if (show === true) {\n this.highlight.setAttribute(\"stroke-opacity\", \"1\");\n } else {\n this.highlight.setAttribute(\"stroke-opacity\", \"0\");\n }\n};\n\nPolymer.minXDist = 30;\nPolymer.prototype.setStickScale = function (scale, svgP) {\n const oldScale = this.stickZoom;\n\n //dist from centre\n const dx = (this.ix - svgP.x);\n const dy = (this.iy - svgP.y);\n\n // new dist from centre\n const nx = dx * scale / oldScale;\n const ny = dy * scale / oldScale;\n\n //required change\n const rx = nx - dx;\n let ry = ny - dy;\n\n if (this.rotation === 0 || this.rotation === 180) {\n ry = 0;\n }\n\n //new pos\n const x = this.ix + rx;\n const y = this.iy + ry;\n\n this.stickZoom = scale;\n this.scale();\n this.setPosition(x, y);\n this.setAllLinkCoordinates();\n};\n\nPolymer.prototype.scale = function () {\n const protLength = (this.size) * this.stickZoom;\n if (this.form === 1) {\n const labelTransform = d3__WEBPACK_IMPORTED_MODULE_0__[\"transform\"](this.labelSVG.getAttribute(\"transform\"));\n const k = this.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate)\n .translate((-(((this.size / 2) * this.stickZoom) + (this.nTerminusFeature ? 25 : 10))), _config__WEBPACK_IMPORTED_MODULE_4__[\"LABEL_Y\"]); //.scale(z).translate(-c.x, -c.y);\n this.labelSVG.transform.baseVal.initialize(this.app.svgElement.createSVGTransformFromMatrix(k));\n\n for (let [annotationType, annotations] of this.annotationSets) {\n if (this.app.annotationSetsShown.get(annotationType) === true) {\n for (let anno of annotations) {\n if (anno.fuzzyStart) {\n anno.fuzzyStart.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n }\n if (anno.certain) {\n let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n let tempEnd = anno.seqDatum.end;\n if (anno.seqDatum.uncertainBegin) {\n tempBegin += 1;\n }\n if (anno.seqDatum.uncertainEnd) {\n tempEnd -= 1;\n }\n anno.certain.setAttribute(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno));\n }\n if (anno.fuzzyEnd) {\n anno.fuzzyEnd.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n }\n }\n }\n }\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.background)\n .attr(\"width\", protLength)\n .attr(\"x\", this.getResXwithStickZoom(0.5));\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.outline)\n .attr(\"width\", protLength)\n .attr(\"x\", this.getResXwithStickZoom(0.5));\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.highlight)\n .attr(\"width\", protLength + 5)\n .attr(\"x\", this.getResXwithStickZoom(0.5) - 2.5);\n\n\n this.setScaleGroup();\n }\n};\n\nPolymer.prototype.setScaleGroup = function () {\n this.upperGroup.appendChild(this.ticks); //will do nothing if this.ticks already appended to this.uppergroup\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.ticks).selectAll(\"*\").remove();\n\n this.scaleLabels = [];\n const ScaleTicksPerLabel = 2; // varies with scale?\n let tick = -1;\n const lastTickX = this.getResXwithStickZoom(this.size);\n\n for (let res = 1; res <= this.size; res++) {\n if (res === 1 ||\n ((res % 100 === 0) && (200 * this.stickZoom > Polymer.minXDist)) ||\n ((res % 10 === 0) && (20 * this.stickZoom > Polymer.minXDist))\n ) {\n const tx = this.getResXwithStickZoom(res);\n if (this.stickZoom >= 8 || res !== 1) {\n tickAt(this, tx);\n }\n tick = (tick + 1) % ScaleTicksPerLabel;\n // does this one get a label?\n if (tick === 0) { // && tx > 20) {\n if ((tx + Polymer.minXDist) < lastTickX) {\n scaleLabelAt(this, res, tx);\n }\n }\n }\n if (this.stickZoom >= 8) {\n const seqLabelGroup = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"g\");\n seqLabelGroup.setAttribute(\"transform\", \"translate(\" + this.getResXwithStickZoom(res) + \" \" + 0 + \")\");\n const seqLabel = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"text\");\n seqLabel.classList.add(\"label\", \"sequence\");\n //css?\n seqLabel.setAttribute(\"x\", \"0\");\n seqLabel.setAttribute(\"y\", \"3\");\n seqLabel.appendChild(document.createTextNode(this.sequence[res - 1]));\n seqLabelGroup.appendChild(seqLabel);\n this.scaleLabels.push(seqLabel);\n this.ticks.appendChild(seqLabelGroup);\n }\n }\n scaleLabelAt(this, this.size, lastTickX);\n if (this.stickZoom >= 8) {\n tickAt(this, lastTickX);\n }\n\n function scaleLabelAt(self, text, tickX) {\n const scaleLabelGroup = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"g\");\n scaleLabelGroup.setAttribute(\"transform\", \"translate(\" + tickX + \" \" + 0 + \")\");\n const scaleLabel = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"text\");\n scaleLabel.classList.add(\"label\", \"scale-label\");\n scaleLabel.setAttribute(\"x\", \"0\");\n scaleLabel.setAttribute(\"y\", Polymer.STICKHEIGHT + 4);\n scaleLabel.appendChild(document.createTextNode(text));\n scaleLabelGroup.appendChild(scaleLabel);\n self.scaleLabels.push(scaleLabel);\n self.ticks.appendChild(scaleLabelGroup);\n }\n\n function tickAt(self, tickX) {\n const tick = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"line\");\n tick.classList.add(\"tick\");\n tick.setAttribute(\"x1\", tickX);\n tick.setAttribute(\"y1\", \"5\");\n tick.setAttribute(\"x2\", tickX);\n tick.setAttribute(\"y2\", \"10\");\n self.ticks.appendChild(tick);\n }\n};\n\nPolymer.prototype.setForm = function (form, svgP) {\n if (this.busy !== true) {\n if (form === 1) {\n if (this.form !== 1) {\n this.toStick();\n }\n } else {\n // if (this.form !== 0) {\n this.toCircle(svgP);\n // var r = this.getSymbolRadius();\n\n }\n // }\n }\n};\n\nPolymer.prototype.toCircle = function (svgP) {\n //svgP = null;// temp hack - you can uncomment this is you experience things 'flying off screen'\n this.busy = true;\n\n const r = this.getSymbolRadius();\n //\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.background).transition()\n .attr(\"x\", -r).attr(\"y\", -r)\n .attr(\"width\", r * 2).attr(\"height\", r * 2)\n .attr(\"rx\", r).attr(\"ry\", r)\n .duration(Polymer.transitionTime);\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.outline).transition()\n .attr(\"x\", -r).attr(\"y\", -r)\n .attr(\"width\", r * 2).attr(\"height\", r * 2)\n .attr(\"rx\", r).attr(\"ry\", r)\n .duration(Polymer.transitionTime);\n // d3.select(this.annotationsSvgGroup).transition()\n // .attr(\"transform\", \"scale(1, 1)\")\n // .duration(Polymer.transitionTime);\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.highlight).transition()\n .attr(\"x\", -r).attr(\"y\", -r)\n .attr(\"width\", r * 2).attr(\"height\", r * 2)\n .attr(\"rx\", r).attr(\"ry\", r)\n .duration(Polymer.transitionTime);\n\n const stickZoomInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](this.stickZoom, 0);\n // var rotationInterpol = d3.interpolate((this.rotation > 180) ? this.rotation - 360 : this.rotation, 0);\n const labelTransform = d3__WEBPACK_IMPORTED_MODULE_0__[\"transform\"](this.labelSVG.getAttribute(\"transform\"));\n const labelStartPoint = labelTransform.translate[0];\n const labelTranslateInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](labelStartPoint, -(r + 5));\n\n let xInterpol = null,\n yInterpol = null;\n if (typeof svgP !== \"undefined\" && svgP !== null) {\n xInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](this.ix, svgP.x);\n yInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](this.iy, svgP.y);\n }\n\n const self = this;\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.ticks).transition().attr(\"opacity\", 0).duration(Polymer.transitionTime / 4)\n .each(\"end\",\n function () {\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this).selectAll(\"*\").remove();\n }\n );\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.highlight).transition()\n .attr(\"width\", (r * 2) + 5).attr(\"height\", (r * 2) + 5)\n .attr(\"x\", -r - 2.5).attr(\"y\", -r - 2.5)\n .attr(\"rx\", r + 2.5).attr(\"ry\", r + 2.5)\n .duration(Polymer.transitionTime);\n\n function changeFuzzyStartToArcPath(anno) {\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](anno.fuzzyStart).attr(\"d\", self.getAnnotationPieSliceArcPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n }\n\n function changeCertainToArcPath(anno) {\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](anno.certain).attr(\"d\", self.getAnnotationPieSliceArcPath(anno.seqDatum.begin, anno.seqDatum.end, anno));\n }\n\n function changeFuzzyEndToArcPath(anno) {\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](anno.fuzzyEnd).attr(\"d\", self.getAnnotationPieSliceArcPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n }\n\n for (let [annotationType, annotations] of this.annotationSets) {\n if (this.app.annotationSetsShown.get(annotationType) === true) {\n for (let anno of annotations) {\n if (anno.fuzzyStart) {\n const fuzzyStart = anno.fuzzyStart;\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](fuzzyStart).transition().attr(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno))\n .duration(Polymer.transitionTime).each(\"end\",\n function () {\n changeFuzzyStartToArcPath(anno);\n }\n );\n }\n\n if (anno.certain) {\n const certain = anno.certain;\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](certain).transition().attr(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.begin, anno.seqDatum.end, anno))\n .duration(Polymer.transitionTime).each(\"end\",\n function () {\n changeCertainToArcPath(anno);\n }\n );\n }\n\n if (anno.fuzzyEnd) {\n const fuzzyEnd = anno.fuzzyEnd;\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](fuzzyEnd).transition().attr(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno))\n .duration(Polymer.transitionTime).each(\"end\",\n function () {\n changeFuzzyEndToArcPath(anno);\n }\n );\n }\n }\n }\n }\n\n const originalStickZoom = this.stickZoom;\n const originalRotation = this.rotation;\n const cubicInOut = d3__WEBPACK_IMPORTED_MODULE_0__[\"ease\"](\"cubic-in-out\");\n d3__WEBPACK_IMPORTED_MODULE_0__[\"timer\"](function (elapsed) {\n return update(elapsed / Polymer.transitionTime);\n });\n\n function update(interp) {\n const labelTransform = d3__WEBPACK_IMPORTED_MODULE_0__[\"transform\"](self.labelSVG.getAttribute(\"transform\"));\n const k = self.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate).translate(labelTranslateInterpol(cubicInOut(interp)), _config__WEBPACK_IMPORTED_MODULE_4__[\"LABEL_Y\"]); //.scale(z).translate(-c.x, -c.y);\n self.labelSVG.transform.baseVal.initialize(self.app.svgElement.createSVGTransformFromMatrix(k));\n //~\n if (xInterpol !== null) {\n self.setPosition(xInterpol(cubicInOut(interp)), yInterpol(cubicInOut(interp)));\n }\n\n self.stickZoom = stickZoomInterpol(cubicInOut(interp));\n self.setAllLinkCoordinates();\n\n if (interp === 1) { // finished - tidy up\n self.form = 0;\n self.checkLinks();\n self.stickZoom = originalStickZoom;\n self.rotation = originalRotation;\n self.busy = false;\n return true;\n } else if (interp > 1) {\n return update(1);\n } else {\n return false;\n }\n }\n};\n\nPolymer.prototype.toStick = function () {\n this.busy = true;\n this.form = 1;\n\n //remove prot-prot links - would it be better if checkLinks did this? - think not\n const c = this.binaryLinks.values().length;\n for (let l = 0; l < c; l++) {\n const link = this.binaryLinks.values()[l];\n //out with the old\n if (link.shown) {\n link.hide();\n }\n }\n\n const protLength = this.size * this.stickZoom;\n const r = this.getSymbolRadius();\n\n const lengthInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"]((2 * r), protLength);\n const stickZoomInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](0, this.stickZoom);\n const labelTranslateInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](-(r + 5), -(((this.size / 2) * this.stickZoom) + (this.nTerminusFeature ? 25 : 10)));\n\n const origStickZoom = this.stickZoom;\n this.stickZoom = 0;\n this.checkLinks(this.binaryLinks);\n this.checkLinks(this.selfLink);\n this.checkLinks(this.sequenceLinks);\n this.stickZoom = origStickZoom;\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.background).transition() //.attr(\"stroke-opacity\", 1)\n .attr(\"height\", Polymer.STICKHEIGHT)\n .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n .attr(\"rx\", 0).attr(\"ry\", 0)\n .duration(Polymer.transitionTime);\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.outline).transition() //.attr(\"stroke-opacity\", 1)\n .attr(\"height\", Polymer.STICKHEIGHT)\n .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n .attr(\"rx\", 0).attr(\"ry\", 0)\n .duration(Polymer.transitionTime);\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.highlight).transition()\n .attr(\"width\", protLength + 5).attr(\"height\", Polymer.STICKHEIGHT + 5)\n .attr(\"x\", this.getResXwithStickZoom(0.5) - 2.5).attr(\"y\", (-Polymer.STICKHEIGHT / 2) - 2.5)\n .attr(\"rx\", 0).attr(\"ry\", 0)\n .duration(Polymer.transitionTime);\n\n for (let [annotationType, annotations] of this.annotationSets) {\n if (this.app.annotationSetsShown.get(annotationType) === true) {\n for (let anno of annotations) {\n if (anno.fuzzyStart) {\n const fuzzyStart = anno.fuzzyStart;\n fuzzyStart.setAttribute(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](fuzzyStart).transition().attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno))\n .duration(Polymer.transitionTime);\n }\n if (anno.certain) {\n const certain = anno.certain;\n let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n let tempEnd = anno.seqDatum.end;\n if (anno.seqDatum.uncertainBegin) {\n tempBegin += 1;\n }\n if (anno.seqDatum.uncertainEnd) {\n tempEnd -= 1;\n }\n\n certain.setAttribute(\"d\", this.getAnnotationPieSliceApproximatePath(tempBegin, tempEnd, anno));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](certain).transition().attr(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno))\n .duration(Polymer.transitionTime);\n }\n if (anno.fuzzyEnd) {\n const fuzzyEnd = anno.fuzzyEnd;\n fuzzyEnd.setAttribute(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](fuzzyEnd).transition().attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno))\n .duration(Polymer.transitionTime);\n }\n }\n }\n }\n\n const self = this;\n const cubicInOut = d3__WEBPACK_IMPORTED_MODULE_0__[\"ease\"](\"cubic-in-out\");\n d3__WEBPACK_IMPORTED_MODULE_0__[\"timer\"](function (elapsed) {\n return update(elapsed / Polymer.transitionTime);\n });\n\n function update(interp) {\n const labelTransform = d3__WEBPACK_IMPORTED_MODULE_0__[\"transform\"](self.labelSVG.getAttribute(\"transform\"));\n const k = self.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate).translate(labelTranslateInterpol(cubicInOut(interp)), _config__WEBPACK_IMPORTED_MODULE_4__[\"LABEL_Y\"]); //.scale(z).translate(-c.x, -c.y);\n self.labelSVG.transform.baseVal.initialize(self.app.svgElement.createSVGTransformFromMatrix(k));\n\n const currentLength = lengthInterpol(cubicInOut(interp));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](self.highlight).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](self.outline).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](self.background).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n self.stickZoom = stickZoomInterpol(cubicInOut(interp));\n self.setAllLinkCoordinates();\n\n if (interp === 1) { // finished - tidy up\n self.busy = false;\n return true;\n } else if (interp > 1) {\n return update(1);\n } else {\n return false;\n }\n }\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.ticks).attr(\"opacity\", 0);\n this.setScaleGroup();\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.ticks).transition().attr(\"opacity\", 1)\n .delay(Polymer.transitionTime * 0.8).duration(Polymer.transitionTime / 2);\n};\n\n\nPolymer.prototype.toStickNoTransition = function () { //todo tidy\n this.busy = true;\n this.form = 1;\n\n //remove prot-prot links - would it be better if checkLinks did this? - think not\n const c = this.binaryLinks.values().length;\n for (let l = 0; l < c; l++) {\n const link = this.binaryLinks.values()[l];\n //out with the old\n if (link.shown) {\n link.hide();\n }\n }\n\n const protLength = this.size * this.stickZoom;\n const r = this.getSymbolRadius();\n\n const lengthInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"]((2 * r), protLength);\n const labelTranslateInterpol = d3__WEBPACK_IMPORTED_MODULE_0__[\"interpolate\"](-(r + 5), -(((this.size / 2) * this.stickZoom) + (this.nTerminusFeature ? 25 : 10)));\n\n this.checkLinks(this.binaryLinks);\n this.checkLinks(this.selfLink);\n this.checkLinks(this.sequenceLinks);\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.background)\n .attr(\"height\", Polymer.STICKHEIGHT)\n .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n .attr(\"rx\", 0).attr(\"ry\", 0);\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.outline)\n .attr(\"height\", Polymer.STICKHEIGHT)\n .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n .attr(\"rx\", 0).attr(\"ry\", 0);\n\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.highlight)\n .attr(\"width\", protLength + 5).attr(\"height\", Polymer.STICKHEIGHT + 5)\n .attr(\"x\", this.getResXwithStickZoom(0.5) - 2.5).attr(\"y\", (-Polymer.STICKHEIGHT / 2) - 2.5)\n .attr(\"rx\", 0).attr(\"ry\", 0);\n\n for (let [annotationType, annotations] of this.annotationSets) {\n if (this.app.annotationSetsShown.get(annotationType) === true) {\n for (let anno of annotations) {\n if (anno.fuzzyStart) {\n const fuzzyStart = anno.fuzzyStart;\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](fuzzyStart).attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n }\n if (anno.certain) {\n let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n let tempEnd = anno.seqDatum.end;\n if (anno.seqDatum.uncertainBegin) {\n tempBegin += 1;\n }\n if (anno.seqDatum.uncertainEnd) {\n tempEnd -= 1;\n }\n anno.certain.setAttribute(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno));\n }\n if (anno.fuzzyEnd) {\n const fuzzyEnd = anno.fuzzyEnd;\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](fuzzyEnd) /*.transition()*/ .attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n }\n }\n }\n }\n\n const self = this;\n\n const labelTransform = d3__WEBPACK_IMPORTED_MODULE_0__[\"transform\"](self.labelSVG.getAttribute(\"transform\"));\n const k = self.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate).translate(labelTranslateInterpol(1), _config__WEBPACK_IMPORTED_MODULE_4__[\"LABEL_Y\"]); //.scale(z).translate(-c.x, -c.y);\n self.labelSVG.transform.baseVal.initialize(self.app.svgElement.createSVGTransformFromMatrix(k));\n\n const currentLength = lengthInterpol(1);\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](self.highlight).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](self.outline).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](self.background).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n self.setAllLinkCoordinates();\n\n this.setScaleGroup();\n d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.ticks).attr(\"opacity\", 1);\n\n self.busy = false;\n};\n\nPolymer.prototype.getResXwithStickZoom = function (r) {\n // if (isNaN(r)) {\n // console.error(\"NOT NUMBER\");\n // }\n if (r === \"n-n\") {\n return (-this.size / 2 * this.stickZoom) - 20;\n } else if (r === \"c-c\") {\n return (this.size / 2 * this.stickZoom) + 20;\n } else {\n return (r - (this.size / 2)) * this.stickZoom;\n }\n};\n\n//calculate the coordinates of a residue (relative to this.util.container)\nPolymer.prototype.getResidueCoordinates = function (r, yOff) {\n if (typeof r === \"undefined\") {\n console.error(\"ERROR: residue number is undefined\");\n }\n let x = this.getResXwithStickZoom(r * 1);// * this.app.z;\n // console.log(\"***\", this.app.z);\n // coz prots don't scale, don't multiple by app.z\n let y;\n if (x !== 0) {\n const l = Math.abs(x);\n const a = Math.acos(x / l);\n const rotRad = (this.rotation / 360) * Math.PI * 2;\n x = l * Math.cos(rotRad + a);\n y = l * Math.sin(rotRad + a);\n if (typeof yOff !== \"undefined\") {\n x += yOff /** this.app.z*/ * Math.cos(rotRad + (Math.PI / 2));\n y += yOff /** this.app.z*/ * Math.sin(rotRad + (Math.PI / 2));\n }\n } else {\n y = yOff;\n }\n x += this.ix;\n y += this.iy;\n return [x, y];\n};\n\nPolymer.prototype.clearPositionalFeatures = function () {\n this.annotations = [];\n this.annotationTypes = [];\n if (this.annotationsSvgGroup) d3__WEBPACK_IMPORTED_MODULE_0__[\"select\"](this.annotationsSvgGroup).selectAll(\"*\").remove();\n};\n\nPolymer.prototype.setPositionalFeatures = function () {\n const self = this;\n\n const toolTipFunc = function (evt) {\n const el = (evt.target.correspondingUseElement) ? evt.target.correspondingUseElement : evt.target;\n self.app.preventDefaultsAndStopPropagation(evt);\n self.app.setTooltip(el.name, el.getAttribute(\"fill\"));\n self.showHighlight(true);\n };\n\n const annotationTypesSet = new Set();\n\n for (let [annotationType, annotationSet] of this.annotationSets) {\n if (this.app.annotationSetsShown.get(annotationType) === true) {\n for (let annotation of annotationSet.values()) {\n\n if (annotation.seqDatum.sequenceDatumString !== \"n-n\" && annotation.seqDatum.sequenceDatumString !== \"c-c\") {\n annotationTypesSet.add(annotation.description);\n }\n }\n }\n }\n this.annotationTypes = Array.from(annotationTypesSet.values()).sort();\n\n for (let [annotationType, annotationSet] of this.annotationSets) {\n if (this.app.annotationSetsShown.get(annotationType) === true) {\n for (let anno of annotationSet.values()) {\n let text = anno.description + \" [\" + (anno.seqDatum ? anno.seqDatum.toString() : anno.seqDatum.begin + \" - \" + anno.seqDatum.end) + \"]\";\n if (anno.description === \"No annotations\") {\n text = \"No annotations\";\n }\n if (anno.seqDatum.uncertainBegin) {\n anno.fuzzyStart = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"path\");\n if (this.form === 0) {\n anno.fuzzyStart.setAttribute(\"d\", this.getAnnotationPieSliceArcPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n } else {\n anno.fuzzyStart.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n }\n anno.fuzzyStart.setAttribute(\"stroke\", \"none\");//-width\", \"1\"); // todo - should be css\n // anno.fuzzyStart.setAttribute(\"fill-opacity\", \"0.6\");\n anno.fuzzyStart.name = text;\n anno.fuzzyStart.onmouseover = toolTipFunc;\n this.annotationsSvgGroup.appendChild(anno.fuzzyStart);\n }\n\n if (anno.seqDatum.begin && anno.seqDatum.end) {\n anno.certain = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"path\");\n let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n let tempEnd = anno.seqDatum.end;\n if (anno.seqDatum.uncertainBegin) {\n tempBegin += 1;\n }\n if (anno.seqDatum.uncertainEnd) {\n tempEnd -= 1;\n }\n if (this.form === 0) {\n anno.certain.setAttribute(\"d\", this.getAnnotationPieSliceArcPath(tempBegin, tempEnd, anno));\n } else {\n anno.certain.setAttribute(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno));\n }\n anno.certain.setAttribute(\"stroke\", \"none\");//-width\", \"1\");\n // anno.certain.setAttribute(\"fill-opacity\", \"0.6\");\n anno.certain.name = text;\n anno.certain.onmouseover = toolTipFunc;\n this.annotationsSvgGroup.appendChild(anno.certain);\n }\n if (anno.seqDatum.uncertainEnd) {\n anno.fuzzyEnd = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_4__[\"svgns\"], \"path\");\n if (this.form === 0) {\n anno.fuzzyEnd.setAttribute(\"d\", this.getAnnotationPieSliceArcPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n } else {\n anno.fuzzyEnd.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n }\n anno.fuzzyEnd.setAttribute(\"stroke\", \"none\"); //-width\", \"1\");\n // anno.fuzzyEnd.setAttribute(\"fill-opacity\", \"0.6\");\n anno.fuzzyEnd.name = text;\n anno.fuzzyEnd.onmouseover = toolTipFunc;\n this.annotationsSvgGroup.appendChild(anno.fuzzyEnd);\n }\n }\n }\n }\n};\n\nPolymer.stepsInArc = 5;\n\nPolymer.prototype.getAnnotationPieSliceArcPath = function (startRes, endRes, annotation) {\n const radius = this.getSymbolRadius();// - 2;\n\n let top, bottom, rungHeight;\n const rung = this.annotationTypes.indexOf(annotation.description);\n // console.log(\"rung\", rung, this.annotationTypes);\n if (rung === -1) {\n bottom = 0;\n top = radius;\n } else {\n //Math.sqrt(this.participant.size / Math.PI) * 0.6\n rungHeight = radius / this.annotationTypes.length;\n bottom = (rung * rungHeight);\n top = bottom + rungHeight;\n //\n // bottom = Math.sqrt(rung / this.annotationTypes.length) * radius;\n // top = Math.sqrt(rung + 1 / this.annotationTypes.length) * radius;\n }\n\n // var startAngle = ((startRes - 1) / this.size) * 360;\n // var endAngle = ((endRes - 1) / this.size) * 360;\n let startAngle, endAngle;\n if (startRes === \"n-n\") {\n startAngle = -20; //((startRes - 1) / this.size) * 360;\n endAngle = 0;//((endRes - 1) / this.size) * 360;\n } else if (endRes === \"c-c\") {\n startAngle = 0;//((startRes - 1) / this.size) * 360;\n endAngle = +20; //((endRes) / this.size) * 360;\n } else {\n startAngle = ((startRes - 1) / this.size) * 360;\n endAngle = ((endRes - 1) / this.size) * 360;\n }\n // const arcStart = trig(radius, startAngle - 90);\n // const arcEnd = trig(radius, endAngle - 90);\n let largeArch = 0;\n if ((endAngle - startAngle) > 180 || (endAngle === startAngle)) {\n largeArch = 1;\n }\n\n const p1 = Object(_config__WEBPACK_IMPORTED_MODULE_4__[\"rotatePointAboutPoint\"])([0, bottom], [0, 0], startAngle - 180);\n const p2 = Object(_config__WEBPACK_IMPORTED_MODULE_4__[\"rotatePointAboutPoint\"])([0, top], [0, 0], startAngle - 180);\n const p3 = Object(_config__WEBPACK_IMPORTED_MODULE_4__[\"rotatePointAboutPoint\"])([0, top], [0, 0], endAngle - 180);\n const p4 = Object(_config__WEBPACK_IMPORTED_MODULE_4__[\"rotatePointAboutPoint\"])([0, bottom], [0, 0], endAngle - 180);\n\n //const r = (bottom + top) / 2;\n const path = \"M\" + p1[0] + \",\" + p1[1] + \" L\" + p2[0] + \",\" + p2[1]\n + \" A\" + top + \",\" + top + \" 0 \" + largeArch + \" 1 \" + p3[0] + \",\" + p3[1] + \" L\" + p4[0] + \",\" + p4[1]\n + \" A\" + bottom + \",\" + bottom + \" 0 \" + largeArch + \" 0 \" + p1[0] + \",\" + p1[1] + \" Z\";\n // console.log(\"**\", path);\n return path;\n // return \"M0,0 L\" + arcStart.x + \",\" + arcStart.y + \" A\" + radius + \",\" +\n // radius + \" 0 \" + largeArch + \" 1 \" + arcEnd.x + \",\" + arcEnd.y + \" Z\";\n};\n\nPolymer.prototype.getAnnotationPieSliceApproximatePath = function (startRes, endRes, annotation) {\n\n // let top, bottom, rungHeight;\n // const rung = this.annotationTypes.indexOf(annotation.description);\n // // console.log(\"rung\", rung, this.annotationTypes);\n // if (rung === -1) {\n // bottom = Polymer.STICKHEIGHT / 2;\n // top = -Polymer.STICKHEIGHT / 2;\n // } else {\n // rungHeight = Polymer.STICKHEIGHT / this.annotationTypes.length;\n // top = (-Polymer.STICKHEIGHT / 2) + (rung * rungHeight);\n // bottom = top + rungHeight;\n // }\n\n //approximate pie slice\n let startAngle, endAngle;\n if (startRes === \"n-n\") {\n startAngle = -20; //((startRes - 1) / this.size) * 360;\n endAngle = 0;//((endRes) / this.size) * 360;\n } else if (endRes === \"c-c\") {\n startAngle = 0;//((startRes - 1) / this.size) * 360;\n endAngle = +20; //((endRes) / this.size) * 360;\n } else {\n startAngle = ((startRes - 1) / this.size) * 360;\n endAngle = ((endRes) / this.size) * 360;\n }\n const pieRadius = this.getSymbolRadius() - 2;\n // var arcStart = Interactor.trig(pieRadius, startAngle - 90);\n // var arcEnd = Interactor.trig(pieRadius, endAngle - 90);\n let approximatePiePath = \"M 0,0\";\n const stepsInArc = 5;\n for (let sia = 0; sia <= Polymer.stepsInArc; sia++) {\n const angle = startAngle + ((endAngle - startAngle) * (sia / stepsInArc));\n const siaCoord = Object(_interactor__WEBPACK_IMPORTED_MODULE_1__[\"trig\"])(pieRadius, angle - 90);\n approximatePiePath += \" L \" + siaCoord.x + \",\" + siaCoord.y;\n }\n approximatePiePath += \" L \" + 0 + \",\" + 0;\n approximatePiePath += \" Z\";\n return approximatePiePath;\n};\n\nPolymer.prototype.getAnnotationRectPath = function (startRes, endRes, annotation) {\n //domain as rectangle path\n let top, bottom, rungHeight;\n const rung = this.annotationTypes.indexOf(annotation.description);\n // console.log(\"rung\", rung, this.annotationTypes);\n if (rung === -1) {\n bottom = Polymer.STICKHEIGHT / 2;\n top = -Polymer.STICKHEIGHT / 2;\n } else {\n rungHeight = Polymer.STICKHEIGHT / this.annotationTypes.length;\n top = (-Polymer.STICKHEIGHT / 2) + (rung * rungHeight);\n bottom = top + rungHeight;\n }\n\n let annotX, annotSize, annotLength;\n if (startRes === \"n-n\") {\n annotX = this.getResXwithStickZoom(0.5) - 20;\n // var annotSize = (1 + (endRes - startRes));\n annotLength = 20;//annotSize * this.stickZoom;\n } else if (endRes === \"c-c\") {\n annotX = this.getResXwithStickZoom(this.size + 0.5);\n // var annotSize = (1 + (endRes - startRes));\n annotLength = 20;//annotSize * this.stickZoom;\n } else {\n annotX = this.getResXwithStickZoom(startRes - 0.5);\n annotSize = (1 + (endRes - startRes));\n annotLength = annotSize * this.stickZoom;\n }\n let rectPath = \"M \" + annotX + \",\" + bottom;\n for (let sia = 0; sia <= Polymer.stepsInArc; sia++) {\n const step = annotX + (annotLength * (sia / Polymer.stepsInArc));\n rectPath += \" L \" + step + \",\" + top;\n }\n rectPath += \" L \" + (annotX + annotLength) + \",\" + bottom +\n \" Z\";\n return rectPath;\n};\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/js/viz/interactor/polymer.js.js","sources":["webpack://complexviewer/./src/js/viz/interactor/polymer.js?04f6"],"sourcesContent":["import * as d3 from \"d3\"; // transitions and other stuff\nimport {Interactor, trig} from \"./interactor\";\nimport {Annotation} from \"./annotation\";\nimport {SequenceDatum} from \"../sequence-datum\";\nimport {svgns, LABEL_Y, rotatePointAboutPoint} from \"../../config\";\n\nPolymer.STICKHEIGHT = 20; //height of stick in pixels\nPolymer.MAXSIZE = 0; // residue count of longest sequence\nPolymer.transitionTime = 650;\n\nexport function Polymer() {\n}\n\nPolymer.prototype = new Interactor();\n\n//sequence = amino acids in UPPERCASE, digits or lowercase can be used for modification info\nPolymer.prototype.setSequence = function (sequence) {\n    //remove modification site info from sequence\n    this.sequence = sequence.replace(/[^A-Z]/g, \"\");\n    this.size = this.sequence.length;\n};\n\nPolymer.prototype.getSymbolRadius = function () {\n    return 15;\n};\n\nPolymer.prototype.showHighlight = function (show) {\n    if (show === true) {\n        this.highlight.setAttribute(\"stroke-opacity\", \"1\");\n    } else {\n        this.highlight.setAttribute(\"stroke-opacity\", \"0\");\n    }\n};\n\nPolymer.minXDist = 30;\nPolymer.prototype.setStickScale = function (scale, svgP) {\n    const oldScale = this.stickZoom;\n\n    //dist from centre\n    const dx = (this.ix - svgP.x);\n    const dy = (this.iy - svgP.y);\n\n    // new dist from centre\n    const nx = dx * scale / oldScale;\n    const ny = dy * scale / oldScale;\n\n    //required change\n    const rx = nx - dx;\n    let ry = ny - dy;\n\n    if (this.rotation === 0 || this.rotation === 180) {\n        ry = 0;\n    }\n\n    //new pos\n    const x = this.ix + rx;\n    const y = this.iy + ry;\n\n    this.stickZoom = scale;\n    this.scale();\n    this.setPosition(x, y);\n    this.setAllLinkCoordinates();\n};\n\nPolymer.prototype.scale = function () {\n    const protLength = (this.size) * this.stickZoom;\n    if (this.form === 1) {\n        const labelTransform = d3.transform(this.labelSVG.getAttribute(\"transform\"));\n        const k = this.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate)\n            .translate((-(((this.size / 2) * this.stickZoom) + (this.nTerminusFeature ? 25 : 10))), LABEL_Y); //.scale(z).translate(-c.x, -c.y);\n        this.labelSVG.transform.baseVal.initialize(this.app.svgElement.createSVGTransformFromMatrix(k));\n\n        for (let [annotationType, annotations] of this.annotationSets) {\n            if (this.app.annotationSetsShown.get(annotationType) === true) {\n                for (let anno of annotations) {\n                    if (anno.fuzzyStart) {\n                        anno.fuzzyStart.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n                    }\n                    if (anno.certain) {\n                        let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n                        let tempEnd = anno.seqDatum.end;\n                        if (anno.seqDatum.uncertainBegin) {\n                            tempBegin += 1;\n                        }\n                        if (anno.seqDatum.uncertainEnd) {\n                            tempEnd -= 1;\n                        }\n                        anno.certain.setAttribute(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno));\n                    }\n                    if (anno.fuzzyEnd) {\n                        anno.fuzzyEnd.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n                    }\n                }\n            }\n        }\n\n        d3.select(this.background)\n            .attr(\"width\", protLength)\n            .attr(\"x\", this.getResXwithStickZoom(0.5));\n\n        d3.select(this.outline)\n            .attr(\"width\", protLength)\n            .attr(\"x\", this.getResXwithStickZoom(0.5));\n\n        d3.select(this.highlight)\n            .attr(\"width\", protLength + 5)\n            .attr(\"x\", this.getResXwithStickZoom(0.5) - 2.5);\n\n\n        this.setScaleGroup();\n    }\n};\n\nPolymer.prototype.setScaleGroup = function () {\n    this.upperGroup.appendChild(this.ticks); //will do nothing if this.ticks already appended to this.uppergroup\n    d3.select(this.ticks).selectAll(\"*\").remove();\n\n    this.scaleLabels = [];\n    const ScaleTicksPerLabel = 2; // varies with scale?\n    let tick = -1;\n    const lastTickX = this.getResXwithStickZoom(this.size);\n\n    for (let res = 1; res <= this.size; res++) {\n        if (res === 1 ||\n            ((res % 100 === 0) && (200 * this.stickZoom > Polymer.minXDist)) ||\n            ((res % 10 === 0) && (20 * this.stickZoom > Polymer.minXDist))\n        ) {\n            const tx = this.getResXwithStickZoom(res);\n            if (this.stickZoom >= 8 || res !== 1) {\n                tickAt(this, tx);\n            }\n            tick = (tick + 1) % ScaleTicksPerLabel;\n            // does this one get a label?\n            if (tick === 0) { // && tx > 20) {\n                if ((tx + Polymer.minXDist) < lastTickX) {\n                    scaleLabelAt(this, res, tx);\n                }\n            }\n        }\n        if (this.stickZoom >= 8) {\n            const seqLabelGroup = document.createElementNS(svgns, \"g\");\n            seqLabelGroup.setAttribute(\"transform\", \"translate(\" + this.getResXwithStickZoom(res) + \" \" + 0 + \")\");\n            const seqLabel = document.createElementNS(svgns, \"text\");\n            seqLabel.classList.add(\"label\", \"sequence\");\n            //css?\n            seqLabel.setAttribute(\"x\", \"0\");\n            seqLabel.setAttribute(\"y\", \"3\");\n            seqLabel.appendChild(document.createTextNode(this.sequence[res - 1]));\n            seqLabelGroup.appendChild(seqLabel);\n            this.scaleLabels.push(seqLabel);\n            this.ticks.appendChild(seqLabelGroup);\n        }\n    }\n    scaleLabelAt(this, this.size, lastTickX);\n    if (this.stickZoom >= 8) {\n        tickAt(this, lastTickX);\n    }\n\n    function scaleLabelAt(self, text, tickX) {\n        const scaleLabelGroup = document.createElementNS(svgns, \"g\");\n        scaleLabelGroup.setAttribute(\"transform\", \"translate(\" + tickX + \" \" + 0 + \")\");\n        const scaleLabel = document.createElementNS(svgns, \"text\");\n        scaleLabel.classList.add(\"label\", \"scale-label\");\n        scaleLabel.setAttribute(\"x\", \"0\");\n        scaleLabel.setAttribute(\"y\", Polymer.STICKHEIGHT + 4);\n        scaleLabel.appendChild(document.createTextNode(text));\n        scaleLabelGroup.appendChild(scaleLabel);\n        self.scaleLabels.push(scaleLabel);\n        self.ticks.appendChild(scaleLabelGroup);\n    }\n\n    function tickAt(self, tickX) {\n        const tick = document.createElementNS(svgns, \"line\");\n        tick.classList.add(\"tick\");\n        tick.setAttribute(\"x1\", tickX);\n        tick.setAttribute(\"y1\", \"5\");\n        tick.setAttribute(\"x2\", tickX);\n        tick.setAttribute(\"y2\", \"10\");\n        self.ticks.appendChild(tick);\n    }\n};\n\nPolymer.prototype.setForm = function (form, svgP) {\n    if (this.busy !== true) {\n        if (form === 1) {\n            if (this.form !== 1) {\n                this.toStick();\n            }\n        } else {\n            // if (this.form !== 0) {\n            this.toCircle(svgP);\n            // var r = this.getSymbolRadius();\n\n        }\n        // }\n    }\n};\n\nPolymer.prototype.toCircle = function (svgP) {\n    //svgP = null;// temp hack - you can uncomment this is you experience things 'flying off screen'\n    this.busy = true;\n\n    const r = this.getSymbolRadius();\n    //\n    d3.select(this.background).transition()\n        .attr(\"x\", -r).attr(\"y\", -r)\n        .attr(\"width\", r * 2).attr(\"height\", r * 2)\n        .attr(\"rx\", r).attr(\"ry\", r)\n        .duration(Polymer.transitionTime);\n    d3.select(this.outline).transition()\n        .attr(\"x\", -r).attr(\"y\", -r)\n        .attr(\"width\", r * 2).attr(\"height\", r * 2)\n        .attr(\"rx\", r).attr(\"ry\", r)\n        .duration(Polymer.transitionTime);\n    // d3.select(this.annotationsSvgGroup).transition()\n    //     .attr(\"transform\", \"scale(1, 1)\")\n    //     .duration(Polymer.transitionTime);\n    d3.select(this.highlight).transition()\n        .attr(\"x\", -r).attr(\"y\", -r)\n        .attr(\"width\", r * 2).attr(\"height\", r * 2)\n        .attr(\"rx\", r).attr(\"ry\", r)\n        .duration(Polymer.transitionTime);\n\n    const stickZoomInterpol = d3.interpolate(this.stickZoom, 0);\n    // var rotationInterpol = d3.interpolate((this.rotation > 180) ? this.rotation - 360 : this.rotation, 0);\n    const labelTransform = d3.transform(this.labelSVG.getAttribute(\"transform\"));\n    const labelStartPoint = labelTransform.translate[0];\n    const labelTranslateInterpol = d3.interpolate(labelStartPoint, -(r + 5));\n\n    let xInterpol = null,\n        yInterpol = null;\n    if (typeof svgP !== \"undefined\" && svgP !== null) {\n        xInterpol = d3.interpolate(this.ix, svgP.x);\n        yInterpol = d3.interpolate(this.iy, svgP.y);\n    }\n\n    const self = this;\n    d3.select(this.ticks).transition().attr(\"opacity\", 0).duration(Polymer.transitionTime / 4)\n        .each(\"end\",\n            function () {\n                d3.select(this).selectAll(\"*\").remove();\n            }\n        );\n\n    d3.select(this.highlight).transition()\n        .attr(\"width\", (r * 2) + 5).attr(\"height\", (r * 2) + 5)\n        .attr(\"x\", -r - 2.5).attr(\"y\", -r - 2.5)\n        .attr(\"rx\", r + 2.5).attr(\"ry\", r + 2.5)\n        .duration(Polymer.transitionTime);\n\n    function changeFuzzyStartToArcPath(anno) {\n        d3.select(anno.fuzzyStart).attr(\"d\", self.getAnnotationPieSliceArcPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n    }\n\n    function changeCertainToArcPath(anno) {\n        d3.select(anno.certain).attr(\"d\", self.getAnnotationPieSliceArcPath(anno.seqDatum.begin, anno.seqDatum.end, anno));\n    }\n\n    function changeFuzzyEndToArcPath(anno) {\n        d3.select(anno.fuzzyEnd).attr(\"d\", self.getAnnotationPieSliceArcPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n    }\n\n    for (let [annotationType, annotations] of this.annotationSets) {\n        if (this.app.annotationSetsShown.get(annotationType) === true) {\n            for (let anno of annotations) {\n                if (anno.fuzzyStart) {\n                    const fuzzyStart = anno.fuzzyStart;\n                    d3.select(fuzzyStart).transition().attr(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno))\n                        .duration(Polymer.transitionTime).each(\"end\",\n                        function () {\n                            changeFuzzyStartToArcPath(anno);\n                        }\n                    );\n                }\n\n                if (anno.certain) {\n                    const certain = anno.certain;\n                    d3.select(certain).transition().attr(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.begin, anno.seqDatum.end, anno))\n                        .duration(Polymer.transitionTime).each(\"end\",\n                        function () {\n                            changeCertainToArcPath(anno);\n                        }\n                    );\n                }\n\n                if (anno.fuzzyEnd) {\n                    const fuzzyEnd = anno.fuzzyEnd;\n                    d3.select(fuzzyEnd).transition().attr(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno))\n                        .duration(Polymer.transitionTime).each(\"end\",\n                        function () {\n                            changeFuzzyEndToArcPath(anno);\n                        }\n                    );\n                }\n            }\n        }\n    }\n\n    const originalStickZoom = this.stickZoom;\n    const originalRotation = this.rotation;\n    const cubicInOut = d3.ease(\"cubic-in-out\");\n    d3.timer(function (elapsed) {\n        return update(elapsed / Polymer.transitionTime);\n    });\n\n    function update(interp) {\n        const labelTransform = d3.transform(self.labelSVG.getAttribute(\"transform\"));\n        const k = self.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate).translate(labelTranslateInterpol(cubicInOut(interp)), LABEL_Y); //.scale(z).translate(-c.x, -c.y);\n        self.labelSVG.transform.baseVal.initialize(self.app.svgElement.createSVGTransformFromMatrix(k));\n        //~\n        if (xInterpol !== null) {\n            self.setPosition(xInterpol(cubicInOut(interp)), yInterpol(cubicInOut(interp)));\n        }\n\n        self.stickZoom = stickZoomInterpol(cubicInOut(interp));\n        self.setAllLinkCoordinates();\n\n        if (interp === 1) { // finished - tidy up\n            self.form = 0;\n            self.checkLinks();\n            self.stickZoom = originalStickZoom;\n            self.rotation = originalRotation;\n            self.busy = false;\n            return true;\n        } else if (interp > 1) {\n            return update(1);\n        } else {\n            return false;\n        }\n    }\n};\n\nPolymer.prototype.toStick = function () {\n    this.busy = true;\n    this.form = 1;\n\n    //remove prot-prot links - would it be better if checkLinks did this? - think not\n    const c = this.binaryLinks.values().length;\n    for (let l = 0; l < c; l++) {\n        const link = this.binaryLinks.values()[l];\n        //out with the old\n        if (link.shown) {\n            link.hide();\n        }\n    }\n\n    const protLength = this.size * this.stickZoom;\n    const r = this.getSymbolRadius();\n\n    const lengthInterpol = d3.interpolate((2 * r), protLength);\n    const stickZoomInterpol = d3.interpolate(0, this.stickZoom);\n    const labelTranslateInterpol = d3.interpolate(-(r + 5), -(((this.size / 2) * this.stickZoom) + (this.nTerminusFeature ? 25 : 10)));\n\n    const origStickZoom = this.stickZoom;\n    this.stickZoom = 0;\n    this.checkLinks(this.binaryLinks);\n    this.checkLinks(this.selfLink);\n    this.checkLinks(this.sequenceLinks);\n    this.stickZoom = origStickZoom;\n\n    d3.select(this.background).transition() //.attr(\"stroke-opacity\", 1)\n        .attr(\"height\", Polymer.STICKHEIGHT)\n        .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n        .attr(\"rx\", 0).attr(\"ry\", 0)\n        .duration(Polymer.transitionTime);\n\n    d3.select(this.outline).transition() //.attr(\"stroke-opacity\", 1)\n        .attr(\"height\", Polymer.STICKHEIGHT)\n        .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n        .attr(\"rx\", 0).attr(\"ry\", 0)\n        .duration(Polymer.transitionTime);\n\n    d3.select(this.highlight).transition()\n        .attr(\"width\", protLength + 5).attr(\"height\", Polymer.STICKHEIGHT + 5)\n        .attr(\"x\", this.getResXwithStickZoom(0.5) - 2.5).attr(\"y\", (-Polymer.STICKHEIGHT / 2) - 2.5)\n        .attr(\"rx\", 0).attr(\"ry\", 0)\n        .duration(Polymer.transitionTime);\n\n    for (let [annotationType, annotations] of this.annotationSets) {\n        if (this.app.annotationSetsShown.get(annotationType) === true) {\n            for (let anno of annotations) {\n                if (anno.fuzzyStart) {\n                    const fuzzyStart = anno.fuzzyStart;\n                    fuzzyStart.setAttribute(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n                    d3.select(fuzzyStart).transition().attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno))\n                        .duration(Polymer.transitionTime);\n                }\n                if (anno.certain) {\n                    const certain = anno.certain;\n                    let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n                    let tempEnd = anno.seqDatum.end;\n                    if (anno.seqDatum.uncertainBegin) {\n                        tempBegin += 1;\n                    }\n                    if (anno.seqDatum.uncertainEnd) {\n                        tempEnd -= 1;\n                    }\n\n                    certain.setAttribute(\"d\", this.getAnnotationPieSliceApproximatePath(tempBegin, tempEnd, anno));\n                    d3.select(certain).transition().attr(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno))\n                        .duration(Polymer.transitionTime);\n                }\n                if (anno.fuzzyEnd) {\n                    const fuzzyEnd = anno.fuzzyEnd;\n                    fuzzyEnd.setAttribute(\"d\", this.getAnnotationPieSliceApproximatePath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n                    d3.select(fuzzyEnd).transition().attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno))\n                        .duration(Polymer.transitionTime);\n                }\n            }\n        }\n    }\n\n    const self = this;\n    const cubicInOut = d3.ease(\"cubic-in-out\");\n    d3.timer(function (elapsed) {\n        return update(elapsed / Polymer.transitionTime);\n    });\n\n    function update(interp) {\n        const labelTransform = d3.transform(self.labelSVG.getAttribute(\"transform\"));\n        const k = self.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate).translate(labelTranslateInterpol(cubicInOut(interp)), LABEL_Y); //.scale(z).translate(-c.x, -c.y);\n        self.labelSVG.transform.baseVal.initialize(self.app.svgElement.createSVGTransformFromMatrix(k));\n\n        const currentLength = lengthInterpol(cubicInOut(interp));\n        d3.select(self.highlight).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n        d3.select(self.outline).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n        d3.select(self.background).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n        self.stickZoom = stickZoomInterpol(cubicInOut(interp));\n        self.setAllLinkCoordinates();\n\n        if (interp === 1) { // finished - tidy up\n            self.busy = false;\n            return true;\n        } else if (interp > 1) {\n            return update(1);\n        } else {\n            return false;\n        }\n    }\n\n    d3.select(this.ticks).attr(\"opacity\", 0);\n    this.setScaleGroup();\n    d3.select(this.ticks).transition().attr(\"opacity\", 1)\n        .delay(Polymer.transitionTime * 0.8).duration(Polymer.transitionTime / 2);\n};\n\n\nPolymer.prototype.toStickNoTransition = function () { //todo tidy\n    this.busy = true;\n    this.form = 1;\n\n    //remove prot-prot links - would it be better if checkLinks did this? - think not\n    const c = this.binaryLinks.values().length;\n    for (let l = 0; l < c; l++) {\n        const link = this.binaryLinks.values()[l];\n        //out with the old\n        if (link.shown) {\n            link.hide();\n        }\n    }\n\n    const protLength = this.size * this.stickZoom;\n    const r = this.getSymbolRadius();\n\n    const lengthInterpol = d3.interpolate((2 * r), protLength);\n    const labelTranslateInterpol = d3.interpolate(-(r + 5), -(((this.size / 2) * this.stickZoom) + (this.nTerminusFeature ? 25 : 10)));\n\n    this.checkLinks(this.binaryLinks);\n    this.checkLinks(this.selfLink);\n    this.checkLinks(this.sequenceLinks);\n\n    d3.select(this.background)\n        .attr(\"height\", Polymer.STICKHEIGHT)\n        .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n        .attr(\"rx\", 0).attr(\"ry\", 0);\n\n    d3.select(this.outline)\n        .attr(\"height\", Polymer.STICKHEIGHT)\n        .attr(\"y\", -Polymer.STICKHEIGHT / 2)\n        .attr(\"rx\", 0).attr(\"ry\", 0);\n\n    d3.select(this.highlight)\n        .attr(\"width\", protLength + 5).attr(\"height\", Polymer.STICKHEIGHT + 5)\n        .attr(\"x\", this.getResXwithStickZoom(0.5) - 2.5).attr(\"y\", (-Polymer.STICKHEIGHT / 2) - 2.5)\n        .attr(\"rx\", 0).attr(\"ry\", 0);\n\n    for (let [annotationType, annotations] of this.annotationSets) {\n        if (this.app.annotationSetsShown.get(annotationType) === true) {\n            for (let anno of annotations) {\n                if (anno.fuzzyStart) {\n                    const fuzzyStart = anno.fuzzyStart;\n                    d3.select(fuzzyStart).attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n                }\n                if (anno.certain) {\n                    let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n                    let tempEnd = anno.seqDatum.end;\n                    if (anno.seqDatum.uncertainBegin) {\n                        tempBegin += 1;\n                    }\n                    if (anno.seqDatum.uncertainEnd) {\n                        tempEnd -= 1;\n                    }\n                    anno.certain.setAttribute(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno));\n                }\n                if (anno.fuzzyEnd) {\n                    const fuzzyEnd = anno.fuzzyEnd;\n                    d3.select(fuzzyEnd) /*.transition()*/ .attr(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n                }\n            }\n        }\n    }\n\n    const self = this;\n\n    const labelTransform = d3.transform(self.labelSVG.getAttribute(\"transform\"));\n    const k = self.app.svgElement.createSVGMatrix().rotate(labelTransform.rotate).translate(labelTranslateInterpol(1), LABEL_Y); //.scale(z).translate(-c.x, -c.y);\n    self.labelSVG.transform.baseVal.initialize(self.app.svgElement.createSVGTransformFromMatrix(k));\n\n    const currentLength = lengthInterpol(1);\n    d3.select(self.highlight).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n    d3.select(self.outline).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n    d3.select(self.background).attr(\"width\", currentLength).attr(\"x\", -(currentLength / 2) + (0.5 * self.stickZoom));\n    self.setAllLinkCoordinates();\n\n    this.setScaleGroup();\n    d3.select(this.ticks).attr(\"opacity\", 1);\n\n    self.busy = false;\n};\n\nPolymer.prototype.getResXwithStickZoom = function (r) {\n    // if (isNaN(r)) {\n    //     console.error(\"NOT NUMBER\");\n    // }\n    if (r === \"n-n\") {\n        return (-this.size / 2 * this.stickZoom) - 20;\n    } else if (r === \"c-c\") {\n        return (this.size / 2 * this.stickZoom) + 20;\n    } else {\n        return (r - (this.size / 2)) * this.stickZoom;\n    }\n};\n\n//calculate the  coordinates of a residue (relative to this.util.container)\nPolymer.prototype.getResidueCoordinates = function (r, yOff) {\n    if (typeof r === \"undefined\") {\n        console.error(\"ERROR: residue number is undefined\");\n    }\n    let x = this.getResXwithStickZoom(r * 1);// * this.app.z;\n    // console.log(\"***\", this.app.z);\n    // coz prots don't scale, don't multiple by app.z\n    let y;\n    if (x !== 0) {\n        const l = Math.abs(x);\n        const a = Math.acos(x / l);\n        const rotRad = (this.rotation / 360) * Math.PI * 2;\n        x = l * Math.cos(rotRad + a);\n        y = l * Math.sin(rotRad + a);\n        if (typeof yOff !== \"undefined\") {\n            x += yOff /** this.app.z*/ * Math.cos(rotRad + (Math.PI / 2));\n            y += yOff /** this.app.z*/ * Math.sin(rotRad + (Math.PI / 2));\n        }\n    } else {\n        y = yOff;\n    }\n    x += this.ix;\n    y += this.iy;\n    return [x, y];\n};\n\nPolymer.prototype.clearPositionalFeatures = function () {\n    this.annotations = [];\n    this.annotationTypes = [];\n    if (this.annotationsSvgGroup) d3.select(this.annotationsSvgGroup).selectAll(\"*\").remove();\n};\n\nPolymer.prototype.setPositionalFeatures = function () {\n    const self = this;\n\n    const toolTipFunc = function (evt) {\n        const el = (evt.target.correspondingUseElement) ? evt.target.correspondingUseElement : evt.target;\n        self.app.preventDefaultsAndStopPropagation(evt);\n        self.app.setTooltip(el.name, el.getAttribute(\"fill\"));\n        self.showHighlight(true);\n    };\n\n    const annotationTypesSet = new Set();\n\n    for (let [annotationType, annotationSet] of this.annotationSets) {\n        if (this.app.annotationSetsShown.get(annotationType) === true) {\n            for (let annotation of annotationSet.values()) {\n\n                if (annotation.seqDatum.sequenceDatumString !== \"n-n\" && annotation.seqDatum.sequenceDatumString !== \"c-c\") {\n                    annotationTypesSet.add(annotation.description);\n                }\n            }\n        }\n    }\n    this.annotationTypes = Array.from(annotationTypesSet.values()).sort();\n\n    for (let [annotationType, annotationSet] of this.annotationSets) {\n        if (this.app.annotationSetsShown.get(annotationType) === true) {\n            for (let anno of annotationSet.values()) {\n                let text = anno.description + \" [\" + (anno.seqDatum ? anno.seqDatum.toString() : anno.seqDatum.begin + \" - \" + anno.seqDatum.end) + \"]\";\n                if (anno.description === \"No annotations\") {\n                    text = \"No annotations\";\n                }\n                if (anno.seqDatum.uncertainBegin) {\n                    anno.fuzzyStart = document.createElementNS(svgns, \"path\");\n                    if (this.form === 0) {\n                        anno.fuzzyStart.setAttribute(\"d\", this.getAnnotationPieSliceArcPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n                    } else {\n                        anno.fuzzyStart.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.uncertainBegin, anno.seqDatum.begin, anno));\n                    }\n                    anno.fuzzyStart.setAttribute(\"stroke\", \"none\");//-width\", \"1\"); // todo - should be css\n                    // anno.fuzzyStart.setAttribute(\"fill-opacity\", \"0.6\");\n                    anno.fuzzyStart.name = text;\n                    anno.fuzzyStart.onmouseover = toolTipFunc;\n                    this.annotationsSvgGroup.appendChild(anno.fuzzyStart);\n                }\n\n                if (anno.seqDatum.begin && anno.seqDatum.end) {\n                    anno.certain = document.createElementNS(svgns, \"path\");\n                    let tempBegin = anno.seqDatum.begin; //todo - might be better to have seperate att in SequenceData for end of uncertain start\n                    let tempEnd = anno.seqDatum.end;\n                    if (anno.seqDatum.uncertainBegin) {\n                        tempBegin += 1;\n                    }\n                    if (anno.seqDatum.uncertainEnd) {\n                        tempEnd -= 1;\n                    }\n                    if (this.form === 0) {\n                        anno.certain.setAttribute(\"d\", this.getAnnotationPieSliceArcPath(tempBegin, tempEnd, anno));\n                    } else {\n                        anno.certain.setAttribute(\"d\", this.getAnnotationRectPath(tempBegin, tempEnd, anno));\n                    }\n                    anno.certain.setAttribute(\"stroke\", \"none\");//-width\", \"1\");\n                    // anno.certain.setAttribute(\"fill-opacity\", \"0.6\");\n                    anno.certain.name = text;\n                    anno.certain.onmouseover = toolTipFunc;\n                    this.annotationsSvgGroup.appendChild(anno.certain);\n                }\n                if (anno.seqDatum.uncertainEnd) {\n                    anno.fuzzyEnd = document.createElementNS(svgns, \"path\");\n                    if (this.form === 0) {\n                        anno.fuzzyEnd.setAttribute(\"d\", this.getAnnotationPieSliceArcPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n                    } else {\n                        anno.fuzzyEnd.setAttribute(\"d\", this.getAnnotationRectPath(anno.seqDatum.end, anno.seqDatum.uncertainEnd, anno));\n                    }\n                    anno.fuzzyEnd.setAttribute(\"stroke\", \"none\"); //-width\", \"1\");\n                    // anno.fuzzyEnd.setAttribute(\"fill-opacity\", \"0.6\");\n                    anno.fuzzyEnd.name = text;\n                    anno.fuzzyEnd.onmouseover = toolTipFunc;\n                    this.annotationsSvgGroup.appendChild(anno.fuzzyEnd);\n                }\n            }\n        }\n    }\n};\n\nPolymer.stepsInArc = 5;\n\nPolymer.prototype.getAnnotationPieSliceArcPath = function (startRes, endRes, annotation) {\n    const radius = this.getSymbolRadius();// - 2;\n\n    let top, bottom, rungHeight;\n    const rung = this.annotationTypes.indexOf(annotation.description);\n    // console.log(\"rung\", rung, this.annotationTypes);\n    if (rung === -1) {\n        bottom = 0;\n        top = radius;\n    } else {\n        //Math.sqrt(this.participant.size / Math.PI) * 0.6\n        rungHeight = radius / this.annotationTypes.length;\n        bottom = (rung * rungHeight);\n        top = bottom + rungHeight;\n        //\n        // bottom = Math.sqrt(rung / this.annotationTypes.length) * radius;\n        // top = Math.sqrt(rung + 1 / this.annotationTypes.length) * radius;\n    }\n\n    // var startAngle = ((startRes - 1) / this.size) * 360;\n    // var endAngle = ((endRes - 1) / this.size) * 360;\n    let startAngle, endAngle;\n    if (startRes === \"n-n\") {\n        startAngle = -20; //((startRes - 1) / this.size) * 360;\n        endAngle = 0;//((endRes - 1) / this.size) * 360;\n    } else if (endRes === \"c-c\") {\n        startAngle = 0;//((startRes - 1) / this.size) * 360;\n        endAngle = +20; //((endRes) / this.size) * 360;\n    } else {\n        startAngle = ((startRes - 1) / this.size) * 360;\n        endAngle = ((endRes - 1) / this.size) * 360;\n    }\n    // const arcStart = trig(radius, startAngle - 90);\n    // const arcEnd = trig(radius, endAngle - 90);\n    let largeArch = 0;\n    if ((endAngle - startAngle) > 180 || (endAngle === startAngle)) {\n        largeArch = 1;\n    }\n\n    const p1 = rotatePointAboutPoint([0, bottom], [0, 0], startAngle - 180);\n    const p2 = rotatePointAboutPoint([0, top], [0, 0], startAngle - 180);\n    const p3 = rotatePointAboutPoint([0, top], [0, 0], endAngle - 180);\n    const p4 = rotatePointAboutPoint([0, bottom], [0, 0], endAngle - 180);\n\n    //const r = (bottom + top) / 2;\n    const path = \"M\" + p1[0] + \",\" + p1[1] + \" L\" + p2[0] + \",\" + p2[1]\n        + \" A\" + top + \",\" + top + \" 0 \" + largeArch + \" 1 \" + p3[0] + \",\" + p3[1] + \" L\" + p4[0] + \",\" + p4[1]\n        + \" A\" + bottom + \",\" + bottom + \" 0 \" + largeArch + \" 0 \" + p1[0] + \",\" + p1[1] + \" Z\";\n    // console.log(\"**\", path);\n    return path;\n    // return \"M0,0 L\" + arcStart.x + \",\" + arcStart.y + \" A\" + radius + \",\" +\n    //     radius + \" 0 \" + largeArch + \" 1 \" + arcEnd.x + \",\" + arcEnd.y + \" Z\";\n};\n\nPolymer.prototype.getAnnotationPieSliceApproximatePath = function (startRes, endRes, annotation) {\n\n    // let top, bottom, rungHeight;\n    // const rung = this.annotationTypes.indexOf(annotation.description);\n    // // console.log(\"rung\", rung, this.annotationTypes);\n    // if (rung === -1) {\n    //     bottom = Polymer.STICKHEIGHT / 2;\n    //     top = -Polymer.STICKHEIGHT / 2;\n    // } else {\n    //     rungHeight = Polymer.STICKHEIGHT / this.annotationTypes.length;\n    //     top = (-Polymer.STICKHEIGHT / 2) + (rung * rungHeight);\n    //     bottom = top + rungHeight;\n    // }\n\n    //approximate pie slice\n    let startAngle, endAngle;\n    if (startRes === \"n-n\") {\n        startAngle = -20; //((startRes - 1) / this.size) * 360;\n        endAngle = 0;//((endRes) / this.size) * 360;\n    } else if (endRes === \"c-c\") {\n        startAngle = 0;//((startRes - 1) / this.size) * 360;\n        endAngle = +20; //((endRes) / this.size) * 360;\n    } else {\n        startAngle = ((startRes - 1) / this.size) * 360;\n        endAngle = ((endRes) / this.size) * 360;\n    }\n    const pieRadius = this.getSymbolRadius() - 2;\n    // var arcStart = Interactor.trig(pieRadius, startAngle - 90);\n    // var arcEnd = Interactor.trig(pieRadius, endAngle - 90);\n    let approximatePiePath = \"M 0,0\";\n    const stepsInArc = 5;\n    for (let sia = 0; sia <= Polymer.stepsInArc; sia++) {\n        const angle = startAngle + ((endAngle - startAngle) * (sia / stepsInArc));\n        const siaCoord = trig(pieRadius, angle - 90);\n        approximatePiePath += \" L \" + siaCoord.x + \",\" + siaCoord.y;\n    }\n    approximatePiePath += \" L \" + 0 + \",\" + 0;\n    approximatePiePath += \"  Z\";\n    return approximatePiePath;\n};\n\nPolymer.prototype.getAnnotationRectPath = function (startRes, endRes, annotation) {\n    //domain as rectangle path\n    let top, bottom, rungHeight;\n    const rung = this.annotationTypes.indexOf(annotation.description);\n    // console.log(\"rung\", rung, this.annotationTypes);\n    if (rung === -1) {\n        bottom = Polymer.STICKHEIGHT / 2;\n        top = -Polymer.STICKHEIGHT / 2;\n    } else {\n        rungHeight = Polymer.STICKHEIGHT / this.annotationTypes.length;\n        top = (-Polymer.STICKHEIGHT / 2) + (rung * rungHeight);\n        bottom = top + rungHeight;\n    }\n\n    let annotX, annotSize, annotLength;\n    if (startRes === \"n-n\") {\n        annotX = this.getResXwithStickZoom(0.5) - 20;\n        // var annotSize = (1 + (endRes - startRes));\n        annotLength = 20;//annotSize * this.stickZoom;\n    } else if (endRes === \"c-c\") {\n        annotX = this.getResXwithStickZoom(this.size + 0.5);\n        // var annotSize = (1 + (endRes - startRes));\n        annotLength = 20;//annotSize * this.stickZoom;\n    } else {\n        annotX = this.getResXwithStickZoom(startRes - 0.5);\n        annotSize = (1 + (endRes - startRes));\n        annotLength = annotSize * this.stickZoom;\n    }\n    let rectPath = \"M \" + annotX + \",\" + bottom;\n    for (let sia = 0; sia <= Polymer.stepsInArc; sia++) {\n        const step = annotX + (annotLength * (sia / Polymer.stepsInArc));\n        rectPath += \" L \" + step + \",\" + top;\n    }\n    rectPath += \" L \" + (annotX + annotLength) + \",\" + bottom +\n        \" Z\";\n    return rectPath;\n};\n"],"mappings":"AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/js/viz/interactor/polymer.js\n"); /***/ }), @@ -1321,7 +1333,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"Protein\", function() { return Protein; });\n/* harmony import */ var _polymer__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./polymer */ \"./src/js/viz/interactor/polymer.js\");\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n\n\n\nfunction Protein(id, /*App*/ app, json, name) {\n this.init(id, app, json, name);\n this.type = \"protein\"; // this isn't absolutely necessary, could do without it\n\n this.upperGroup = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"g\");\n this.rotation = 0;\n this.stickZoom = 1;\n this.form = 0; // 0 = blob, 1 = stick\n\n //make highlight\n this.highlight = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"rect\");\n this.highlight.setAttribute(\"stroke\", _config__WEBPACK_IMPORTED_MODULE_1__[\"highlightColour\"]);\n this.highlight.setAttribute(\"stroke-width\", \"5\");\n this.highlight.setAttribute(\"fill\", \"none\");\n this.upperGroup.appendChild(this.highlight);\n\n //make background\n //http://stackoverflow.com/questions/17437408/how-do-i-change-a-circle-to-a-square-using-d3\n this.background = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"rect\");\n this.background.setAttribute(\"fill\", \"#FFFFFF\");\n this.upperGroup.appendChild(this.background);\n //create label - we will move this svg element around when protein form changes\n this.initLabel();\n //ticks (and amino acid letters)\n this.ticks = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"g\");\n //svg group for annotations\n this.annotationsSvgGroup = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"g\");\n this.annotationsSvgGroup.setAttribute(\"opacity\", \"1\");\n this.upperGroup.appendChild(this.annotationsSvgGroup);\n\n //make outline\n this.outline = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"rect\");\n this.outline.setAttribute(\"stroke\", \"black\");\n this.outline.setAttribute(\"stroke-width\", \"1\");\n this.outline.setAttribute(\"fill\", \"none\");\n this.upperGroup.appendChild(this.outline);\n\n this.scaleLabels = [];\n\n //since form is set to 0, make this a circle, this stuff is equivalent to\n // end result of toCircle but without transition\n const r = this.getSymbolRadius();\n\n this.outline.setAttribute(\"x\", -r);\n this.outline.setAttribute(\"y\", -r);\n this.outline.setAttribute(\"width\", r * 2);\n this.outline.setAttribute(\"height\", r * 2);\n this.outline.setAttribute(\"rx\", r);\n this.outline.setAttribute(\"ry\", r);\n\n this.background.setAttribute(\"x\", -r);\n this.background.setAttribute(\"y\", -r);\n this.background.setAttribute(\"width\", r * 2);\n this.background.setAttribute(\"height\", r * 2);\n this.background.setAttribute(\"rx\", r);\n this.background.setAttribute(\"ry\", r);\n\n this.annotationsSvgGroup.setAttribute(\"transform\", \"scale(1, 1)\");\n\n this.highlight.setAttribute(\"width\", (r * 2) + 5);\n this.highlight.setAttribute(\"height\", (r * 2) + 5);\n this.highlight.setAttribute(\"x\", -r - 2.5);\n this.highlight.setAttribute(\"y\", -r - 2.5);\n this.highlight.setAttribute(\"rx\", r + 2.5);\n this.highlight.setAttribute(\"ry\", r + 2.5);\n this.highlight.setAttribute(\"stroke-opacity\", \"0\");\n\n this.labelSVG.setAttribute(\"transform\", \"translate(\" + (-(r + 5)) + \",\" + \"-5)\");\n\n this.initListeners();\n\n const self = this;\n Object.defineProperty(this, \"height\", {\n get: function height() {\n return self.form == 1? 120:40;\n //return 160;\n }\n });\n\n this.showHighlight(false);\n}\n\nProtein.prototype = new _polymer__WEBPACK_IMPORTED_MODULE_0__[\"Polymer\"]();\n\n/*\nProtein.prototype.showData = function(evt) {\n const url = \"http://www.uniprot.org/uniprot/\" + this.json.identifier.id;\n window.open(url, '_blank');\n}\n*/\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvdml6L2ludGVyYWN0b3IvcHJvdGVpbi5qcy5qcyIsInNvdXJjZXMiOlsid2VicGFjazovL2NvbXBsZXh2aWV3ZXIvLi9zcmMvanMvdml6L2ludGVyYWN0b3IvcHJvdGVpbi5qcz9kMmFiIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7UG9seW1lcn0gZnJvbSBcIi4vcG9seW1lclwiO1xuaW1wb3J0IHtzdmducywgaGlnaGxpZ2h0Q29sb3VyfSBmcm9tIFwiLi4vLi4vY29uZmlnXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiBQcm90ZWluKGlkLCAvKkFwcCovIGFwcCwganNvbiwgbmFtZSkge1xuICAgIHRoaXMuaW5pdChpZCwgYXBwLCBqc29uLCBuYW1lKTtcbiAgICB0aGlzLnR5cGUgPSBcInByb3RlaW5cIjsgLy8gdGhpcyBpc24ndCBhYnNvbHV0ZWx5IG5lY2Vzc2FyeSwgY291bGQgZG8gd2l0aG91dCBpdFxuXG4gICAgdGhpcy51cHBlckdyb3VwID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudE5TKHN2Z25zLCBcImdcIik7XG4gICAgdGhpcy5yb3RhdGlvbiA9IDA7XG4gICAgdGhpcy5zdGlja1pvb20gPSAxO1xuICAgIHRoaXMuZm9ybSA9IDA7IC8vIDAgPSBibG9iLCAxID0gc3RpY2tcblxuICAgIC8vbWFrZSBoaWdobGlnaHRcbiAgICB0aGlzLmhpZ2hsaWdodCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnROUyhzdmducywgXCJyZWN0XCIpO1xuICAgIHRoaXMuaGlnaGxpZ2h0LnNldEF0dHJpYnV0ZShcInN0cm9rZVwiLCBoaWdobGlnaHRDb2xvdXIpO1xuICAgIHRoaXMuaGlnaGxpZ2h0LnNldEF0dHJpYnV0ZShcInN0cm9rZS13aWR0aFwiLCBcIjVcIik7XG4gICAgdGhpcy5oaWdobGlnaHQuc2V0QXR0cmlidXRlKFwiZmlsbFwiLCBcIm5vbmVcIik7XG4gICAgdGhpcy51cHBlckdyb3VwLmFwcGVuZENoaWxkKHRoaXMuaGlnaGxpZ2h0KTtcblxuICAgIC8vbWFrZSBiYWNrZ3JvdW5kXG4gICAgLy9odHRwOi8vc3RhY2tvdmVyZmxvdy5jb20vcXVlc3Rpb25zLzE3NDM3NDA4L2hvdy1kby1pLWNoYW5nZS1hLWNpcmNsZS10by1hLXNxdWFyZS11c2luZy1kM1xuICAgIHRoaXMuYmFja2dyb3VuZCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnROUyhzdmducywgXCJyZWN0XCIpO1xuICAgIHRoaXMuYmFja2dyb3VuZC5zZXRBdHRyaWJ1dGUoXCJmaWxsXCIsIFwiI0ZGRkZGRlwiKTtcbiAgICB0aGlzLnVwcGVyR3JvdXAuYXBwZW5kQ2hpbGQodGhpcy5iYWNrZ3JvdW5kKTtcbiAgICAvL2NyZWF0ZSBsYWJlbCAtIHdlIHdpbGwgbW92ZSB0aGlzIHN2ZyBlbGVtZW50IGFyb3VuZCB3aGVuIHByb3RlaW4gZm9ybSBjaGFuZ2VzXG4gICAgdGhpcy5pbml0TGFiZWwoKTtcbiAgICAvL3RpY2tzIChhbmQgYW1pbm8gYWNpZCBsZXR0ZXJzKVxuICAgIHRoaXMudGlja3MgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50TlMoc3ZnbnMsIFwiZ1wiKTtcbiAgICAvL3N2ZyBncm91cCBmb3IgYW5ub3RhdGlvbnNcbiAgICB0aGlzLmFubm90YXRpb25zU3ZnR3JvdXAgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50TlMoc3ZnbnMsIFwiZ1wiKTtcbiAgICB0aGlzLmFubm90YXRpb25zU3ZnR3JvdXAuc2V0QXR0cmlidXRlKFwib3BhY2l0eVwiLCBcIjFcIik7XG4gICAgdGhpcy51cHBlckdyb3VwLmFwcGVuZENoaWxkKHRoaXMuYW5ub3RhdGlvbnNTdmdHcm91cCk7XG5cbiAgICAvL21ha2Ugb3V0bGluZVxuICAgIHRoaXMub3V0bGluZSA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnROUyhzdmducywgXCJyZWN0XCIpO1xuICAgIHRoaXMub3V0bGluZS5zZXRBdHRyaWJ1dGUoXCJzdHJva2VcIiwgXCJibGFja1wiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwic3Ryb2tlLXdpZHRoXCIsIFwiMVwiKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwiZmlsbFwiLCBcIm5vbmVcIik7XG4gICAgdGhpcy51cHBlckdyb3VwLmFwcGVuZENoaWxkKHRoaXMub3V0bGluZSk7XG5cbiAgICB0aGlzLnNjYWxlTGFiZWxzID0gW107XG5cbiAgICAvL3NpbmNlIGZvcm0gaXMgc2V0IHRvIDAsIG1ha2UgdGhpcyBhIGNpcmNsZSwgdGhpcyBzdHVmZiBpcyBlcXVpdmFsZW50IHRvXG4gICAgLy8gZW5kIHJlc3VsdCBvZiB0b0NpcmNsZSBidXQgd2l0aG91dCB0cmFuc2l0aW9uXG4gICAgY29uc3QgciA9IHRoaXMuZ2V0U3ltYm9sUmFkaXVzKCk7XG5cbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwieFwiLCAtcik7XG4gICAgdGhpcy5vdXRsaW5lLnNldEF0dHJpYnV0ZShcInlcIiwgLXIpO1xuICAgIHRoaXMub3V0bGluZS5zZXRBdHRyaWJ1dGUoXCJ3aWR0aFwiLCByICogMik7XG4gICAgdGhpcy5vdXRsaW5lLnNldEF0dHJpYnV0ZShcImhlaWdodFwiLCByICogMik7XG4gICAgdGhpcy5vdXRsaW5lLnNldEF0dHJpYnV0ZShcInJ4XCIsIHIpO1xuICAgIHRoaXMub3V0bGluZS5zZXRBdHRyaWJ1dGUoXCJyeVwiLCByKTtcblxuICAgIHRoaXMuYmFja2dyb3VuZC5zZXRBdHRyaWJ1dGUoXCJ4XCIsIC1yKTtcbiAgICB0aGlzLmJhY2tncm91bmQuc2V0QXR0cmlidXRlKFwieVwiLCAtcik7XG4gICAgdGhpcy5iYWNrZ3JvdW5kLnNldEF0dHJpYnV0ZShcIndpZHRoXCIsIHIgKiAyKTtcbiAgICB0aGlzLmJhY2tncm91bmQuc2V0QXR0cmlidXRlKFwiaGVpZ2h0XCIsIHIgKiAyKTtcbiAgICB0aGlzLmJhY2tncm91bmQuc2V0QXR0cmlidXRlKFwicnhcIiwgcik7XG4gICAgdGhpcy5iYWNrZ3JvdW5kLnNldEF0dHJpYnV0ZShcInJ5XCIsIHIpO1xuXG4gICAgdGhpcy5hbm5vdGF0aW9uc1N2Z0dyb3VwLnNldEF0dHJpYnV0ZShcInRyYW5zZm9ybVwiLCBcInNjYWxlKDEsIDEpXCIpO1xuXG4gICAgdGhpcy5oaWdobGlnaHQuc2V0QXR0cmlidXRlKFwid2lkdGhcIiwgKHIgKiAyKSArIDUpO1xuICAgIHRoaXMuaGlnaGxpZ2h0LnNldEF0dHJpYnV0ZShcImhlaWdodFwiLCAociAqIDIpICsgNSk7XG4gICAgdGhpcy5oaWdobGlnaHQuc2V0QXR0cmlidXRlKFwieFwiLCAtciAtIDIuNSk7XG4gICAgdGhpcy5oaWdobGlnaHQuc2V0QXR0cmlidXRlKFwieVwiLCAtciAtIDIuNSk7XG4gICAgdGhpcy5oaWdobGlnaHQuc2V0QXR0cmlidXRlKFwicnhcIiwgciArIDIuNSk7XG4gICAgdGhpcy5oaWdobGlnaHQuc2V0QXR0cmlidXRlKFwicnlcIiwgciArIDIuNSk7XG4gICAgdGhpcy5oaWdobGlnaHQuc2V0QXR0cmlidXRlKFwic3Ryb2tlLW9wYWNpdHlcIiwgXCIwXCIpO1xuXG4gICAgdGhpcy5sYWJlbFNWRy5zZXRBdHRyaWJ1dGUoXCJ0cmFuc2Zvcm1cIiwgXCJ0cmFuc2xhdGUoXCIgKyAoLShyICsgNSkpICsgXCIsXCIgKyBcIi01KVwiKTtcblxuICAgIHRoaXMuaW5pdExpc3RlbmVycygpO1xuXG4gICAgY29uc3Qgc2VsZiA9IHRoaXM7XG4gICAgT2JqZWN0LmRlZmluZVByb3BlcnR5KHRoaXMsIFwiaGVpZ2h0XCIsIHtcbiAgICAgICAgZ2V0OiBmdW5jdGlvbiBoZWlnaHQoKSB7XG4gICAgICAgICAgICByZXR1cm4gc2VsZi5mb3JtID09IDE/IDEyMDo0MDtcbiAgICAgICAgICAgIC8vcmV0dXJuIDE2MDtcbiAgICAgICAgfVxuICAgIH0pO1xuXG4gICAgdGhpcy5zaG93SGlnaGxpZ2h0KGZhbHNlKTtcbn1cblxuUHJvdGVpbi5wcm90b3R5cGUgPSBuZXcgUG9seW1lcigpO1xuXG4vKlxuUHJvdGVpbi5wcm90b3R5cGUuc2hvd0RhdGEgPSBmdW5jdGlvbihldnQpIHtcbiAgICBjb25zdCB1cmwgPSBcImh0dHA6Ly93d3cudW5pcHJvdC5vcmcvdW5pcHJvdC9cIiArIHRoaXMuanNvbi5pZGVudGlmaWVyLmlkO1xuICAgIHdpbmRvdy5vcGVuKHVybCwgJ19ibGFuaycpO1xufVxuKi9cbiJdLCJtYXBwaW5ncyI6IkFBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7Iiwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./src/js/viz/interactor/protein.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"Protein\", function() { return Protein; });\n/* harmony import */ var _polymer__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./polymer */ \"./src/js/viz/interactor/polymer.js\");\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n\n\n\nfunction Protein(id, /*App*/ app, json, name) {\n this.init(id, app, json, name);\n this.type = \"protein\"; // this isn't absolutely necessary, could do without it\n\n this.upperGroup = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"g\");\n this.rotation = 0;\n this.stickZoom = 1;\n this.form = 0; // 0 = blob, 1 = stick\n\n //make highlight\n this.highlight = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"rect\");\n this.highlight.classList.add(\"highlight\", \"participant-highlight\");\n this.upperGroup.appendChild(this.highlight);\n\n //make background\n //http://stackoverflow.com/questions/17437408/how-do-i-change-a-circle-to-a-square-using-d3\n this.background = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"rect\");\n this.background.setAttribute(\"fill\", \"#FFFFFF\");\n this.upperGroup.appendChild(this.background);\n //create label - we will move this svg element around when protein form changes\n this.initLabel();\n //ticks (and amino acid letters)\n this.ticks = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"g\");\n //svg group for annotations\n this.annotationsSvgGroup = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"g\");\n this.annotationsSvgGroup.setAttribute(\"opacity\", \"1\");\n this.upperGroup.appendChild(this.annotationsSvgGroup);\n\n //make outline\n this.outline = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"rect\");\n // css...\n this.outline.setAttribute(\"stroke\", \"black\");\n this.outline.setAttribute(\"stroke-width\", \"1\");\n this.outline.setAttribute(\"fill\", \"none\");\n this.upperGroup.appendChild(this.outline);\n\n this.scaleLabels = [];\n\n //since form is set to 0, make this a circle, this stuff is equivalent to\n // end result of toCircle but without transition\n const r = this.getSymbolRadius();\n\n this.outline.setAttribute(\"x\", -r);\n this.outline.setAttribute(\"y\", -r);\n this.outline.setAttribute(\"width\", r * 2);\n this.outline.setAttribute(\"height\", r * 2);\n this.outline.setAttribute(\"rx\", r);\n this.outline.setAttribute(\"ry\", r);\n\n this.background.setAttribute(\"x\", -r);\n this.background.setAttribute(\"y\", -r);\n this.background.setAttribute(\"width\", r * 2);\n this.background.setAttribute(\"height\", r * 2);\n this.background.setAttribute(\"rx\", r);\n this.background.setAttribute(\"ry\", r);\n\n this.annotationsSvgGroup.setAttribute(\"transform\", \"scale(1, 1)\");\n\n this.highlight.setAttribute(\"width\", (r * 2) + 5);\n this.highlight.setAttribute(\"height\", (r * 2) + 5);\n this.highlight.setAttribute(\"x\", -r - 2.5);\n this.highlight.setAttribute(\"y\", -r - 2.5);\n this.highlight.setAttribute(\"rx\", r + 2.5);\n this.highlight.setAttribute(\"ry\", r + 2.5);\n this.highlight.setAttribute(\"stroke-opacity\", \"0\");\n\n this.labelSVG.setAttribute(\"transform\", \"translate(\" + (-(r + 5)) + \",\" + \"-5)\");\n\n this.initListeners();\n\n const self = this;\n Object.defineProperty(this, \"height\", {\n get: function height() {\n return self.form == 1? 120:40;\n //return 160;\n }\n });\n\n this.showHighlight(false);\n}\n\nProtein.prototype = new _polymer__WEBPACK_IMPORTED_MODULE_0__[\"Polymer\"]();\n\n/*\nProtein.prototype.showData = function(evt) {\n const url = \"http://www.uniprot.org/uniprot/\" + this.json.identifier.id;\n window.open(url, '_blank');\n}\n*/\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvdml6L2ludGVyYWN0b3IvcHJvdGVpbi5qcy5qcyIsInNvdXJjZXMiOlsid2VicGFjazovL2NvbXBsZXh2aWV3ZXIvLi9zcmMvanMvdml6L2ludGVyYWN0b3IvcHJvdGVpbi5qcz9kMmFiIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7UG9seW1lcn0gZnJvbSBcIi4vcG9seW1lclwiO1xuaW1wb3J0IHtzdmduc30gZnJvbSBcIi4uLy4uL2NvbmZpZ1wiO1xuXG5leHBvcnQgZnVuY3Rpb24gUHJvdGVpbihpZCwgLypBcHAqLyBhcHAsIGpzb24sIG5hbWUpIHtcbiAgICB0aGlzLmluaXQoaWQsIGFwcCwganNvbiwgbmFtZSk7XG4gICAgdGhpcy50eXBlID0gXCJwcm90ZWluXCI7IC8vIHRoaXMgaXNuJ3QgYWJzb2x1dGVseSBuZWNlc3NhcnksIGNvdWxkIGRvIHdpdGhvdXQgaXRcblxuICAgIHRoaXMudXBwZXJHcm91cCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnROUyhzdmducywgXCJnXCIpO1xuICAgIHRoaXMucm90YXRpb24gPSAwO1xuICAgIHRoaXMuc3RpY2tab29tID0gMTtcbiAgICB0aGlzLmZvcm0gPSAwOyAvLyAwID0gYmxvYiwgMSA9IHN0aWNrXG5cbiAgICAvL21ha2UgaGlnaGxpZ2h0XG4gICAgdGhpcy5oaWdobGlnaHQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50TlMoc3ZnbnMsIFwicmVjdFwiKTtcbiAgICB0aGlzLmhpZ2hsaWdodC5jbGFzc0xpc3QuYWRkKFwiaGlnaGxpZ2h0XCIsIFwicGFydGljaXBhbnQtaGlnaGxpZ2h0XCIpO1xuICAgIHRoaXMudXBwZXJHcm91cC5hcHBlbmRDaGlsZCh0aGlzLmhpZ2hsaWdodCk7XG5cbiAgICAvL21ha2UgYmFja2dyb3VuZFxuICAgIC8vaHR0cDovL3N0YWNrb3ZlcmZsb3cuY29tL3F1ZXN0aW9ucy8xNzQzNzQwOC9ob3ctZG8taS1jaGFuZ2UtYS1jaXJjbGUtdG8tYS1zcXVhcmUtdXNpbmctZDNcbiAgICB0aGlzLmJhY2tncm91bmQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50TlMoc3ZnbnMsIFwicmVjdFwiKTtcbiAgICB0aGlzLmJhY2tncm91bmQuc2V0QXR0cmlidXRlKFwiZmlsbFwiLCBcIiNGRkZGRkZcIik7XG4gICAgdGhpcy51cHBlckdyb3VwLmFwcGVuZENoaWxkKHRoaXMuYmFja2dyb3VuZCk7XG4gICAgLy9jcmVhdGUgbGFiZWwgLSB3ZSB3aWxsIG1vdmUgdGhpcyBzdmcgZWxlbWVudCBhcm91bmQgd2hlbiBwcm90ZWluIGZvcm0gY2hhbmdlc1xuICAgIHRoaXMuaW5pdExhYmVsKCk7XG4gICAgLy90aWNrcyAoYW5kIGFtaW5vIGFjaWQgbGV0dGVycylcbiAgICB0aGlzLnRpY2tzID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudE5TKHN2Z25zLCBcImdcIik7XG4gICAgLy9zdmcgZ3JvdXAgZm9yIGFubm90YXRpb25zXG4gICAgdGhpcy5hbm5vdGF0aW9uc1N2Z0dyb3VwID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudE5TKHN2Z25zLCBcImdcIik7XG4gICAgdGhpcy5hbm5vdGF0aW9uc1N2Z0dyb3VwLnNldEF0dHJpYnV0ZShcIm9wYWNpdHlcIiwgXCIxXCIpO1xuICAgIHRoaXMudXBwZXJHcm91cC5hcHBlbmRDaGlsZCh0aGlzLmFubm90YXRpb25zU3ZnR3JvdXApO1xuXG4gICAgLy9tYWtlIG91dGxpbmVcbiAgICB0aGlzLm91dGxpbmUgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50TlMoc3ZnbnMsIFwicmVjdFwiKTtcbiAgICAvLyBjc3MuLi5cbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwic3Ryb2tlXCIsIFwiYmxhY2tcIik7XG4gICAgdGhpcy5vdXRsaW5lLnNldEF0dHJpYnV0ZShcInN0cm9rZS13aWR0aFwiLCBcIjFcIik7XG4gICAgdGhpcy5vdXRsaW5lLnNldEF0dHJpYnV0ZShcImZpbGxcIiwgXCJub25lXCIpO1xuICAgIHRoaXMudXBwZXJHcm91cC5hcHBlbmRDaGlsZCh0aGlzLm91dGxpbmUpO1xuXG4gICAgdGhpcy5zY2FsZUxhYmVscyA9IFtdO1xuXG4gICAgLy9zaW5jZSBmb3JtIGlzIHNldCB0byAwLCBtYWtlIHRoaXMgYSBjaXJjbGUsIHRoaXMgc3R1ZmYgaXMgZXF1aXZhbGVudCB0b1xuICAgIC8vIGVuZCByZXN1bHQgb2YgdG9DaXJjbGUgYnV0IHdpdGhvdXQgdHJhbnNpdGlvblxuICAgIGNvbnN0IHIgPSB0aGlzLmdldFN5bWJvbFJhZGl1cygpO1xuXG4gICAgdGhpcy5vdXRsaW5lLnNldEF0dHJpYnV0ZShcInhcIiwgLXIpO1xuICAgIHRoaXMub3V0bGluZS5zZXRBdHRyaWJ1dGUoXCJ5XCIsIC1yKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwid2lkdGhcIiwgciAqIDIpO1xuICAgIHRoaXMub3V0bGluZS5zZXRBdHRyaWJ1dGUoXCJoZWlnaHRcIiwgciAqIDIpO1xuICAgIHRoaXMub3V0bGluZS5zZXRBdHRyaWJ1dGUoXCJyeFwiLCByKTtcbiAgICB0aGlzLm91dGxpbmUuc2V0QXR0cmlidXRlKFwicnlcIiwgcik7XG5cbiAgICB0aGlzLmJhY2tncm91bmQuc2V0QXR0cmlidXRlKFwieFwiLCAtcik7XG4gICAgdGhpcy5iYWNrZ3JvdW5kLnNldEF0dHJpYnV0ZShcInlcIiwgLXIpO1xuICAgIHRoaXMuYmFja2dyb3VuZC5zZXRBdHRyaWJ1dGUoXCJ3aWR0aFwiLCByICogMik7XG4gICAgdGhpcy5iYWNrZ3JvdW5kLnNldEF0dHJpYnV0ZShcImhlaWdodFwiLCByICogMik7XG4gICAgdGhpcy5iYWNrZ3JvdW5kLnNldEF0dHJpYnV0ZShcInJ4XCIsIHIpO1xuICAgIHRoaXMuYmFja2dyb3VuZC5zZXRBdHRyaWJ1dGUoXCJyeVwiLCByKTtcblxuICAgIHRoaXMuYW5ub3RhdGlvbnNTdmdHcm91cC5zZXRBdHRyaWJ1dGUoXCJ0cmFuc2Zvcm1cIiwgXCJzY2FsZSgxLCAxKVwiKTtcblxuICAgIHRoaXMuaGlnaGxpZ2h0LnNldEF0dHJpYnV0ZShcIndpZHRoXCIsIChyICogMikgKyA1KTtcbiAgICB0aGlzLmhpZ2hsaWdodC5zZXRBdHRyaWJ1dGUoXCJoZWlnaHRcIiwgKHIgKiAyKSArIDUpO1xuICAgIHRoaXMuaGlnaGxpZ2h0LnNldEF0dHJpYnV0ZShcInhcIiwgLXIgLSAyLjUpO1xuICAgIHRoaXMuaGlnaGxpZ2h0LnNldEF0dHJpYnV0ZShcInlcIiwgLXIgLSAyLjUpO1xuICAgIHRoaXMuaGlnaGxpZ2h0LnNldEF0dHJpYnV0ZShcInJ4XCIsIHIgKyAyLjUpO1xuICAgIHRoaXMuaGlnaGxpZ2h0LnNldEF0dHJpYnV0ZShcInJ5XCIsIHIgKyAyLjUpO1xuICAgIHRoaXMuaGlnaGxpZ2h0LnNldEF0dHJpYnV0ZShcInN0cm9rZS1vcGFjaXR5XCIsIFwiMFwiKTtcblxuICAgIHRoaXMubGFiZWxTVkcuc2V0QXR0cmlidXRlKFwidHJhbnNmb3JtXCIsIFwidHJhbnNsYXRlKFwiICsgKC0ociArIDUpKSArIFwiLFwiICsgXCItNSlcIik7XG5cbiAgICB0aGlzLmluaXRMaXN0ZW5lcnMoKTtcblxuICAgIGNvbnN0IHNlbGYgPSB0aGlzO1xuICAgIE9iamVjdC5kZWZpbmVQcm9wZXJ0eSh0aGlzLCBcImhlaWdodFwiLCB7XG4gICAgICAgIGdldDogZnVuY3Rpb24gaGVpZ2h0KCkge1xuICAgICAgICAgICAgcmV0dXJuIHNlbGYuZm9ybSA9PSAxPyAxMjA6NDA7XG4gICAgICAgICAgICAvL3JldHVybiAxNjA7XG4gICAgICAgIH1cbiAgICB9KTtcblxuICAgIHRoaXMuc2hvd0hpZ2hsaWdodChmYWxzZSk7XG59XG5cblByb3RlaW4ucHJvdG90eXBlID0gbmV3IFBvbHltZXIoKTtcblxuLypcblByb3RlaW4ucHJvdG90eXBlLnNob3dEYXRhID0gZnVuY3Rpb24oZXZ0KSB7XG4gICAgY29uc3QgdXJsID0gXCJodHRwOi8vd3d3LnVuaXByb3Qub3JnL3VuaXByb3QvXCIgKyB0aGlzLmpzb24uaWRlbnRpZmllci5pZDtcbiAgICB3aW5kb3cub3Blbih1cmwsICdfYmxhbmsnKTtcbn1cbiovXG4iXSwibWFwcGluZ3MiOiJBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTsiLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./src/js/viz/interactor/protein.js\n"); /***/ }), @@ -1357,7 +1369,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"FeatureLink\", function() { return FeatureLink; });\n/* harmony import */ var _link__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./link */ \"./src/js/viz/link/link.js\");\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n\n\n// import * as Point2D from \"point2d\";\n// import * as Intersection from \"intersectionjs\";\n\nfunction FeatureLink(id, fromFeatPos, toFeatPos, app) {\n this.init(id, fromFeatPos, toFeatPos, app);\n}\n\nFeatureLink.prototype = new _link__WEBPACK_IMPORTED_MODULE_0__[\"Link\"]();\n\nFeatureLink.prototype.init = function (id, fromFeatPos, toFeatPos, app) {\n this.id = id;\n this.app = app;\n this.fromSequenceData = fromFeatPos;\n this.toSequenceData = toFeatPos;\n\n this.participants = [this.fromSequenceData[0].participant, this.toSequenceData[0].participant]; //*\n // *potentially, this over simplifies the situation,\n // but there is a workaround in way ReadMiJson init's links so OK for now\n\n this.glyph = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"path\");\n this.uncertainGlyph = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"path\");\n // this.highlightGlyph = document.createElementNS(svgns, \"path\");\n this.glyph.setAttribute(\"stroke-linecap\", \"round\");\n this.uncertainGlyph.setAttribute(\"stroke-linecap\", \"round\");\n // this.highlightGlyph.setAttribute(\"stroke-linecap\", \"round\");\n this.glyph.setAttribute(\"class\", \"link\");\n this.glyph.setAttribute(\"fill\", \"black\");//\"#E08214\");\n this.glyph.setAttribute(\"opacity\", \"0.6\");\n this.glyph.setAttribute(\"stroke\", \"black\");//\"\"#A08214\");// // TODO: will look better with this line partly removed\n this.glyph.setAttribute(\"stroke-opacity\", \"0.6\");\n this.glyph.setAttribute(\"stroke-width\", \"1\");\n this.uncertainGlyph.setAttribute(\"class\", \"link\");\n this.uncertainGlyph.setAttribute(\"fill\", \"black\");//url('#checkers_uncertain')\");//\"#A01284\");\n this.uncertainGlyph.setAttribute(\"stroke\", \"black\");//\"none\");//\"#A01284\");\n this.uncertainGlyph.setAttribute(\"stroke-opacity\", \"0.2\");\n this.uncertainGlyph.setAttribute(\"fill-opacity\", \"0.2\");\n // this.highlightGlyph.setAttribute(\"class\", \"link\");\n // this.highlightGlyph.setAttribute(\"fill\", \"none\");\n // this.highlightGlyph.setAttribute(\"stroke\", highlightColour);\n // this.highlightGlyph.setAttribute(\"stroke-width\", \"10\");\n // this.highlightGlyph.setAttribute(\"stroke-opacity\", \"0\");\n\n //set the events for it\n const self = this;\n this.uncertainGlyph.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.uncertainGlyph.onmouseover = function (evt) {\n self.mouseOver(evt);\n };\n this.uncertainGlyph.onmouseout = function (evt) {\n self.mouseOut(evt);\n };\n this.glyph.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.glyph.onmouseover = function (evt) {\n self.mouseOver(evt);\n };\n this.glyph.onmouseout = function (evt) {\n self.mouseOut(evt);\n };\n // this.highlightGlyph.onmousedown = function (evt) {\n // self.mouseDown(evt);\n // };\n // this.highlightGlyph.onmouseover = function (evt) {\n // self.mouseOver(evt);\n // };\n // this.highlightGlyph.onmouseout = function (evt) {\n // self.mouseOut(evt);\n // };\n};\n\n//andAlternatives means highlight alternative links in case of site ambiguity\n// FeatureLink.prototype.showHighlight = function (show) {\n// if (show) {\n// this.highlightGlyph.setAttribute(\"stroke-opacity\", \"1\");\n// } else {\n// this.highlightGlyph.setAttribute(\"stroke-opacity\", \"0\");\n// }\n// };\n\n//used when filter changed\nFeatureLink.prototype.check = function () {\n if (this.anyParticipantIsBar() === true) {\n this.show();\n return true;\n } else {\n this.hide();\n return false;\n }\n};\n\nFeatureLink.prototype.anyParticipantIsBar = function () {\n const ic = this.participants.length;\n for (let i = 0; i < ic; i++) {\n if (this.participants[i].form === 1) {\n return true;\n }\n }\n return false;\n};\n\nFeatureLink.prototype.show = function () {\n // //this.glyph.setAttribute(\"stroke-width\", this.util.z * xiNET.linkWidth);\n // this.uncertainGlyph.setAttribute(\"stroke-width\", this.util.z * 10);\n // this.highlightGlyph.setAttribute(\"stroke-width\", this.util.z * 10);\n this.setLinkCoordinates();\n let containingGroup = this.app.res_resLinks;\n if (this.participants[0] === this.participants[1]) {\n containingGroup = this.app.selfRes_resLinks;\n }\n // containingGroup.appendChild(this.highlightGlyph);\n containingGroup.appendChild(this.glyph);\n containingGroup.appendChild(this.uncertainGlyph);\n};\n\nFeatureLink.prototype.hide = function () {\n this.glyph.remove();\n this.uncertainGlyph.remove();\n};\n\n// update the links(polygons/lines) to fit to the protein\nFeatureLink.prototype.setLinkCoordinates = function () {\n function isNumber(thing) {\n return (!isNaN(parseFloat(thing)) && isFinite(thing));\n }\n\n function getSegment(midPoint, controlPoint, startRes, endRes, participant, yOffset, originPoint) {\n let startPoint, endPoint;\n if (!participant.form) { // tests if form = undefined or 0 //TODO: maybe change this, its confusing\n startPoint = participant.getPosition(originPoint);\n endPoint = startPoint;\n } else {\n startPoint = participant.getResidueCoordinates(startRes, yOffset);\n endPoint = participant.getResidueCoordinates(endRes, yOffset);\n }\n return \" Q\" + controlPoint[0] + \",\" + controlPoint[1] + \" \" + startPoint[0] + \",\" + startPoint[1] +\n \" L\" + endPoint[0] + \",\" + endPoint[1] +\n \" Q\" + controlPoint[0] + \",\" + controlPoint[1] + \" \" + midPoint[0] + \",\" + midPoint[1];\n }\n\n function sequenceDataMidPoint(sequenceData, participant) {\n //get the smallest start and the biggest end\n let lowestLinkedRes = null,\n highestLinkedRes = null;\n const sdCount = sequenceData.length;\n for (let s = 0; s < sdCount; s++) {\n const seqDatum = sequenceData[s];\n if (!isNaN(parseFloat(seqDatum.begin)) && isFinite(seqDatum.begin)) {\n // noinspection PointlessArithmeticExpressionJS\n const start = seqDatum.begin * 1; // the * 1 is necessary (type conversion)\n if (lowestLinkedRes === null || start < lowestLinkedRes) {\n lowestLinkedRes = start;\n }\n }\n if (!isNaN(parseFloat(seqDatum.uncertainBegin)) && isFinite(seqDatum.uncertainBegin)) {\n const uncertainBegin = seqDatum.uncertainBegin * 1;\n if (lowestLinkedRes === null || uncertainBegin < lowestLinkedRes) {\n lowestLinkedRes = uncertainBegin;\n }\n }\n if (!isNaN(parseFloat(seqDatum.end)) && isFinite(seqDatum.end)) {\n const end = seqDatum.end * 1;\n if (highestLinkedRes === null || end > highestLinkedRes) {\n highestLinkedRes = end;\n }\n }\n if (!isNaN(parseFloat(seqDatum.uncertainEnd)) && isFinite(seqDatum.uncertainEnd)) {\n const uncertainEnd = seqDatum.uncertainEnd * 1;\n if (highestLinkedRes === null || uncertainEnd > highestLinkedRes) {\n highestLinkedRes = uncertainEnd;\n }\n }\n }\n return participant.getResidueCoordinates((lowestLinkedRes + highestLinkedRes) / 2, 0);\n }\n\n const fromParticipant = this.fromSequenceData[0].participant;\n const toParticipant = this.toSequenceData[0].participant;\n //calculate mid points of from and to sequence data\n let fMid, tMid;\n\n if (fromParticipant.form) {\n fMid = sequenceDataMidPoint(this.fromSequenceData, fromParticipant);\n }\n if (toParticipant.form) {\n tMid = sequenceDataMidPoint(this.toSequenceData, toParticipant);\n }\n if (!fromParticipant.form) { // if not (undefined or 0)\n fMid = fromParticipant.getPosition(tMid);//toOriginPoint);\n }\n if (!toParticipant.form) {// if not (undefined or 0)\n tMid = toParticipant.getPosition(fMid);//fromOriginPoint);\n }\n\n const fromOriginPoint = fMid;//null;//[fromParticipant.cy, fromParticipant.cy];\n const toOriginPoint = tMid;//null;//[toParticipant.cy, toParticipant.cy];\n\n // if (!fromParticipant.form) { // if not (undefined or 0)\n // fMid = fromParticipant.getPosition();//toOriginPoint);\n // } else {\n // fMid = sequenceDataMidPoint(this.fromSequenceData, fromParticipant);\n // }\n // if (!toParticipant.form) {// if not (undefined or 0)\n // tMid = toParticipant.getPosition();//fromOriginPoint);\n // } else {\n // tMid = sequenceDataMidPoint(this.toSequenceData, toParticipant);\n // }\n\n //calculate angle from fromParticipant mid point to toParticipant mid point\n const deltaX = fMid[0] - tMid[0];\n const deltaY = fMid[1] - tMid[1];\n const angleBetweenMidPoints = Math.atan2(deltaY, deltaX);\n //todo: tidy up trig code so everything is always in radian\n let abmpDeg = angleBetweenMidPoints / (2 * Math.PI) * 360;\n if (abmpDeg < 0) {\n abmpDeg += 360;\n }\n\n //out is value we use to decide which side of bar the link glyph is drawn\n //first for 'from' participant\n let out = (abmpDeg - fromParticipant.rotation);\n if (out < 0) {\n out += 360;\n }\n let fyOffset = 10;\n if (out < 180) {\n fyOffset = -10;\n }\n let fRotRad = (fromParticipant.rotation / 360) * Math.PI * 2;\n if (out > 180) {\n fRotRad -= Math.PI;\n }\n //now for 'to' participant\n out = (abmpDeg - toParticipant.rotation);\n if (out < 0) {\n out += 360;\n }\n let tyOffset = 10;\n if (out > 180) {\n tyOffset = -10;\n }\n let tRotRad = (toParticipant.rotation / 360) * Math.PI * 2;\n if (out < 180) {\n tRotRad -= Math.PI;\n }\n\n let ftMid = [fMid[0] + (30 * Math.sin(fRotRad) * this.app.z),\n fMid[1] - (30 * Math.cos(fRotRad) * this.app.z)\n ];\n if (!fromParticipant.form) { // if not (undefined or 0)\n ftMid = fMid;\n }\n\n let ttMid = [tMid[0] + (30 * Math.sin(tRotRad) * this.app.z),\n tMid[1] - (30 * Math.cos(tRotRad) * this.app.z)\n ];\n if (!toParticipant.form) { // if not (undefined or 0)\n ttMid = tMid;\n }\n\n const triPointMid = [(ftMid[0] + ttMid[0]) / 2, (ftMid[1] + ttMid[1]) / 2];\n const fSDCount = this.fromSequenceData.length;\n const tSDCount = this.toSequenceData.length;\n let seqDatum;//, highlightStartRes, highlightEndRes;\n let glyphPath = \"M\" + triPointMid[0] + \",\" + triPointMid[1];\n let uncertainGlyphPath = \"M\" + triPointMid[0] + \",\" + triPointMid[1];\n // let highlightGlyphPath = \"M\" + triPointMid[0] + \",\" + triPointMid[1];\n for (let f = 0; f < fSDCount; f++) {\n seqDatum = this.fromSequenceData[f];\n if (isNumber(seqDatum.begin) && isNumber(seqDatum.end)) {\n glyphPath += getSegment(triPointMid, ftMid, seqDatum.begin, seqDatum.end, fromParticipant, fyOffset, toOriginPoint);\n }\n // highlightStartRes = seqDatum.begin;\n // highlightEndRes = seqDatum.end;\n if (isNumber(seqDatum.uncertainBegin)) {\n uncertainGlyphPath += getSegment(triPointMid, ftMid,\n seqDatum.uncertainBegin, seqDatum.begin, fromParticipant, fyOffset, toOriginPoint);\n // highlightStartRes = seqDatum.uncertainBegin;\n }\n if (isNumber(seqDatum.uncertainEnd)) {\n uncertainGlyphPath += getSegment(triPointMid, ftMid,\n seqDatum.end, seqDatum.uncertainEnd, fromParticipant, fyOffset, toOriginPoint);\n // highlightEndRes = seqDatum.uncertainEnd;\n }\n // highlightGlyphPath += getPathSegments(triPointMid, ftMid,\n // highlightStartRes, highlightEndRes, fromParticipant, fyOffset);\n }\n for (let t = 0; t < tSDCount; t++) {\n seqDatum = this.toSequenceData[t];\n if (isNumber(seqDatum.begin) && isNumber(seqDatum.end)) {\n glyphPath += getSegment(triPointMid, ttMid, seqDatum.begin, seqDatum.end, toParticipant, tyOffset, fromOriginPoint);\n }\n // highlightStartRes = seqDatum.begin;\n // highlightEndRes = seqDatum.end;\n if (isNumber(seqDatum.uncertainBegin)) {\n uncertainGlyphPath += getSegment(triPointMid, ttMid,\n seqDatum.uncertainBegin, seqDatum.begin, toParticipant, tyOffset, fromOriginPoint);\n // highlightStartRes = seqDatum.uncertainBegin;\n }\n if (isNumber(seqDatum.uncertainEnd)) {\n uncertainGlyphPath += getSegment(triPointMid, ttMid,\n seqDatum.end, seqDatum.uncertainEnd, toParticipant, tyOffset, fromOriginPoint);\n // highlightEndRes = seqDatum.uncertainEnd;\n }\n // highlightGlyphPath += getPathSegments(triPointMid, ttMid,\n // highlightStartRes, highlightEndRes, toParticipant, tyOffset);\n }\n\n if (!this.glyph) {\n this.initSVG();\n }\n\n this.glyph.setAttribute(\"d\", glyphPath);\n this.uncertainGlyph.setAttribute(\"d\", uncertainGlyphPath);\n // this.highlightGlyph.setAttribute(\"d\", highlightGlyphPath);\n};\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/js/viz/link/feature-link.js.js","sources":["webpack://complexviewer/./src/js/viz/link/feature-link.js?1a50"],"sourcesContent":["import {Link} from \"./link\";\nimport {svgns} from \"../../config\";\n// import * as Point2D from \"point2d\";\n// import * as Intersection from \"intersectionjs\";\n\nexport function FeatureLink(id, fromFeatPos, toFeatPos, app) {\n    this.init(id, fromFeatPos, toFeatPos, app);\n}\n\nFeatureLink.prototype = new Link();\n\nFeatureLink.prototype.init = function (id, fromFeatPos, toFeatPos, app) {\n    this.id = id;\n    this.app = app;\n    this.fromSequenceData = fromFeatPos;\n    this.toSequenceData = toFeatPos;\n\n    this.participants = [this.fromSequenceData[0].participant, this.toSequenceData[0].participant]; //*\n    // *potentially, this over simplifies the situation,\n    // but there is a workaround in way ReadMiJson init's links so OK for now\n\n    this.glyph = document.createElementNS(svgns, \"path\");\n    this.uncertainGlyph = document.createElementNS(svgns, \"path\");\n    // this.highlightGlyph = document.createElementNS(svgns, \"path\");\n    this.glyph.setAttribute(\"stroke-linecap\", \"round\");\n    this.uncertainGlyph.setAttribute(\"stroke-linecap\", \"round\");\n    // this.highlightGlyph.setAttribute(\"stroke-linecap\", \"round\");\n    this.glyph.setAttribute(\"class\", \"link\");\n    this.glyph.setAttribute(\"fill\", \"black\");//\"#E08214\");\n    this.glyph.setAttribute(\"opacity\", \"0.6\");\n    this.glyph.setAttribute(\"stroke\", \"black\");//\"\"#A08214\");// // TODO: will look better with this line partly removed\n    this.glyph.setAttribute(\"stroke-opacity\", \"0.6\");\n    this.glyph.setAttribute(\"stroke-width\", \"1\");\n    this.uncertainGlyph.setAttribute(\"class\", \"link\");\n    this.uncertainGlyph.setAttribute(\"fill\", \"black\");//url('#checkers_uncertain')\");//\"#A01284\");\n    this.uncertainGlyph.setAttribute(\"stroke\", \"black\");//\"none\");//\"#A01284\");\n    this.uncertainGlyph.setAttribute(\"stroke-opacity\", \"0.2\");\n    this.uncertainGlyph.setAttribute(\"fill-opacity\", \"0.2\");\n    // this.highlightGlyph.setAttribute(\"class\", \"link\");\n    // this.highlightGlyph.setAttribute(\"fill\", \"none\");\n    // this.highlightGlyph.setAttribute(\"stroke\", highlightColour);\n    // this.highlightGlyph.setAttribute(\"stroke-width\", \"10\");\n    // this.highlightGlyph.setAttribute(\"stroke-opacity\", \"0\");\n\n    //set the events for it\n    const self = this;\n    this.uncertainGlyph.onmousedown = function (evt) {\n        self.mouseDown(evt);\n    };\n    this.uncertainGlyph.onmouseover = function (evt) {\n        self.mouseOver(evt);\n    };\n    this.uncertainGlyph.onmouseout = function (evt) {\n        self.mouseOut(evt);\n    };\n    this.glyph.onmousedown = function (evt) {\n        self.mouseDown(evt);\n    };\n    this.glyph.onmouseover = function (evt) {\n        self.mouseOver(evt);\n    };\n    this.glyph.onmouseout = function (evt) {\n        self.mouseOut(evt);\n    };\n    // this.highlightGlyph.onmousedown = function (evt) {\n    //     self.mouseDown(evt);\n    // };\n    // this.highlightGlyph.onmouseover = function (evt) {\n    //     self.mouseOver(evt);\n    // };\n    // this.highlightGlyph.onmouseout = function (evt) {\n    //     self.mouseOut(evt);\n    // };\n};\n\n//andAlternatives means highlight alternative links in case of site ambiguity\n// FeatureLink.prototype.showHighlight = function (show) {\n//     if (show) {\n//         this.highlightGlyph.setAttribute(\"stroke-opacity\", \"1\");\n//     } else {\n//         this.highlightGlyph.setAttribute(\"stroke-opacity\", \"0\");\n//     }\n// };\n\n//used when filter changed\nFeatureLink.prototype.check = function () {\n    if (this.anyParticipantIsBar() === true) {\n        this.show();\n        return true;\n    } else {\n        this.hide();\n        return false;\n    }\n};\n\nFeatureLink.prototype.anyParticipantIsBar = function () {\n    const ic = this.participants.length;\n    for (let i = 0; i < ic; i++) {\n        if (this.participants[i].form === 1) {\n            return true;\n        }\n    }\n    return false;\n};\n\nFeatureLink.prototype.show = function () {\n    // //this.glyph.setAttribute(\"stroke-width\", this.util.z * xiNET.linkWidth);\n    // this.uncertainGlyph.setAttribute(\"stroke-width\", this.util.z * 10);\n    // this.highlightGlyph.setAttribute(\"stroke-width\", this.util.z * 10);\n    this.setLinkCoordinates();\n    let containingGroup = this.app.res_resLinks;\n    if (this.participants[0] === this.participants[1]) {\n        containingGroup = this.app.selfRes_resLinks;\n    }\n    // containingGroup.appendChild(this.highlightGlyph);\n    containingGroup.appendChild(this.glyph);\n    containingGroup.appendChild(this.uncertainGlyph);\n};\n\nFeatureLink.prototype.hide = function () {\n    this.glyph.remove();\n    this.uncertainGlyph.remove();\n};\n\n// update the links(polygons/lines) to fit to the protein\nFeatureLink.prototype.setLinkCoordinates = function () {\n    function isNumber(thing) {\n        return (!isNaN(parseFloat(thing)) && isFinite(thing));\n    }\n\n    function getSegment(midPoint, controlPoint, startRes, endRes, participant, yOffset, originPoint) {\n        let startPoint, endPoint;\n        if (!participant.form) { // tests if form = undefined or 0 //TODO: maybe change this, its confusing\n            startPoint = participant.getPosition(originPoint);\n            endPoint = startPoint;\n        } else {\n            startPoint = participant.getResidueCoordinates(startRes, yOffset);\n            endPoint = participant.getResidueCoordinates(endRes, yOffset);\n        }\n        return \" Q\" + controlPoint[0] + \",\" + controlPoint[1] + \" \" + startPoint[0] + \",\" + startPoint[1] +\n            \" L\" + endPoint[0] + \",\" + endPoint[1] +\n            \" Q\" + controlPoint[0] + \",\" + controlPoint[1] + \" \" + midPoint[0] + \",\" + midPoint[1];\n    }\n\n    function sequenceDataMidPoint(sequenceData, participant) {\n        //get the smallest start and the biggest end\n        let lowestLinkedRes = null,\n            highestLinkedRes = null;\n        const sdCount = sequenceData.length;\n        for (let s = 0; s < sdCount; s++) {\n            const seqDatum = sequenceData[s];\n            if (!isNaN(parseFloat(seqDatum.begin)) && isFinite(seqDatum.begin)) {\n                // noinspection PointlessArithmeticExpressionJS\n                const start = seqDatum.begin * 1; // the * 1 is necessary (type conversion)\n                if (lowestLinkedRes === null || start < lowestLinkedRes) {\n                    lowestLinkedRes = start;\n                }\n            }\n            if (!isNaN(parseFloat(seqDatum.uncertainBegin)) && isFinite(seqDatum.uncertainBegin)) {\n                const uncertainBegin = seqDatum.uncertainBegin * 1;\n                if (lowestLinkedRes === null || uncertainBegin < lowestLinkedRes) {\n                    lowestLinkedRes = uncertainBegin;\n                }\n            }\n            if (!isNaN(parseFloat(seqDatum.end)) && isFinite(seqDatum.end)) {\n                const end = seqDatum.end * 1;\n                if (highestLinkedRes === null || end > highestLinkedRes) {\n                    highestLinkedRes = end;\n                }\n            }\n            if (!isNaN(parseFloat(seqDatum.uncertainEnd)) && isFinite(seqDatum.uncertainEnd)) {\n                const uncertainEnd = seqDatum.uncertainEnd * 1;\n                if (highestLinkedRes === null || uncertainEnd > highestLinkedRes) {\n                    highestLinkedRes = uncertainEnd;\n                }\n            }\n        }\n        return participant.getResidueCoordinates((lowestLinkedRes + highestLinkedRes) / 2, 0);\n    }\n\n    const fromParticipant = this.fromSequenceData[0].participant;\n    const toParticipant = this.toSequenceData[0].participant;\n    //calculate mid points of from and to sequence data\n    let fMid, tMid;\n\n    if (fromParticipant.form)  {\n        fMid = sequenceDataMidPoint(this.fromSequenceData, fromParticipant);\n    }\n    if (toParticipant.form)  {\n        tMid = sequenceDataMidPoint(this.toSequenceData, toParticipant);\n    }\n    if (!fromParticipant.form) { // if not (undefined or 0)\n        fMid = fromParticipant.getPosition(tMid);//toOriginPoint);\n    }\n    if (!toParticipant.form) {// if not (undefined or 0)\n        tMid = toParticipant.getPosition(fMid);//fromOriginPoint);\n    }\n\n    const fromOriginPoint = fMid;//null;//[fromParticipant.cy, fromParticipant.cy];\n    const toOriginPoint = tMid;//null;//[toParticipant.cy, toParticipant.cy];\n\n    // if (!fromParticipant.form) { // if not (undefined or 0)\n    //     fMid = fromParticipant.getPosition();//toOriginPoint);\n    // } else {\n    //     fMid = sequenceDataMidPoint(this.fromSequenceData, fromParticipant);\n    // }\n    // if (!toParticipant.form) {// if not (undefined or 0)\n    //     tMid = toParticipant.getPosition();//fromOriginPoint);\n    // } else {\n    //     tMid = sequenceDataMidPoint(this.toSequenceData, toParticipant);\n    // }\n\n    //calculate angle from fromParticipant mid point to toParticipant mid point\n    const deltaX = fMid[0] - tMid[0];\n    const deltaY = fMid[1] - tMid[1];\n    const angleBetweenMidPoints = Math.atan2(deltaY, deltaX);\n    //todo: tidy up trig code so everything is always in radian\n    let abmpDeg = angleBetweenMidPoints / (2 * Math.PI) * 360;\n    if (abmpDeg < 0) {\n        abmpDeg += 360;\n    }\n\n    //out is value we use to decide which side of bar the link glyph is drawn\n    //first for 'from' participant\n    let out = (abmpDeg - fromParticipant.rotation);\n    if (out < 0) {\n        out += 360;\n    }\n    let fyOffset = 10;\n    if (out < 180) {\n        fyOffset = -10;\n    }\n    let fRotRad = (fromParticipant.rotation / 360) * Math.PI * 2;\n    if (out > 180) {\n        fRotRad -= Math.PI;\n    }\n    //now for 'to' participant\n    out = (abmpDeg - toParticipant.rotation);\n    if (out < 0) {\n        out += 360;\n    }\n    let tyOffset = 10;\n    if (out > 180) {\n        tyOffset = -10;\n    }\n    let tRotRad = (toParticipant.rotation / 360) * Math.PI * 2;\n    if (out < 180) {\n        tRotRad -= Math.PI;\n    }\n\n    let ftMid = [fMid[0] + (30 * Math.sin(fRotRad) * this.app.z),\n        fMid[1] - (30 * Math.cos(fRotRad) * this.app.z)\n    ];\n    if (!fromParticipant.form) { // if not (undefined or 0)\n        ftMid = fMid;\n    }\n\n    let ttMid = [tMid[0] + (30 * Math.sin(tRotRad) * this.app.z),\n        tMid[1] - (30 * Math.cos(tRotRad) * this.app.z)\n    ];\n    if (!toParticipant.form) { // if not (undefined or 0)\n        ttMid = tMid;\n    }\n\n    const triPointMid = [(ftMid[0] + ttMid[0]) / 2, (ftMid[1] + ttMid[1]) / 2];\n    const fSDCount = this.fromSequenceData.length;\n    const tSDCount = this.toSequenceData.length;\n    let seqDatum;//, highlightStartRes, highlightEndRes;\n    let glyphPath = \"M\" + triPointMid[0] + \",\" + triPointMid[1];\n    let uncertainGlyphPath = \"M\" + triPointMid[0] + \",\" + triPointMid[1];\n    // let highlightGlyphPath = \"M\" + triPointMid[0] + \",\" + triPointMid[1];\n    for (let f = 0; f < fSDCount; f++) {\n        seqDatum = this.fromSequenceData[f];\n        if (isNumber(seqDatum.begin)  && isNumber(seqDatum.end)) {\n            glyphPath += getSegment(triPointMid, ftMid, seqDatum.begin, seqDatum.end, fromParticipant, fyOffset, toOriginPoint);\n        }\n        // highlightStartRes = seqDatum.begin;\n        // highlightEndRes = seqDatum.end;\n        if (isNumber(seqDatum.uncertainBegin)) {\n            uncertainGlyphPath += getSegment(triPointMid, ftMid,\n                seqDatum.uncertainBegin, seqDatum.begin, fromParticipant, fyOffset, toOriginPoint);\n            // highlightStartRes = seqDatum.uncertainBegin;\n        }\n        if (isNumber(seqDatum.uncertainEnd)) {\n            uncertainGlyphPath += getSegment(triPointMid, ftMid,\n                seqDatum.end, seqDatum.uncertainEnd, fromParticipant, fyOffset, toOriginPoint);\n            // highlightEndRes = seqDatum.uncertainEnd;\n        }\n        // highlightGlyphPath += getPathSegments(triPointMid, ftMid,\n        //     highlightStartRes, highlightEndRes, fromParticipant, fyOffset);\n    }\n    for (let t = 0; t < tSDCount; t++) {\n        seqDatum = this.toSequenceData[t];\n        if (isNumber(seqDatum.begin) && isNumber(seqDatum.end)) {\n            glyphPath += getSegment(triPointMid, ttMid, seqDatum.begin, seqDatum.end, toParticipant, tyOffset, fromOriginPoint);\n        }\n        // highlightStartRes = seqDatum.begin;\n        // highlightEndRes = seqDatum.end;\n        if (isNumber(seqDatum.uncertainBegin)) {\n            uncertainGlyphPath += getSegment(triPointMid, ttMid,\n                seqDatum.uncertainBegin, seqDatum.begin, toParticipant, tyOffset, fromOriginPoint);\n            // highlightStartRes = seqDatum.uncertainBegin;\n        }\n        if (isNumber(seqDatum.uncertainEnd)) {\n            uncertainGlyphPath += getSegment(triPointMid, ttMid,\n                seqDatum.end, seqDatum.uncertainEnd, toParticipant, tyOffset, fromOriginPoint);\n            // highlightEndRes = seqDatum.uncertainEnd;\n        }\n        // highlightGlyphPath += getPathSegments(triPointMid, ttMid,\n        //     highlightStartRes, highlightEndRes, toParticipant, tyOffset);\n    }\n\n    if (!this.glyph) {\n        this.initSVG();\n    }\n\n    this.glyph.setAttribute(\"d\", glyphPath);\n    this.uncertainGlyph.setAttribute(\"d\", uncertainGlyphPath);\n    // this.highlightGlyph.setAttribute(\"d\", highlightGlyphPath);\n};\n"],"mappings":"AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/js/viz/link/feature-link.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"FeatureLink\", function() { return FeatureLink; });\n/* harmony import */ var _link__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./link */ \"./src/js/viz/link/link.js\");\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n\n\n// import * as Point2D from \"point2d\";\n// import * as Intersection from \"intersectionjs\";\n\nfunction FeatureLink(id, fromFeatPos, toFeatPos, app) {\n this.init(id, fromFeatPos, toFeatPos, app);\n}\n\nFeatureLink.prototype = new _link__WEBPACK_IMPORTED_MODULE_0__[\"Link\"]();\n\nFeatureLink.prototype.init = function (id, fromFeatPos, toFeatPos, app) {\n this.id = id;\n this.app = app;\n this.fromSequenceData = fromFeatPos;\n this.toSequenceData = toFeatPos;\n\n this.participants = [this.fromSequenceData[0].participant, this.toSequenceData[0].participant]; //*\n // *potentially, this over simplifies the situation,\n // but there is a workaround in way ReadMiJson init's links so OK for now\n\n this.glyph = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"path\");\n this.uncertainGlyph = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_1__[\"svgns\"], \"path\");\n this.glyph.classList.add(\"link\", \"feature-link\", \"certain-link\");\n this.uncertainGlyph.classList.add(\"link\", \"feature-link\", \"uncertain-link\");\n\n //set the events for it\n const self = this;\n this.uncertainGlyph.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.uncertainGlyph.onmouseover = function (evt) {\n self.mouseOver(evt);\n };\n this.uncertainGlyph.onmouseout = function (evt) {\n self.mouseOut(evt);\n };\n this.glyph.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.glyph.onmouseover = function (evt) {\n self.mouseOver(evt);\n };\n this.glyph.onmouseout = function (evt) {\n self.mouseOut(evt);\n };\n // this.highlightGlyph.onmousedown = function (evt) {\n // self.mouseDown(evt);\n // };\n // this.highlightGlyph.onmouseover = function (evt) {\n // self.mouseOver(evt);\n // };\n // this.highlightGlyph.onmouseout = function (evt) {\n // self.mouseOut(evt);\n // };\n};\n\n//andAlternatives means highlight alternative links in case of site ambiguity\n// FeatureLink.prototype.showHighlight = function (show) {\n// if (show) {\n// this.highlightGlyph.setAttribute(\"stroke-opacity\", \"1\");\n// } else {\n// this.highlightGlyph.setAttribute(\"stroke-opacity\", \"0\");\n// }\n// };\n\n//used when filter changed\nFeatureLink.prototype.check = function () {\n if (this.anyParticipantIsBar() === true) {\n this.show();\n return true;\n } else {\n this.hide();\n return false;\n }\n};\n\nFeatureLink.prototype.anyParticipantIsBar = function () {\n const ic = this.participants.length;\n for (let i = 0; i < ic; i++) {\n if (this.participants[i].form === 1) {\n return true;\n }\n }\n return false;\n};\n\nFeatureLink.prototype.show = function () {\n // //this.glyph.setAttribute(\"stroke-width\", this.util.z * xiNET.linkWidth);\n // this.uncertainGlyph.setAttribute(\"stroke-width\", this.util.z * 10);\n // this.highlightGlyph.setAttribute(\"stroke-width\", this.util.z * 10);\n this.setLinkCoordinates();\n let containingGroup = this.app.res_resLinks;\n if (this.participants[0] === this.participants[1]) {\n containingGroup = this.app.selfRes_resLinks;\n }\n // containingGroup.appendChild(this.highlightGlyph);\n containingGroup.appendChild(this.glyph);\n containingGroup.appendChild(this.uncertainGlyph);\n};\n\nFeatureLink.prototype.hide = function () {\n this.glyph.remove();\n this.uncertainGlyph.remove();\n};\n\n// update the links(polygons/lines) to fit to the protein\nFeatureLink.prototype.setLinkCoordinates = function () {\n function isNumber(thing) {\n return (!isNaN(parseFloat(thing)) && isFinite(thing));\n }\n\n function getSegment(midPoint, controlPoint, startRes, endRes, participant, yOffset, originPoint) {\n let startPoint, endPoint;\n if (!participant.form) { // tests if form = undefined or 0 //TODO: maybe change this, its confusing\n startPoint = participant.getPosition(originPoint);\n endPoint = startPoint;\n } else {\n startPoint = participant.getResidueCoordinates(startRes, yOffset);\n endPoint = participant.getResidueCoordinates(endRes, yOffset);\n }\n return \" Q\" + controlPoint[0] + \",\" + controlPoint[1] + \" \" + startPoint[0] + \",\" + startPoint[1] +\n \" L\" + endPoint[0] + \",\" + endPoint[1] +\n \" Q\" + controlPoint[0] + \",\" + controlPoint[1] + \" \" + midPoint[0] + \",\" + midPoint[1];\n }\n\n function sequenceDataMidPoint(sequenceData, participant) {\n //get the smallest start and the biggest end\n let lowestLinkedRes = null,\n highestLinkedRes = null;\n const sdCount = sequenceData.length;\n for (let s = 0; s < sdCount; s++) {\n const seqDatum = sequenceData[s];\n if (!isNaN(parseFloat(seqDatum.begin)) && isFinite(seqDatum.begin)) {\n // noinspection PointlessArithmeticExpressionJS\n const start = seqDatum.begin * 1; // the * 1 is necessary (type conversion)\n if (lowestLinkedRes === null || start < lowestLinkedRes) {\n lowestLinkedRes = start;\n }\n }\n if (!isNaN(parseFloat(seqDatum.uncertainBegin)) && isFinite(seqDatum.uncertainBegin)) {\n const uncertainBegin = seqDatum.uncertainBegin * 1;\n if (lowestLinkedRes === null || uncertainBegin < lowestLinkedRes) {\n lowestLinkedRes = uncertainBegin;\n }\n }\n if (!isNaN(parseFloat(seqDatum.end)) && isFinite(seqDatum.end)) {\n const end = seqDatum.end * 1;\n if (highestLinkedRes === null || end > highestLinkedRes) {\n highestLinkedRes = end;\n }\n }\n if (!isNaN(parseFloat(seqDatum.uncertainEnd)) && isFinite(seqDatum.uncertainEnd)) {\n const uncertainEnd = seqDatum.uncertainEnd * 1;\n if (highestLinkedRes === null || uncertainEnd > highestLinkedRes) {\n highestLinkedRes = uncertainEnd;\n }\n }\n }\n return participant.getResidueCoordinates((lowestLinkedRes + highestLinkedRes) / 2, 0);\n }\n\n const fromParticipant = this.fromSequenceData[0].participant;\n const toParticipant = this.toSequenceData[0].participant;\n //calculate mid points of from and to sequence data\n let fMid, tMid;\n\n if (fromParticipant.form) {\n fMid = sequenceDataMidPoint(this.fromSequenceData, fromParticipant);\n }\n if (toParticipant.form) {\n tMid = sequenceDataMidPoint(this.toSequenceData, toParticipant);\n }\n if (!fromParticipant.form) { // if not (undefined or 0)\n fMid = fromParticipant.getPosition(tMid);//toOriginPoint);\n }\n if (!toParticipant.form) {// if not (undefined or 0)\n tMid = toParticipant.getPosition(fMid);//fromOriginPoint);\n }\n\n const fromOriginPoint = fMid;//null;//[fromParticipant.cy, fromParticipant.cy];\n const toOriginPoint = tMid;//null;//[toParticipant.cy, toParticipant.cy];\n\n // if (!fromParticipant.form) { // if not (undefined or 0)\n // fMid = fromParticipant.getPosition();//toOriginPoint);\n // } else {\n // fMid = sequenceDataMidPoint(this.fromSequenceData, fromParticipant);\n // }\n // if (!toParticipant.form) {// if not (undefined or 0)\n // tMid = toParticipant.getPosition();//fromOriginPoint);\n // } else {\n // tMid = sequenceDataMidPoint(this.toSequenceData, toParticipant);\n // }\n\n //calculate angle from fromParticipant mid point to toParticipant mid point\n const deltaX = fMid[0] - tMid[0];\n const deltaY = fMid[1] - tMid[1];\n const angleBetweenMidPoints = Math.atan2(deltaY, deltaX);\n //todo: tidy up trig code so everything is always in radian\n let abmpDeg = angleBetweenMidPoints / (2 * Math.PI) * 360;\n if (abmpDeg < 0) {\n abmpDeg += 360;\n }\n\n //out is value we use to decide which side of bar the link glyph is drawn\n //first for 'from' participant\n let out = (abmpDeg - fromParticipant.rotation);\n if (out < 0) {\n out += 360;\n }\n let fyOffset = 10;\n if (out < 180) {\n fyOffset = -10;\n }\n let fRotRad = (fromParticipant.rotation / 360) * Math.PI * 2;\n if (out > 180) {\n fRotRad -= Math.PI;\n }\n //now for 'to' participant\n out = (abmpDeg - toParticipant.rotation);\n if (out < 0) {\n out += 360;\n }\n let tyOffset = 10;\n if (out > 180) {\n tyOffset = -10;\n }\n let tRotRad = (toParticipant.rotation / 360) * Math.PI * 2;\n if (out < 180) {\n tRotRad -= Math.PI;\n }\n\n let ftMid = [fMid[0] + (30 * Math.sin(fRotRad) * this.app.z),\n fMid[1] - (30 * Math.cos(fRotRad) * this.app.z)\n ];\n if (!fromParticipant.form) { // if not (undefined or 0)\n ftMid = fMid;\n }\n\n let ttMid = [tMid[0] + (30 * Math.sin(tRotRad) * this.app.z),\n tMid[1] - (30 * Math.cos(tRotRad) * this.app.z)\n ];\n if (!toParticipant.form) { // if not (undefined or 0)\n ttMid = tMid;\n }\n\n const triPointMid = [(ftMid[0] + ttMid[0]) / 2, (ftMid[1] + ttMid[1]) / 2];\n const fSDCount = this.fromSequenceData.length;\n const tSDCount = this.toSequenceData.length;\n let seqDatum;//, highlightStartRes, highlightEndRes;\n let glyphPath = \"M\" + triPointMid[0] + \",\" + triPointMid[1];\n let uncertainGlyphPath = \"M\" + triPointMid[0] + \",\" + triPointMid[1];\n // let highlightGlyphPath = \"M\" + triPointMid[0] + \",\" + triPointMid[1];\n for (let f = 0; f < fSDCount; f++) {\n seqDatum = this.fromSequenceData[f];\n if (isNumber(seqDatum.begin) && isNumber(seqDatum.end) || fromParticipant.type === \"complex\") {\n glyphPath += getSegment(triPointMid, ftMid, seqDatum.begin, seqDatum.end, fromParticipant, fyOffset, toOriginPoint);\n }\n // highlightStartRes = seqDatum.begin;\n // highlightEndRes = seqDatum.end;\n if (isNumber(seqDatum.uncertainBegin)) {\n uncertainGlyphPath += getSegment(triPointMid, ftMid,\n seqDatum.uncertainBegin, seqDatum.begin, fromParticipant, fyOffset, toOriginPoint);\n // highlightStartRes = seqDatum.uncertainBegin;\n }\n if (isNumber(seqDatum.uncertainEnd)) {\n uncertainGlyphPath += getSegment(triPointMid, ftMid,\n seqDatum.end, seqDatum.uncertainEnd, fromParticipant, fyOffset, toOriginPoint);\n // highlightEndRes = seqDatum.uncertainEnd;\n }\n // highlightGlyphPath += getPathSegments(triPointMid, ftMid,\n // highlightStartRes, highlightEndRes, fromParticipant, fyOffset);\n }\n for (let t = 0; t < tSDCount; t++) {\n seqDatum = this.toSequenceData[t];\n if (isNumber(seqDatum.begin) && isNumber(seqDatum.end) || toParticipant.type === \"complex\") {\n glyphPath += getSegment(triPointMid, ttMid, seqDatum.begin, seqDatum.end, toParticipant, tyOffset, fromOriginPoint);\n }\n // highlightStartRes = seqDatum.begin;\n // highlightEndRes = seqDatum.end;\n if (isNumber(seqDatum.uncertainBegin)) {\n uncertainGlyphPath += getSegment(triPointMid, ttMid,\n seqDatum.uncertainBegin, seqDatum.begin, toParticipant, tyOffset, fromOriginPoint);\n // highlightStartRes = seqDatum.uncertainBegin;\n }\n if (isNumber(seqDatum.uncertainEnd)) {\n uncertainGlyphPath += getSegment(triPointMid, ttMid,\n seqDatum.end, seqDatum.uncertainEnd, toParticipant, tyOffset, fromOriginPoint);\n // highlightEndRes = seqDatum.uncertainEnd;\n }\n // highlightGlyphPath += getPathSegments(triPointMid, ttMid,\n // highlightStartRes, highlightEndRes, toParticipant, tyOffset);\n }\n\n if (!this.glyph) {\n this.initSVG();\n }\n\n this.glyph.setAttribute(\"d\", glyphPath);\n this.uncertainGlyph.setAttribute(\"d\", uncertainGlyphPath);\n // this.highlightGlyph.setAttribute(\"d\", highlightGlyphPath);\n};\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/js/viz/link/feature-link.js.js","sources":["webpack://complexviewer/./src/js/viz/link/feature-link.js?1a50"],"sourcesContent":["import {Link} from \"./link\";\nimport {svgns} from \"../../config\";\n// import * as Point2D from \"point2d\";\n// import * as Intersection from \"intersectionjs\";\n\nexport function FeatureLink(id, fromFeatPos, toFeatPos, app) {\n    this.init(id, fromFeatPos, toFeatPos, app);\n}\n\nFeatureLink.prototype = new Link();\n\nFeatureLink.prototype.init = function (id, fromFeatPos, toFeatPos, app) {\n    this.id = id;\n    this.app = app;\n    this.fromSequenceData = fromFeatPos;\n    this.toSequenceData = toFeatPos;\n\n    this.participants = [this.fromSequenceData[0].participant, this.toSequenceData[0].participant]; //*\n    // *potentially, this over simplifies the situation,\n    // but there is a workaround in way ReadMiJson init's links so OK for now\n\n    this.glyph = document.createElementNS(svgns, \"path\");\n    this.uncertainGlyph = document.createElementNS(svgns, \"path\");\n    this.glyph.classList.add(\"link\", \"feature-link\", \"certain-link\");\n    this.uncertainGlyph.classList.add(\"link\", \"feature-link\", \"uncertain-link\");\n\n    //set the events for it\n    const self = this;\n    this.uncertainGlyph.onmousedown = function (evt) {\n        self.mouseDown(evt);\n    };\n    this.uncertainGlyph.onmouseover = function (evt) {\n        self.mouseOver(evt);\n    };\n    this.uncertainGlyph.onmouseout = function (evt) {\n        self.mouseOut(evt);\n    };\n    this.glyph.onmousedown = function (evt) {\n        self.mouseDown(evt);\n    };\n    this.glyph.onmouseover = function (evt) {\n        self.mouseOver(evt);\n    };\n    this.glyph.onmouseout = function (evt) {\n        self.mouseOut(evt);\n    };\n    // this.highlightGlyph.onmousedown = function (evt) {\n    //     self.mouseDown(evt);\n    // };\n    // this.highlightGlyph.onmouseover = function (evt) {\n    //     self.mouseOver(evt);\n    // };\n    // this.highlightGlyph.onmouseout = function (evt) {\n    //     self.mouseOut(evt);\n    // };\n};\n\n//andAlternatives means highlight alternative links in case of site ambiguity\n// FeatureLink.prototype.showHighlight = function (show) {\n//     if (show) {\n//         this.highlightGlyph.setAttribute(\"stroke-opacity\", \"1\");\n//     } else {\n//         this.highlightGlyph.setAttribute(\"stroke-opacity\", \"0\");\n//     }\n// };\n\n//used when filter changed\nFeatureLink.prototype.check = function () {\n    if (this.anyParticipantIsBar() === true) {\n        this.show();\n        return true;\n    } else {\n        this.hide();\n        return false;\n    }\n};\n\nFeatureLink.prototype.anyParticipantIsBar = function () {\n    const ic = this.participants.length;\n    for (let i = 0; i < ic; i++) {\n        if (this.participants[i].form === 1) {\n            return true;\n        }\n    }\n    return false;\n};\n\nFeatureLink.prototype.show = function () {\n    // //this.glyph.setAttribute(\"stroke-width\", this.util.z * xiNET.linkWidth);\n    // this.uncertainGlyph.setAttribute(\"stroke-width\", this.util.z * 10);\n    // this.highlightGlyph.setAttribute(\"stroke-width\", this.util.z * 10);\n    this.setLinkCoordinates();\n    let containingGroup = this.app.res_resLinks;\n    if (this.participants[0] === this.participants[1]) {\n        containingGroup = this.app.selfRes_resLinks;\n    }\n    // containingGroup.appendChild(this.highlightGlyph);\n    containingGroup.appendChild(this.glyph);\n    containingGroup.appendChild(this.uncertainGlyph);\n};\n\nFeatureLink.prototype.hide = function () {\n    this.glyph.remove();\n    this.uncertainGlyph.remove();\n};\n\n// update the links(polygons/lines) to fit to the protein\nFeatureLink.prototype.setLinkCoordinates = function () {\n    function isNumber(thing) {\n        return (!isNaN(parseFloat(thing)) && isFinite(thing));\n    }\n\n    function getSegment(midPoint, controlPoint, startRes, endRes, participant, yOffset, originPoint) {\n        let startPoint, endPoint;\n        if (!participant.form) { // tests if form = undefined or 0 //TODO: maybe change this, its confusing\n            startPoint = participant.getPosition(originPoint);\n            endPoint = startPoint;\n        } else {\n            startPoint = participant.getResidueCoordinates(startRes, yOffset);\n            endPoint = participant.getResidueCoordinates(endRes, yOffset);\n        }\n        return \" Q\" + controlPoint[0] + \",\" + controlPoint[1] + \" \" + startPoint[0] + \",\" + startPoint[1] +\n            \" L\" + endPoint[0] + \",\" + endPoint[1] +\n            \" Q\" + controlPoint[0] + \",\" + controlPoint[1] + \" \" + midPoint[0] + \",\" + midPoint[1];\n    }\n\n    function sequenceDataMidPoint(sequenceData, participant) {\n        //get the smallest start and the biggest end\n        let lowestLinkedRes = null,\n            highestLinkedRes = null;\n        const sdCount = sequenceData.length;\n        for (let s = 0; s < sdCount; s++) {\n            const seqDatum = sequenceData[s];\n            if (!isNaN(parseFloat(seqDatum.begin)) && isFinite(seqDatum.begin)) {\n                // noinspection PointlessArithmeticExpressionJS\n                const start = seqDatum.begin * 1; // the * 1 is necessary (type conversion)\n                if (lowestLinkedRes === null || start < lowestLinkedRes) {\n                    lowestLinkedRes = start;\n                }\n            }\n            if (!isNaN(parseFloat(seqDatum.uncertainBegin)) && isFinite(seqDatum.uncertainBegin)) {\n                const uncertainBegin = seqDatum.uncertainBegin * 1;\n                if (lowestLinkedRes === null || uncertainBegin < lowestLinkedRes) {\n                    lowestLinkedRes = uncertainBegin;\n                }\n            }\n            if (!isNaN(parseFloat(seqDatum.end)) && isFinite(seqDatum.end)) {\n                const end = seqDatum.end * 1;\n                if (highestLinkedRes === null || end > highestLinkedRes) {\n                    highestLinkedRes = end;\n                }\n            }\n            if (!isNaN(parseFloat(seqDatum.uncertainEnd)) && isFinite(seqDatum.uncertainEnd)) {\n                const uncertainEnd = seqDatum.uncertainEnd * 1;\n                if (highestLinkedRes === null || uncertainEnd > highestLinkedRes) {\n                    highestLinkedRes = uncertainEnd;\n                }\n            }\n        }\n        return participant.getResidueCoordinates((lowestLinkedRes + highestLinkedRes) / 2, 0);\n    }\n\n    const fromParticipant = this.fromSequenceData[0].participant;\n    const toParticipant = this.toSequenceData[0].participant;\n    //calculate mid points of from and to sequence data\n    let fMid, tMid;\n\n    if (fromParticipant.form)  {\n        fMid = sequenceDataMidPoint(this.fromSequenceData, fromParticipant);\n    }\n    if (toParticipant.form)  {\n        tMid = sequenceDataMidPoint(this.toSequenceData, toParticipant);\n    }\n    if (!fromParticipant.form) { // if not (undefined or 0)\n        fMid = fromParticipant.getPosition(tMid);//toOriginPoint);\n    }\n    if (!toParticipant.form) {// if not (undefined or 0)\n        tMid = toParticipant.getPosition(fMid);//fromOriginPoint);\n    }\n\n    const fromOriginPoint = fMid;//null;//[fromParticipant.cy, fromParticipant.cy];\n    const toOriginPoint = tMid;//null;//[toParticipant.cy, toParticipant.cy];\n\n    // if (!fromParticipant.form) { // if not (undefined or 0)\n    //     fMid = fromParticipant.getPosition();//toOriginPoint);\n    // } else {\n    //     fMid = sequenceDataMidPoint(this.fromSequenceData, fromParticipant);\n    // }\n    // if (!toParticipant.form) {// if not (undefined or 0)\n    //     tMid = toParticipant.getPosition();//fromOriginPoint);\n    // } else {\n    //     tMid = sequenceDataMidPoint(this.toSequenceData, toParticipant);\n    // }\n\n    //calculate angle from fromParticipant mid point to toParticipant mid point\n    const deltaX = fMid[0] - tMid[0];\n    const deltaY = fMid[1] - tMid[1];\n    const angleBetweenMidPoints = Math.atan2(deltaY, deltaX);\n    //todo: tidy up trig code so everything is always in radian\n    let abmpDeg = angleBetweenMidPoints / (2 * Math.PI) * 360;\n    if (abmpDeg < 0) {\n        abmpDeg += 360;\n    }\n\n    //out is value we use to decide which side of bar the link glyph is drawn\n    //first for 'from' participant\n    let out = (abmpDeg - fromParticipant.rotation);\n    if (out < 0) {\n        out += 360;\n    }\n    let fyOffset = 10;\n    if (out < 180) {\n        fyOffset = -10;\n    }\n    let fRotRad = (fromParticipant.rotation / 360) * Math.PI * 2;\n    if (out > 180) {\n        fRotRad -= Math.PI;\n    }\n    //now for 'to' participant\n    out = (abmpDeg - toParticipant.rotation);\n    if (out < 0) {\n        out += 360;\n    }\n    let tyOffset = 10;\n    if (out > 180) {\n        tyOffset = -10;\n    }\n    let tRotRad = (toParticipant.rotation / 360) * Math.PI * 2;\n    if (out < 180) {\n        tRotRad -= Math.PI;\n    }\n\n    let ftMid = [fMid[0] + (30 * Math.sin(fRotRad) * this.app.z),\n        fMid[1] - (30 * Math.cos(fRotRad) * this.app.z)\n    ];\n    if (!fromParticipant.form) { // if not (undefined or 0)\n        ftMid = fMid;\n    }\n\n    let ttMid = [tMid[0] + (30 * Math.sin(tRotRad) * this.app.z),\n        tMid[1] - (30 * Math.cos(tRotRad) * this.app.z)\n    ];\n    if (!toParticipant.form) { // if not (undefined or 0)\n        ttMid = tMid;\n    }\n\n    const triPointMid = [(ftMid[0] + ttMid[0]) / 2, (ftMid[1] + ttMid[1]) / 2];\n    const fSDCount = this.fromSequenceData.length;\n    const tSDCount = this.toSequenceData.length;\n    let seqDatum;//, highlightStartRes, highlightEndRes;\n    let glyphPath = \"M\" + triPointMid[0] + \",\" + triPointMid[1];\n    let uncertainGlyphPath = \"M\" + triPointMid[0] + \",\" + triPointMid[1];\n    // let highlightGlyphPath = \"M\" + triPointMid[0] + \",\" + triPointMid[1];\n    for (let f = 0; f < fSDCount; f++) {\n        seqDatum = this.fromSequenceData[f];\n        if (isNumber(seqDatum.begin)  && isNumber(seqDatum.end) || fromParticipant.type === \"complex\") {\n            glyphPath += getSegment(triPointMid, ftMid, seqDatum.begin, seqDatum.end, fromParticipant, fyOffset, toOriginPoint);\n        }\n        // highlightStartRes = seqDatum.begin;\n        // highlightEndRes = seqDatum.end;\n        if (isNumber(seqDatum.uncertainBegin)) {\n            uncertainGlyphPath += getSegment(triPointMid, ftMid,\n                seqDatum.uncertainBegin, seqDatum.begin, fromParticipant, fyOffset, toOriginPoint);\n            // highlightStartRes = seqDatum.uncertainBegin;\n        }\n        if (isNumber(seqDatum.uncertainEnd)) {\n            uncertainGlyphPath += getSegment(triPointMid, ftMid,\n                seqDatum.end, seqDatum.uncertainEnd, fromParticipant, fyOffset, toOriginPoint);\n            // highlightEndRes = seqDatum.uncertainEnd;\n        }\n        // highlightGlyphPath += getPathSegments(triPointMid, ftMid,\n        //     highlightStartRes, highlightEndRes, fromParticipant, fyOffset);\n    }\n    for (let t = 0; t < tSDCount; t++) {\n        seqDatum = this.toSequenceData[t];\n        if (isNumber(seqDatum.begin) && isNumber(seqDatum.end) || toParticipant.type === \"complex\") {\n            glyphPath += getSegment(triPointMid, ttMid, seqDatum.begin, seqDatum.end, toParticipant, tyOffset, fromOriginPoint);\n        }\n        // highlightStartRes = seqDatum.begin;\n        // highlightEndRes = seqDatum.end;\n        if (isNumber(seqDatum.uncertainBegin)) {\n            uncertainGlyphPath += getSegment(triPointMid, ttMid,\n                seqDatum.uncertainBegin, seqDatum.begin, toParticipant, tyOffset, fromOriginPoint);\n            // highlightStartRes = seqDatum.uncertainBegin;\n        }\n        if (isNumber(seqDatum.uncertainEnd)) {\n            uncertainGlyphPath += getSegment(triPointMid, ttMid,\n                seqDatum.end, seqDatum.uncertainEnd, toParticipant, tyOffset, fromOriginPoint);\n            // highlightEndRes = seqDatum.uncertainEnd;\n        }\n        // highlightGlyphPath += getPathSegments(triPointMid, ttMid,\n        //     highlightStartRes, highlightEndRes, toParticipant, tyOffset);\n    }\n\n    if (!this.glyph) {\n        this.initSVG();\n    }\n\n    this.glyph.setAttribute(\"d\", glyphPath);\n    this.uncertainGlyph.setAttribute(\"d\", uncertainGlyphPath);\n    // this.highlightGlyph.setAttribute(\"d\", highlightGlyphPath);\n};\n"],"mappings":"AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/js/viz/link/feature-link.js\n"); /***/ }), @@ -1369,7 +1381,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"Link\", function() { return Link; });\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n\nfunction Link () {}\n\nLink.prototype.highlightParticipants = function (show) {\n for (let participant of this.participants) {\n participant.showHighlight(show);\n }\n};\n\nLink.prototype.initSVG = function () {\n this.line.setAttribute(\"class\", \"link\");\n this.line.setAttribute(\"fill\", \"none\");\n this.line.setAttribute(\"stroke\", \"black\");\n this.line.setAttribute(\"stroke-width\", \"1\");\n this.line.setAttribute(\"stroke-linecap\", \"round\");\n this.highlightLine.setAttribute(\"class\", \"link\");\n this.highlightLine.setAttribute(\"fill\", \"none\");\n this.highlightLine.setAttribute(\"stroke\", _config__WEBPACK_IMPORTED_MODULE_0__[\"highlightColour\"]);\n this.highlightLine.setAttribute(\"stroke-width\", \"10\");\n this.highlightLine.setAttribute(\"stroke-linecap\", \"round\");\n this.highlightLine.setAttribute(\"stroke-opacity\", \"0\");\n //set the events for it\n const self = this;\n this.line.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.line.onmouseover = function (evt) {\n self.mouseOver(evt);\n };\n this.line.onmouseout = function (evt) {\n self.mouseOut(evt);\n };\n // this.line.ontouchstart = function (evt) {\n // self.touchStart(evt);\n // };\n\n this.highlightLine.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.highlightLine.onmouseover = function (evt) {\n self.mouseOver(evt);\n };\n this.highlightLine.onmouseout = function (evt) {\n self.mouseOut(evt);\n };\n // this.highlightLine.ontouchstart = function (evt) {\n // self.touchStart(evt);\n // };\n};\n// event handler for starting dragging or rotation (or flipping internal links)\nLink.prototype.mouseDown = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt); //see MouseEvents.js\n //stop layout\n this.app.d3cola.stop();\n this.app.dragElement = this;\n //store start location\n const p = this.app.getEventPoint(evt); // seems to be correct, see above\n this.app.dragStart = this.app.mouseToSVG(p.x, p.y);\n return false;\n};\n\nLink.prototype.mouseOver = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt);\n this.app.setTooltip(this.getToolTip(), this.color);\n return false;\n};\n\nLink.prototype.getToolTip = function () {\n return this.id;\n};\n\nLink.prototype.mouseOut = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt);\n this.app.hideTooltip();\n return false;\n};\n\n/*\nLink.prototype.touchStart = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt); //see MouseEvents.js\n //if a force layout exists then stop it\n if (this.app.layout !== undefined) {\n this.app.layout.stop();\n }\n this.app.dragElement = this;\n this.app.clearSelection();\n // this.setSelected(true);\n //store start location\n const p = this.app.getTouchEventPoint(evt); // seems to be correct, see above\n this.app.dragStart = this.app.mouseToSVG(p.x, p.y);\n //~ this.showData();\n return false;\n};*/\n\n//used by BinaryLink and UnaryLink\nLink.prototype.hide = function () {\n this.highlightLine.remove();\n this.line.remove();\n};\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvdml6L2xpbmsvbGluay5qcy5qcyIsInNvdXJjZXMiOlsid2VicGFjazovL2NvbXBsZXh2aWV3ZXIvLi9zcmMvanMvdml6L2xpbmsvbGluay5qcz8xMmU5Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7aGlnaGxpZ2h0Q29sb3VyfSBmcm9tIFwiLi4vLi4vY29uZmlnXCI7XG5leHBvcnQgZnVuY3Rpb24gTGluayAoKSB7fVxuXG5MaW5rLnByb3RvdHlwZS5oaWdobGlnaHRQYXJ0aWNpcGFudHMgPSBmdW5jdGlvbiAoc2hvdykge1xuICAgIGZvciAobGV0IHBhcnRpY2lwYW50IG9mIHRoaXMucGFydGljaXBhbnRzKSB7XG4gICAgICAgIHBhcnRpY2lwYW50LnNob3dIaWdobGlnaHQoc2hvdyk7XG4gICAgfVxufTtcblxuTGluay5wcm90b3R5cGUuaW5pdFNWRyA9IGZ1bmN0aW9uICgpIHtcbiAgICB0aGlzLmxpbmUuc2V0QXR0cmlidXRlKFwiY2xhc3NcIiwgXCJsaW5rXCIpO1xuICAgIHRoaXMubGluZS5zZXRBdHRyaWJ1dGUoXCJmaWxsXCIsIFwibm9uZVwiKTtcbiAgICB0aGlzLmxpbmUuc2V0QXR0cmlidXRlKFwic3Ryb2tlXCIsIFwiYmxhY2tcIik7XG4gICAgdGhpcy5saW5lLnNldEF0dHJpYnV0ZShcInN0cm9rZS13aWR0aFwiLCBcIjFcIik7XG4gICAgdGhpcy5saW5lLnNldEF0dHJpYnV0ZShcInN0cm9rZS1saW5lY2FwXCIsIFwicm91bmRcIik7XG4gICAgdGhpcy5oaWdobGlnaHRMaW5lLnNldEF0dHJpYnV0ZShcImNsYXNzXCIsIFwibGlua1wiKTtcbiAgICB0aGlzLmhpZ2hsaWdodExpbmUuc2V0QXR0cmlidXRlKFwiZmlsbFwiLCBcIm5vbmVcIik7XG4gICAgdGhpcy5oaWdobGlnaHRMaW5lLnNldEF0dHJpYnV0ZShcInN0cm9rZVwiLCBoaWdobGlnaHRDb2xvdXIpO1xuICAgIHRoaXMuaGlnaGxpZ2h0TGluZS5zZXRBdHRyaWJ1dGUoXCJzdHJva2Utd2lkdGhcIiwgXCIxMFwiKTtcbiAgICB0aGlzLmhpZ2hsaWdodExpbmUuc2V0QXR0cmlidXRlKFwic3Ryb2tlLWxpbmVjYXBcIiwgXCJyb3VuZFwiKTtcbiAgICB0aGlzLmhpZ2hsaWdodExpbmUuc2V0QXR0cmlidXRlKFwic3Ryb2tlLW9wYWNpdHlcIiwgXCIwXCIpO1xuICAgIC8vc2V0IHRoZSBldmVudHMgZm9yIGl0XG4gICAgY29uc3Qgc2VsZiA9IHRoaXM7XG4gICAgdGhpcy5saW5lLm9ubW91c2Vkb3duID0gZnVuY3Rpb24gKGV2dCkge1xuICAgICAgICBzZWxmLm1vdXNlRG93bihldnQpO1xuICAgIH07XG4gICAgdGhpcy5saW5lLm9ubW91c2VvdmVyID0gZnVuY3Rpb24gKGV2dCkge1xuICAgICAgICBzZWxmLm1vdXNlT3ZlcihldnQpO1xuICAgIH07XG4gICAgdGhpcy5saW5lLm9ubW91c2VvdXQgPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgICAgIHNlbGYubW91c2VPdXQoZXZ0KTtcbiAgICB9O1xuICAgIC8vIHRoaXMubGluZS5vbnRvdWNoc3RhcnQgPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgLy8gICAgIHNlbGYudG91Y2hTdGFydChldnQpO1xuICAgIC8vIH07XG5cbiAgICB0aGlzLmhpZ2hsaWdodExpbmUub25tb3VzZWRvd24gPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgICAgIHNlbGYubW91c2VEb3duKGV2dCk7XG4gICAgfTtcbiAgICB0aGlzLmhpZ2hsaWdodExpbmUub25tb3VzZW92ZXIgPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgICAgIHNlbGYubW91c2VPdmVyKGV2dCk7XG4gICAgfTtcbiAgICB0aGlzLmhpZ2hsaWdodExpbmUub25tb3VzZW91dCA9IGZ1bmN0aW9uIChldnQpIHtcbiAgICAgICAgc2VsZi5tb3VzZU91dChldnQpO1xuICAgIH07XG4gICAgLy8gdGhpcy5oaWdobGlnaHRMaW5lLm9udG91Y2hzdGFydCA9IGZ1bmN0aW9uIChldnQpIHtcbiAgICAvLyAgICAgc2VsZi50b3VjaFN0YXJ0KGV2dCk7XG4gICAgLy8gfTtcbn07XG4vLyBldmVudCBoYW5kbGVyIGZvciBzdGFydGluZyBkcmFnZ2luZyBvciByb3RhdGlvbiAob3IgZmxpcHBpbmcgaW50ZXJuYWwgbGlua3MpXG5MaW5rLnByb3RvdHlwZS5tb3VzZURvd24gPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgdGhpcy5hcHAucHJldmVudERlZmF1bHRzQW5kU3RvcFByb3BhZ2F0aW9uKGV2dCk7IC8vc2VlIE1vdXNlRXZlbnRzLmpzXG4gICAgLy9zdG9wIGxheW91dFxuICAgIHRoaXMuYXBwLmQzY29sYS5zdG9wKCk7XG4gICAgdGhpcy5hcHAuZHJhZ0VsZW1lbnQgPSB0aGlzO1xuICAgIC8vc3RvcmUgc3RhcnQgbG9jYXRpb25cbiAgICBjb25zdCBwID0gdGhpcy5hcHAuZ2V0RXZlbnRQb2ludChldnQpOyAvLyBzZWVtcyB0byBiZSBjb3JyZWN0LCBzZWUgYWJvdmVcbiAgICB0aGlzLmFwcC5kcmFnU3RhcnQgPSB0aGlzLmFwcC5tb3VzZVRvU1ZHKHAueCwgcC55KTtcbiAgICByZXR1cm4gZmFsc2U7XG59O1xuXG5MaW5rLnByb3RvdHlwZS5tb3VzZU92ZXIgPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgdGhpcy5hcHAucHJldmVudERlZmF1bHRzQW5kU3RvcFByb3BhZ2F0aW9uKGV2dCk7XG4gICAgdGhpcy5hcHAuc2V0VG9vbHRpcCh0aGlzLmdldFRvb2xUaXAoKSwgdGhpcy5jb2xvcik7XG4gICAgcmV0dXJuIGZhbHNlO1xufTtcblxuTGluay5wcm90b3R5cGUuZ2V0VG9vbFRpcCA9IGZ1bmN0aW9uICgpIHtcbiAgICByZXR1cm4gdGhpcy5pZDtcbn07XG5cbkxpbmsucHJvdG90eXBlLm1vdXNlT3V0ID0gZnVuY3Rpb24gKGV2dCkge1xuICAgIHRoaXMuYXBwLnByZXZlbnREZWZhdWx0c0FuZFN0b3BQcm9wYWdhdGlvbihldnQpO1xuICAgIHRoaXMuYXBwLmhpZGVUb29sdGlwKCk7XG4gICAgcmV0dXJuIGZhbHNlO1xufTtcblxuLypcbkxpbmsucHJvdG90eXBlLnRvdWNoU3RhcnQgPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgdGhpcy5hcHAucHJldmVudERlZmF1bHRzQW5kU3RvcFByb3BhZ2F0aW9uKGV2dCk7IC8vc2VlIE1vdXNlRXZlbnRzLmpzXG4gICAgLy9pZiBhIGZvcmNlIGxheW91dCBleGlzdHMgdGhlbiBzdG9wIGl0XG4gICAgaWYgKHRoaXMuYXBwLmxheW91dCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgIHRoaXMuYXBwLmxheW91dC5zdG9wKCk7XG4gICAgfVxuICAgIHRoaXMuYXBwLmRyYWdFbGVtZW50ID0gdGhpcztcbiAgICB0aGlzLmFwcC5jbGVhclNlbGVjdGlvbigpO1xuICAgIC8vICAgIHRoaXMuc2V0U2VsZWN0ZWQodHJ1ZSk7XG4gICAgLy9zdG9yZSBzdGFydCBsb2NhdGlvblxuICAgIGNvbnN0IHAgPSB0aGlzLmFwcC5nZXRUb3VjaEV2ZW50UG9pbnQoZXZ0KTsgLy8gc2VlbXMgdG8gYmUgY29ycmVjdCwgc2VlIGFib3ZlXG4gICAgdGhpcy5hcHAuZHJhZ1N0YXJ0ID0gdGhpcy5hcHAubW91c2VUb1NWRyhwLngsIHAueSk7XG4gICAgLy9+IHRoaXMuc2hvd0RhdGEoKTtcbiAgICByZXR1cm4gZmFsc2U7XG59OyovXG5cbi8vdXNlZCBieSBCaW5hcnlMaW5rIGFuZCBVbmFyeUxpbmtcbkxpbmsucHJvdG90eXBlLmhpZGUgPSBmdW5jdGlvbiAoKSB7XG4gICAgdGhpcy5oaWdobGlnaHRMaW5lLnJlbW92ZSgpO1xuICAgIHRoaXMubGluZS5yZW1vdmUoKTtcbn07XG4iXSwibWFwcGluZ3MiOiJBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTsiLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./src/js/viz/link/link.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"Link\", function() { return Link; });\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n\nfunction Link () {}\n\nLink.prototype.highlightParticipants = function (show) {\n for (let participant of this.participants) {\n participant.showHighlight(show);\n }\n};\n\nLink.prototype.initSVG = function () {\n this.line.classList.add(\"link\",\"link-line\");//, \"certain-link\");\n this.highlightLine.classList.add(\"link\", \"highlight\", \"link-highlight\");\n //set the events for it\n const self = this;\n this.line.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.line.onmouseover = function (evt) {\n self.mouseOver(evt);\n };\n this.line.onmouseout = function (evt) {\n self.mouseOut(evt);\n };\n // this.line.ontouchstart = function (evt) {\n // self.touchStart(evt);\n // };\n\n this.highlightLine.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.highlightLine.onmouseover = function (evt) {\n self.mouseOver(evt);\n };\n this.highlightLine.onmouseout = function (evt) {\n self.mouseOut(evt);\n };\n // this.highlightLine.ontouchstart = function (evt) {\n // self.touchStart(evt);\n // };\n};\n// event handler for starting dragging or rotation (or flipping internal links)\nLink.prototype.mouseDown = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt); //see MouseEvents.js\n //stop layout\n this.app.d3cola.stop();\n this.app.dragElement = this;\n //store start location\n const p = this.app.getEventPoint(evt); // seems to be correct, see above\n this.app.dragStart = this.app.mouseToSVG(p.x, p.y);\n return false;\n};\n\nLink.prototype.mouseOver = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt);\n this.app.setTooltip(this.getToolTip(), this.color);\n return false;\n};\n\nLink.prototype.getToolTip = function () {\n return this.id;\n};\n\nLink.prototype.mouseOut = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt);\n this.app.hideTooltip();\n return false;\n};\n\n/*\nLink.prototype.touchStart = function (evt) {\n this.app.preventDefaultsAndStopPropagation(evt); //see MouseEvents.js\n //if a force layout exists then stop it\n if (this.app.layout !== undefined) {\n this.app.layout.stop();\n }\n this.app.dragElement = this;\n this.app.clearSelection();\n // this.setSelected(true);\n //store start location\n const p = this.app.getTouchEventPoint(evt); // seems to be correct, see above\n this.app.dragStart = this.app.mouseToSVG(p.x, p.y);\n //~ this.showData();\n return false;\n};*/\n\n//used by BinaryLink and UnaryLink\nLink.prototype.hide = function () {\n this.highlightLine.remove();\n this.line.remove();\n};\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvdml6L2xpbmsvbGluay5qcy5qcyIsInNvdXJjZXMiOlsid2VicGFjazovL2NvbXBsZXh2aWV3ZXIvLi9zcmMvanMvdml6L2xpbmsvbGluay5qcz8xMmU5Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7aGlnaGxpZ2h0Q29sb3VyfSBmcm9tIFwiLi4vLi4vY29uZmlnXCI7XG5leHBvcnQgZnVuY3Rpb24gTGluayAoKSB7fVxuXG5MaW5rLnByb3RvdHlwZS5oaWdobGlnaHRQYXJ0aWNpcGFudHMgPSBmdW5jdGlvbiAoc2hvdykge1xuICAgIGZvciAobGV0IHBhcnRpY2lwYW50IG9mIHRoaXMucGFydGljaXBhbnRzKSB7XG4gICAgICAgIHBhcnRpY2lwYW50LnNob3dIaWdobGlnaHQoc2hvdyk7XG4gICAgfVxufTtcblxuTGluay5wcm90b3R5cGUuaW5pdFNWRyA9IGZ1bmN0aW9uICgpIHtcbiAgICB0aGlzLmxpbmUuY2xhc3NMaXN0LmFkZChcImxpbmtcIixcImxpbmstbGluZVwiKTsvLywgXCJjZXJ0YWluLWxpbmtcIik7XG4gICAgdGhpcy5oaWdobGlnaHRMaW5lLmNsYXNzTGlzdC5hZGQoXCJsaW5rXCIsIFwiaGlnaGxpZ2h0XCIsIFwibGluay1oaWdobGlnaHRcIik7XG4gICAgLy9zZXQgdGhlIGV2ZW50cyBmb3IgaXRcbiAgICBjb25zdCBzZWxmID0gdGhpcztcbiAgICB0aGlzLmxpbmUub25tb3VzZWRvd24gPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgICAgIHNlbGYubW91c2VEb3duKGV2dCk7XG4gICAgfTtcbiAgICB0aGlzLmxpbmUub25tb3VzZW92ZXIgPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgICAgIHNlbGYubW91c2VPdmVyKGV2dCk7XG4gICAgfTtcbiAgICB0aGlzLmxpbmUub25tb3VzZW91dCA9IGZ1bmN0aW9uIChldnQpIHtcbiAgICAgICAgc2VsZi5tb3VzZU91dChldnQpO1xuICAgIH07XG4gICAgLy8gdGhpcy5saW5lLm9udG91Y2hzdGFydCA9IGZ1bmN0aW9uIChldnQpIHtcbiAgICAvLyAgICAgc2VsZi50b3VjaFN0YXJ0KGV2dCk7XG4gICAgLy8gfTtcblxuICAgIHRoaXMuaGlnaGxpZ2h0TGluZS5vbm1vdXNlZG93biA9IGZ1bmN0aW9uIChldnQpIHtcbiAgICAgICAgc2VsZi5tb3VzZURvd24oZXZ0KTtcbiAgICB9O1xuICAgIHRoaXMuaGlnaGxpZ2h0TGluZS5vbm1vdXNlb3ZlciA9IGZ1bmN0aW9uIChldnQpIHtcbiAgICAgICAgc2VsZi5tb3VzZU92ZXIoZXZ0KTtcbiAgICB9O1xuICAgIHRoaXMuaGlnaGxpZ2h0TGluZS5vbm1vdXNlb3V0ID0gZnVuY3Rpb24gKGV2dCkge1xuICAgICAgICBzZWxmLm1vdXNlT3V0KGV2dCk7XG4gICAgfTtcbiAgICAvLyB0aGlzLmhpZ2hsaWdodExpbmUub250b3VjaHN0YXJ0ID0gZnVuY3Rpb24gKGV2dCkge1xuICAgIC8vICAgICBzZWxmLnRvdWNoU3RhcnQoZXZ0KTtcbiAgICAvLyB9O1xufTtcbi8vIGV2ZW50IGhhbmRsZXIgZm9yIHN0YXJ0aW5nIGRyYWdnaW5nIG9yIHJvdGF0aW9uIChvciBmbGlwcGluZyBpbnRlcm5hbCBsaW5rcylcbkxpbmsucHJvdG90eXBlLm1vdXNlRG93biA9IGZ1bmN0aW9uIChldnQpIHtcbiAgICB0aGlzLmFwcC5wcmV2ZW50RGVmYXVsdHNBbmRTdG9wUHJvcGFnYXRpb24oZXZ0KTsgLy9zZWUgTW91c2VFdmVudHMuanNcbiAgICAvL3N0b3AgbGF5b3V0XG4gICAgdGhpcy5hcHAuZDNjb2xhLnN0b3AoKTtcbiAgICB0aGlzLmFwcC5kcmFnRWxlbWVudCA9IHRoaXM7XG4gICAgLy9zdG9yZSBzdGFydCBsb2NhdGlvblxuICAgIGNvbnN0IHAgPSB0aGlzLmFwcC5nZXRFdmVudFBvaW50KGV2dCk7IC8vIHNlZW1zIHRvIGJlIGNvcnJlY3QsIHNlZSBhYm92ZVxuICAgIHRoaXMuYXBwLmRyYWdTdGFydCA9IHRoaXMuYXBwLm1vdXNlVG9TVkcocC54LCBwLnkpO1xuICAgIHJldHVybiBmYWxzZTtcbn07XG5cbkxpbmsucHJvdG90eXBlLm1vdXNlT3ZlciA9IGZ1bmN0aW9uIChldnQpIHtcbiAgICB0aGlzLmFwcC5wcmV2ZW50RGVmYXVsdHNBbmRTdG9wUHJvcGFnYXRpb24oZXZ0KTtcbiAgICB0aGlzLmFwcC5zZXRUb29sdGlwKHRoaXMuZ2V0VG9vbFRpcCgpLCB0aGlzLmNvbG9yKTtcbiAgICByZXR1cm4gZmFsc2U7XG59O1xuXG5MaW5rLnByb3RvdHlwZS5nZXRUb29sVGlwID0gZnVuY3Rpb24gKCkge1xuICAgIHJldHVybiB0aGlzLmlkO1xufTtcblxuTGluay5wcm90b3R5cGUubW91c2VPdXQgPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgdGhpcy5hcHAucHJldmVudERlZmF1bHRzQW5kU3RvcFByb3BhZ2F0aW9uKGV2dCk7XG4gICAgdGhpcy5hcHAuaGlkZVRvb2x0aXAoKTtcbiAgICByZXR1cm4gZmFsc2U7XG59O1xuXG4vKlxuTGluay5wcm90b3R5cGUudG91Y2hTdGFydCA9IGZ1bmN0aW9uIChldnQpIHtcbiAgICB0aGlzLmFwcC5wcmV2ZW50RGVmYXVsdHNBbmRTdG9wUHJvcGFnYXRpb24oZXZ0KTsgLy9zZWUgTW91c2VFdmVudHMuanNcbiAgICAvL2lmIGEgZm9yY2UgbGF5b3V0IGV4aXN0cyB0aGVuIHN0b3AgaXRcbiAgICBpZiAodGhpcy5hcHAubGF5b3V0ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgdGhpcy5hcHAubGF5b3V0LnN0b3AoKTtcbiAgICB9XG4gICAgdGhpcy5hcHAuZHJhZ0VsZW1lbnQgPSB0aGlzO1xuICAgIHRoaXMuYXBwLmNsZWFyU2VsZWN0aW9uKCk7XG4gICAgLy8gICAgdGhpcy5zZXRTZWxlY3RlZCh0cnVlKTtcbiAgICAvL3N0b3JlIHN0YXJ0IGxvY2F0aW9uXG4gICAgY29uc3QgcCA9IHRoaXMuYXBwLmdldFRvdWNoRXZlbnRQb2ludChldnQpOyAvLyBzZWVtcyB0byBiZSBjb3JyZWN0LCBzZWUgYWJvdmVcbiAgICB0aGlzLmFwcC5kcmFnU3RhcnQgPSB0aGlzLmFwcC5tb3VzZVRvU1ZHKHAueCwgcC55KTtcbiAgICAvL34gdGhpcy5zaG93RGF0YSgpO1xuICAgIHJldHVybiBmYWxzZTtcbn07Ki9cblxuLy91c2VkIGJ5IEJpbmFyeUxpbmsgYW5kIFVuYXJ5TGlua1xuTGluay5wcm90b3R5cGUuaGlkZSA9IGZ1bmN0aW9uICgpIHtcbiAgICB0aGlzLmhpZ2hsaWdodExpbmUucmVtb3ZlKCk7XG4gICAgdGhpcy5saW5lLnJlbW92ZSgpO1xufTtcbiJdLCJtYXBwaW5ncyI6IkFBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOyIsInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./src/js/viz/link/link.js\n"); /***/ }), @@ -1381,7 +1393,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"NaryLink\", function() { return NaryLink; });\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! d3 */ \"./node_modules/d3/d3.js\");\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(d3__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _link__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./link */ \"./src/js/viz/link/link.js\");\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n //used for d3.geom.hull\n\n\n\n//NaryLink.naryColors; // init'ed in clear function of util\nNaryLink.orbitNodes = 20;\nNaryLink.orbitRadius = 20;\n\nfunction NaryLink(id, app) {\n this.id = id;\n this.participants = [];\n this.sequenceLinks = new Map();\n this.binaryLinks = new Map();\n this.unaryLinks = new Map();\n this.app = app;\n // this.tooltip = this.id;\n this.initSVG();\n}\n\nNaryLink.prototype = new _link__WEBPACK_IMPORTED_MODULE_1__[\"Link\"]();\n\n/*\nNaryLink.prototype.getTotalParticipantCount = function () {\n let result = 0;\n const c = this.participants.length;\n for (let p = 0; p < c; p++) {\n const participant = this.participants[p];\n //console.log(\"! \" + typeof participant);\n if (participant.type !== \"complex\") {\n result++;\n } else {\n result += participant.naryLink.getTotalParticipantCount();\n }\n }\n return result;\n};\n*/\n\nNaryLink.prototype.initSVG = function () {\n this.path = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_2__[\"svgns\"], \"path\");\n this.color = NaryLink.naryColors(this.id);\n this.path.setAttribute(\"fill\", this.color);\n //set the events for it\n const self = this;\n this.path.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.path.onmouseover = function (evt) {\n self.mouseOver(evt);\n };\n this.path.onmouseout = function (evt) {\n self.mouseOut(evt);\n };\n // this.path.ontouchstart = function (evt) {\n // self.touchStart(evt);\n // };\n};\n\nNaryLink.prototype.showHighlight = function (show) {\n this.highlightParticipants(show);\n};\n\nNaryLink.prototype.check = function () {\n this.show();\n return true;\n};\n\nNaryLink.prototype.show = function () {\n // this.path.setAttribute(\"stroke-width\", this.app.z);\n this.setLinkCoordinates();\n this.app.naryLinks.appendChild(this.path);\n};\n\nNaryLink.prototype.hide = function () {\n};\n\nNaryLink.prototype.setLinkCoordinates = function (dontPropogate) {\n // Uses d3.geom.hull to calculate a bounding path around an array of vertices\n const calculateHullPath = function (values) {\n const hullPath = d3__WEBPACK_IMPORTED_MODULE_0__[\"geom\"].hull(values);\n self.hull = hullPath; //hack?\n return \"M\" + hullPath.join(\"L\") + \"Z\";\n };\n const self = this; // TODO: - tidy hack above?\n this.mapped = this.orbitNodes(this.getMappedCoordinates());\n const hullValues = calculateHullPath(this.mapped);\n if (hullValues) {\n this.path.setAttribute(\"d\", hullValues);\n }\n if (this.complex && !dontPropogate) {\n this.complex.setAllLinkCoordinates();\n }\n};\n\nNaryLink.prototype.getMappedCoordinates = function () {\n const participants = this.participants;\n let mapped = [];\n const ic = participants.length;\n for (let i = 0; i < ic; i++) {\n const participant = participants[i];\n if (participant.type === \"complex\") {\n mapped = mapped.concat(this.orbitNodes(participant.naryLink.getMappedCoordinates()));\n } else if (participant.form === 1) {\n const start = participant.getResidueCoordinates(0);\n const end = participant.getResidueCoordinates(participant.size);\n if (!isNaN(start[0]) && !isNaN(start[1]) &&\n !isNaN(end[0]) && !isNaN(end[1])) {\n mapped.push(start);\n mapped.push(end);\n } else {\n mapped.push(participant.getPosition());\n }\n } else {\n mapped.push(participant.getPosition());\n }\n }\n return mapped;\n};\n\n//'orbit' nodes - several nodes around participant positions to give margin\nNaryLink.prototype.orbitNodes = function (mapped) {\n\n\n const orbitNodes = [];\n const mc = mapped.length;\n for (let mi = 0; mi < mc; mi++) {\n const m = mapped[mi];\n for (let o = 0; o < NaryLink.orbitNodes; o++) {\n const angle = (360 / NaryLink.orbitNodes) * o;\n const p = [m[0] + NaryLink.orbitRadius, m[1]];\n orbitNodes.push(Object(_config__WEBPACK_IMPORTED_MODULE_2__[\"rotatePointAboutPoint\"])(p, m, angle));\n }\n }\n return orbitNodes;\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvdml6L2xpbmsvbmFyeS1saW5rLmpzLmpzIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vY29tcGxleHZpZXdlci8uL3NyYy9qcy92aXovbGluay9uYXJ5LWxpbmsuanM/NmYzYiJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBkMyBmcm9tIFwiZDNcIjsgLy91c2VkIGZvciBkMy5nZW9tLmh1bGxcbmltcG9ydCB7TGlua30gZnJvbSBcIi4vbGlua1wiO1xuaW1wb3J0IHtzdmducywgcm90YXRlUG9pbnRBYm91dFBvaW50fSBmcm9tIFwiLi4vLi4vY29uZmlnXCI7XG5cbi8vTmFyeUxpbmsubmFyeUNvbG9yczsgLy8gaW5pdCdlZCBpbiBjbGVhciBmdW5jdGlvbiBvZiB1dGlsXG5OYXJ5TGluay5vcmJpdE5vZGVzID0gMjA7XG5OYXJ5TGluay5vcmJpdFJhZGl1cyA9IDIwO1xuXG5leHBvcnQgZnVuY3Rpb24gTmFyeUxpbmsoaWQsIGFwcCkge1xuICAgIHRoaXMuaWQgPSBpZDtcbiAgICB0aGlzLnBhcnRpY2lwYW50cyA9IFtdO1xuICAgIHRoaXMuc2VxdWVuY2VMaW5rcyA9IG5ldyBNYXAoKTtcbiAgICB0aGlzLmJpbmFyeUxpbmtzID0gbmV3IE1hcCgpO1xuICAgIHRoaXMudW5hcnlMaW5rcyA9IG5ldyBNYXAoKTtcbiAgICB0aGlzLmFwcCA9IGFwcDtcbiAgICAvLyB0aGlzLnRvb2x0aXAgPSB0aGlzLmlkO1xuICAgIHRoaXMuaW5pdFNWRygpO1xufVxuXG5OYXJ5TGluay5wcm90b3R5cGUgPSBuZXcgTGluaygpO1xuXG4vKlxuTmFyeUxpbmsucHJvdG90eXBlLmdldFRvdGFsUGFydGljaXBhbnRDb3VudCA9IGZ1bmN0aW9uICgpIHtcbiAgICBsZXQgcmVzdWx0ID0gMDtcbiAgICBjb25zdCBjID0gdGhpcy5wYXJ0aWNpcGFudHMubGVuZ3RoO1xuICAgIGZvciAobGV0IHAgPSAwOyBwIDwgYzsgcCsrKSB7XG4gICAgICAgIGNvbnN0IHBhcnRpY2lwYW50ID0gdGhpcy5wYXJ0aWNpcGFudHNbcF07XG4gICAgICAgIC8vY29uc29sZS5sb2coXCIhIFwiICsgdHlwZW9mIHBhcnRpY2lwYW50KTtcbiAgICAgICAgaWYgKHBhcnRpY2lwYW50LnR5cGUgIT09IFwiY29tcGxleFwiKSB7XG4gICAgICAgICAgICByZXN1bHQrKztcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHJlc3VsdCArPSBwYXJ0aWNpcGFudC5uYXJ5TGluay5nZXRUb3RhbFBhcnRpY2lwYW50Q291bnQoKTtcbiAgICAgICAgfVxuICAgIH1cbiAgICByZXR1cm4gcmVzdWx0O1xufTtcbiovXG5cbk5hcnlMaW5rLnByb3RvdHlwZS5pbml0U1ZHID0gZnVuY3Rpb24gKCkge1xuICAgIHRoaXMucGF0aCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnROUyhzdmducywgXCJwYXRoXCIpO1xuICAgIHRoaXMuY29sb3IgPSBOYXJ5TGluay5uYXJ5Q29sb3JzKHRoaXMuaWQpO1xuICAgIHRoaXMucGF0aC5zZXRBdHRyaWJ1dGUoXCJmaWxsXCIsIHRoaXMuY29sb3IpO1xuICAgIC8vc2V0IHRoZSBldmVudHMgZm9yIGl0XG4gICAgY29uc3Qgc2VsZiA9IHRoaXM7XG4gICAgdGhpcy5wYXRoLm9ubW91c2Vkb3duID0gZnVuY3Rpb24gKGV2dCkge1xuICAgICAgICBzZWxmLm1vdXNlRG93bihldnQpO1xuICAgIH07XG4gICAgdGhpcy5wYXRoLm9ubW91c2VvdmVyID0gZnVuY3Rpb24gKGV2dCkge1xuICAgICAgICBzZWxmLm1vdXNlT3ZlcihldnQpO1xuICAgIH07XG4gICAgdGhpcy5wYXRoLm9ubW91c2VvdXQgPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgICAgIHNlbGYubW91c2VPdXQoZXZ0KTtcbiAgICB9O1xuICAgIC8vIHRoaXMucGF0aC5vbnRvdWNoc3RhcnQgPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgLy8gICAgIHNlbGYudG91Y2hTdGFydChldnQpO1xuICAgIC8vIH07XG59O1xuXG5OYXJ5TGluay5wcm90b3R5cGUuc2hvd0hpZ2hsaWdodCA9IGZ1bmN0aW9uIChzaG93KSB7XG4gICAgdGhpcy5oaWdobGlnaHRQYXJ0aWNpcGFudHMoc2hvdyk7XG59O1xuXG5OYXJ5TGluay5wcm90b3R5cGUuY2hlY2sgPSBmdW5jdGlvbiAoKSB7XG4gICAgdGhpcy5zaG93KCk7XG4gICAgcmV0dXJuIHRydWU7XG59O1xuXG5OYXJ5TGluay5wcm90b3R5cGUuc2hvdyA9IGZ1bmN0aW9uICgpIHtcbiAgICAvLyB0aGlzLnBhdGguc2V0QXR0cmlidXRlKFwic3Ryb2tlLXdpZHRoXCIsIHRoaXMuYXBwLnopO1xuICAgIHRoaXMuc2V0TGlua0Nvb3JkaW5hdGVzKCk7XG4gICAgdGhpcy5hcHAubmFyeUxpbmtzLmFwcGVuZENoaWxkKHRoaXMucGF0aCk7XG59O1xuXG5OYXJ5TGluay5wcm90b3R5cGUuaGlkZSA9IGZ1bmN0aW9uICgpIHtcbn07XG5cbk5hcnlMaW5rLnByb3RvdHlwZS5zZXRMaW5rQ29vcmRpbmF0ZXMgPSBmdW5jdGlvbiAoZG9udFByb3BvZ2F0ZSkge1xuICAgIC8vIFVzZXMgZDMuZ2VvbS5odWxsIHRvIGNhbGN1bGF0ZSBhIGJvdW5kaW5nIHBhdGggYXJvdW5kIGFuIGFycmF5IG9mIHZlcnRpY2VzXG4gICAgY29uc3QgY2FsY3VsYXRlSHVsbFBhdGggPSBmdW5jdGlvbiAodmFsdWVzKSB7XG4gICAgICAgIGNvbnN0IGh1bGxQYXRoID0gZDMuZ2VvbS5odWxsKHZhbHVlcyk7XG4gICAgICAgIHNlbGYuaHVsbCA9IGh1bGxQYXRoOyAvL2hhY2s/XG4gICAgICAgIHJldHVybiBcIk1cIiArIGh1bGxQYXRoLmpvaW4oXCJMXCIpICsgXCJaXCI7XG4gICAgfTtcbiAgICBjb25zdCBzZWxmID0gdGhpczsgLy8gVE9ETzogLSB0aWR5IGhhY2sgYWJvdmU/XG4gICAgdGhpcy5tYXBwZWQgPSB0aGlzLm9yYml0Tm9kZXModGhpcy5nZXRNYXBwZWRDb29yZGluYXRlcygpKTtcbiAgICBjb25zdCBodWxsVmFsdWVzID0gY2FsY3VsYXRlSHVsbFBhdGgodGhpcy5tYXBwZWQpO1xuICAgIGlmIChodWxsVmFsdWVzKSB7XG4gICAgICAgIHRoaXMucGF0aC5zZXRBdHRyaWJ1dGUoXCJkXCIsIGh1bGxWYWx1ZXMpO1xuICAgIH1cbiAgICBpZiAodGhpcy5jb21wbGV4ICYmICFkb250UHJvcG9nYXRlKSB7XG4gICAgICAgIHRoaXMuY29tcGxleC5zZXRBbGxMaW5rQ29vcmRpbmF0ZXMoKTtcbiAgICB9XG59O1xuXG5OYXJ5TGluay5wcm90b3R5cGUuZ2V0TWFwcGVkQ29vcmRpbmF0ZXMgPSBmdW5jdGlvbiAoKSB7XG4gICAgY29uc3QgcGFydGljaXBhbnRzID0gdGhpcy5wYXJ0aWNpcGFudHM7XG4gICAgbGV0IG1hcHBlZCA9IFtdO1xuICAgIGNvbnN0IGljID0gcGFydGljaXBhbnRzLmxlbmd0aDtcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IGljOyBpKyspIHtcbiAgICAgICAgY29uc3QgcGFydGljaXBhbnQgPSBwYXJ0aWNpcGFudHNbaV07XG4gICAgICAgIGlmIChwYXJ0aWNpcGFudC50eXBlID09PSBcImNvbXBsZXhcIikge1xuICAgICAgICAgICAgbWFwcGVkID0gbWFwcGVkLmNvbmNhdCh0aGlzLm9yYml0Tm9kZXMocGFydGljaXBhbnQubmFyeUxpbmsuZ2V0TWFwcGVkQ29vcmRpbmF0ZXMoKSkpO1xuICAgICAgICB9IGVsc2UgaWYgKHBhcnRpY2lwYW50LmZvcm0gPT09IDEpIHtcbiAgICAgICAgICAgIGNvbnN0IHN0YXJ0ID0gcGFydGljaXBhbnQuZ2V0UmVzaWR1ZUNvb3JkaW5hdGVzKDApO1xuICAgICAgICAgICAgY29uc3QgZW5kID0gcGFydGljaXBhbnQuZ2V0UmVzaWR1ZUNvb3JkaW5hdGVzKHBhcnRpY2lwYW50LnNpemUpO1xuICAgICAgICAgICAgaWYgKCFpc05hTihzdGFydFswXSkgJiYgIWlzTmFOKHN0YXJ0WzFdKSAmJlxuICAgICAgICAgICAgICAgICFpc05hTihlbmRbMF0pICYmICFpc05hTihlbmRbMV0pKSB7XG4gICAgICAgICAgICAgICAgbWFwcGVkLnB1c2goc3RhcnQpO1xuICAgICAgICAgICAgICAgIG1hcHBlZC5wdXNoKGVuZCk7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgIG1hcHBlZC5wdXNoKHBhcnRpY2lwYW50LmdldFBvc2l0aW9uKCkpO1xuICAgICAgICAgICAgfVxuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgbWFwcGVkLnB1c2gocGFydGljaXBhbnQuZ2V0UG9zaXRpb24oKSk7XG4gICAgICAgIH1cbiAgICB9XG4gICAgcmV0dXJuIG1hcHBlZDtcbn07XG5cbi8vJ29yYml0JyBub2RlcyAtIHNldmVyYWwgbm9kZXMgYXJvdW5kIHBhcnRpY2lwYW50IHBvc2l0aW9ucyB0byBnaXZlIG1hcmdpblxuTmFyeUxpbmsucHJvdG90eXBlLm9yYml0Tm9kZXMgPSBmdW5jdGlvbiAobWFwcGVkKSB7XG5cblxuICAgIGNvbnN0IG9yYml0Tm9kZXMgPSBbXTtcbiAgICBjb25zdCBtYyA9IG1hcHBlZC5sZW5ndGg7XG4gICAgZm9yIChsZXQgbWkgPSAwOyBtaSA8IG1jOyBtaSsrKSB7XG4gICAgICAgIGNvbnN0IG0gPSBtYXBwZWRbbWldO1xuICAgICAgICBmb3IgKGxldCBvID0gMDsgbyA8IE5hcnlMaW5rLm9yYml0Tm9kZXM7IG8rKykge1xuICAgICAgICAgICAgY29uc3QgYW5nbGUgPSAoMzYwIC8gTmFyeUxpbmsub3JiaXROb2RlcykgKiBvO1xuICAgICAgICAgICAgY29uc3QgcCA9IFttWzBdICsgTmFyeUxpbmsub3JiaXRSYWRpdXMsIG1bMV1dO1xuICAgICAgICAgICAgb3JiaXROb2Rlcy5wdXNoKHJvdGF0ZVBvaW50QWJvdXRQb2ludChwLCBtLCBhbmdsZSkpO1xuICAgICAgICB9XG4gICAgfVxuICAgIHJldHVybiBvcmJpdE5vZGVzO1xufTsiXSwibWFwcGluZ3MiOiJBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSIsInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./src/js/viz/link/nary-link.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"NaryLink\", function() { return NaryLink; });\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! d3 */ \"./node_modules/d3/d3.js\");\n/* harmony import */ var d3__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(d3__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _link__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./link */ \"./src/js/viz/link/link.js\");\n/* harmony import */ var _config__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../../config */ \"./src/js/config.js\");\n //used for d3.geom.hull\n\n\n\n//NaryLink.naryColors; // init'ed in clear function of util\nNaryLink.orbitNodes = 20;\nNaryLink.orbitRadius = 22;\n\nfunction NaryLink(id, app) {\n this.id = id;\n this.participants = [];\n this.sequenceLinks = new Map();\n this.binaryLinks = new Map();\n this.unaryLinks = new Map();\n this.app = app;\n // this.tooltip = this.id;\n this.initSVG();\n}\n\nNaryLink.prototype = new _link__WEBPACK_IMPORTED_MODULE_1__[\"Link\"]();\n\n/*\nNaryLink.prototype.getTotalParticipantCount = function () {\n let result = 0;\n const c = this.participants.length;\n for (let p = 0; p < c; p++) {\n const participant = this.participants[p];\n //console.log(\"! \" + typeof participant);\n if (participant.type !== \"complex\") {\n result++;\n } else {\n result += participant.naryLink.getTotalParticipantCount();\n }\n }\n return result;\n};\n*/\n\nNaryLink.prototype.initSVG = function () {\n this.path = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_2__[\"svgns\"], \"path\");\n this.color = NaryLink.naryColors(this.id);\n this.path.setAttribute(\"fill\", this.color);\n //set the events for it\n const self = this;\n this.path.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.path.onmouseover = function (evt) {\n self.mouseOver(evt);\n };\n this.path.onmouseout = function (evt) {\n self.mouseOut(evt);\n };\n // this.path.ontouchstart = function (evt) {\n // self.touchStart(evt);\n // };\n // todo - prob better way todo this\n this.path2 = document.createElementNS(_config__WEBPACK_IMPORTED_MODULE_2__[\"svgns\"], \"path\");\n this.path2.setAttribute(\"fill\", \"none\");\n //set the events for it\n this.path2.onmousedown = function (evt) {\n self.mouseDown(evt);\n };\n this.path2.onmouseover = function (evt) {\n self.mouseOver(evt);\n };\n this.path2.onmouseout = function (evt) {\n self.mouseOut(evt);\n };\n // this.path2.ontouchstart = function (evt) {\n // self.touchStart(evt);\n // };\n};\n\nNaryLink.prototype.showHighlight = function (show) {\n this.highlightParticipants(show);\n};\n\nNaryLink.prototype.check = function () {\n this.show();\n return true;\n};\n\nNaryLink.prototype.show = function () {\n // this.path.setAttribute(\"stroke-width\", this.app.z);\n this.setLinkCoordinates();\n this.app.naryLinks.appendChild(this.path);\n this.app.naryLinks.appendChild(this.path2);\n};\n\nNaryLink.prototype.hide = function () {\n};\n\nNaryLink.prototype.setLinkCoordinates = function (dontPropogate) {\n // Uses d3.geom.hull to calculate a bounding path around an array of vertices\n const calculateHullPath = function (values) {\n const hullPath = d3__WEBPACK_IMPORTED_MODULE_0__[\"geom\"].hull(values);\n self.hull = hullPath; //hack?\n return \"M\" + hullPath.join(\"L\") + \"Z\";\n };\n const self = this; // TODO: - tidy hack above?\n this.mapped = this.orbitNodes(this.getMappedCoordinates());\n const hullValues = calculateHullPath(this.mapped);\n if (hullValues) {\n this.path.setAttribute(\"d\", hullValues);\n this.path2.setAttribute(\"d\", hullValues);\n }\n if (this.complex && !dontPropogate) {\n this.complex.setAllLinkCoordinates();\n }\n};\n\nNaryLink.prototype.getMappedCoordinates = function () {\n const participants = this.participants;\n let mapped = [];\n const ic = participants.length;\n for (let i = 0; i < ic; i++) {\n const participant = participants[i];\n if (participant.type === \"complex\") {\n mapped = mapped.concat(this.orbitNodes(participant.naryLink.getMappedCoordinates()));\n } else if (participant.form === 1) {\n const start = participant.getResidueCoordinates(0);\n const end = participant.getResidueCoordinates(participant.size);\n if (!isNaN(start[0]) && !isNaN(start[1]) &&\n !isNaN(end[0]) && !isNaN(end[1])) {\n mapped.push(start);\n mapped.push(end);\n } else {\n mapped.push(participant.getPosition());\n }\n } else {\n mapped.push(participant.getPosition());\n }\n }\n return mapped;\n};\n\n//'orbit' nodes - several nodes around participant positions to give margin\nNaryLink.prototype.orbitNodes = function (mapped) {\n\n\n const orbitNodes = [];\n const mc = mapped.length;\n for (let mi = 0; mi < mc; mi++) {\n const m = mapped[mi];\n for (let o = 0; o < NaryLink.orbitNodes; o++) {\n const angle = (360 / NaryLink.orbitNodes) * o;\n const p = [m[0] + NaryLink.orbitRadius, m[1]];\n orbitNodes.push(Object(_config__WEBPACK_IMPORTED_MODULE_2__[\"rotatePointAboutPoint\"])(p, m, angle));\n }\n }\n return orbitNodes;\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvanMvdml6L2xpbmsvbmFyeS1saW5rLmpzLmpzIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vY29tcGxleHZpZXdlci8uL3NyYy9qcy92aXovbGluay9uYXJ5LWxpbmsuanM/NmYzYiJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBkMyBmcm9tIFwiZDNcIjsgLy91c2VkIGZvciBkMy5nZW9tLmh1bGxcbmltcG9ydCB7TGlua30gZnJvbSBcIi4vbGlua1wiO1xuaW1wb3J0IHtzdmducywgcm90YXRlUG9pbnRBYm91dFBvaW50fSBmcm9tIFwiLi4vLi4vY29uZmlnXCI7XG5cbi8vTmFyeUxpbmsubmFyeUNvbG9yczsgLy8gaW5pdCdlZCBpbiBjbGVhciBmdW5jdGlvbiBvZiB1dGlsXG5OYXJ5TGluay5vcmJpdE5vZGVzID0gMjA7XG5OYXJ5TGluay5vcmJpdFJhZGl1cyA9IDIyO1xuXG5leHBvcnQgZnVuY3Rpb24gTmFyeUxpbmsoaWQsIGFwcCkge1xuICAgIHRoaXMuaWQgPSBpZDtcbiAgICB0aGlzLnBhcnRpY2lwYW50cyA9IFtdO1xuICAgIHRoaXMuc2VxdWVuY2VMaW5rcyA9IG5ldyBNYXAoKTtcbiAgICB0aGlzLmJpbmFyeUxpbmtzID0gbmV3IE1hcCgpO1xuICAgIHRoaXMudW5hcnlMaW5rcyA9IG5ldyBNYXAoKTtcbiAgICB0aGlzLmFwcCA9IGFwcDtcbiAgICAvLyB0aGlzLnRvb2x0aXAgPSB0aGlzLmlkO1xuICAgIHRoaXMuaW5pdFNWRygpO1xufVxuXG5OYXJ5TGluay5wcm90b3R5cGUgPSBuZXcgTGluaygpO1xuXG4vKlxuTmFyeUxpbmsucHJvdG90eXBlLmdldFRvdGFsUGFydGljaXBhbnRDb3VudCA9IGZ1bmN0aW9uICgpIHtcbiAgICBsZXQgcmVzdWx0ID0gMDtcbiAgICBjb25zdCBjID0gdGhpcy5wYXJ0aWNpcGFudHMubGVuZ3RoO1xuICAgIGZvciAobGV0IHAgPSAwOyBwIDwgYzsgcCsrKSB7XG4gICAgICAgIGNvbnN0IHBhcnRpY2lwYW50ID0gdGhpcy5wYXJ0aWNpcGFudHNbcF07XG4gICAgICAgIC8vY29uc29sZS5sb2coXCIhIFwiICsgdHlwZW9mIHBhcnRpY2lwYW50KTtcbiAgICAgICAgaWYgKHBhcnRpY2lwYW50LnR5cGUgIT09IFwiY29tcGxleFwiKSB7XG4gICAgICAgICAgICByZXN1bHQrKztcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHJlc3VsdCArPSBwYXJ0aWNpcGFudC5uYXJ5TGluay5nZXRUb3RhbFBhcnRpY2lwYW50Q291bnQoKTtcbiAgICAgICAgfVxuICAgIH1cbiAgICByZXR1cm4gcmVzdWx0O1xufTtcbiovXG5cbk5hcnlMaW5rLnByb3RvdHlwZS5pbml0U1ZHID0gZnVuY3Rpb24gKCkge1xuICAgIHRoaXMucGF0aCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnROUyhzdmducywgXCJwYXRoXCIpO1xuICAgIHRoaXMuY29sb3IgPSBOYXJ5TGluay5uYXJ5Q29sb3JzKHRoaXMuaWQpO1xuICAgIHRoaXMucGF0aC5zZXRBdHRyaWJ1dGUoXCJmaWxsXCIsIHRoaXMuY29sb3IpO1xuICAgIC8vc2V0IHRoZSBldmVudHMgZm9yIGl0XG4gICAgY29uc3Qgc2VsZiA9IHRoaXM7XG4gICAgdGhpcy5wYXRoLm9ubW91c2Vkb3duID0gZnVuY3Rpb24gKGV2dCkge1xuICAgICAgICBzZWxmLm1vdXNlRG93bihldnQpO1xuICAgIH07XG4gICAgdGhpcy5wYXRoLm9ubW91c2VvdmVyID0gZnVuY3Rpb24gKGV2dCkge1xuICAgICAgICBzZWxmLm1vdXNlT3ZlcihldnQpO1xuICAgIH07XG4gICAgdGhpcy5wYXRoLm9ubW91c2VvdXQgPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgICAgIHNlbGYubW91c2VPdXQoZXZ0KTtcbiAgICB9O1xuICAgIC8vIHRoaXMucGF0aC5vbnRvdWNoc3RhcnQgPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgLy8gICAgIHNlbGYudG91Y2hTdGFydChldnQpO1xuICAgIC8vIH07XG4gICAgLy8gdG9kbyAtIHByb2IgYmV0dGVyIHdheSB0b2RvIHRoaXNcbiAgICB0aGlzLnBhdGgyID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudE5TKHN2Z25zLCBcInBhdGhcIik7XG4gICAgdGhpcy5wYXRoMi5zZXRBdHRyaWJ1dGUoXCJmaWxsXCIsIFwibm9uZVwiKTtcbiAgICAvL3NldCB0aGUgZXZlbnRzIGZvciBpdFxuICAgIHRoaXMucGF0aDIub25tb3VzZWRvd24gPSBmdW5jdGlvbiAoZXZ0KSB7XG4gICAgICAgIHNlbGYubW91c2VEb3duKGV2dCk7XG4gICAgfTtcbiAgICB0aGlzLnBhdGgyLm9ubW91c2VvdmVyID0gZnVuY3Rpb24gKGV2dCkge1xuICAgICAgICBzZWxmLm1vdXNlT3ZlcihldnQpO1xuICAgIH07XG4gICAgdGhpcy5wYXRoMi5vbm1vdXNlb3V0ID0gZnVuY3Rpb24gKGV2dCkge1xuICAgICAgICBzZWxmLm1vdXNlT3V0KGV2dCk7XG4gICAgfTtcbiAgICAvLyB0aGlzLnBhdGgyLm9udG91Y2hzdGFydCA9IGZ1bmN0aW9uIChldnQpIHtcbiAgICAvLyAgICAgc2VsZi50b3VjaFN0YXJ0KGV2dCk7XG4gICAgLy8gfTtcbn07XG5cbk5hcnlMaW5rLnByb3RvdHlwZS5zaG93SGlnaGxpZ2h0ID0gZnVuY3Rpb24gKHNob3cpIHtcbiAgICB0aGlzLmhpZ2hsaWdodFBhcnRpY2lwYW50cyhzaG93KTtcbn07XG5cbk5hcnlMaW5rLnByb3RvdHlwZS5jaGVjayA9IGZ1bmN0aW9uICgpIHtcbiAgICB0aGlzLnNob3coKTtcbiAgICByZXR1cm4gdHJ1ZTtcbn07XG5cbk5hcnlMaW5rLnByb3RvdHlwZS5zaG93ID0gZnVuY3Rpb24gKCkge1xuICAgIC8vIHRoaXMucGF0aC5zZXRBdHRyaWJ1dGUoXCJzdHJva2Utd2lkdGhcIiwgdGhpcy5hcHAueik7XG4gICAgdGhpcy5zZXRMaW5rQ29vcmRpbmF0ZXMoKTtcbiAgICB0aGlzLmFwcC5uYXJ5TGlua3MuYXBwZW5kQ2hpbGQodGhpcy5wYXRoKTtcbiAgICB0aGlzLmFwcC5uYXJ5TGlua3MuYXBwZW5kQ2hpbGQodGhpcy5wYXRoMik7XG59O1xuXG5OYXJ5TGluay5wcm90b3R5cGUuaGlkZSA9IGZ1bmN0aW9uICgpIHtcbn07XG5cbk5hcnlMaW5rLnByb3RvdHlwZS5zZXRMaW5rQ29vcmRpbmF0ZXMgPSBmdW5jdGlvbiAoZG9udFByb3BvZ2F0ZSkge1xuICAgIC8vIFVzZXMgZDMuZ2VvbS5odWxsIHRvIGNhbGN1bGF0ZSBhIGJvdW5kaW5nIHBhdGggYXJvdW5kIGFuIGFycmF5IG9mIHZlcnRpY2VzXG4gICAgY29uc3QgY2FsY3VsYXRlSHVsbFBhdGggPSBmdW5jdGlvbiAodmFsdWVzKSB7XG4gICAgICAgIGNvbnN0IGh1bGxQYXRoID0gZDMuZ2VvbS5odWxsKHZhbHVlcyk7XG4gICAgICAgIHNlbGYuaHVsbCA9IGh1bGxQYXRoOyAvL2hhY2s/XG4gICAgICAgIHJldHVybiBcIk1cIiArIGh1bGxQYXRoLmpvaW4oXCJMXCIpICsgXCJaXCI7XG4gICAgfTtcbiAgICBjb25zdCBzZWxmID0gdGhpczsgLy8gVE9ETzogLSB0aWR5IGhhY2sgYWJvdmU/XG4gICAgdGhpcy5tYXBwZWQgPSB0aGlzLm9yYml0Tm9kZXModGhpcy5nZXRNYXBwZWRDb29yZGluYXRlcygpKTtcbiAgICBjb25zdCBodWxsVmFsdWVzID0gY2FsY3VsYXRlSHVsbFBhdGgodGhpcy5tYXBwZWQpO1xuICAgIGlmIChodWxsVmFsdWVzKSB7XG4gICAgICAgIHRoaXMucGF0aC5zZXRBdHRyaWJ1dGUoXCJkXCIsIGh1bGxWYWx1ZXMpO1xuICAgICAgICB0aGlzLnBhdGgyLnNldEF0dHJpYnV0ZShcImRcIiwgaHVsbFZhbHVlcyk7XG4gICAgfVxuICAgIGlmICh0aGlzLmNvbXBsZXggJiYgIWRvbnRQcm9wb2dhdGUpIHtcbiAgICAgICAgdGhpcy5jb21wbGV4LnNldEFsbExpbmtDb29yZGluYXRlcygpO1xuICAgIH1cbn07XG5cbk5hcnlMaW5rLnByb3RvdHlwZS5nZXRNYXBwZWRDb29yZGluYXRlcyA9IGZ1bmN0aW9uICgpIHtcbiAgICBjb25zdCBwYXJ0aWNpcGFudHMgPSB0aGlzLnBhcnRpY2lwYW50cztcbiAgICBsZXQgbWFwcGVkID0gW107XG4gICAgY29uc3QgaWMgPSBwYXJ0aWNpcGFudHMubGVuZ3RoO1xuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgaWM7IGkrKykge1xuICAgICAgICBjb25zdCBwYXJ0aWNpcGFudCA9IHBhcnRpY2lwYW50c1tpXTtcbiAgICAgICAgaWYgKHBhcnRpY2lwYW50LnR5cGUgPT09IFwiY29tcGxleFwiKSB7XG4gICAgICAgICAgICBtYXBwZWQgPSBtYXBwZWQuY29uY2F0KHRoaXMub3JiaXROb2RlcyhwYXJ0aWNpcGFudC5uYXJ5TGluay5nZXRNYXBwZWRDb29yZGluYXRlcygpKSk7XG4gICAgICAgIH0gZWxzZSBpZiAocGFydGljaXBhbnQuZm9ybSA9PT0gMSkge1xuICAgICAgICAgICAgY29uc3Qgc3RhcnQgPSBwYXJ0aWNpcGFudC5nZXRSZXNpZHVlQ29vcmRpbmF0ZXMoMCk7XG4gICAgICAgICAgICBjb25zdCBlbmQgPSBwYXJ0aWNpcGFudC5nZXRSZXNpZHVlQ29vcmRpbmF0ZXMocGFydGljaXBhbnQuc2l6ZSk7XG4gICAgICAgICAgICBpZiAoIWlzTmFOKHN0YXJ0WzBdKSAmJiAhaXNOYU4oc3RhcnRbMV0pICYmXG4gICAgICAgICAgICAgICAgIWlzTmFOKGVuZFswXSkgJiYgIWlzTmFOKGVuZFsxXSkpIHtcbiAgICAgICAgICAgICAgICBtYXBwZWQucHVzaChzdGFydCk7XG4gICAgICAgICAgICAgICAgbWFwcGVkLnB1c2goZW5kKTtcbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgbWFwcGVkLnB1c2gocGFydGljaXBhbnQuZ2V0UG9zaXRpb24oKSk7XG4gICAgICAgICAgICB9XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICBtYXBwZWQucHVzaChwYXJ0aWNpcGFudC5nZXRQb3NpdGlvbigpKTtcbiAgICAgICAgfVxuICAgIH1cbiAgICByZXR1cm4gbWFwcGVkO1xufTtcblxuLy8nb3JiaXQnIG5vZGVzIC0gc2V2ZXJhbCBub2RlcyBhcm91bmQgcGFydGljaXBhbnQgcG9zaXRpb25zIHRvIGdpdmUgbWFyZ2luXG5OYXJ5TGluay5wcm90b3R5cGUub3JiaXROb2RlcyA9IGZ1bmN0aW9uIChtYXBwZWQpIHtcblxuXG4gICAgY29uc3Qgb3JiaXROb2RlcyA9IFtdO1xuICAgIGNvbnN0IG1jID0gbWFwcGVkLmxlbmd0aDtcbiAgICBmb3IgKGxldCBtaSA9IDA7IG1pIDwgbWM7IG1pKyspIHtcbiAgICAgICAgY29uc3QgbSA9IG1hcHBlZFttaV07XG4gICAgICAgIGZvciAobGV0IG8gPSAwOyBvIDwgTmFyeUxpbmsub3JiaXROb2RlczsgbysrKSB7XG4gICAgICAgICAgICBjb25zdCBhbmdsZSA9ICgzNjAgLyBOYXJ5TGluay5vcmJpdE5vZGVzKSAqIG87XG4gICAgICAgICAgICBjb25zdCBwID0gW21bMF0gKyBOYXJ5TGluay5vcmJpdFJhZGl1cywgbVsxXV07XG4gICAgICAgICAgICBvcmJpdE5vZGVzLnB1c2gocm90YXRlUG9pbnRBYm91dFBvaW50KHAsIG0sIGFuZ2xlKSk7XG4gICAgICAgIH1cbiAgICB9XG4gICAgcmV0dXJuIG9yYml0Tm9kZXM7XG59OyJdLCJtYXBwaW5ncyI6IkFBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./src/js/viz/link/nary-link.js\n"); /***/ }), diff --git a/index.html b/index.html index 3674613..4c024bf 100644 --- a/index.html +++ b/index.html @@ -168,7 +168,7 @@

> 6), + 0x80 | (charcode & 0x3f)); + } else if (charcode < 0xd800 || charcode >= 0xe000) { + array.push(0xe0 | (charcode >> 12), + 0x80 | ((charcode >> 6) & 0x3f), + 0x80 | (charcode & 0x3f)); + } + // surrogate pair + else { + i++; + // UTF-16 encodes 0x10000-0x10FFFF by + // subtracting 0x10000 and splitting the + // 20 bits of 0x0-0xFFFFF into two halves + charcode = 0x10000 + (((charcode & 0x3ff) << 10) | + (binary.charCodeAt(i) & 0x3ff)); + array.push(0xf0 | (charcode >> 18), + 0x80 | ((charcode >> 12) & 0x3f), + 0x80 | ((charcode >> 6) & 0x3f), + 0x80 | (charcode & 0x3f)); + } + } + } + + return new Blob([new Uint8Array(array)], { + type: newContentType + }); + } + + var blob = dataURItoBlob(content); + + if (navigator.msSaveOrOpenBlob) { + navigator.msSaveOrOpenBlob(blob, fileName); + } else { + var a = document.createElement("a"); + a.href = window.URL.createObjectURL(blob); + // Give filename you wish to download + a.download = fileName; + a.style.display = "none"; + document.body.appendChild(a); + a.click(); + document.body.removeChild(a); + window.URL.revokeObjectURL(a.href); // clear up url reference to blob so it can be g.c.'ed + } + + blob = null; } //]]> diff --git a/src/css/xinet.css b/src/css/xinet.css index 68101ca..ec88cd8 100644 --- a/src/css/xinet.css +++ b/src/css/xinet.css @@ -1,13 +1,3 @@ -.protein { - cursor: crosshair; -} - -.link { - /* - cursor: crosshair; - */ -} - /*you need this to stop horrible looking flickering of text as you drag*/ svg { -webkit-user-select: none; @@ -15,21 +5,110 @@ svg { user-select: none; } -.xlv_text { +.highlight { + stroke: #ffff99; +} + +.link { + stroke-linecap: round; + stroke: black; +} + +.certain-link { + opacity: 0.6; + stroke-opacity: 0.6; +} + +.uncertain-link { + opacity: 0.2; + stroke-opacity: 0.6; +} + +.link-line { + stroke-width: 1; +} + +.link-highlight { + stroke-width: 10; + stroke-opacity: 0; +} + +.complex-outline { + stroke: white; + stroke-linejoin: round; + stroke-width: 7; +} + +.linked-complex { + stroke: black; + stroke-linejoin: round; + stroke-width: 1; +} + +feature-link { + fill: black; +} + +.protein { + cursor: crosshair; +} + +/*todo - seperate out outline*/ +.label { + font-size: 10pt; /*font-weight: bold;*/ + + /*color: #fff;*/ + /*text-shadow: white 0px 0px 1px;*/ + -webkit-font-smoothing: antialiased; + + /*text-shadow: #fff 0px 0px 1px, #fff 0px 0px 1px, #fff 0px 0px 1px,*/ + /*#fff 0px 0px 1px, #fff 0px 0px 1px, #fff 0px 0px 1px;*/ + text-shadow: -1px -1px 0 white, 1px -1px 0 white, -1px 1px 0 white, 1px 1px 0 white; - /* -2px -1px 0 white, - 2px -1px 0 white, - -2px 1px 0 white, - 2px 1px 0 white;*/ + /*-2px -1px 0 white,*/ + /* 2px -1px 0 white,*/ + /* -2px 1px 0 white,*/ + /* 2px 1px 0 white;*/ + fill: black; + text-anchor: end; } -.proteinLabel { - font-size: 10pt; - /*font-weight: bold;*/ +.tooltip{ + text-anchor: start; +} + +.outline { + stroke: black; + stroke-width: 1; + stroke-opacity: 1; + fill: white; + fill-opacity: 1; +} + +.participant-highlight { + stroke-width: 5; + fill: none; +} + +/*for protein bar scale*/ +.tick { + stroke: black; +} + +.tick-labels { + font-size: 8pt; + text-anchor: middle; +} + +/*not working right*/ +.sequence { + font-family: 'Courier New', monospace; + font-size: 10px; + text-anchor: middle; } .custom-menu-margin { @@ -43,9 +122,6 @@ svg { overflow: hidden; border: 1px solid #CCC; white-space: nowrap; - /* - font-family: sans-serif; -*/ background: #FFF; color: #333; list-style: none; @@ -70,3 +146,16 @@ svg { display: inline; padding-left: 10px; } + +.tooltip-background { + fill-opacity: 0.75; + stroke-opacity: 1; + stroke-width: 1; +} + +.tooltip-sub-background { + fill: white; + stroke: white; + opacity: 1; + stroke-width: 1; +} \ No newline at end of file diff --git a/src/js/annotations.js b/src/js/annotations.js index d109900..de51838 100644 --- a/src/js/annotations.js +++ b/src/js/annotations.js @@ -20,7 +20,7 @@ export function fetchAnnotations(annotationChoice, /*App*/ app, callback) { let protsAnnotated = 0; const molCount = proteins.length; - if (annotationChoice === "INTERACTOR") { + if (annotationChoice === "INTERACTOR") { //todo - move this out of here if (app.proteinCount < 21) { for (let prot of proteins) { const annotation = new Annotation(prot.json.label, new SequenceDatum(null, 1 + "-" + prot.size)); @@ -33,7 +33,7 @@ export function fetchAnnotations(annotationChoice, /*App*/ app, callback) { } // app.annotationSetsShown.set("INTERACTOR", true); } else { - alert("Too many (> 20) - can't color by interactor."); + // alert("Too many (> 20) - can't color by interactor."); // people are gong to complain about why arent my interactor colours showing up } callback(); } else if (annotationChoice.toUpperCase() === "SUPERFAMILY") { diff --git a/src/js/app.js b/src/js/app.js index 0e4990a..6a4f8a6 100644 --- a/src/js/app.js +++ b/src/js/app.js @@ -6,6 +6,7 @@ import * as d3_chromatic from "d3-scale-chromatic"; import * as cola from "./cola"; import {readMijson} from "./read-mijson"; import {chooseColors, fetchAnnotations} from "./annotations"; +import {svgUtils} from "./svgexp"; // import {SymbolKey} from "./symbol-key"; import * as ColorSchemeKey from "./color-scheme-key"; @@ -131,44 +132,6 @@ export function App(/*HTMLDivElement*/networkDiv) { const svg = d3.select(this.svgElement); this.defs = svg.append("defs"); - this.createHatchedFill("checkers_uncertain", "black"); - - //markers - const data = [{ - id: 1, - name: "diamond", - path: "M 0,-7.0710768 L 0,7.0710589 L 7.0710462,0 z", - viewbox: "-15 -15 25 25", - transform: "scale(1.5) translate(-5,0)", - color: "black" - }]; - - this.defs.selectAll("marker") - .data(data) - .enter() - .append("svg:marker") - .attr("id", function (d) { - return "marker_" + d.name; - }) - .attr("markerHeight", 15) - .attr("markerWidth", 15) - .attr("markerUnits", "userSpaceOnUse") - .attr("orient", "auto") - .attr("refX", 0) - .attr("refY", 0) - .attr("viewBox", function (d) { - return d.viewbox; - }) - .append("svg:path") - .attr("d", function (d) { - return d.path; - }) - .attr("fill", function (d) { - return d.color; - }) - .attr("transform", function (d) { - return d.transform; - }); this.acknowledgement = document.createElementNS(svgns, "g"); const ackText = document.createElementNS(svgns, "text"); @@ -176,7 +139,6 @@ export function App(/*HTMLDivElement*/networkDiv) { + version + "by Rappsilber Laboratory"; this.acknowledgement.appendChild(ackText); - ackText.classList.add("xlv_text"); ackText.setAttribute("font-size", "8pt"); this.svgElement.appendChild(this.acknowledgement); @@ -215,24 +177,16 @@ export function App(/*HTMLDivElement*/networkDiv) { this.tooltip = document.createElementNS(svgns, "text"); this.tooltip.setAttribute("x", "0"); this.tooltip.setAttribute("y", "0"); - this.tooltip.setAttribute("class", "xlv_text"); const tooltipTextNode = document.createTextNode("tooltip"); + this.tooltip.classList.add("label", "tooltip"); this.tooltip.appendChild(tooltipTextNode); this.tooltip_bg = document.createElementNS(svgns, "rect"); - this.tooltip_bg.setAttribute("class", "tooltip_bg"); - - this.tooltip_bg.setAttribute("fill-opacity", "0.75"); - this.tooltip_bg.setAttribute("stroke-opacity", "1"); - this.tooltip_bg.setAttribute("stroke-width", "1"); + this.tooltip_bg.classList.add("tooltip-background"); this.tooltip_subBg = document.createElementNS(svgns, "rect"); - this.tooltip_subBg.setAttribute("fill", "white"); - this.tooltip_subBg.setAttribute("stroke", "white"); - this.tooltip_subBg.setAttribute("class", "tooltip_bg"); - this.tooltip_subBg.setAttribute("opacity", "1"); - this.tooltip_subBg.setAttribute("stroke-width", "1"); + this.tooltip_subBg.classList.add("tooltip-sub-background"); this.svgElement.appendChild(this.tooltip_subBg); this.svgElement.appendChild(this.tooltip_bg); @@ -242,28 +196,32 @@ export function App(/*HTMLDivElement*/networkDiv) { } App.prototype.createHatchedFill = function (name, color) { - const pattern = this.defs.append("pattern") - .attr("id", name) - .attr("patternUnits", "userSpaceOnUse") - .attr("x", 0) - .attr("y", 0) - .attr("width", 12) - .attr("height", 12) - .attr("patternTransform", "rotate(45)"); - - pattern.append("rect") - .attr("x", 0) - .attr("y", 2) - .attr("width", 12) - .attr("height", 4) - .attr("fill", color); - - pattern.append("rect") - .attr("x", 0) - .attr("y", 8) - .attr("width", 12) - .attr("height", 4) - .attr("fill", color); + if (!this.checkedHatchNames.has(name)) { + const pattern = this.defs.append("pattern") + .attr("id", name) + .attr("patternUnits", "userSpaceOnUse") + .attr("x", 0) + .attr("y", 0) + .attr("width", 12) + .attr("height", 12) + .attr("patternTransform", "rotate(45)"); + + pattern.append("rect") + .attr("x", 0) + .attr("y", 2) + .attr("width", 12) + .attr("height", 4) + .attr("fill", color); + + pattern.append("rect") + .attr("x", 0) + .attr("y", 8) + .attr("width", 12) + .attr("height", 4) + .attr("fill", color); + + this.checkedHatchNames.add(name); + } }; App.prototype.clear = function () { @@ -272,6 +230,8 @@ App.prototype.clear = function () { this.annotationSetsShown = new Map(); // this.annotationSetsShown.set("MI FEATURES", true); + this.checkedHatchNames = new Set(); + //lighten colors const complexColors = []; for (let c of d3_chromatic.schemePastel2) {//colorbrewer.Pastel2[8]) { @@ -318,16 +278,10 @@ App.prototype.collapseProtein = function () { App.prototype.init = function () { this.d3cola.stop(); - // let i = 0; for (let participant of this.participants.values()) { if (participant.type != "complex") { - // let pos = rotatePointAboutPoint([0, -500], [0, 0], (360 / this.participants.size * i)); - participant.setPosition(-500, -500);//pos[0], pos[1]); - // if (participant.type === "protein") { - // participant.setPositionalFeatures(); - // } + participant.setPosition(-500, -500); } - // i++; } this.updateAnnotations(); this.checkLinks(); //totally needed, not sure why tbh todo - check this out @@ -739,11 +693,6 @@ App.prototype.autoLayout = function () { } } - // if (preRun) { - // layoutObj.nodes = layoutObj.nodes.concat(prunedOut); - // } - - // self.d3cola.convergenceThreshold = 0.01; //console.log("groups", groups); delete self.d3cola._lastStress; delete self.d3cola._alpha; @@ -789,7 +738,7 @@ App.prototype.autoLayout = function () { const startTime = Date.now(); self.d3cola.symmetricDiffLinkLengths(linkLength) .on("tick", function () { - if (Date.now() - startTime > 400) {//!preRun) { + if (Date.now() - startTime > 750) {//!preRun) { const nodes = self.d3cola.nodes(); for (let node of nodes) { node.setPosition(node.x, node.y); @@ -835,10 +784,6 @@ App.prototype.autoLayout = function () { // // for (let p of layoutObj.nodes) { // // p.fixed = 1; // // } - // // let nodesExceptComplexes = Array.from(self.participants.values()); - // // allNodesExceptComplexes = allNodesExceptComplexes.filter(function (value) { - // // return value.type !== "complex"; - // // }); doLayout(allNodesExceptComplexes, false); } else { for (let node of nodes) { @@ -856,16 +801,29 @@ App.prototype.autoLayout = function () { } }; -App.prototype.getSVG = function () { //todo - update after styling of svg is moved to css - let svgXml = this.svgElement.outerHTML.replace(//i, ""); //take out white background fill - const viewBox = "viewBox=\"0 0 " + this.svgElement.parentNode.clientWidth + " " + this.svgElement.parentNode.clientHeight + "\" "; - svgXml = svgXml.replace("" + - "" + - svgXml; }; +// App.prototype.getSVG = function () { +// let svgXml = this.svgElement.outerHTML.replace(//i, ""); //take out white background fill +// const viewBox = "viewBox=\"0 0 " + this.svgElement.parentNode.clientWidth + " " + this.svgElement.parentNode.clientHeight + "\" "; +// svgXml = svgXml.replace("" + +// "" + +// svgXml; +// }; + // transform the mouse-position into a position on the svg App.prototype.mouseToSVG = function (x, y) { const p = this.svgElement.createSVGPoint(); @@ -880,29 +838,33 @@ App.prototype.readMIJSON = function (miJson, expand = true) { }; App.prototype.checkLinks = function () { - function checkAll(linkMap) { - for (let link of linkMap.values()) { - link.check(); - } + for (let link of this.allNaryLinks.values()) { + link.check(); + } + for (let link of this.allBinaryLinks.values()) { + link.check(); + } + for (let link of this.allUnaryLinks.values()) { + link.check(); + } + for (let link of this.allSequenceLinks.values()) { + link.check(); } - - checkAll(this.allNaryLinks); - checkAll(this.allBinaryLinks); - checkAll(this.allUnaryLinks); - checkAll(this.allSequenceLinks); }; App.prototype.setAllLinkCoordinates = function () { - function setAll(linkMap) { - for (let link of linkMap.values()) { - link.setLinkCoordinates(true); // true means don't propogate changes from naryLink up to complex, everythings getting refreshed anyway - } + for (let link of this.allNaryLinks.values()) { + link.setLinkCoordinates(); + } + for (let link of this.allBinaryLinks.values()) { + link.setLinkCoordinates(); + } + for (let link of this.allUnaryLinks.values()) { + link.setLinkCoordinates(); + } + for (let link of this.allSequenceLinks.values()) { + link.setLinkCoordinates(); } - - setAll(this.allNaryLinks); - setAll(this.allBinaryLinks); - setAll(this.allUnaryLinks); - setAll(this.allSequenceLinks); }; App.prototype.showTooltip = function (p) { @@ -944,6 +906,7 @@ App.prototype.setTooltip = function (text, color) { this.tooltip_bg.setAttribute("fill", "white"); this.tooltip_bg.setAttribute("stroke", "grey"); } + // todo - whats this height for? this.tooltip_bg.setAttribute("height", "28"); this.tooltip_subBg.setAttribute("height", "28"); this.tooltip_bg.setAttribute("display", "block"); @@ -1044,21 +1007,18 @@ App.prototype.expandAll = function () { //from noe App.prototype.expandAndCollapseSelection = function (moleculesSelected) { - const molecules = this.molecules.values(); - for (var m = 0; m < molecules.length; m++) { - var molecule = molecules[m]; - var molecule_id = molecule.json.identifier.id; + for (let participant of this.participants.values()) { + const molecule_id = participant.json.identifier.id; if (moleculesSelected.includes(molecule_id)) { - if (molecule.form === 0) { - molecule.setForm(1); + if (participant.form === 0) { + participant.setForm(1); } - } else if (molecule.form === 1) { - molecule.setForm(0); + } else if (participant.form === 1) { + participant.setForm(0); } } }; - // export function makeSymbolKey(targetDiv){ // new SymbolKey(targetDiv); -// } +// } \ No newline at end of file diff --git a/src/js/color-scheme-key.js b/src/js/color-scheme-key.js index 4dc7b9b..b4a0e4c 100644 --- a/src/js/color-scheme-key.js +++ b/src/js/color-scheme-key.js @@ -16,7 +16,7 @@ export function update(/*HTMLDivElement*/ div, /*App*/app) { tc1.style.backgroundColor = ccRange[i % 6]; const tc2 = tr.insertCell(); tc2.textContent = ccDomain[i]; - console.log(i + " " + ccDomain[i] + " " + ccRange[i]); + //console.log(i + " " + ccDomain[i] + " " + ccRange[i]); } div.appendChild(complexColorTable); diff --git a/src/js/config.js b/src/js/config.js index 7f5815c..1eef32c 100644 --- a/src/js/config.js +++ b/src/js/config.js @@ -1,8 +1,6 @@ export const svgns = "http://www.w3.org/2000/svg";//, // namespace for svg elements // xlinkNS: 'http://www.w3.org/1999/xlink', // namespace for xlink, for use/defs elements -export const highlightColour = "#ffff99"; //, //"#fdc086"); // todo use css - export const LABEL_Y = -5; // todo this isn't needed // selectedColour: '#ffff99', // diff --git a/src/js/svgexp.js b/src/js/svgexp.js new file mode 100644 index 0000000..8ca2827 --- /dev/null +++ b/src/js/svgexp.js @@ -0,0 +1,268 @@ +/** + * Created by cs22 on 04/12/14. + */ + +export const svgUtils = { + + capture: function (svgElems) { + return svgElems.map (function(svg) { return svgUtils.makeSVGDoc (svg); }); + }, + + getAllSVGElements: function () { + // search through all document objects, including those in iframes + var allIFrames = [].slice.apply (document.getElementsByTagName('iframe')); + var docs = [document]; + allIFrames.forEach (function (iframe) { + try { + docs.push (iframe.contentDocument || iframe.contentWindow.document); + } + catch (e) { + console.log ("Protected cross-domain IFrame", iframe); + } + }); + + var allSvgs = []; + docs.forEach (function(doc) { + var allDocSvgs = [].slice.apply (doc.getElementsByTagName('svg')); + allSvgs.push.apply (allSvgs, allDocSvgs); + }); + return allSvgs; + }, + + + makeSVGDoc: function (svgElem) { + // clone node + var cloneSVG = svgElem.cloneNode (true); + var ownerDoc = cloneSVG.ownerDocument || document; + svgUtils.pruneInvisibleSubtrees (cloneSVG, svgElem); + + // find all styles inherited/referenced at or below this node + var styles = svgUtils.usedStyles (svgElem, true, true); + + // collect relevant info on parent chain of svg node + var predecessorInfo = svgUtils.parentChain (svgElem, styles); + + var addDummy = function (dummySVGElem, cloneSVG, origSVG, transferAttr) { + dummySVGElem.appendChild (cloneSVG); + Object.keys(transferAttr).forEach (function (attr) { + var val = cloneSVG.getAttribute (attr) || cloneSVG.style [attr] || svgUtils.getComputedStyleCssText (origSVG, attr); + if (val != null) { + dummySVGElem.setAttribute (attr, val); + var attrVal = transferAttr[attr]; + if (attrVal.replace) { + cloneSVG.setAttribute (attr, attrVal.replace); + } else if (attrVal.delete) { + cloneSVG.removeAttribute (attr); + } + } + }); + }; + + // make a chain of dummy svg nodes to include classes / ids of parent chain of our original svg + // this means any styles referenced within the svg that depend on the presence of these classes/ids are fired + var transferAttr = {width: {replace: "100%"}, height: {replace: "100%"}, xmlns: {delete: true}}; + var parentAdded = false; + for (var p = 0; p < predecessorInfo.length; p++) { + var pinf = predecessorInfo [p]; + //var dummySVGElem = ownerDoc.createElement ("svg"); + var dummySVGElem = ownerDoc.createElementNS ("http://www.w3.org/2000/svg", "svg"); + var empty = true; + Object.keys(pinf).forEach (function (key) { + if (pinf[key]) { + dummySVGElem.setAttribute (key, pinf[key]); + empty = false; + } + }); + // If the dummy svg has no relevant id, classes or computed style then ignore it, otherwise make it the new root + if (!empty) { + addDummy (dummySVGElem, cloneSVG, svgElem, transferAttr); + cloneSVG = dummySVGElem; + parentAdded = true; + } + } + + // if no dummy parent added in previous section, but our svg isn't root then add one as placeholder + if (svgElem.parentNode != null && !parentAdded) { + var dummySVGElem = ownerDoc.createElementNS ("http://www.w3.org/2000/svg", "svg"); + addDummy (dummySVGElem, cloneSVG, svgElem, transferAttr); + cloneSVG = dummySVGElem; + parentAdded = true; + } + + // Copy svg's computed style (it's style context) if a dummy parent node has been introduced + if (parentAdded) { + cloneSVG.setAttribute ("style", svgUtils.getComputedStyleCssText (svgElem)); + } + + cloneSVG.setAttribute ("version", "1.1"); + //cloneSVG.setAttribute ("xmlns", "http://www.w3.org/2000/svg"); // XMLSerializer does this + //cloneSVG.setAttribute ("xmlns:xlink", "http://www.w3.org/1999/xlink"); // when I used setAttributeNS it ballsed up + // however using these attributeNS calls work, and stops errors in IE11. Win. + cloneSVG.setAttributeNS ("http://www.w3.org/2000/xmlns/", "xmlns", "http://www.w3.org/2000/svg"); // XMLSerializer does this + cloneSVG.setAttributeNS ("http://www.w3.org/2000/xmlns/", "xmlns:xlink", "http://www.w3.org/1999/xlink"); // when I used setAttributeNS it ballsed up + + + var styleElem = ownerDoc.createElement ("style"); + styleElem.setAttribute ("type", "text/css"); + var styleText = ownerDoc.createTextNode (styles.join("\n")); + styleElem.appendChild (styleText); + cloneSVG.insertBefore (styleElem, cloneSVG.firstChild); + + return cloneSVG; + }, + + // Because firefox returns cssText as empty + // https://bugzilla.mozilla.org/show_bug.cgi?id=137687 + getComputedStyleCssText: function (element, field) { + var style = window.getComputedStyle(element); + if (field) { + return style[field]; + } + + if (style.cssText != "") { + return style.cssText; + } + + var cssText = ""; + for (var i = 0; i < style.length; i++) { + var styleName = style[i]; + var propVal = style.getPropertyValue(styleName); + cssText += styleName + ": " + propVal + "; "; + } + + return cssText; + }, + + doPruneInvisible: true, + + pruneConditionSets: [{"display": "none"}, {"visibility": "hidden"}, {"opacity": "0"}, {"fill-opacity": "0", "stroke-opacity": "0"}, {"fill-opacity": "0", "stroke": "none"}, {"fill": "none", "stroke-opacity": "0"}], + + pruneInvisibleSubtrees: function (clonedElement, matchingOriginalElement) { + if (svgUtils.doPruneInvisible) { + var style = window.getComputedStyle (matchingOriginalElement); // cloned (unattached) nodes in chrome at least don't have computed styles + var prune = false; + + svgUtils.pruneConditionSets.forEach (function (conditionSet) { + if (!prune) { + var allConditionsMet = true; + Object.keys(conditionSet).forEach (function (condition) { + var condVal = conditionSet[condition]; + var eStyle = style[condition]; + var eAttr = matchingOriginalElement.getAttribute(condition); + if (!(eStyle === condVal || (!eStyle && eAttr === condVal))) { + allConditionsMet = false; + } + }); + prune = allConditionsMet; + } + }); + if (prune && clonedElement.parentNode) { + clonedElement.parentNode.removeChild (clonedElement); + //console.log ("removed", clonedElement); + } else { + var clonedChildren = clonedElement.children; + var matchingOriginalChildren = matchingOriginalElement.children; + //console.log ("kept", clonedElement, style.display, style.visibility, style.opacity, style["stroke-opacity"], style["fill-opacity"], style); + //console.log (element, "children", children); + if (clonedChildren && clonedChildren.length) { + // count backwards because removing a child will break the 'i' counter if we go forwards + // e.g. if children=[A,B,C,D] and i=2, if we delete[C] then children becomes [A,B,D], + // and when i then increments to 3, expecting D, instead we find the end of loop, and don't test D + // PS. And if we fixed that we'd then need a separate counter for the original child elements anyways so backwards it is + for (var i = clonedChildren.length; --i >= 0;) { + svgUtils.pruneInvisibleSubtrees (clonedChildren[i], matchingOriginalChildren[i]); + } + } + } + } + }, + + parentChain: function (elem, styles) { + // Capture id / classes of svg's parent chain. + var ownerDoc = elem.ownerDocument || document; + var elemArr = []; + while (elem.parentNode !== ownerDoc && elem.parentNode !== null) { + elem = elem.parentNode; + elemArr.push ({id: elem.id, class: elem.getAttribute("class") || ""}); + } + + // see if id or element class are referenced in any styles collected below the svg node + // if not, null the id / class as they're not going to be relevant + elemArr.forEach (function (elemData) { + var presences = {id: false, class: false}; + var classes = elemData.class.split(" ").filter(function(a) { return a.length > 0; }); // v1.13: may be multiple classes in a containing class attribute + styles.forEach (function (style) { + for (var c = 0; c < classes.length; c++) { + if (style.indexOf ("."+classes[c]) >= 0) { + presences.class = true; + break; // no need to keep looking through rest of classtypes if one is needed + } + } + if (elemData.id && style.indexOf ("#"+elemData.id) >= 0) { + presences.id = true; + } + }); + Object.keys(presences).forEach (function (presence) { + if (!presences[presence]) { elemData[presence] = undefined; } + }); + }); + + return elemArr; + }, + + // code adapted from user adardesign's answer in http://stackoverflow.com/questions/13204785/is-it-possible-to-read-the-styles-of-css-classes-not-being-used-in-the-dom-using + usedStyles: function (elem, subtree, both) { + var needed = [], rule; + var ownerDoc = elem.ownerDocument || document; + var CSSSheets = ownerDoc.styleSheets; + + for(var j=0; j < CSSSheets.length; j++){ + // stop accessing empty style sheets (1.15), catch security exceptions (1.20) + try{ + if (CSSSheets[j].cssRules == null) { + continue; + } + } catch (err) { + continue; + } + + for(var i=0; i < CSSSheets[j].cssRules.length; i++){ + rule = CSSSheets[j].cssRules[i]; + var match = false; + // Issue reported, css rule '[ng:cloak], [ng-cloak], [data-ng-cloak], [x-ng-cloak], .ng-cloak, .x-ng-cloak, .ng-hide:not(.ng-hide-animate)' gives error + // It's the [ng:cloak] bit that does the damage + // Fix found from https://github.com/exupero/saveSvgAsPng/issues/11 - but the css rule isn't applied + try { + if (subtree) { + match = elem.querySelectorAll(rule.selectorText).length > 0; + } + if (!subtree || both) { + match |= elem.matches(rule.selectorText); + } + } + catch (err) { + console.warn ("CSS selector error: "+rule.selectorText+". Often angular issue.", err); + } + if (match) { needed.push (rule.cssText); } + } + } + + return needed; + }, + + makeXMLStr: function (xmls, svgDoc) { + var xmlStr = xmls.serializeToString(svgDoc); + // serializing adds an xmlns attribute to the style element ('cos it thinks we want xhtml), which knackers it for inkscape, here we chop it out + xmlStr = xmlStr.split("xmlns=\"http://www.w3.org/1999/xhtml\"").join(""); + return xmlStr; + }, + + // saveSVGDocs: function (svgDocs) { + // var xmls = new XMLSerializer(); + // svgDocs.forEach (function (svgDoc, i) { + // var xmlStr = svgUtils.makeXMLStr (xmls, svgDoc); + // var blob = new Blob([xmlStr], {type: "image/svg+xml"}); + // saveAs(blob, "saved"+i+".svg"); + // }); + // }, +}; \ No newline at end of file diff --git a/src/js/viz/interactor/annotation.js b/src/js/viz/interactor/annotation.js index 90f8413..05cd5d3 100644 --- a/src/js/viz/interactor/annotation.js +++ b/src/js/viz/interactor/annotation.js @@ -1,6 +1,6 @@ //constructor for annotations export function Annotation(annotationName, seqDatum) { - console.log("**", annotationName, seqDatum); + // console.log("**", annotationName, seqDatum); this.description = annotationName.trim(); this.seqDatum = seqDatum; } diff --git a/src/js/viz/interactor/complex.js b/src/js/viz/interactor/complex.js index 387895b..72f875f 100644 --- a/src/js/viz/interactor/complex.js +++ b/src/js/viz/interactor/complex.js @@ -5,7 +5,7 @@ import * as Intersection from "intersectionjs"; export function Complex(id, app) { this.init(id, app); this.type = "complex"; - this.padding = 20; + this.padding = 22; // const self = this; // // its bad if you end up with these getting called @@ -25,22 +25,17 @@ export function Complex(id, app) { Complex.prototype = new Interactor(); -// Complex.prototype = { -// get width() { -// return this.naryLink.path.getBBox().width; -// }, -// get height() { -// return this.naryLink.path.getBBox().height; -// }, -// }; - Complex.prototype.initLink = function (naryLink) { this.naryLink = naryLink; - naryLink.path.setAttribute("stroke", "white"); - naryLink.path.setAttribute("stroke-linejoin", "round"); - naryLink.path.setAttribute("stroke-width", 4); + this.naryLink.path.classList.add("complex-outline"); }; +Complex.prototype.setLinked = function () { + + this.naryLink.path2.classList.add("linked-complex"); +}; + + Complex.prototype.getPosition = function (originPoint) { const mapped = this.naryLink.mapped;//getMappedCoordinates(); const mc = mapped.length; @@ -65,6 +60,7 @@ Complex.prototype.getPosition = function (originPoint) { if (intersect.points[0]) { return [intersect.points[0].x, intersect.points[0].y]; } + this.setLinked(); } return center; }; diff --git a/src/js/viz/interactor/interactor.js b/src/js/viz/interactor/interactor.js index df81e64..81aa5af 100644 --- a/src/js/viz/interactor/interactor.js +++ b/src/js/viz/interactor/interactor.js @@ -19,9 +19,10 @@ Interactor.prototype.init = function (id, app, json, name){ this.json = json; this.name = name; - //annotations indexed by annotation set name ("MI FEATURES", "SUPERFAMILY", etc) - this.annotationSets = new Map(); + //todo - think 'type' should be a property here (except for complex, can just return json.type.name) + //annotations indexed by annotation set name ("MIFEATURES", "SUPERFAMILY", etc) + this.annotationSets = new Map(); //links this.naryLinks = new Map(); this.binaryLinks = new Map(); @@ -29,16 +30,10 @@ Interactor.prototype.init = function (id, app, json, name){ }; Interactor.prototype.initLabel = function (){ - this.labelSVG = document.createElementNS(svgns, "text"); - this.labelSVG.setAttribute("text-anchor", "end"); - this.labelSVG.setAttribute("fill", "black"); - this.labelSVG.setAttribute("x", "0"); + this.labelSVG.setAttribute("x", "0"); // css? this.labelSVG.setAttribute("y", "10"); - this.labelSVG.setAttribute("class", "xlv_text proteinLabel"); - this.labelSVG.setAttribute("font-family", "Arial"); - this.labelSVG.setAttribute("font-size", "16"); - + this.labelSVG.classList.add("label"); //choose label text if (this.name) { this.labelText = this.name; @@ -48,7 +43,6 @@ Interactor.prototype.initLabel = function (){ if (this.labelText.length > 25) { this.labelText = this.labelText.substr(0, 16) + "..."; } - this.labelText = this.name; this.labelTextNode = document.createTextNode(this.labelText); this.labelSVG.appendChild(this.labelTextNode); @@ -58,12 +52,7 @@ Interactor.prototype.initLabel = function (){ }; Interactor.prototype.initOutline = function (){ - this.outline.setAttribute("stroke", "black"); - this.outline.setAttribute("stroke-width", "1"); - this.outline.setAttribute("stroke-opacity", "1"); - this.outline.setAttribute("fill-opacity", "1"); - this.outline.setAttribute("fill", "#ffffff"); - //append outline + this.outline.classList.add("outline"); this.upperGroup.appendChild(this.outline); }; @@ -227,6 +216,3 @@ export function trig (radius, angleDegrees) { y: (radius * Math.sin(radians)) }; } -// -// Interactor.prototype.setForm = function () { -// }; diff --git a/src/js/viz/interactor/molecule-set.js b/src/js/viz/interactor/molecule-set.js index d20068c..0c70a59 100644 --- a/src/js/viz/interactor/molecule-set.js +++ b/src/js/viz/interactor/molecule-set.js @@ -12,6 +12,7 @@ export function MoleculeSet(id, app, json, name) { this.outline.setAttribute("height", "20"); this.outline.setAttribute("rx", "5"); this.outline.setAttribute("ry", "5"); + //todo - css... (initOutline hasn't been called so it doesn't have outlin in its classList) this.outline.setAttribute("stroke", "black"); this.outline.setAttribute("stroke-width", "4"); this.outline.setAttribute("stroke-opacity", "1"); diff --git a/src/js/viz/interactor/polymer.js b/src/js/viz/interactor/polymer.js index 0e5486f..5ef3247 100644 --- a/src/js/viz/interactor/polymer.js +++ b/src/js/viz/interactor/polymer.js @@ -141,9 +141,8 @@ Polymer.prototype.setScaleGroup = function () { const seqLabelGroup = document.createElementNS(svgns, "g"); seqLabelGroup.setAttribute("transform", "translate(" + this.getResXwithStickZoom(res) + " " + 0 + ")"); const seqLabel = document.createElementNS(svgns, "text"); - seqLabel.setAttribute("font-family", "'Courier New', monospace"); - seqLabel.setAttribute("font-size", "10px"); - seqLabel.setAttribute("text-anchor", "middle"); + seqLabel.classList.add("label", "sequence"); + //css? seqLabel.setAttribute("x", "0"); seqLabel.setAttribute("y", "3"); seqLabel.appendChild(document.createTextNode(this.sequence[res - 1])); @@ -161,10 +160,7 @@ Polymer.prototype.setScaleGroup = function () { const scaleLabelGroup = document.createElementNS(svgns, "g"); scaleLabelGroup.setAttribute("transform", "translate(" + tickX + " " + 0 + ")"); const scaleLabel = document.createElementNS(svgns, "text"); - scaleLabel.setAttribute("class", "xlv_text"); - // scaleLabel.setAttribute("font-family", "'Courier New', monospace"); - scaleLabel.setAttribute("font-size", "8pt"); // todo css... - scaleLabel.setAttribute("text-anchor", "middle"); + scaleLabel.classList.add("label", "scale-label"); scaleLabel.setAttribute("x", "0"); scaleLabel.setAttribute("y", Polymer.STICKHEIGHT + 4); scaleLabel.appendChild(document.createTextNode(text)); @@ -175,11 +171,11 @@ Polymer.prototype.setScaleGroup = function () { function tickAt(self, tickX) { const tick = document.createElementNS(svgns, "line"); + tick.classList.add("tick"); tick.setAttribute("x1", tickX); tick.setAttribute("y1", "5"); tick.setAttribute("x2", tickX); tick.setAttribute("y2", "10"); - tick.setAttribute("stroke", "black"); self.ticks.appendChild(tick); } }; @@ -712,7 +708,7 @@ Polymer.prototype.getAnnotationPieSliceArcPath = function (startRes, endRes, ann const path = "M" + p1[0] + "," + p1[1] + " L" + p2[0] + "," + p2[1] + " A" + top + "," + top + " 0 " + largeArch + " 1 " + p3[0] + "," + p3[1] + " L" + p4[0] + "," + p4[1] + " A" + bottom + "," + bottom + " 0 " + largeArch + " 0 " + p1[0] + "," + p1[1] + " Z"; - console.log("**", path); + // console.log("**", path); return path; // return "M0,0 L" + arcStart.x + "," + arcStart.y + " A" + radius + "," + // radius + " 0 " + largeArch + " 1 " + arcEnd.x + "," + arcEnd.y + " Z"; diff --git a/src/js/viz/interactor/protein.js b/src/js/viz/interactor/protein.js index fafff5b..d12566f 100644 --- a/src/js/viz/interactor/protein.js +++ b/src/js/viz/interactor/protein.js @@ -1,5 +1,5 @@ import {Polymer} from "./polymer"; -import {svgns, highlightColour} from "../../config"; +import {svgns} from "../../config"; export function Protein(id, /*App*/ app, json, name) { this.init(id, app, json, name); @@ -12,9 +12,7 @@ export function Protein(id, /*App*/ app, json, name) { //make highlight this.highlight = document.createElementNS(svgns, "rect"); - this.highlight.setAttribute("stroke", highlightColour); - this.highlight.setAttribute("stroke-width", "5"); - this.highlight.setAttribute("fill", "none"); + this.highlight.classList.add("highlight", "participant-highlight"); this.upperGroup.appendChild(this.highlight); //make background @@ -33,6 +31,7 @@ export function Protein(id, /*App*/ app, json, name) { //make outline this.outline = document.createElementNS(svgns, "rect"); + // css... this.outline.setAttribute("stroke", "black"); this.outline.setAttribute("stroke-width", "1"); this.outline.setAttribute("fill", "none"); diff --git a/src/js/viz/link/feature-link.js b/src/js/viz/link/feature-link.js index 4a24f3b..f9ff631 100644 --- a/src/js/viz/link/feature-link.js +++ b/src/js/viz/link/feature-link.js @@ -21,26 +21,8 @@ FeatureLink.prototype.init = function (id, fromFeatPos, toFeatPos, app) { this.glyph = document.createElementNS(svgns, "path"); this.uncertainGlyph = document.createElementNS(svgns, "path"); - // this.highlightGlyph = document.createElementNS(svgns, "path"); - this.glyph.setAttribute("stroke-linecap", "round"); - this.uncertainGlyph.setAttribute("stroke-linecap", "round"); - // this.highlightGlyph.setAttribute("stroke-linecap", "round"); - this.glyph.setAttribute("class", "link"); - this.glyph.setAttribute("fill", "black");//"#E08214"); - this.glyph.setAttribute("opacity", "0.6"); - this.glyph.setAttribute("stroke", "black");//""#A08214");// // TODO: will look better with this line partly removed - this.glyph.setAttribute("stroke-opacity", "0.6"); - this.glyph.setAttribute("stroke-width", "1"); - this.uncertainGlyph.setAttribute("class", "link"); - this.uncertainGlyph.setAttribute("fill", "black");//url('#checkers_uncertain')");//"#A01284"); - this.uncertainGlyph.setAttribute("stroke", "black");//"none");//"#A01284"); - this.uncertainGlyph.setAttribute("stroke-opacity", "0.2"); - this.uncertainGlyph.setAttribute("fill-opacity", "0.2"); - // this.highlightGlyph.setAttribute("class", "link"); - // this.highlightGlyph.setAttribute("fill", "none"); - // this.highlightGlyph.setAttribute("stroke", highlightColour); - // this.highlightGlyph.setAttribute("stroke-width", "10"); - // this.highlightGlyph.setAttribute("stroke-opacity", "0"); + this.glyph.classList.add("link", "feature-link", "certain-link"); + this.uncertainGlyph.classList.add("link", "feature-link", "uncertain-link"); //set the events for it const self = this; @@ -271,7 +253,7 @@ FeatureLink.prototype.setLinkCoordinates = function () { // let highlightGlyphPath = "M" + triPointMid[0] + "," + triPointMid[1]; for (let f = 0; f < fSDCount; f++) { seqDatum = this.fromSequenceData[f]; - if (isNumber(seqDatum.begin) && isNumber(seqDatum.end)) { + if (isNumber(seqDatum.begin) && isNumber(seqDatum.end) || fromParticipant.type === "complex") { glyphPath += getSegment(triPointMid, ftMid, seqDatum.begin, seqDatum.end, fromParticipant, fyOffset, toOriginPoint); } // highlightStartRes = seqDatum.begin; @@ -291,7 +273,7 @@ FeatureLink.prototype.setLinkCoordinates = function () { } for (let t = 0; t < tSDCount; t++) { seqDatum = this.toSequenceData[t]; - if (isNumber(seqDatum.begin) && isNumber(seqDatum.end)) { + if (isNumber(seqDatum.begin) && isNumber(seqDatum.end) || toParticipant.type === "complex") { glyphPath += getSegment(triPointMid, ttMid, seqDatum.begin, seqDatum.end, toParticipant, tyOffset, fromOriginPoint); } // highlightStartRes = seqDatum.begin; diff --git a/src/js/viz/link/link.js b/src/js/viz/link/link.js index fdaafb3..fe6b36c 100644 --- a/src/js/viz/link/link.js +++ b/src/js/viz/link/link.js @@ -8,17 +8,8 @@ Link.prototype.highlightParticipants = function (show) { }; Link.prototype.initSVG = function () { - this.line.setAttribute("class", "link"); - this.line.setAttribute("fill", "none"); - this.line.setAttribute("stroke", "black"); - this.line.setAttribute("stroke-width", "1"); - this.line.setAttribute("stroke-linecap", "round"); - this.highlightLine.setAttribute("class", "link"); - this.highlightLine.setAttribute("fill", "none"); - this.highlightLine.setAttribute("stroke", highlightColour); - this.highlightLine.setAttribute("stroke-width", "10"); - this.highlightLine.setAttribute("stroke-linecap", "round"); - this.highlightLine.setAttribute("stroke-opacity", "0"); + this.line.classList.add("link","link-line");//, "certain-link"); + this.highlightLine.classList.add("link", "highlight", "link-highlight"); //set the events for it const self = this; this.line.onmousedown = function (evt) { diff --git a/src/js/viz/link/nary-link.js b/src/js/viz/link/nary-link.js index 80c8de8..6df479a 100644 --- a/src/js/viz/link/nary-link.js +++ b/src/js/viz/link/nary-link.js @@ -4,7 +4,7 @@ import {svgns, rotatePointAboutPoint} from "../../config"; //NaryLink.naryColors; // init'ed in clear function of util NaryLink.orbitNodes = 20; -NaryLink.orbitRadius = 20; +NaryLink.orbitRadius = 22; export function NaryLink(id, app) { this.id = id; @@ -54,6 +54,22 @@ NaryLink.prototype.initSVG = function () { // this.path.ontouchstart = function (evt) { // self.touchStart(evt); // }; + // todo - prob better way todo this + this.path2 = document.createElementNS(svgns, "path"); + this.path2.setAttribute("fill", "none"); + //set the events for it + this.path2.onmousedown = function (evt) { + self.mouseDown(evt); + }; + this.path2.onmouseover = function (evt) { + self.mouseOver(evt); + }; + this.path2.onmouseout = function (evt) { + self.mouseOut(evt); + }; + // this.path2.ontouchstart = function (evt) { + // self.touchStart(evt); + // }; }; NaryLink.prototype.showHighlight = function (show) { @@ -69,6 +85,7 @@ NaryLink.prototype.show = function () { // this.path.setAttribute("stroke-width", this.app.z); this.setLinkCoordinates(); this.app.naryLinks.appendChild(this.path); + this.app.naryLinks.appendChild(this.path2); }; NaryLink.prototype.hide = function () { @@ -86,6 +103,7 @@ NaryLink.prototype.setLinkCoordinates = function (dontPropogate) { const hullValues = calculateHullPath(this.mapped); if (hullValues) { this.path.setAttribute("d", hullValues); + this.path2.setAttribute("d", hullValues); } if (this.complex && !dontPropogate) { this.complex.setAllLinkCoordinates();