diff --git a/dist/smiles-drawer.js b/dist/smiles-drawer.js index 81965bb5..3dd1f0bc 100644 --- a/dist/smiles-drawer.js +++ b/dist/smiles-drawer.js @@ -1983,6 +1983,12 @@ class Drawer { fontSizeLarge: 5, fontSizeSmall: 3, padding: 20.0, + experimentalSSSR: false, + kkThreshold: 0.1, + kkInnerThreshold: 0.1, + kkMaxIteration: 20000, + kkMaxInnerIteration: 50, + kkMaxEnergy: 1e9, themes: { dark: { C: '#fff', @@ -2194,6 +2200,12 @@ class Drawer { this.drawEdges(this.opts.debug); this.drawVertices(this.opts.debug); this.canvasWrapper.reset(); + + if (this.opts.debug) { + console.log(this.graph); + console.log(this.rings); + console.log(this.ringConnections); + } } } /** @@ -2544,7 +2556,7 @@ class Drawer { } // Get the rings in the graph (the SSSR) - let rings = SSSR.getRings(this.graph); + let rings = SSSR.getRings(this.graph, this.opts.experimentalSSSR); if (rings === null) { return; @@ -2759,6 +2771,7 @@ class Drawer { let ring = new Ring([...ringMembers]); + this.addRing(ring); ring.isBridged = true; ring.neighbours = [...neighbours]; @@ -2766,8 +2779,6 @@ class Drawer { ring.rings.push(this.getRing(ringIds[i]).clone()); } - this.addRing(ring); - for (var i = 0; i < ring.members.length; i++) { this.graph.vertices[ring.members[i]].value.bridgedRing = ring.id; } // Atoms inside the ring are no longer part of a ring but are now @@ -3594,7 +3605,7 @@ class Drawer { if (ring.isBridged) { - this.graph.kkLayout(ring.members.slice(), center, startVertex.id, ring, this.opts.bondLength); + this.graph.kkLayout(ring.members.slice(), center, startVertex.id, ring, this.opts.bondLength, this.opts.kkThreshold, this.opts.kkInnerThreshold, this.opts.kkMaxIteration, this.opts.kkMaxInnerIteration, this.opts.kkMaxEnergy); ring.positioned = true; // Update the center of the bridged ring this.setRingCenter(ring); @@ -5488,6 +5499,7 @@ class Graph { } /** * Positiones the (sub)graph using Kamada and Kawais algorithm for drawing general undirected graphs. https://pdfs.semanticscholar.org/b8d3/bca50ccc573c5cb99f7d201e8acce6618f04.pdf + * There are undocumented layout parameters. They are undocumented for a reason, so be very careful. * * @param {Number[]} vertexIds An array containing vertexIds to be placed using the force based layout. * @param {Vector2} center The center of the layout. @@ -5496,7 +5508,7 @@ class Graph { */ - kkLayout(vertexIds, center, startVertexId, ring, bondLength) { + kkLayout(vertexIds, center, startVertexId, ring, bondLength, threshold = 0.1, innerThreshold = 0.1, maxIteration = 2000, maxInnerIteration = 50, maxEnergy = 1e9) { let edgeStrength = bondLength; // Add vertices that are directly connected to the ring var i = vertexIds.length; @@ -5698,15 +5710,9 @@ class Graph { arrEnergySumX[index] = dEX; arrEnergySumY[index] = dEY; - }; // Setting parameters + }; // Setting up variables for the while loops - let threshold = 0.1; - let innerThreshold = 0.1; - let maxIteration = 2000; - let maxInnerIteration = 50; - let maxEnergy = 1e9; // Setting up variables for the while loops - let maxEnergyId = 0; let dEX = 0.0; let dEY = 0.0; @@ -8643,9 +8649,10 @@ class SSSR { * Returns an array containing arrays, each representing a ring from the smallest set of smallest rings in the graph. * * @param {Graph} graph A Graph object. + * @param {Boolean} [experimental=false] Whether or not to use experimental SSSR. * @returns {Array[]} An array containing arrays, each representing a ring from the smallest set of smallest rings in the group. */ - static getRings(graph) { + static getRings(graph, experimental = false) { let adjacencyMatrix = graph.getComponentsAdjacencyMatrix(); if (adjacencyMatrix.length === 0) { @@ -8702,6 +8709,10 @@ class SSSR { continue; } + if (experimental) { + nSssr = 999; + } + let { d, pe, @@ -8721,8 +8732,12 @@ class SSSR { rings.push(ring); } - } + } // So, for some reason, this would return three rings for C1CCCC2CC1CCCC2, which is wrong + // As I don't have time to fix this properly, it will stay in. I'm sorry next person who works + // on it. At that point it might be best to reimplement the whole SSSR thing... + + rings.pop(); return rings; } /** diff --git a/dist/smiles-drawer.min.js b/dist/smiles-drawer.min.js index cc5767d0..0c4f4814 100644 --- a/dist/smiles-drawer.min.js +++ b/dist/smiles-drawer.min.js @@ -1,2 +1,2 @@ -(function(){function s(g,e,n){function t(o,i){if(!e[o]){if(!g[o]){var d="function"==typeof require&&require;if(!i&&d)return d(o,!0);if(r)return r(o,!0);var l=new Error("Cannot find module '"+o+"'");throw l.code="MODULE_NOT_FOUND",l}var a=e[o]={exports:{}};g[o][0].call(a.exports,function(e){var i=g[o][1][e];return t(i||e)},a,a.exports,s,g,e,n)}return e[o].exports}for(var r="function"==typeof require&&require,o=0;o>>0,o=arguments[1],a=o>>0,s=0>a?n(r+a,0):i(a,r),g=arguments[2],d=void 0===g?r:g>>0,l=0>d?n(r+d,0):i(d,r);si.x&&(o=i.x),a>i.y&&(a=i.y)}var g=this.opts.padding;t+=g,r+=g,o-=g,a-=g,this.drawingWidth=t-o,this.drawingHeight=r-a;var d=this.canvas.offsetWidth/this.drawingWidth,l=this.canvas.offsetHeight/this.drawingHeight,h=di&&(n.globalAlpha=i),n.strokeStyle=h,n.stroke(),n.restore()}drawWedge(e,i=1){if(isNaN(e.from.x)||isNaN(e.from.y)||isNaN(e.to.x)||isNaN(e.to.y))return;let n=this.ctx,o=this.offsetX,s=this.offsetY,g=e.clone().shorten(5),d=g.getLeftVector().clone(),h=g.getRightVector().clone();d.x+=o,d.y+=s,h.x+=o,h.y+=s,d=e.getLeftVector().clone(),h=e.getRightVector().clone(),d.x+=o,d.y+=s,h.x+=o,h.y+=s,n.save();let c=a.normals(d,h);c[0].normalize(),c[1].normalize();let p=e.getRightChiral(),f=d,m=h;p&&(f=h,m=d);let b=a.add(f,a.multiplyScalar(c[0],this.halfBondThickness)),t=a.add(m,a.multiplyScalar(c[0],1.5+this.halfBondThickness)),u=a.add(m,a.multiplyScalar(c[1],1.5+this.halfBondThickness)),v=a.add(f,a.multiplyScalar(c[1],this.halfBondThickness));n.beginPath(),n.moveTo(b.x,b.y),n.lineTo(t.x,t.y),n.lineTo(u.x,u.y),n.lineTo(v.x,v.y);let x=this.ctx.createRadialGradient(h.x,h.y,this.opts.bondLength,h.x,h.y,0);x.addColorStop(.4,this.getColor(e.getLeftElement())||this.getColor("C")),x.addColorStop(.6,this.getColor(e.getRightElement())||this.getColor("C")),n.fillStyle=x,n.fill(),n.restore()}drawDashedWedge(e){if(isNaN(e.from.x)||isNaN(e.from.y)||isNaN(e.to.x)||isNaN(e.to.y))return;let i=this.ctx,n=this.offsetX,o=this.offsetY,s=e.getLeftVector().clone(),g=e.getRightVector().clone();s.x+=n,s.y+=o,g.x+=n,g.y+=o,i.save();let r=a.normals(s,g);r[0].normalize(),r[1].normalize();let d,l,h,c,p=e.getRightChiral(),u=e.clone();p?(d=g,l=s,u.shortenRight(1),h=u.getRightVector().clone(),c=u.getLeftVector().clone()):(d=s,l=g,u.shortenLeft(1),h=u.getLeftVector().clone(),c=u.getRightVector().clone()),h.x+=n,h.y+=o,c.x+=n,c.y+=o;let v=a.subtract(l,d).normalize();i.strokeStyle=this.getColor("C"),i.lineCap="round",i.lineWidth=this.opts.bondThickness,i.beginPath();let f=e.getLength(),m=1.25/(f/(3*this.opts.bondThickness)),b=!1;for(var x=0;1>x;x+=m){let t=a.multiplyScalar(v,x*f),n=a.add(d,t),o=1.5*x,s=a.multiplyScalar(r[0],o);!b&&.5this.opts.fontSizeLarge?b.width:this.opts.fontSizeLarge;x/=1.5,h.globalCompositeOperation="destination-out",h.beginPath(),h.arc(e+c,t+p,x,0,o.twoPI,!0),h.closePath(),h.fill(),h.globalCompositeOperation="source-over";let y=-b.width/2,S=-b.width/2;h.fillStyle=this.getColor(i),h.fillText(i,e+c+y,t+this.opts.halfFontSizeLarge+p),y+=b.width,g&&(h.font=this.fontSmall,h.fillText(u,e+c+y,t-this.opts.fifthFontSizeSmall+p),y+=v),0n&&(r=t.sourceId,o=t.targetId);let a=this.getSubtreeOverlapScore(o,r,e.vertexScores);if(a.value>this.opts.overlapSensitivity){let t=this.graph.vertices[r],i=this.graph.vertices[o],n=i.getNeighbours(r);if(1===n.length){let e=this.graph.vertices[n[0]],r=e.position.getRotateAwayFromAngle(t.position,i.position,d.toRad(120));this.rotateSubtree(e.id,i.id,r,i.position);let o=this.getOverlapScore().total;o>this.totalOverlapScore?this.rotateSubtree(e.id,i.id,-r,i.position):this.totalOverlapScore=o}else if(2===n.length){if(0!==i.value.rings.length&&0!==t.value.rings.length)continue;let e=this.graph.vertices[n[0]],r=this.graph.vertices[n[1]];if(1===e.value.rings.length&&1===r.value.rings.length){if(e.value.rings[0]!==r.value.rings[0])continue;}else if(0!==e.value.rings.length||0!==r.value.rings.length)continue;else{let n=e.position.getRotateAwayFromAngle(t.position,i.position,d.toRad(120)),o=r.position.getRotateAwayFromAngle(t.position,i.position,d.toRad(120));this.rotateSubtree(e.id,i.id,n,i.position),this.rotateSubtree(r.id,i.id,o,i.position);let a=this.getOverlapScore().total;a>this.totalOverlapScore?(this.rotateSubtree(e.id,i.id,-n,i.position),this.rotateSubtree(r.id,i.id,-o,i.position)):this.totalOverlapScore=a}}e=this.getOverlapScore()}}}this.resolveSecondaryOverlaps(e.scores),this.opts.isomeric&&this.annotateStereochemistry(),this.opts.compactDrawing&&"default"===this.opts.atomVisualization&&this.initPseudoElements(),this.rotateDrawing(),this.canvasWrapper.scale(this.graph.vertices),this.drawEdges(this.opts.debug),this.drawVertices(this.opts.debug),this.canvasWrapper.reset()}}edgeRingCount(e){let t=this.graph.edges[e],i=this.graph.vertices[t.sourceId],n=this.graph.vertices[t.targetId];return s(i.value.rings.length,n.value.rings.length)}getBridgedRings(){let e=[];for(var t=0;tn&&(n=s,e=r,t=o)}}let s=-h.subtract(this.graph.vertices[e].position,this.graph.vertices[t].position).angle();if(!isNaN(s)){let e=s%.523599;.2617995>e?s-=e:s+=.523599-e;for(var r=0;r{if(n.has(i)){let e=n.get(i);t+=i+(1e.value.getRingbondCount()||1>t.value.getRingbondCount())return null;for(var n=0;ni.value.rings.length&&!i.value.bridgedRing||i.value.bridgedRing&&2>i.value.originalRings.length)&&(t.value.isDrawn=!1)}}getBridgedRingRings(e){let t=[],o=this,a=function(e){let r=o.getRing(e);t.push(e);for(var s=0;sr&&(r=t,o=e)}return o}getVerticesAt(e,t,n){let r=[];for(var o=0;on;){let i=this.graph.vertices[n],a=this.graph.vertices[o];if(!i.value.isDrawn||!a.value.isDrawn)continue;let s=h.subtract(i.position,a.position).lengthSq();if(sc[1]?0:1,sideCount:d,position:d[0]>d[1]?0:1,anCount:a,bnCount:s}}setRingCenter(e){let t=e.getSize(),n=new h(0,0);for(var r=0;rt.sideCount[1]){p[0].multiplyScalar(i.opts.bondSpacing),p[1].multiplyScalar(i.opts.bondSpacing);let e=new c(h.add(d,p[0]),h.add(a,p[0]),s,g);e.shorten(this.opts.bondLength-this.opts.shortBondLength*this.opts.bondLength),this.canvasWrapper.drawLine(e),this.canvasWrapper.drawLine(new c(d,a,s,g))}else if(t.sideCount[0]t.totalSideCount[1]){p[0].multiplyScalar(i.opts.bondSpacing),p[1].multiplyScalar(i.opts.bondSpacing);let e=new c(h.add(d,p[0]),h.add(a,p[0]),s,g);e.shorten(this.opts.bondLength-this.opts.shortBondLength*this.opts.bondLength),this.canvasWrapper.drawLine(e),this.canvasWrapper.drawLine(new c(d,a,s,g))}else if(t.totalSideCount[0]<=t.totalSideCount[1]){p[0].multiplyScalar(i.opts.bondSpacing),p[1].multiplyScalar(i.opts.bondSpacing);let e=new c(h.add(d,p[1]),h.add(a,p[1]),s,g);e.shorten(this.opts.bondLength-this.opts.shortBondLength*this.opts.bondLength),this.canvasWrapper.drawLine(e),this.canvasWrapper.drawLine(new c(d,a,s,g))}else;}else if("#"===n.bondType){p[0].multiplyScalar(i.opts.bondSpacing/1.5),p[1].multiplyScalar(i.opts.bondSpacing/1.5);let e=new c(h.add(d,p[0]),h.add(a,p[0]),s,g),t=new c(h.add(d,p[1]),h.add(a,p[1]),s,g);this.canvasWrapper.drawLine(e),this.canvasWrapper.drawLine(t),this.canvasWrapper.drawLine(new c(d,a,s,g))}else if("."===n.bondType);else{let e=r.value.isStereoCenter,t=o.value.isStereoCenter;"up"===n.wedge?this.canvasWrapper.drawWedge(new c(d,a,s,g,e,t)):"down"===n.wedge?this.canvasWrapper.drawDashedWedge(new c(d,a,s,g,e,t)):this.canvasWrapper.drawLine(new c(d,a,s,g,e,t))}if(t){let t=h.midpoint(d,a);this.canvasWrapper.drawDebugText(t.x,t.y,"e: "+e)}}drawVertices(e){for(var t=Math.abs,n=this.graph.vertices.length,n=0;nt(o-r)&&this.canvasWrapper.drawPoint(i.position.x,i.position.y,d)}if(e){let e="v: "+i.id+" "+l.print(r.ringbonds);this.canvasWrapper.drawDebugText(i.position.x,i.position.y,e)}else;}if(this.opts.debug)for(var n=0;ne.rings.length)&&(o.angle=p,o.positioned=!0)},v,o?o.id:null);e.positioned=!0,e.center=t;for(var f=0;fh.subtract(t,g[0]).lengthSq()&&(c=g[1]);let p=h.subtract(o.position,c),u=h.subtract(a.position,c);-1===p.clockwise(u)?!i.positioned&&this.createRing(i,c,o,a):!i.positioned&&this.createRing(i,c,a,o)}else if(1===n.length){e.isSpiro=!0,i.isSpiro=!0;let o=this.graph.vertices[n[0]],a=h.subtract(t,o.position);a.invert(),a.normalize();let s=d.polyCircumradius(this.opts.bondLength,i.getSize());a.multiplyScalar(s),a.add(o.position),i.positioned||this.createRing(i,a,o)}}for(var f=0;fn.opts.overlapSensitivity&&(r+=t,a++);let s=n.graph.vertices[e.id].position.clone();s.multiplyScalar(t),o.add(s)}}),o.divide(r),{value:r/a,center:o}}getCurrentCenterOfMass(){let e=new h(0,0),t=0;for(var n=0;ng&&(this.rotateSubtree(e.id,t.common.id,2*n,t.common.position),this.rotateSubtree(i.id,t.common.id,-2*n,t.common.position))}else 1!==t.vertices.length||2!==t.rings.length}}resolveSecondaryOverlaps(e){for(var t=0;tthis.opts.overlapSensitivity){let i=this.graph.vertices[e[t].id];if(i.isTerminal()){let e=this.getClosestVertex(i);if(e){let t=null;t=e.isTerminal()?0===e.id?this.graph.vertices[1].position:e.previousPosition:0===e.id?this.graph.vertices[1].position:e.position;let n=0===i.id?this.graph.vertices[1].position:i.previousPosition;i.position.rotateAwayFrom(t,n,d.toRad(20))}}}}getLastVertexWithAngle(e){let t=0,i=null;for(;!t&&e;)i=this.graph.vertices[e],t=i.angle,e=i.parentVertexId;return i}createNextBond(e,t=null,n=0,r=!1,o=!1){if(e.positioned&&!o)return;let c=!1;if(t){let i=this.graph.getEdge(e.id,t.id);("/"===i.bondType||"\\"===i.bondType)&&1==++this.doubleBondConfigCount%2&&null===this.doubleBondConfig&&(this.doubleBondConfig=i.bondType,c=!0,null===t.parentVertexId&&e.value.branchBond&&("/"===this.doubleBondConfig?this.doubleBondConfig="\\":"\\"===this.doubleBondConfig&&(this.doubleBondConfig="/")))}if(!o)if(!t){let t=new h(this.opts.bondLength,0);t.rotate(d.toRad(-60)),e.previousPosition=t,e.setPosition(this.opts.bondLength,0),e.angle=d.toRad(-60),null===e.value.bridgedRing&&(e.positioned=!0)}else if(0n?g(-1.0472,n):1.0472;else if(!n){let t=this.getLastVertexWithAngle(e.id);n=t.angle,n||(n=1.0472)}if(t&&!c){let t=this.graph.getEdge(e.id,i.id).bondType;"/"===t?("/"===this.doubleBondConfig||"\\"===this.doubleBondConfig&&(n=-n),this.doubleBondConfig=null):"\\"===t&&("/"===this.doubleBondConfig?n=-n:"\\"===this.doubleBondConfig,this.doubleBondConfig=null)}i.angle=r?n:-n,this.createNextBond(i,e,u+i.angle)}}else if(2===o.length){let i=e.angle;i||(i=1.0472);let n=this.graph.getTreeDepth(o[0],e.id),a=this.graph.getTreeDepth(o[1],e.id),s=this.graph.vertices[o[0]],g=this.graph.vertices[o[1]];s.value.subtreeDepth=n,g.value.subtreeDepth=a;let r=this.graph.getTreeDepth(t?t.id:null,e.id);t&&(t.value.subtreeDepth=r);let d=0,l=1;"C"===g.value.element&&"C"!==s.value.element&&1n?(d=1,l=0):"C"!==g.value.element&&"C"===s.value.element&&1a?(d=0,l=1):a>n&&(d=1,l=0);let h=this.graph.vertices[o[d]],c=this.graph.vertices[o[l]],p=this.graph.getEdge(e.id,h.id),v=this.graph.getEdge(e.id,c.id),f=!1;ri&&n>a?(g=this.graph.vertices[o[1]],s=this.graph.vertices[o[0]],l=this.graph.vertices[o[2]]):a>i&&a>n&&(g=this.graph.vertices[o[2]],s=this.graph.vertices[o[0]],l=this.graph.vertices[o[1]]),t&&1>t.value.rings.length&&1>g.value.rings.length&&1>s.value.rings.length&&1>l.value.rings.length&&1===this.graph.getTreeDepth(s.id,e.id)&&1===this.graph.getTreeDepth(l.id,e.id)&&1t&&i>n&&i>r?(a=this.graph.vertices[o[1]],s=this.graph.vertices[o[0]],g=this.graph.vertices[o[2]],l=this.graph.vertices[o[3]]):n>t&&n>i&&n>r?(a=this.graph.vertices[o[2]],s=this.graph.vertices[o[0]],g=this.graph.vertices[o[1]],l=this.graph.vertices[o[3]]):r>t&&r>i&&r>n&&(a=this.graph.vertices[o[3]],s=this.graph.vertices[o[0]],g=this.graph.vertices[o[1]],l=this.graph.vertices[o[2]]),a.angle=-d.toRad(36),s.angle=d.toRad(36),g.angle=-d.toRad(108),l.angle=d.toRad(108),this.createNextBond(a,e,u+a.angle),this.createNextBond(s,e,u+s.angle),this.createNextBond(g,e,u+g.angle),this.createNextBond(l,e,u+l.angle)}}}getCommonRingbondNeighbour(e){let t=e.neighbours;for(var n=0;ng&&(g=s[t][1].length);for(var n=0;nl&&(l=s[t][1][n].length)}for(var t=0;tt[1][i][n])return-1;if(e[1][i][n]t[0]?-1:e[0]i.getNeighbourCount()||0o)continue;let s=null;for(var t=0;tn[r][a]+n[a][o]&&(n[r][o]=n[r][a]+n[a][o]);return n}getSubgraphDistanceMatrix(e){let t=e.length,n=this.getSubgraphAdjacencyMatrix(e),r=Array(t);for(var o=0;or[o][s]+r[s][a]&&(r[o][a]=r[o][s]+r[s][a]);return r}getAdjacencyList(){let e=this.vertices.length,t=Array(e);for(var n=0;nr&&(r=i)}return r+1}traverseTree(e,t,n,r=g,o=!1,a=1,s=null){var g=Number.MAX_SAFE_INTEGER;if(null===s&&(s=new Uint8Array(this.vertices.length)),a>r+1||1===s[e])return;s[e]=1;let d=this.vertices[e],l=d.getNeighbours(t);(!o||1e&&!1===y[h]&&(e=r,t=h,i=o,n=a)}return[t,e,i,n]},k=function(e,t,i){let o=0,a=0,s=0,g=b[e],d=x[e],c=S[e],u=C[e];for(h=p;h--;){if(h===e)continue;let t=b[h],i=x[h],r=c[h],l=u[h],p=(g-t)*(g-t),v=1/n(p+(d-i)*(d-i),1.5);o+=l*(1-r*(d-i)*(d-i)*v),a+=l*(1-r*p*v),s+=l*(r*(g-t)*(d-i)*v)}0==o&&(o=.1),0===a&&(a=.1),0===s&&(s=.1);let v=t/o+i/s;v/=s/o-a/s;let l=-(s*v+t)/o;b[e]+=l,x[e]+=v;let f=R[e];t=0,i=0,g=b[e],d=x[e];let m,y,j,I,B;for(h=p;h--;)e!==h&&(m=b[h],y=x[h],j=f[h][0],I=f[h][1],B=1/r((g-m)*(g-m)+(d-y)*(d-y)),l=u[h]*(g-m-c[h]*(g-m)*B),v=u[h]*(d-y-c[h]*(d-y)*B),f[h]=[l,v],t+=l,i+=v,A[h]+=l-j,T[h]+=v-I);A[e]=t,T[e]=i},w=1e9,P=0,L=0,E=0,O=0,H=0,D=0;for(;w>.1&&2e3>H;)for(H++,[P,w,L,E]=B(),O=w,D=0;O>.1&&50>D;)D++,k(P,L,E),[O,L,E]=I(P);for(h=p;h--;){let t=e[h],i=this.vertices[t];i.position.x=b[h],i.position.y=x[h],i.positioned=!0,i.forcePositioned=!0}}_bridgeDfs(e,t,n,r,a,s,g){t[e]=!0,n[e]=r[e]=++this._time;for(var d=0;dn[e]&&g.push([e,i]))}}static getConnectedComponents(e){let t=e.length,i=Array(t),n=[],r=0;i.fill(!1);for(var o=0;oVe&&(Ve=He,ze=[]),ze.push(e))}function g(){var e,t,i,n,r,o,a,s,c,p;if(e=He,t=He,i=l(),i!==B){for(n=[],r=d();r!==B;)n.push(r),r=d();if(n!==B){for(r=[],o=He,a=h(),a===B&&(a=null),a===B?(He=o,o=B):(s=m(),s===B?(He=o,o=B):(a=[a,s],o=a));o!==B;)r.push(o),o=He,a=h(),a===B&&(a=null),a===B?(He=o,o=B):(s=m(),s===B?(He=o,o=B):(a=[a,s],o=a));if(r!==B){for(o=[],a=d();a!==B;)o.push(a),a=d();if(o===B)He=t,t=B;else if(a=h(),a===B&&(a=null),a===B)He=t,t=B;else if(s=g(),s===B&&(s=null),s!==B){for(c=[],p=d();p!==B;)c.push(p),p=d();c===B?(He=t,t=B):(i=[i,n,r,o,a,s,c],t=i)}else He=t,t=B}else He=t,t=B}else He=t,t=B}else He=t,t=B;return t!==B&&(De=e,t=P(t)),e=t,e}function d(){var e,i,n,r,o,a;return e=He,i=He,40===t.charCodeAt(He)?(n="(",He++):(n=B,s(L)),n===B?(He=i,i=B):(r=h(),r===B&&(r=null),r===B?(He=i,i=B):(o=g(),o===B?(He=i,i=B):(41===t.charCodeAt(He)?(a=")",He++):(a=B,s(E)),a===B?(He=i,i=B):(n=[n,r,o,a],i=n)))),i!==B&&(De=e,i=O(i)),e=i,e}function l(){var e,t;return e=He,t=p(),t===B&&(t=u(),t===B&&(t=c(),t===B&&(t=v()))),t!==B&&(De=e,t=H(t)),e=t,e}function h(){var e,i;return e=He,D.test(t.charAt(He))?(i=t.charAt(He),He++):(i=B,s(N)),i!==B&&(De=e,i=V(i)),e=i,e}function c(){var e,i,n,r,o,a,g,d,l,h;return e=He,i=He,91===t.charCodeAt(He)?(n="[",He++):(n=B,s(z)),n===B?(He=i,i=B):(r=A(),r===B&&(r=null),r===B?(He=i,i=B):("se"===t.substr(He,2)?(o="se",He+=2):(o=B,s(W)),o===B&&("as"===t.substr(He,2)?(o="as",He+=2):(o=B,s(F)),o===B&&(o=u(),o===B&&(o=f(),o===B&&(o=v())))),o===B?(He=i,i=B):(a=b(),a===B&&(a=null),a===B?(He=i,i=B):(g=C(),g===B&&(g=null),g===B?(He=i,i=B):(d=x(),d===B&&(d=null),d===B?(He=i,i=B):(l=R(),l===B&&(l=null),l===B?(He=i,i=B):(93===t.charCodeAt(He)?(h="]",He++):(h=B,s(M)),h===B?(He=i,i=B):(n=[n,r,o,a,g,d,l,h],i=n)))))))),i!==B&&(De=e,i=q(i)),e=i,e}function p(){var e,i,n,r;return e=He,i=He,66===t.charCodeAt(He)?(n="B",He++):(n=B,s(_)),n===B?(He=i,i=B):(114===t.charCodeAt(He)?(r="r",He++):(r=B,s(U)),r===B&&(r=null),r===B?(He=i,i=B):(n=[n,r],i=n)),i===B&&(i=He,67===t.charCodeAt(He)?(n="C",He++):(n=B,s(G)),n===B?(He=i,i=B):(108===t.charCodeAt(He)?(r="l",He++):(r=B,s(X)),r===B&&(r=null),r===B?(He=i,i=B):(n=[n,r],i=n)),i===B&&(Y.test(t.charAt(He))?(i=t.charAt(He),He++):(i=B,s(K)))),i!==B&&(De=e,i=Z(i)),e=i,e}function u(){var e,i;return e=He,$.test(t.charAt(He))?(i=t.charAt(He),He++):(i=B,s(J)),i!==B&&(De=e,i=H(i)),e=i,e}function v(){var e,i;return e=He,42===t.charCodeAt(He)?(i="*",He++):(i=B,s(Q)),i!==B&&(De=e,i=ee(i)),e=i,e}function f(){var e,i,n,r;return e=He,i=He,te.test(t.charAt(He))?(n=t.charAt(He),He++):(n=B,s(ie)),n===B?(He=i,i=B):(ne.test(t.charAt(He))?(r=t.charAt(He),He++):(r=B,s(re)),r===B&&(r=null),r===B?(He=i,i=B):(n=[n,r],i=n)),i!==B&&(De=e,i=oe(i)),e=i,e}function m(){var e,i,n,r,o;return e=He,i=He,37===t.charCodeAt(He)?(n="%",He++):(n=B,s(ae)),n===B?(He=i,i=B):(se.test(t.charAt(He))?(r=t.charAt(He),He++):(r=B,s(ge)),r===B?(He=i,i=B):(de.test(t.charAt(He))?(o=t.charAt(He),He++):(o=B,s(le)),o===B?(He=i,i=B):(n=[n,r,o],i=n))),i===B&&(de.test(t.charAt(He))?(i=t.charAt(He),He++):(i=B,s(le))),i!==B&&(De=e,i=he(i)),e=i,e}function b(){var e,i,n,r,o,a,g;return e=He,i=He,64===t.charCodeAt(He)?(n="@",He++):(n=B,s(ce)),n===B?(He=i,i=B):(64===t.charCodeAt(He)?(r="@",He++):(r=B,s(ce)),r===B&&(r=He,"TH"===t.substr(He,2)?(o="TH",He+=2):(o=B,s(pe)),o===B?(He=r,r=B):(ue.test(t.charAt(He))?(a=t.charAt(He),He++):(a=B,s(ve)),a===B?(He=r,r=B):(o=[o,a],r=o)),r===B&&(r=He,"AL"===t.substr(He,2)?(o="AL",He+=2):(o=B,s(fe)),o===B?(He=r,r=B):(ue.test(t.charAt(He))?(a=t.charAt(He),He++):(a=B,s(ve)),a===B?(He=r,r=B):(o=[o,a],r=o)),r===B&&(r=He,"SP"===t.substr(He,2)?(o="SP",He+=2):(o=B,s(me)),o===B?(He=r,r=B):(be.test(t.charAt(He))?(a=t.charAt(He),He++):(a=B,s(xe)),a===B?(He=r,r=B):(o=[o,a],r=o)),r===B&&(r=He,"TB"===t.substr(He,2)?(o="TB",He+=2):(o=B,s(ye)),o===B?(He=r,r=B):(se.test(t.charAt(He))?(a=t.charAt(He),He++):(a=B,s(ge)),a===B?(He=r,r=B):(de.test(t.charAt(He))?(g=t.charAt(He),He++):(g=B,s(le)),g===B&&(g=null),g===B?(He=r,r=B):(o=[o,a,g],r=o))),r===B&&(r=He,"OH"===t.substr(He,2)?(o="OH",He+=2):(o=B,s(Se)),o===B?(He=r,r=B):(se.test(t.charAt(He))?(a=t.charAt(He),He++):(a=B,s(ge)),a===B?(He=r,r=B):(de.test(t.charAt(He))?(g=t.charAt(He),He++):(g=B,s(le)),g===B&&(g=null),g===B?(He=r,r=B):(o=[o,a,g],r=o)))))))),r===B&&(r=null),r===B?(He=i,i=B):(n=[n,r],i=n)),i!==B&&(De=e,i=Ce(i)),e=i,e}function x(){var e,t;return e=He,t=y(),t===B&&(t=S()),t!==B&&(De=e,t=Re(t)),e=t,e}function y(){var e,i,n,r,o,a;return e=He,i=He,43===t.charCodeAt(He)?(n="+",He++):(n=B,s(Ae)),n===B?(He=i,i=B):(43===t.charCodeAt(He)?(r="+",He++):(r=B,s(Ae)),r===B&&(r=He,se.test(t.charAt(He))?(o=t.charAt(He),He++):(o=B,s(ge)),o===B?(He=r,r=B):(de.test(t.charAt(He))?(a=t.charAt(He),He++):(a=B,s(le)),a===B&&(a=null),a===B?(He=r,r=B):(o=[o,a],r=o))),r===B&&(r=null),r===B?(He=i,i=B):(n=[n,r],i=n)),i!==B&&(De=e,i=je(i)),e=i,e}function S(){var e,i,n,r,o,a;return e=He,i=He,45===t.charCodeAt(He)?(n="-",He++):(n=B,s(Te)),n===B?(He=i,i=B):(45===t.charCodeAt(He)?(r="-",He++):(r=B,s(Te)),r===B&&(r=He,se.test(t.charAt(He))?(o=t.charAt(He),He++):(o=B,s(ge)),o===B?(He=r,r=B):(de.test(t.charAt(He))?(a=t.charAt(He),He++):(a=B,s(le)),a===B&&(a=null),a===B?(He=r,r=B):(o=[o,a],r=o))),r===B&&(r=null),r===B?(He=i,i=B):(n=[n,r],i=n)),i!==B&&(De=e,i=Ie(i)),e=i,e}function C(){var e,i,n,r;return e=He,i=He,72===t.charCodeAt(He)?(n="H",He++):(n=B,s(Be)),n===B?(He=i,i=B):(de.test(t.charAt(He))?(r=t.charAt(He),He++):(r=B,s(le)),r===B&&(r=null),r===B?(He=i,i=B):(n=[n,r],i=n)),i!==B&&(De=e,i=ke(i)),e=i,e}function R(){var e,i,n,r,o,a,g;if(e=He,i=He,58===t.charCodeAt(He)?(n=":",He++):(n=B,s(we)),n!==B){if(r=He,se.test(t.charAt(He))?(o=t.charAt(He),He++):(o=B,s(ge)),o!==B){for(a=[],de.test(t.charAt(He))?(g=t.charAt(He),He++):(g=B,s(le));g!==B;)a.push(g),de.test(t.charAt(He))?(g=t.charAt(He),He++):(g=B,s(le));a===B?(He=r,r=B):(o=[o,a],r=o)}else He=r,r=B;r===B&&(Pe.test(t.charAt(He))?(r=t.charAt(He),He++):(r=B,s(Le))),r===B?(He=i,i=B):(n=[n,r],i=n)}else He=i,i=B;return i!==B&&(De=e,i=Ee(i)),e=i,e}function A(){var e,i,n,r,o;return e=He,i=He,se.test(t.charAt(He))?(n=t.charAt(He),He++):(n=B,s(ge)),n===B?(He=i,i=B):(de.test(t.charAt(He))?(r=t.charAt(He),He++):(r=B,s(le)),r===B&&(r=null),r===B?(He=i,i=B):(de.test(t.charAt(He))?(o=t.charAt(He),He++):(o=B,s(le)),o===B&&(o=null),o===B?(He=i,i=B):(n=[n,r,o],i=n))),i!==B&&(De=e,i=Oe(i)),e=i,e}i=void 0===i?{}:i;var j=t.split("(").length-1,T=t.split(")").length-1;if(j!==T)throw function(t,i){return new e(t,null,null,i)}("The number of opening parentheses does not match the number of closing parentheses.",0);var I,B={},k={chain:g},w=g,P=function(e){for(var t=[],n=[],r=0;ro;){let a=r;t(a),r=e[r].getNextInRing(e,this.id,n),n=a,r==i&&(r=null),o++}}getOrderedNeighbours(e){let t=Array(this.neighbours.length);for(let n,r=0;rt){var g,d,h;if(e===t+1)for(s[c][p]=[a[c][p].length],g=a[c][p].length;g--;)for(s[c][p][g]=[a[c][p][g].length],d=a[c][p][g].length;d--;)for(s[c][p][g][d]=[a[c][p][g][d].length],h=a[c][p][g][d].length;h--;)s[c][p][g][d][h]=[a[c][p][g][d][0],a[c][p][g][d][1]];else s[c][p]=[];for(o[c][p]=t,a[c][p]=[[]],g=a[c][u][0].length;g--;)a[c][p][0].push(a[c][u][0][g]);for(g=a[u][p][0].length;g--;)a[c][p][0].push(a[u][p][0][g])}else if(e===t){if(a[c][u].length&&a[u][p].length){var g;if(a[c][p].length){let e=[];for(g=a[c][u][0].length;g--;)e.push(a[c][u][0][g]);for(g=a[u][p][0].length;g--;)e.push(a[u][p][0][g]);a[c][p].push(e)}else{let e=[];for(g=a[c][u][0].length;g--;)e.push(a[c][u][0][g]);for(g=a[u][p][0].length;g--;)e.push(a[u][p][0][g]);a[c][p][0]=e}}}else if(e===t-1){var g;if(s[c][p].length){let e=[];for(g=a[c][u][0].length;g--;)e.push(a[c][u][0][g]);for(g=a[u][p][0].length;g--;)e.push(a[u][p][0][g]);s[c][p].push(e)}else{let e=[];for(g=a[c][u][0].length;g--;)e.push(a[c][u][0][g]);for(g=a[u][p][0].length;g--;)e.push(a[u][p][0][g]);s[c][p][0]=e}}}return{d:o,pe:a,pe_prime:s}}static getRingCandidates(e,t,n){let r=e.length,o=[],a=0;for(let s=0;sg)return d}else for(let t,i=0;ig)return d}return d}static getEdgeCount(e){let t=0,n=e.length;for(var r=n-1;r--;)for(var o=n;o--;)1===e[r][o]&&t++;return t}static getEdgeList(e){let t=e.length,n=[];for(var r=t-1;r--;)for(var o=t;o--;)1===e[r][o]&&n.push([r,o]);return n}static bondsToAtoms(e){let t=new Set;for(var n=e.length;n--;)t.add(e[n][0]),t.add(e[n][1]);return t}static getBondCount(e,t){let i=0;for(let n of e)for(let r of e)n!==r&&(i+=t[n][r]);return i/2}static pathSetsContain(e,t,n,o,a,s){for(var g=e.length;g--;){if(r.isSupersetOf(t,e[g]))return!0;if(e[g].size===t.size&&r.areSetsEqual(e[g],t))return!0}let d=0,l=!1;for(g=n.length;g--;)for(var h=o.length;h--;)(n[g][0]===o[h][0]&&n[g][1]===o[h][1]||n[g][1]===o[h][0]&&n[g][0]===o[h][1])&&d++,d===n.length&&(l=!0);let c=!1;if(l)for(let e of t)if(s[e]i)return-1;return t===i?0:1}relativeClockwise(e,t){let i=(this.y-e.y)*(t.x-e.x),n=(this.x-e.x)*(t.y-e.y);if(i>n)return-1;return i===n?0:1}rotate(e){let t=new g(0,0),i=r(e),o=n(e);return t.x=this.x*i-this.y*o,t.y=this.x*o+this.y*i,this.x=t.x,this.y=t.y,this}rotateAround(e,t){let i=n(e),o=r(e);this.x-=t.x,this.y-=t.y;let a=this.x*o-this.y*i,s=this.x*i+this.y*o;return this.x=a+t.x,this.y=s+t.y,this}rotateTo(e,t,i=0){this.x+=.001,this.y-=.001;let n=g.subtract(this,t),r=g.subtract(e,t),o=g.angle(r,n);return this.rotateAround(o+i,t),this}rotateAwayFrom(e,t,i){this.rotateAround(i,t);let n=this.distanceSq(e);this.rotateAround(-2*i,t);let r=this.distanceSq(e);rr?i:-i}getRotateToAngle(e,t){let i=g.subtract(this,t),n=g.subtract(e,t),r=g.angle(n,i);return Number.isNaN(r)?0:r}isInPolygon(e){let t=!1;for(let n=0,i=e.length-1;nthis.y!=e[i].y>this.y&&this.x<(e[i].x-e[n].x)*(this.y-e[n].y)/(e[i].y-e[n].y)+e[n].x&&(t=!t);return t}length(){return o(this.x*this.x+this.y*this.y)}lengthSq(){return this.x*this.x+this.y*this.y}normalize(){return this.divide(this.length()),this}normalized(){return g.divideScalar(this,this.length())}whichSide(e,t){return(this.x-e.x)*(t.y-e.y)-(this.y-e.y)*(t.x-e.x)}sameSideAs(e,t,i){let n=this.whichSide(e,t),r=i.whichSide(e,t);return 0>n&&0>r||0==n&&0==r||0this.neighbours.length?i=2:i=3),null===this.value.bracket.hcount&&0===t&&(i=1),null===this.value.bracket.hcount&&1===t&&(3>this.neighbours.length?i=1:i=2),this.neighbours.splice(i,0,e)}else this.neighbours.push(e);this.neighbourCount++}setParentVertexId(e){this.neighbourCount++,this.parentVertexId=e,this.neighbours.push(e)}isTerminal(){return!!this.value.hasAttachedPseudoElements||null===this.parentVertexId&&2>this.children.length||0===this.children.length}clone(){let e=new g(this.value,this.position.x,this.position.y);return e.id=this.id,e.previousPosition=new a(this.previousPosition.x,this.previousPosition.y),e.parentVertexId=this.parentVertexId,e.children=o.clone(this.children),e.spanningTreeChildren=o.clone(this.spanningTreeChildren),e.edges=o.clone(this.edges),e.positioned=this.positioned,e.angle=this.angle,e.forcePositioned=this.forcePositioned,e}equals(e){return this.id===e.id}getAngle(e=null,t=!1){let i=null;return i=e?a.subtract(this.position,e):a.subtract(this.position,this.previousPosition),t?r.toDeg(i.angle()):i.angle()}getTextDirection(e){let t=this.getDrawnNeighbours(e),o=[];for(let n=0;n>>0,o=arguments[1],a=o>>0,s=0>a?i(r+a,0):n(a,r),g=arguments[2],d=void 0===g?r:g>>0,l=0>d?i(r+d,0):n(d,r);sn.x&&(o=n.x),a>n.y&&(a=n.y)}var g=this.opts.padding;t+=g,r+=g,o-=g,a-=g,this.drawingWidth=t-o,this.drawingHeight=r-a;var d=this.canvas.offsetWidth/this.drawingWidth,l=this.canvas.offsetHeight/this.drawingHeight,h=dn&&(i.globalAlpha=n),i.strokeStyle=h,i.stroke(),i.restore()}drawWedge(e,n=1){if(isNaN(e.from.x)||isNaN(e.from.y)||isNaN(e.to.x)||isNaN(e.to.y))return;let i=this.ctx,a=this.offsetX,s=this.offsetY,g=e.clone().shorten(5),d=g.getLeftVector().clone(),h=g.getRightVector().clone();d.x+=a,d.y+=s,h.x+=a,h.y+=s,d=e.getLeftVector().clone(),h=e.getRightVector().clone(),d.x+=a,d.y+=s,h.x+=a,h.y+=s,i.save();let c=o.normals(d,h);c[0].normalize(),c[1].normalize();let p=e.getRightChiral(),f=d,m=h;p&&(f=h,m=d);let b=o.add(f,o.multiplyScalar(c[0],this.halfBondThickness)),t=o.add(m,o.multiplyScalar(c[0],1.5+this.halfBondThickness)),u=o.add(m,o.multiplyScalar(c[1],1.5+this.halfBondThickness)),v=o.add(f,o.multiplyScalar(c[1],this.halfBondThickness));i.beginPath(),i.moveTo(b.x,b.y),i.lineTo(t.x,t.y),i.lineTo(u.x,u.y),i.lineTo(v.x,v.y);let x=this.ctx.createRadialGradient(h.x,h.y,this.opts.bondLength,h.x,h.y,0);x.addColorStop(.4,this.getColor(e.getLeftElement())||this.getColor("C")),x.addColorStop(.6,this.getColor(e.getRightElement())||this.getColor("C")),i.fillStyle=x,i.fill(),i.restore()}drawDashedWedge(e){if(isNaN(e.from.x)||isNaN(e.from.y)||isNaN(e.to.x)||isNaN(e.to.y))return;let n=this.ctx,i=this.offsetX,a=this.offsetY,s=e.getLeftVector().clone(),g=e.getRightVector().clone();s.x+=i,s.y+=a,g.x+=i,g.y+=a,n.save();let r=o.normals(s,g);r[0].normalize(),r[1].normalize();let d,l,h,c,p=e.getRightChiral(),u=e.clone();p?(d=g,l=s,u.shortenRight(1),h=u.getRightVector().clone(),c=u.getLeftVector().clone()):(d=s,l=g,u.shortenLeft(1),h=u.getLeftVector().clone(),c=u.getRightVector().clone()),h.x+=i,h.y+=a,c.x+=i,c.y+=a;let v=o.subtract(l,d).normalize();n.strokeStyle=this.getColor("C"),n.lineCap="round",n.lineWidth=this.opts.bondThickness,n.beginPath();let f=e.getLength(),m=1.25/(f/(3*this.opts.bondThickness)),b=!1;for(var x=0;1>x;x+=m){let t=o.multiplyScalar(v,x*f),i=o.add(d,t),a=1.5*x,s=o.multiplyScalar(r[0],a);!b&&.5this.opts.fontSizeLarge?b.width:this.opts.fontSizeLarge;x/=1.5,h.globalCompositeOperation="destination-out",h.beginPath(),h.arc(e+c,t+p,x,0,i.twoPI,!0),h.closePath(),h.fill(),h.globalCompositeOperation="source-over";let y=-b.width/2,S=-b.width/2;h.fillStyle=this.getColor(n),h.fillText(n,e+c+y,t+this.opts.halfFontSizeLarge+p),y+=b.width,g&&(h.font=this.fontSmall,h.fillText(u,e+c+y,t-this.opts.fifthFontSizeSmall+p),y+=v),0i&&(r=t.sourceId,o=t.targetId);let a=this.getSubtreeOverlapScore(o,r,e.vertexScores);if(a.value>this.opts.overlapSensitivity){let t=this.graph.vertices[r],n=this.graph.vertices[o],i=n.getNeighbours(r);if(1===i.length){let e=this.graph.vertices[i[0]],r=e.position.getRotateAwayFromAngle(t.position,n.position,d.toRad(120));this.rotateSubtree(e.id,n.id,r,n.position);let o=this.getOverlapScore().total;o>this.totalOverlapScore?this.rotateSubtree(e.id,n.id,-r,n.position):this.totalOverlapScore=o}else if(2===i.length){if(0!==n.value.rings.length&&0!==t.value.rings.length)continue;let e=this.graph.vertices[i[0]],r=this.graph.vertices[i[1]];if(1===e.value.rings.length&&1===r.value.rings.length){if(e.value.rings[0]!==r.value.rings[0])continue;}else if(0!==e.value.rings.length||0!==r.value.rings.length)continue;else{let i=e.position.getRotateAwayFromAngle(t.position,n.position,d.toRad(120)),o=r.position.getRotateAwayFromAngle(t.position,n.position,d.toRad(120));this.rotateSubtree(e.id,n.id,i,n.position),this.rotateSubtree(r.id,n.id,o,n.position);let a=this.getOverlapScore().total;a>this.totalOverlapScore?(this.rotateSubtree(e.id,n.id,-i,n.position),this.rotateSubtree(r.id,n.id,-o,n.position)):this.totalOverlapScore=a}}e=this.getOverlapScore()}}}this.resolveSecondaryOverlaps(e.scores),this.opts.isomeric&&this.annotateStereochemistry(),this.opts.compactDrawing&&"default"===this.opts.atomVisualization&&this.initPseudoElements(),this.rotateDrawing(),this.canvasWrapper.scale(this.graph.vertices),this.drawEdges(this.opts.debug),this.drawVertices(this.opts.debug),this.canvasWrapper.reset(),this.opts.debug&&(console.log(this.graph),console.log(this.rings),console.log(this.ringConnections))}}edgeRingCount(e){let t=this.graph.edges[e],n=this.graph.vertices[t.sourceId],i=this.graph.vertices[t.targetId];return s(n.value.rings.length,i.value.rings.length)}getBridgedRings(){let e=[];for(var t=0;tn&&(n=s,e=r,t=o)}}let s=-h.subtract(this.graph.vertices[e].position,this.graph.vertices[t].position).angle();if(!isNaN(s)){let e=s%.523599;.2617995>e?s-=e:s+=.523599-e;for(var r=0;r{if(n.has(i)){let e=n.get(i);t+=i+(1e.value.getRingbondCount()||1>t.value.getRingbondCount())return null;for(var n=0;nn.value.rings.length&&!n.value.bridgedRing||n.value.bridgedRing&&2>n.value.originalRings.length)&&(t.value.isDrawn=!1)}}getBridgedRingRings(e){let t=[],o=this,a=function(e){let r=o.getRing(e);t.push(e);for(var s=0;sr&&(r=t,o=e)}return o}getVerticesAt(e,t,n){let r=[];for(var o=0;on;){let i=this.graph.vertices[n],a=this.graph.vertices[o];if(!i.value.isDrawn||!a.value.isDrawn)continue;let s=h.subtract(i.position,a.position).lengthSq();if(sc[1]?0:1,sideCount:d,position:d[0]>d[1]?0:1,anCount:a,bnCount:s}}setRingCenter(e){let t=e.getSize(),n=new h(0,0);for(var r=0;rt.sideCount[1]){p[0].multiplyScalar(n.opts.bondSpacing),p[1].multiplyScalar(n.opts.bondSpacing);let e=new c(h.add(d,p[0]),h.add(a,p[0]),s,g);e.shorten(this.opts.bondLength-this.opts.shortBondLength*this.opts.bondLength),this.canvasWrapper.drawLine(e),this.canvasWrapper.drawLine(new c(d,a,s,g))}else if(t.sideCount[0]t.totalSideCount[1]){p[0].multiplyScalar(n.opts.bondSpacing),p[1].multiplyScalar(n.opts.bondSpacing);let e=new c(h.add(d,p[0]),h.add(a,p[0]),s,g);e.shorten(this.opts.bondLength-this.opts.shortBondLength*this.opts.bondLength),this.canvasWrapper.drawLine(e),this.canvasWrapper.drawLine(new c(d,a,s,g))}else if(t.totalSideCount[0]<=t.totalSideCount[1]){p[0].multiplyScalar(n.opts.bondSpacing),p[1].multiplyScalar(n.opts.bondSpacing);let e=new c(h.add(d,p[1]),h.add(a,p[1]),s,g);e.shorten(this.opts.bondLength-this.opts.shortBondLength*this.opts.bondLength),this.canvasWrapper.drawLine(e),this.canvasWrapper.drawLine(new c(d,a,s,g))}else;}else if("#"===i.bondType){p[0].multiplyScalar(n.opts.bondSpacing/1.5),p[1].multiplyScalar(n.opts.bondSpacing/1.5);let e=new c(h.add(d,p[0]),h.add(a,p[0]),s,g),t=new c(h.add(d,p[1]),h.add(a,p[1]),s,g);this.canvasWrapper.drawLine(e),this.canvasWrapper.drawLine(t),this.canvasWrapper.drawLine(new c(d,a,s,g))}else if("."===i.bondType);else{let e=r.value.isStereoCenter,t=o.value.isStereoCenter;"up"===i.wedge?this.canvasWrapper.drawWedge(new c(d,a,s,g,e,t)):"down"===i.wedge?this.canvasWrapper.drawDashedWedge(new c(d,a,s,g,e,t)):this.canvasWrapper.drawLine(new c(d,a,s,g,e,t))}if(t){let t=h.midpoint(d,a);this.canvasWrapper.drawDebugText(t.x,t.y,"e: "+e)}}drawVertices(e){for(var t=Math.abs,n=this.graph.vertices.length,n=0;nt(o-r)&&this.canvasWrapper.drawPoint(i.position.x,i.position.y,d)}if(e){let e="v: "+i.id+" "+l.print(r.ringbonds);this.canvasWrapper.drawDebugText(i.position.x,i.position.y,e)}else;}if(this.opts.debug)for(var n=0;ne.rings.length)&&(o.angle=p,o.positioned=!0)},v,o?o.id:null);e.positioned=!0,e.center=t;for(var f=0;fh.subtract(t,g[0]).lengthSq()&&(c=g[1]);let p=h.subtract(o.position,c),u=h.subtract(a.position,c);-1===p.clockwise(u)?!n.positioned&&this.createRing(n,c,o,a):!n.positioned&&this.createRing(n,c,a,o)}else if(1===i.length){e.isSpiro=!0,n.isSpiro=!0;let o=this.graph.vertices[i[0]],a=h.subtract(t,o.position);a.invert(),a.normalize();let s=d.polyCircumradius(this.opts.bondLength,n.getSize());a.multiplyScalar(s),a.add(o.position),n.positioned||this.createRing(n,a,o)}}for(var f=0;fi.opts.overlapSensitivity&&(r+=t,a++);let s=i.graph.vertices[e.id].position.clone();s.multiplyScalar(t),o.add(s)}}),o.divide(r),{value:r/a,center:o}}getCurrentCenterOfMass(){let e=new h(0,0),t=0;for(var n=0;ng&&(this.rotateSubtree(e.id,t.common.id,2*i,t.common.position),this.rotateSubtree(n.id,t.common.id,-2*i,t.common.position))}else 1!==t.vertices.length||2!==t.rings.length}}resolveSecondaryOverlaps(e){for(var t=0;tthis.opts.overlapSensitivity){let n=this.graph.vertices[e[t].id];if(n.isTerminal()){let e=this.getClosestVertex(n);if(e){let t=null;t=e.isTerminal()?0===e.id?this.graph.vertices[1].position:e.previousPosition:0===e.id?this.graph.vertices[1].position:e.position;let i=0===n.id?this.graph.vertices[1].position:n.previousPosition;n.position.rotateAwayFrom(t,i,d.toRad(20))}}}}getLastVertexWithAngle(e){let t=0,n=null;for(;!t&&e;)n=this.graph.vertices[e],t=n.angle,e=n.parentVertexId;return n}createNextBond(e,t=null,n=0,r=!1,o=!1){if(e.positioned&&!o)return;let c=!1;if(t){let n=this.graph.getEdge(e.id,t.id);("/"===n.bondType||"\\"===n.bondType)&&1==++this.doubleBondConfigCount%2&&null===this.doubleBondConfig&&(this.doubleBondConfig=n.bondType,c=!0,null===t.parentVertexId&&e.value.branchBond&&("/"===this.doubleBondConfig?this.doubleBondConfig="\\":"\\"===this.doubleBondConfig&&(this.doubleBondConfig="/")))}if(!o)if(!t){let t=new h(this.opts.bondLength,0);t.rotate(d.toRad(-60)),e.previousPosition=t,e.setPosition(this.opts.bondLength,0),e.angle=d.toRad(-60),null===e.value.bridgedRing&&(e.positioned=!0)}else if(0i?g(-1.0472,i):1.0472;else if(!i){let t=this.getLastVertexWithAngle(e.id);i=t.angle,i||(i=1.0472)}if(t&&!c){let t=this.graph.getEdge(e.id,n.id).bondType;"/"===t?("/"===this.doubleBondConfig||"\\"===this.doubleBondConfig&&(i=-i),this.doubleBondConfig=null):"\\"===t&&("/"===this.doubleBondConfig?i=-i:"\\"===this.doubleBondConfig,this.doubleBondConfig=null)}n.angle=r?i:-i,this.createNextBond(n,e,u+n.angle)}}else if(2===o.length){let n=e.angle;n||(n=1.0472);let i=this.graph.getTreeDepth(o[0],e.id),a=this.graph.getTreeDepth(o[1],e.id),s=this.graph.vertices[o[0]],g=this.graph.vertices[o[1]];s.value.subtreeDepth=i,g.value.subtreeDepth=a;let r=this.graph.getTreeDepth(t?t.id:null,e.id);t&&(t.value.subtreeDepth=r);let d=0,l=1;"C"===g.value.element&&"C"!==s.value.element&&1i?(d=1,l=0):"C"!==g.value.element&&"C"===s.value.element&&1a?(d=0,l=1):a>i&&(d=1,l=0);let h=this.graph.vertices[o[d]],c=this.graph.vertices[o[l]],p=this.graph.getEdge(e.id,h.id),v=this.graph.getEdge(e.id,c.id),f=!1;rn&&i>a?(g=this.graph.vertices[o[1]],s=this.graph.vertices[o[0]],l=this.graph.vertices[o[2]]):a>n&&a>i&&(g=this.graph.vertices[o[2]],s=this.graph.vertices[o[0]],l=this.graph.vertices[o[1]]),t&&1>t.value.rings.length&&1>g.value.rings.length&&1>s.value.rings.length&&1>l.value.rings.length&&1===this.graph.getTreeDepth(s.id,e.id)&&1===this.graph.getTreeDepth(l.id,e.id)&&1t&&n>i&&n>r?(a=this.graph.vertices[o[1]],s=this.graph.vertices[o[0]],g=this.graph.vertices[o[2]],l=this.graph.vertices[o[3]]):i>t&&i>n&&i>r?(a=this.graph.vertices[o[2]],s=this.graph.vertices[o[0]],g=this.graph.vertices[o[1]],l=this.graph.vertices[o[3]]):r>t&&r>n&&r>i&&(a=this.graph.vertices[o[3]],s=this.graph.vertices[o[0]],g=this.graph.vertices[o[1]],l=this.graph.vertices[o[2]]),a.angle=-d.toRad(36),s.angle=d.toRad(36),g.angle=-d.toRad(108),l.angle=d.toRad(108),this.createNextBond(a,e,u+a.angle),this.createNextBond(s,e,u+s.angle),this.createNextBond(g,e,u+g.angle),this.createNextBond(l,e,u+l.angle)}}}getCommonRingbondNeighbour(e){let t=e.neighbours;for(var n=0;ng&&(g=s[t][1].length);for(var n=0;nl&&(l=s[t][1][n].length)}for(var t=0;tt[1][n][i])return-1;if(e[1][n][i]t[0]?-1:e[0]n.getNeighbourCount()||0o)continue;let s=null;for(var t=0;tn[r][a]+n[a][o]&&(n[r][o]=n[r][a]+n[a][o]);return n}getSubgraphDistanceMatrix(e){let t=e.length,n=this.getSubgraphAdjacencyMatrix(e),r=Array(t);for(var o=0;or[o][s]+r[s][a]&&(r[o][a]=r[o][s]+r[s][a]);return r}getAdjacencyList(){let e=this.vertices.length,t=Array(e);for(var n=0;nr&&(r=i)}return r+1}traverseTree(e,t,n,r=g,o=!1,a=1,s=null){var g=Number.MAX_SAFE_INTEGER;if(null===s&&(s=new Uint8Array(this.vertices.length)),a>r+1||1===s[e])return;s[e]=1;let d=this.vertices[e],l=d.getNeighbours(t);(!o||1e&&!1===A[v]&&(e=r,t=v,n=o,i=a)}return[t,e,n,i]},E=function(e,t,i){let o=0,a=0,s=0,g=C[e],d=R[e],h=T[e],c=k[e];for(v=b;v--;){if(v===e)continue;let t=C[v],i=R[v],r=h[v],l=c[v],p=(g-t)*(g-t),u=1/n(p+(d-i)*(d-i),1.5);o+=l*(1-r*(d-i)*(d-i)*u),a+=l*(1-r*p*u),s+=l*(r*(g-t)*(d-i)*u)}0==o&&(o=.1),0===a&&(a=.1),0===s&&(s=.1);let p=t/o+i/s;p/=s/o-a/s;let l=-(s*p+t)/o;C[e]+=l,R[e]+=p;let u=I[e];t=0,i=0,g=C[e],d=R[e];let f,m,x,y,S;for(v=b;v--;)e!==v&&(f=C[v],m=R[v],x=u[v][0],y=u[v][1],S=1/r((g-f)*(g-f)+(d-m)*(d-m)),l=c[v]*(g-f-h[v]*(g-f)*S),p=c[v]*(d-m-h[v]*(d-m)*S),u[v]=[l,p],t+=l,i+=p,B[v]+=l-x,w[v]+=p-y);B[e]=t,w[e]=i},O=0,H=0,D=0,N=0,V=0,z=0;for(;u>l&&c>V;)for(V++,[O,u,H,D]=L(),N=u,z=0;N>h&&p>z;)z++,E(O,H,D),[N,H,D]=P(O);for(v=b;v--;){let t=e[v],n=this.vertices[t];n.position.x=C[v],n.position.y=R[v],n.positioned=!0,n.forcePositioned=!0}}_bridgeDfs(e,t,n,r,a,s,g){t[e]=!0,n[e]=r[e]=++this._time;for(var d=0;dn[e]&&g.push([e,i]))}}static getConnectedComponents(e){let t=e.length,n=Array(t),i=[],r=0;n.fill(!1);for(var o=0;oVe&&(Ve=He,ze=[]),ze.push(e))}function g(){var e,t,n,i,r,o,a,s,c,p;if(e=He,t=He,n=l(),n!==I){for(i=[],r=d();r!==I;)i.push(r),r=d();if(i!==I){for(r=[],o=He,a=h(),a===I&&(a=null),a===I?(He=o,o=I):(s=m(),s===I?(He=o,o=I):(a=[a,s],o=a));o!==I;)r.push(o),o=He,a=h(),a===I&&(a=null),a===I?(He=o,o=I):(s=m(),s===I?(He=o,o=I):(a=[a,s],o=a));if(r!==I){for(o=[],a=d();a!==I;)o.push(a),a=d();if(o===I)He=t,t=I;else if(a=h(),a===I&&(a=null),a===I)He=t,t=I;else if(s=g(),s===I&&(s=null),s!==I){for(c=[],p=d();p!==I;)c.push(p),p=d();c===I?(He=t,t=I):(n=[n,i,r,o,a,s,c],t=n)}else He=t,t=I}else He=t,t=I}else He=t,t=I}else He=t,t=I;return t!==I&&(De=e,t=P(t)),e=t,e}function d(){var e,n,i,r,o,a;return e=He,n=He,40===t.charCodeAt(He)?(i="(",He++):(i=I,s(L)),i===I?(He=n,n=I):(r=h(),r===I&&(r=null),r===I?(He=n,n=I):(o=g(),o===I?(He=n,n=I):(41===t.charCodeAt(He)?(a=")",He++):(a=I,s(E)),a===I?(He=n,n=I):(i=[i,r,o,a],n=i)))),n!==I&&(De=e,n=O(n)),e=n,e}function l(){var e,t;return e=He,t=p(),t===I&&(t=u(),t===I&&(t=c(),t===I&&(t=v()))),t!==I&&(De=e,t=H(t)),e=t,e}function h(){var e,n;return e=He,D.test(t.charAt(He))?(n=t.charAt(He),He++):(n=I,s(N)),n!==I&&(De=e,n=V(n)),e=n,e}function c(){var e,n,i,r,o,a,g,d,l,h;return e=He,n=He,91===t.charCodeAt(He)?(i="[",He++):(i=I,s(z)),i===I?(He=n,n=I):(r=A(),r===I&&(r=null),r===I?(He=n,n=I):("se"===t.substr(He,2)?(o="se",He+=2):(o=I,s(W)),o===I&&("as"===t.substr(He,2)?(o="as",He+=2):(o=I,s(F)),o===I&&(o=u(),o===I&&(o=f(),o===I&&(o=v())))),o===I?(He=n,n=I):(a=b(),a===I&&(a=null),a===I?(He=n,n=I):(g=C(),g===I&&(g=null),g===I?(He=n,n=I):(d=x(),d===I&&(d=null),d===I?(He=n,n=I):(l=R(),l===I&&(l=null),l===I?(He=n,n=I):(93===t.charCodeAt(He)?(h="]",He++):(h=I,s(M)),h===I?(He=n,n=I):(i=[i,r,o,a,g,d,l,h],n=i)))))))),n!==I&&(De=e,n=q(n)),e=n,e}function p(){var e,n,i,r;return e=He,n=He,66===t.charCodeAt(He)?(i="B",He++):(i=I,s(_)),i===I?(He=n,n=I):(114===t.charCodeAt(He)?(r="r",He++):(r=I,s(U)),r===I&&(r=null),r===I?(He=n,n=I):(i=[i,r],n=i)),n===I&&(n=He,67===t.charCodeAt(He)?(i="C",He++):(i=I,s(G)),i===I?(He=n,n=I):(108===t.charCodeAt(He)?(r="l",He++):(r=I,s(X)),r===I&&(r=null),r===I?(He=n,n=I):(i=[i,r],n=i)),n===I&&(Y.test(t.charAt(He))?(n=t.charAt(He),He++):(n=I,s(K)))),n!==I&&(De=e,n=Z(n)),e=n,e}function u(){var e,n;return e=He,$.test(t.charAt(He))?(n=t.charAt(He),He++):(n=I,s(J)),n!==I&&(De=e,n=H(n)),e=n,e}function v(){var e,n;return e=He,42===t.charCodeAt(He)?(n="*",He++):(n=I,s(Q)),n!==I&&(De=e,n=ee(n)),e=n,e}function f(){var e,n,i,r;return e=He,n=He,te.test(t.charAt(He))?(i=t.charAt(He),He++):(i=I,s(ne)),i===I?(He=n,n=I):(ie.test(t.charAt(He))?(r=t.charAt(He),He++):(r=I,s(re)),r===I&&(r=null),r===I?(He=n,n=I):(i=[i,r],n=i)),n!==I&&(De=e,n=oe(n)),e=n,e}function m(){var e,n,i,r,o;return e=He,n=He,37===t.charCodeAt(He)?(i="%",He++):(i=I,s(ae)),i===I?(He=n,n=I):(se.test(t.charAt(He))?(r=t.charAt(He),He++):(r=I,s(ge)),r===I?(He=n,n=I):(de.test(t.charAt(He))?(o=t.charAt(He),He++):(o=I,s(le)),o===I?(He=n,n=I):(i=[i,r,o],n=i))),n===I&&(de.test(t.charAt(He))?(n=t.charAt(He),He++):(n=I,s(le))),n!==I&&(De=e,n=he(n)),e=n,e}function b(){var e,n,i,r,o,a,g;return e=He,n=He,64===t.charCodeAt(He)?(i="@",He++):(i=I,s(ce)),i===I?(He=n,n=I):(64===t.charCodeAt(He)?(r="@",He++):(r=I,s(ce)),r===I&&(r=He,"TH"===t.substr(He,2)?(o="TH",He+=2):(o=I,s(pe)),o===I?(He=r,r=I):(ue.test(t.charAt(He))?(a=t.charAt(He),He++):(a=I,s(ve)),a===I?(He=r,r=I):(o=[o,a],r=o)),r===I&&(r=He,"AL"===t.substr(He,2)?(o="AL",He+=2):(o=I,s(fe)),o===I?(He=r,r=I):(ue.test(t.charAt(He))?(a=t.charAt(He),He++):(a=I,s(ve)),a===I?(He=r,r=I):(o=[o,a],r=o)),r===I&&(r=He,"SP"===t.substr(He,2)?(o="SP",He+=2):(o=I,s(me)),o===I?(He=r,r=I):(be.test(t.charAt(He))?(a=t.charAt(He),He++):(a=I,s(xe)),a===I?(He=r,r=I):(o=[o,a],r=o)),r===I&&(r=He,"TB"===t.substr(He,2)?(o="TB",He+=2):(o=I,s(ye)),o===I?(He=r,r=I):(se.test(t.charAt(He))?(a=t.charAt(He),He++):(a=I,s(ge)),a===I?(He=r,r=I):(de.test(t.charAt(He))?(g=t.charAt(He),He++):(g=I,s(le)),g===I&&(g=null),g===I?(He=r,r=I):(o=[o,a,g],r=o))),r===I&&(r=He,"OH"===t.substr(He,2)?(o="OH",He+=2):(o=I,s(Se)),o===I?(He=r,r=I):(se.test(t.charAt(He))?(a=t.charAt(He),He++):(a=I,s(ge)),a===I?(He=r,r=I):(de.test(t.charAt(He))?(g=t.charAt(He),He++):(g=I,s(le)),g===I&&(g=null),g===I?(He=r,r=I):(o=[o,a,g],r=o)))))))),r===I&&(r=null),r===I?(He=n,n=I):(i=[i,r],n=i)),n!==I&&(De=e,n=Ce(n)),e=n,e}function x(){var e,t;return e=He,t=y(),t===I&&(t=S()),t!==I&&(De=e,t=Re(t)),e=t,e}function y(){var e,n,i,r,o,a;return e=He,n=He,43===t.charCodeAt(He)?(i="+",He++):(i=I,s(Ae)),i===I?(He=n,n=I):(43===t.charCodeAt(He)?(r="+",He++):(r=I,s(Ae)),r===I&&(r=He,se.test(t.charAt(He))?(o=t.charAt(He),He++):(o=I,s(ge)),o===I?(He=r,r=I):(de.test(t.charAt(He))?(a=t.charAt(He),He++):(a=I,s(le)),a===I&&(a=null),a===I?(He=r,r=I):(o=[o,a],r=o))),r===I&&(r=null),r===I?(He=n,n=I):(i=[i,r],n=i)),n!==I&&(De=e,n=je(n)),e=n,e}function S(){var e,n,i,r,o,a;return e=He,n=He,45===t.charCodeAt(He)?(i="-",He++):(i=I,s(Te)),i===I?(He=n,n=I):(45===t.charCodeAt(He)?(r="-",He++):(r=I,s(Te)),r===I&&(r=He,se.test(t.charAt(He))?(o=t.charAt(He),He++):(o=I,s(ge)),o===I?(He=r,r=I):(de.test(t.charAt(He))?(a=t.charAt(He),He++):(a=I,s(le)),a===I&&(a=null),a===I?(He=r,r=I):(o=[o,a],r=o))),r===I&&(r=null),r===I?(He=n,n=I):(i=[i,r],n=i)),n!==I&&(De=e,n=ke(n)),e=n,e}function C(){var e,n,i,r;return e=He,n=He,72===t.charCodeAt(He)?(i="H",He++):(i=I,s(Ie)),i===I?(He=n,n=I):(de.test(t.charAt(He))?(r=t.charAt(He),He++):(r=I,s(le)),r===I&&(r=null),r===I?(He=n,n=I):(i=[i,r],n=i)),n!==I&&(De=e,n=Be(n)),e=n,e}function R(){var e,n,i,r,o,a,g;if(e=He,n=He,58===t.charCodeAt(He)?(i=":",He++):(i=I,s(we)),i!==I){if(r=He,se.test(t.charAt(He))?(o=t.charAt(He),He++):(o=I,s(ge)),o!==I){for(a=[],de.test(t.charAt(He))?(g=t.charAt(He),He++):(g=I,s(le));g!==I;)a.push(g),de.test(t.charAt(He))?(g=t.charAt(He),He++):(g=I,s(le));a===I?(He=r,r=I):(o=[o,a],r=o)}else He=r,r=I;r===I&&(Pe.test(t.charAt(He))?(r=t.charAt(He),He++):(r=I,s(Le))),r===I?(He=n,n=I):(i=[i,r],n=i)}else He=n,n=I;return n!==I&&(De=e,n=Ee(n)),e=n,e}function A(){var e,n,i,r,o;return e=He,n=He,se.test(t.charAt(He))?(i=t.charAt(He),He++):(i=I,s(ge)),i===I?(He=n,n=I):(de.test(t.charAt(He))?(r=t.charAt(He),He++):(r=I,s(le)),r===I&&(r=null),r===I?(He=n,n=I):(de.test(t.charAt(He))?(o=t.charAt(He),He++):(o=I,s(le)),o===I&&(o=null),o===I?(He=n,n=I):(i=[i,r,o],n=i))),n!==I&&(De=e,n=Oe(n)),e=n,e}n=void 0===n?{}:n;var j=t.split("(").length-1,T=t.split(")").length-1;if(j!==T)throw function(t,n){return new e(t,null,null,n)}("The number of opening parentheses does not match the number of closing parentheses.",0);var k,I={},B={chain:g},w=g,P=function(e){for(var t=[],n=[],r=0;ro;){let a=r;t(a),r=e[r].getNextInRing(e,this.id,i),i=a,r==n&&(r=null),o++}}getOrderedNeighbours(e){let t=Array(this.neighbours.length);for(let n,r=0;rt){var g,d,h;if(e===t+1)for(s[c][p]=[a[c][p].length],g=a[c][p].length;g--;)for(s[c][p][g]=[a[c][p][g].length],d=a[c][p][g].length;d--;)for(s[c][p][g][d]=[a[c][p][g][d].length],h=a[c][p][g][d].length;h--;)s[c][p][g][d][h]=[a[c][p][g][d][0],a[c][p][g][d][1]];else s[c][p]=[];for(o[c][p]=t,a[c][p]=[[]],g=a[c][u][0].length;g--;)a[c][p][0].push(a[c][u][0][g]);for(g=a[u][p][0].length;g--;)a[c][p][0].push(a[u][p][0][g])}else if(e===t){if(a[c][u].length&&a[u][p].length){var g;if(a[c][p].length){let e=[];for(g=a[c][u][0].length;g--;)e.push(a[c][u][0][g]);for(g=a[u][p][0].length;g--;)e.push(a[u][p][0][g]);a[c][p].push(e)}else{let e=[];for(g=a[c][u][0].length;g--;)e.push(a[c][u][0][g]);for(g=a[u][p][0].length;g--;)e.push(a[u][p][0][g]);a[c][p][0]=e}}}else if(e===t-1){var g;if(s[c][p].length){let e=[];for(g=a[c][u][0].length;g--;)e.push(a[c][u][0][g]);for(g=a[u][p][0].length;g--;)e.push(a[u][p][0][g]);s[c][p].push(e)}else{let e=[];for(g=a[c][u][0].length;g--;)e.push(a[c][u][0][g]);for(g=a[u][p][0].length;g--;)e.push(a[u][p][0][g]);s[c][p][0]=e}}}return{d:o,pe:a,pe_prime:s}}static getRingCandidates(e,t,n){let r=e.length,o=[],a=0;for(let s=0;sg)return d}else for(let t,i=0;ig)return d}return d}static getEdgeCount(e){let t=0,n=e.length;for(var r=n-1;r--;)for(var o=n;o--;)1===e[r][o]&&t++;return t}static getEdgeList(e){let t=e.length,n=[];for(var r=t-1;r--;)for(var o=t;o--;)1===e[r][o]&&n.push([r,o]);return n}static bondsToAtoms(e){let t=new Set;for(var n=e.length;n--;)t.add(e[n][0]),t.add(e[n][1]);return t}static getBondCount(e,t){let n=0;for(let i of e)for(let r of e)i!==r&&(n+=t[i][r]);return n/2}static pathSetsContain(e,t,n,o,a,s){for(var g=e.length;g--;){if(r.isSupersetOf(t,e[g]))return!0;if(e[g].size===t.size&&r.areSetsEqual(e[g],t))return!0}let d=0,l=!1;for(g=n.length;g--;)for(var h=o.length;h--;)(n[g][0]===o[h][0]&&n[g][1]===o[h][1]||n[g][1]===o[h][0]&&n[g][0]===o[h][1])&&d++,d===n.length&&(l=!0);let c=!1;if(l)for(let e of t)if(s[e]n)return-1;return t===n?0:1}relativeClockwise(e,t){let n=(this.y-e.y)*(t.x-e.x),i=(this.x-e.x)*(t.y-e.y);if(n>i)return-1;return n===i?0:1}rotate(e){let t=new g(0,0),n=r(e),o=i(e);return t.x=this.x*n-this.y*o,t.y=this.x*o+this.y*n,this.x=t.x,this.y=t.y,this}rotateAround(e,t){let n=i(e),o=r(e);this.x-=t.x,this.y-=t.y;let a=this.x*o-this.y*n,s=this.x*n+this.y*o;return this.x=a+t.x,this.y=s+t.y,this}rotateTo(e,t,n=0){this.x+=.001,this.y-=.001;let i=g.subtract(this,t),r=g.subtract(e,t),o=g.angle(r,i);return this.rotateAround(o+n,t),this}rotateAwayFrom(e,t,n){this.rotateAround(n,t);let i=this.distanceSq(e);this.rotateAround(-2*n,t);let r=this.distanceSq(e);rr?n:-n}getRotateToAngle(e,t){let n=g.subtract(this,t),i=g.subtract(e,t),r=g.angle(i,n);return Number.isNaN(r)?0:r}isInPolygon(e){let t=!1;for(let n=0,i=e.length-1;nthis.y!=e[i].y>this.y&&this.x<(e[i].x-e[n].x)*(this.y-e[n].y)/(e[i].y-e[n].y)+e[n].x&&(t=!t);return t}length(){return o(this.x*this.x+this.y*this.y)}lengthSq(){return this.x*this.x+this.y*this.y}normalize(){return this.divide(this.length()),this}normalized(){return g.divideScalar(this,this.length())}whichSide(e,t){return(this.x-e.x)*(t.y-e.y)-(this.y-e.y)*(t.x-e.x)}sameSideAs(e,t,n){let i=this.whichSide(e,t),r=n.whichSide(e,t);return 0>i&&0>r||0==i&&0==r||0this.neighbours.length?n=2:n=3),null===this.value.bracket.hcount&&0===t&&(n=1),null===this.value.bracket.hcount&&1===t&&(3>this.neighbours.length?n=1:n=2),this.neighbours.splice(n,0,e)}else this.neighbours.push(e);this.neighbourCount++}setParentVertexId(e){this.neighbourCount++,this.parentVertexId=e,this.neighbours.push(e)}isTerminal(){return!!this.value.hasAttachedPseudoElements||null===this.parentVertexId&&2>this.children.length||0===this.children.length}clone(){let e=new g(this.value,this.position.x,this.position.y);return e.id=this.id,e.previousPosition=new a(this.previousPosition.x,this.previousPosition.y),e.parentVertexId=this.parentVertexId,e.children=o.clone(this.children),e.spanningTreeChildren=o.clone(this.spanningTreeChildren),e.edges=o.clone(this.edges),e.positioned=this.positioned,e.angle=this.angle,e.forcePositioned=this.forcePositioned,e}equals(e){return this.id===e.id}getAngle(e=null,t=!1){let n=null;return n=e?a.subtract(this.position,e):a.subtract(this.position,this.previousPosition),t?r.toDeg(n.angle()):n.angle()}getTextDirection(e){let t=this.getDrawnNeighbours(e),o=[];for(let n=0;n>> 0;\r\n\r\n // Steps 6-7.\r\n var start = arguments[1];\r\n var relativeStart = start >> 0;\r\n\r\n // Step 8.\r\n var k = relativeStart < 0 ?\r\n Math.max(len + relativeStart, 0) :\r\n Math.min(relativeStart, len);\r\n\r\n // Steps 9-10.\r\n var end = arguments[2];\r\n var relativeEnd = end === undefined ?\r\n len : end >> 0;\r\n\r\n // Step 11.\r\n var final = relativeEnd < 0 ?\r\n Math.max(len + relativeEnd, 0) :\r\n Math.min(relativeEnd, len);\r\n\r\n // Step 12.\r\n while (k < final) {\r\n O[k] = value;\r\n k++;\r\n }\r\n\r\n // Step 13.\r\n return O;\r\n }\r\n});\r\n}\r\n\r\nmodule.exports = SmilesDrawer;","//@ts-check\r\n\r\n/** \r\n * A static class containing helper functions for array-related tasks. \r\n */\r\nclass ArrayHelper {\r\n /**\r\n * Clone an array or an object. If an object is passed, a shallow clone will be created.\r\n *\r\n * @static\r\n * @param {*} arr The array or object to be cloned.\r\n * @returns {*} A clone of the array or object.\r\n */\r\n static clone(arr) {\r\n let out = Array.isArray(arr) ? Array() : {};\r\n \r\n for (let key in arr) {\r\n let value = arr[key];\r\n \r\n if (typeof value.clone === 'function') {\r\n out[key] = value.clone();\r\n }\r\n else {\r\n out[key] = (typeof value === 'object') ? ArrayHelper.clone(value) : value;\r\n }\r\n }\r\n \r\n return out;\r\n }\r\n\r\n /**\r\n * Returns a boolean indicating whether or not the two arrays contain the same elements.\r\n * Only supports 1d, non-nested arrays.\r\n *\r\n * @static\r\n * @param {Array} arrA An array.\r\n * @param {Array} arrB An array.\r\n * @returns {Boolean} A boolean indicating whether or not the two arrays contain the same elements.\r\n */\r\n static equals(arrA, arrB) {\r\n if (arrA.length !== arrB.length) {\r\n return false;\r\n }\r\n\r\n let tmpA = arrA.slice().sort();\r\n let tmpB = arrB.slice().sort();\r\n\r\n for (var i = 0; i < tmpA.length; i++) {\r\n if (tmpA[i] !== tmpB[i]) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Returns a string representation of an array. If the array contains objects with an id property, the id property is printed for each of the elements.\r\n *\r\n * @static\r\n * @param {Object[]} arr An array.\r\n * @param {*} arr[].id If the array contains an object with the property 'id', the properties value is printed. Else, the array elements value is printend.\r\n * @returns {String} A string representation of the array.\r\n */\r\n static print(arr) {\r\n if (arr.length == 0) {\r\n return '';\r\n }\r\n\r\n let s = '(';\r\n\r\n for (let i = 0; i < arr.length; i++) {\r\n s += arr[i].id ? arr[i].id + ', ' : arr[i] + ', ';\r\n }\r\n\r\n s = s.substring(0, s.length - 2);\r\n\r\n return s + ')';\r\n }\r\n\r\n /**\r\n * Run a function for each element in the array. The element is supplied as an argument for the callback function\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {Function} callback The callback function that is called for each element.\r\n */\r\n static each(arr, callback) {\r\n for (let i = 0; i < arr.length; i++) {\r\n callback(arr[i]);\r\n }\r\n }\r\n\r\n /**\r\n * Return the array element from an array containing objects, where a property of the object is set to a given value.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {(String|Number)} property A property contained within an object in the array.\r\n * @param {(String|Number)} value The value of the property.\r\n * @returns {*} The array element matching the value.\r\n */\r\n static get(arr, property, value) {\r\n for (let i = 0; i < arr.length; i++) {\r\n if (arr[i][property] == value) {\r\n return arr[i];\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Checks whether or not an array contains a given value. the options object passed as a second argument can contain three properties. value: The value to be searched for. property: The property that is to be searched for a given value. func: A function that is used as a callback to return either true or false in order to do a custom comparison.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {Object} options See method description.\r\n * @param {*} options.value The value for which to check.\r\n * @param {String} [options.property=undefined] The property on which to check.\r\n * @param {Function} [options.func=undefined] A custom property function.\r\n * @returns {Boolean} A boolean whether or not the array contains a value.\r\n */\r\n static contains(arr, options) {\r\n if (!options.property && !options.func) {\r\n for (let i = 0; i < arr.length; i++) {\r\n if (arr[i] == options.value) {\r\n return true;\r\n }\r\n }\r\n } else if (options.func) {\r\n for (let i = 0; i < arr.length; i++) {\r\n if (options.func(arr[i])) {\r\n return true;\r\n }\r\n }\r\n } else {\r\n for (let i = 0; i < arr.length; i++) {\r\n if (arr[i][options.property] == options.value) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Returns an array containing the intersection between two arrays. That is, values that are common to both arrays.\r\n *\r\n * @static\r\n * @param {Array} arrA An array.\r\n * @param {Array} arrB An array.\r\n * @returns {Array} The intersecting vlaues.\r\n */\r\n static intersection(arrA, arrB) {\r\n let intersection = new Array();\r\n \r\n for (let i = 0; i < arrA.length; i++) {\r\n for (let j = 0; j < arrB.length; j++) {\r\n if (arrA[i] === arrB[j]) {\r\n intersection.push(arrA[i]);\r\n }\r\n }\r\n }\r\n\r\n return intersection;\r\n }\r\n\r\n /**\r\n * Returns an array of unique elements contained in an array.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @returns {Array} An array of unique elements contained within the array supplied as an argument.\r\n */\r\n static unique(arr) {\r\n let contains = {};\r\n return arr.filter(function (i) {\r\n // using !== instead of hasOwnProperty (http://andrew.hedges.name/experiments/in/)\r\n return contains[i] !== undefined ? false : (contains[i] = true);\r\n });\r\n }\r\n\r\n /**\r\n * Count the number of occurences of a value in an array.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {*} value A value to be counted.\r\n * @returns {Number} The number of occurences of a value in the array.\r\n */\r\n static count(arr, value) {\r\n let count = 0;\r\n\r\n for (let i = 0; i < arr.length; i++) {\r\n if (arr[i] === value) {\r\n count++;\r\n }\r\n }\r\n\r\n return count;\r\n }\r\n\r\n /**\r\n * Toggles the value of an array. If a value is not contained in an array, the array returned will contain all the values of the original array including the value. If a value is contained in an array, the array returned will contain all the values of the original array excluding the value.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {*} value A value to be toggled.\r\n * @returns {Array} The toggled array.\r\n */\r\n static toggle(arr, value) {\r\n let newArr = Array();\r\n\r\n let removed = false;\r\n for (let i = 0; i < arr.length; i++) {\r\n // Do not copy value if it exists\r\n if (arr[i] !== value) {\r\n newArr.push(arr[i]);\r\n } else {\r\n // The element was not copied to the new array, which\r\n // means it was removed\r\n removed = true;\r\n }\r\n }\r\n\r\n // If the element was not removed, then it was not in the array\r\n // so add it\r\n if (!removed) {\r\n newArr.push(value);\r\n }\r\n\r\n return newArr;\r\n }\r\n\r\n /**\r\n * Remove a value from an array.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {*} value A value to be removed.\r\n * @returns {Array} A new array with the element with a given value removed.\r\n */\r\n static remove(arr, value) {\r\n let tmp = Array();\r\n\r\n for (let i = 0; i < arr.length; i++) {\r\n if (arr[i] !== value) {\r\n tmp.push(arr[i]);\r\n }\r\n }\r\n\r\n return tmp;\r\n }\r\n\r\n /**\r\n * Remove a value from an array with unique values.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {*} value A value to be removed.\r\n * @returns {Array} An array with the element with a given value removed.\r\n */\r\n static removeUnique(arr, value) {\r\n let index = arr.indexOf(value);\r\n\r\n if (index > -1) {\r\n arr.splice(index, 1);\r\n }\r\n\r\n return arr;\r\n }\r\n\r\n /**\r\n * Remove all elements contained in one array from another array.\r\n *\r\n * @static\r\n * @param {Array} arrA The array to be filtered.\r\n * @param {Array} arrB The array containing elements that will be removed from the other array.\r\n * @returns {Array} The filtered array.\r\n */\r\n static removeAll(arrA, arrB) {\r\n return arrA.filter(function (item) {\r\n return arrB.indexOf(item) === -1;\r\n });\r\n }\r\n\r\n /**\r\n * Merges two arrays and returns the result. The first array will be appended to the second array.\r\n *\r\n * @static\r\n * @param {Array} arrA An array.\r\n * @param {Array} arrB An array.\r\n * @returns {Array} The merged array.\r\n */\r\n static merge(arrA, arrB) {\r\n let arr = new Array(arrA.length + arrB.length);\r\n\r\n for (let i = 0; i < arrA.length; i++) {\r\n arr[i] = arrA[i];\r\n }\r\n\r\n for (let i = 0; i < arrB.length; i++) {\r\n arr[arrA.length + i] = arrB[i];\r\n }\r\n\r\n return arr;\r\n }\r\n\r\n /**\r\n * Checks whether or not an array contains all the elements of another array, without regard to the order.\r\n *\r\n * @static\r\n * @param {Array} arrA An array.\r\n * @param {Array} arrB An array.\r\n * @returns {Boolean} A boolean indicating whether or not both array contain the same elements.\r\n */\r\n static containsAll(arrA, arrB) {\r\n let containing = 0;\r\n for (let i = 0; i < arrA.length; i++) {\r\n for (let j = 0; j < arrB.length; j++) {\r\n if (arrA[i] === arrB[j]) {\r\n containing++;\r\n }\r\n }\r\n }\r\n\r\n return containing === arrB.length;\r\n }\r\n \r\n /**\r\n * Sort an array of atomic number information. Where the number is indicated as x, x.y, x.y.z, ...\r\n *\r\n * @param {Object[]} arr An array of vertex ids with their associated atomic numbers.\r\n * @param {Number} arr[].vertexId A vertex id.\r\n * @param {String} arr[].atomicNumber The atomic number associated with the vertex id.\r\n * @returns {Object[]} The array sorted by atomic number. Example of an array entry: { atomicNumber: 2, vertexId: 5 }.\r\n */\r\n static sortByAtomicNumberDesc(arr) {\r\n let map = arr.map(function(e, i) {\r\n return { index: i, value: e.atomicNumber.split('.').map(Number) };\r\n });\r\n\r\n map.sort(function(a, b) {\r\n let min = Math.min(b.value.length, a.value.length);\r\n let i = 0;\r\n \r\n while(i < min && b.value[i] === a.value[i]) {\r\n i++;\r\n }\r\n\r\n return i === min ? b.value.length - a.value.length : b.value[i] - a.value[i];\r\n });\r\n\r\n return map.map(function(e) {\r\n return arr[e.index];\r\n });\r\n }\r\n\r\n /**\r\n * Copies a an n-dimensional array.\r\n * \r\n * @param {Array} arr The array to be copied.\r\n * @returns {Array} The copy.\r\n */\r\n static deepCopy(arr) {\r\n let newArr = Array();\r\n\r\n for (let i = 0; i < arr.length; i++) {\r\n let item = arr[i];\r\n\r\n if (item instanceof Array) {\r\n newArr[i] = ArrayHelper.deepCopy(item);\r\n } else {\r\n newArr[i] = item;\r\n }\r\n }\r\n\r\n return newArr;\r\n }\r\n\r\n}\r\n\r\nmodule.exports = ArrayHelper;","//@ts-check\r\nconst ArrayHelper = require('./ArrayHelper')\r\nconst Vertex = require('./Vertex')\r\nconst Ring = require('./Ring')\r\n\r\n/** \r\n * A class representing an atom.\r\n * \r\n * @property {String} element The element symbol of this atom. Single-letter symbols are always uppercase. Examples: H, C, F, Br, Si, ...\r\n * @property {Boolean} drawExplicit A boolean indicating whether or not this atom is drawn explicitly (for example, a carbon atom). This overrides the default behaviour.\r\n * @property {Object[]} ringbonds An array containing the ringbond ids and bond types as specified in the original SMILE.\r\n * @property {String} branchBond The branch bond as defined in the SMILES.\r\n * @property {Number} ringbonds[].id The ringbond id as defined in the SMILES.\r\n * @property {String} ringbonds[].bondType The bond type of the ringbond as defined in the SMILES.\r\n * @property {Number[]} rings The ids of rings which contain this atom.\r\n * @property {String} bondType The bond type associated with this array. Examples: -, =, #, ...\r\n * @property {Boolean} isBridge A boolean indicating whether or not this atom is part of a bridge in a bridged ring (contained by the largest ring).\r\n * @property {Boolean} isBridgeNode A boolean indicating whether or not this atom is a bridge node (a member of the largest ring in a bridged ring which is connected to a bridge-atom).\r\n * @property {Number[]} originalRings Used to back up rings when they are replaced by a bridged ring.\r\n * @property {Number} bridgedRing The id of the bridged ring if the atom is part of a bridged ring.\r\n * @property {Number[]} anchoredRings The ids of the rings that are anchored to this atom. The centers of anchored rings are translated when this atom is translated.\r\n * @property {Object} bracket If this atom is defined as a bracket atom in the original SMILES, this object contains all the bracket information. Example: { hcount: {Number}, charge: ['--', '-', '+', '++'], isotope: {Number} }.\r\n * @property {Number} plane Specifies on which \"plane\" the atoms is in stereochemical deptictions (-1 back, 0 middle, 1 front).\r\n * @property {Object[]} attachedPseudoElements A map with containing information for pseudo elements or concatinated elements. The key is comprised of the element symbol and the hydrogen count.\r\n * @property {String} attachedPseudoElement[].element The element symbol.\r\n * @property {Number} attachedPseudoElement[].count The number of occurences that match the key.\r\n * @property {Number} attachedPseudoElement[].hyrogenCount The number of hydrogens attached to each atom matching the key.\r\n * @property {Boolean} hasAttachedPseudoElements A boolean indicating whether or not this attom will be drawn with an attached pseudo element or concatinated elements.\r\n * @property {Boolean} isDrawn A boolean indicating whether or not this atom is drawn. In contrast to drawExplicit, the bond is drawn neither.\r\n * @property {Boolean} isConnectedToRing A boolean indicating whether or not this atom is directly connected (but not a member of) a ring.\r\n * @property {String[]} neighbouringElements An array containing the element symbols of neighbouring atoms.\r\n * @property {Boolean} isPartOfAromaticRing A boolean indicating whether or not this atom is part of an explicitly defined aromatic ring. Example: c1ccccc1.\r\n * @property {Number} bondCount The number of bonds in which this atom is participating.\r\n * @property {String} chirality The chirality of this atom if it is a stereocenter (R or S).\r\n * @property {Number} priority The priority of this atom acording to the CIP rules, where 0 is the highest priority.\r\n * @property {Boolean} mainChain A boolean indicating whether or not this atom is part of the main chain (used for chirality).\r\n * @property {String} hydrogenDirection The direction of the hydrogen, either up or down. Only for stereocenters with and explicit hydrogen.\r\n * @property {Number} subtreeDepth The depth of the subtree coming from a stereocenter.\r\n */\r\nclass Atom {\r\n /**\r\n * The constructor of the class Atom.\r\n *\r\n * @param {String} element The one-letter code of the element.\r\n * @param {String} [bondType='-'] The type of the bond associated with this atom.\r\n */\r\n constructor(element, bondType = '-') {\r\n this.element = element.length === 1 ? element.toUpperCase() : element;\r\n this.drawExplicit = false;\r\n this.ringbonds = Array();\r\n this.rings = Array();\r\n this.bondType = bondType;\r\n this.branchBond = null;\r\n this.isBridge = false;\r\n this.isBridgeNode = false;\r\n this.originalRings = Array();\r\n this.bridgedRing = null;\r\n this.anchoredRings = Array();\r\n this.bracket = null;\r\n this.plane = 0;\r\n this.attachedPseudoElements = {};\r\n this.hasAttachedPseudoElements = false;\r\n this.isDrawn = true;\r\n this.isConnectedToRing = false;\r\n this.neighbouringElements = Array();\r\n this.isPartOfAromaticRing = element !== this.element;\r\n this.bondCount = 0;\r\n this.chirality = '';\r\n this.isStereoCenter = false;\r\n this.priority = 0;\r\n this.mainChain = false;\r\n this.hydrogenDirection = 'down';\r\n this.subtreeDepth = 1;\r\n this.hasHydrogen = false;\r\n }\r\n\r\n /**\r\n * Adds a neighbouring element to this atom.\r\n * \r\n * @param {String} element A string representing an element.\r\n */\r\n addNeighbouringElement(element) {\r\n this.neighbouringElements.push(element);\r\n }\r\n\r\n /**\r\n * Attaches a pseudo element (e.g. Ac) to the atom.\r\n * @param {String} element The element identifier (e.g. Br, C, ...).\r\n * @param {String} previousElement The element that is part of the main chain (not the terminals that are converted to the pseudo element or concatinated).\r\n * @param {Number} [hydrogenCount=0] The number of hydrogens for the element.\r\n * @param {Number} [charge=0] The charge for the element.\r\n */\r\n attachPseudoElement(element, previousElement, hydrogenCount = 0, charge = 0) {\r\n if (hydrogenCount === null) {\r\n hydrogenCount = 0;\r\n }\r\n\r\n if (charge === null) {\r\n charge = 0;\r\n }\r\n\r\n let key = hydrogenCount + element + charge;\r\n\r\n if (this.attachedPseudoElements[key]) {\r\n this.attachedPseudoElements[key].count += 1;\r\n } else {\r\n this.attachedPseudoElements[key] = {\r\n element: element,\r\n count: 1,\r\n hydrogenCount: hydrogenCount,\r\n previousElement: previousElement,\r\n charge: charge\r\n };\r\n }\r\n\r\n this.hasAttachedPseudoElements = true;\r\n }\r\n\r\n /**\r\n * Returns the attached pseudo elements sorted by hydrogen count (ascending).\r\n *\r\n * @returns {Object} The sorted attached pseudo elements.\r\n */\r\n getAttachedPseudoElements() {\r\n let ordered = {};\r\n let that = this;\r\n\r\n Object.keys(this.attachedPseudoElements).sort().forEach(function (key) {\r\n ordered[key] = that.attachedPseudoElements[key];\r\n });\r\n\r\n return ordered;\r\n }\r\n\r\n /**\r\n * Returns the number of attached pseudo elements.\r\n *\r\n * @returns {Number} The number of attached pseudo elements.\r\n */\r\n getAttachedPseudoElementsCount() {\r\n return Object.keys(this.attachedPseudoElements).length;\r\n }\r\n\r\n /**\r\n * Returns whether this atom is a heteroatom (not C and not H).\r\n *\r\n * @returns {Boolean} A boolean indicating whether this atom is a heteroatom.\r\n */\r\n isHeteroAtom() {\r\n return this.element !== 'C' && this.element !== 'H';\r\n }\r\n\r\n /**\r\n * Defines this atom as the anchor for a ring. When doing repositionings of the vertices and the vertex associated with this atom is moved, the center of this ring is moved as well.\r\n *\r\n * @param {Number} ringId A ring id.\r\n */\r\n addAnchoredRing(ringId) {\r\n if (!ArrayHelper.contains(this.anchoredRings, {\r\n value: ringId\r\n })) {\r\n this.anchoredRings.push(ringId);\r\n }\r\n }\r\n\r\n /**\r\n * Returns the number of ringbonds (breaks in rings to generate the MST of the smiles) within this atom is connected to.\r\n *\r\n * @returns {Number} The number of ringbonds this atom is connected to.\r\n */\r\n getRingbondCount() {\r\n return this.ringbonds.length;\r\n }\r\n\r\n /**\r\n * Backs up the current rings.\r\n */\r\n backupRings() {\r\n this.originalRings = Array(this.rings.length);\r\n\r\n for (let i = 0; i < this.rings.length; i++) {\r\n this.originalRings[i] = this.rings[i];\r\n }\r\n }\r\n\r\n /**\r\n * Restores the most recent backed up rings.\r\n */\r\n restoreRings() {\r\n this.rings = Array(this.originalRings.length);\r\n\r\n for (let i = 0; i < this.originalRings.length; i++) {\r\n this.rings[i] = this.originalRings[i];\r\n }\r\n }\r\n\r\n /**\r\n * Checks whether or not two atoms share a common ringbond id. A ringbond is a break in a ring created when generating the spanning tree of a structure.\r\n *\r\n * @param {Atom} atomA An atom.\r\n * @param {Atom} atomB An atom.\r\n * @returns {Boolean} A boolean indicating whether or not two atoms share a common ringbond.\r\n */\r\n haveCommonRingbond(atomA, atomB) {\r\n for (let i = 0; i < atomA.ringbonds.length; i++) {\r\n for (let j = 0; j < atomB.ringbonds.length; j++) {\r\n if (atomA.ringbonds[i].id == atomB.ringbonds[j].id) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Check whether or not the neighbouring elements of this atom equal the supplied array.\r\n * \r\n * @param {String[]} arr An array containing all the elements that are neighbouring this atom. E.g. ['C', 'O', 'O', 'N']\r\n * @returns {Boolean} A boolean indicating whether or not the neighbours match the supplied array of elements.\r\n */\r\n neighbouringElementsEqual(arr) {\r\n if (arr.length !== this.neighbouringElements.length) {\r\n return false;\r\n }\r\n\r\n arr.sort();\r\n this.neighbouringElements.sort();\r\n\r\n for (var i = 0; i < this.neighbouringElements.length; i++) {\r\n if (arr[i] !== this.neighbouringElements[i]) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Get the atomic number of this atom.\r\n * \r\n * @returns {Number} The atomic number of this atom.\r\n */\r\n getAtomicNumber() {\r\n return Atom.atomicNumbers[this.element];\r\n }\r\n\r\n /**\r\n * Get the maximum number of bonds for this atom.\r\n * \r\n * @returns {Number} The maximum number of bonds of this atom.\r\n */\r\n getMaxBonds() {\r\n return Atom.maxBonds[this.element];\r\n }\r\n\r\n /**\r\n * A map mapping element symbols to their maximum bonds.\r\n */\r\n static get maxBonds() {\r\n return {\r\n 'H': 1,\r\n 'C': 4,\r\n 'N': 3,\r\n 'O': 2,\r\n 'P': 3,\r\n 'S': 2,\r\n 'B': 3,\r\n 'F': 1,\r\n 'I': 1,\r\n 'Cl': 1,\r\n 'Br': 1\r\n };\r\n }\r\n\r\n /**\r\n * A map mapping element symbols to the atomic number.\r\n */\r\n static get atomicNumbers() {\r\n return {\r\n 'H': 1,\r\n 'He': 2,\r\n 'Li': 3,\r\n 'Be': 4,\r\n 'B': 5,\r\n 'b': 5,\r\n 'C': 6,\r\n 'c': 6,\r\n 'N': 7,\r\n 'n': 7,\r\n 'O': 8,\r\n 'o': 8,\r\n 'F': 9,\r\n 'Ne': 10,\r\n 'Na': 11,\r\n 'Mg': 12,\r\n 'Al': 13,\r\n 'Si': 14,\r\n 'P': 15,\r\n 'p': 15,\r\n 'S': 16,\r\n 's': 16,\r\n 'Cl': 17,\r\n 'Ar': 18,\r\n 'K': 19,\r\n 'Ca': 20,\r\n 'Sc': 21,\r\n 'Ti': 22,\r\n 'V': 23,\r\n 'Cr': 24,\r\n 'Mn': 25,\r\n 'Fe': 26,\r\n 'Co': 27,\r\n 'Ni': 28,\r\n 'Cu': 29,\r\n 'Zn': 30,\r\n 'Ga': 31,\r\n 'Ge': 32,\r\n 'As': 33,\r\n 'Se': 34,\r\n 'Br': 35,\r\n 'Kr': 36,\r\n 'Rb': 37,\r\n 'Sr': 38,\r\n 'Y': 39,\r\n 'Zr': 40,\r\n 'Nb': 41,\r\n 'Mo': 42,\r\n 'Tc': 43,\r\n 'Ru': 44,\r\n 'Rh': 45,\r\n 'Pd': 46,\r\n 'Ag': 47,\r\n 'Cd': 48,\r\n 'In': 49,\r\n 'Sn': 50,\r\n 'Sb': 51,\r\n 'Te': 52,\r\n 'I': 53,\r\n 'Xe': 54,\r\n 'Cs': 55,\r\n 'Ba': 56,\r\n 'La': 57,\r\n 'Ce': 58,\r\n 'Pr': 59,\r\n 'Nd': 60,\r\n 'Pm': 61,\r\n 'Sm': 62,\r\n 'Eu': 63,\r\n 'Gd': 64,\r\n 'Tb': 65,\r\n 'Dy': 66,\r\n 'Ho': 67,\r\n 'Er': 68,\r\n 'Tm': 69,\r\n 'Yb': 70,\r\n 'Lu': 71,\r\n 'Hf': 72,\r\n 'Ta': 73,\r\n 'W': 74,\r\n 'Re': 75,\r\n 'Os': 76,\r\n 'Ir': 77,\r\n 'Pt': 78,\r\n 'Au': 79,\r\n 'Hg': 80,\r\n 'Tl': 81,\r\n 'Pb': 82,\r\n 'Bi': 83,\r\n 'Po': 84,\r\n 'At': 85,\r\n 'Rn': 86,\r\n 'Fr': 87,\r\n 'Ra': 88,\r\n 'Ac': 89,\r\n 'Th': 90,\r\n 'Pa': 91,\r\n 'U': 92,\r\n 'Np': 93,\r\n 'Pu': 94,\r\n 'Am': 95,\r\n 'Cm': 96,\r\n 'Bk': 97,\r\n 'Cf': 98,\r\n 'Es': 99,\r\n 'Fm': 100,\r\n 'Md': 101,\r\n 'No': 102,\r\n 'Lr': 103,\r\n 'Rf': 104,\r\n 'Db': 105,\r\n 'Sg': 106,\r\n 'Bh': 107,\r\n 'Hs': 108,\r\n 'Mt': 109,\r\n 'Ds': 110,\r\n 'Rg': 111,\r\n 'Cn': 112,\r\n 'Uut': 113,\r\n 'Uuq': 114,\r\n 'Uup': 115,\r\n 'Uuh': 116,\r\n 'Uus': 117,\r\n 'Uuo': 118\r\n };\r\n }\r\n\r\n /**\r\n * A map mapping element symbols to the atomic mass.\r\n */\r\n static get mass() {\r\n return {\r\n 'H': 1,\r\n 'He': 2,\r\n 'Li': 3,\r\n 'Be': 4,\r\n 'B': 5,\r\n 'b': 5,\r\n 'C': 6,\r\n 'c': 6,\r\n 'N': 7,\r\n 'n': 7,\r\n 'O': 8,\r\n 'o': 8,\r\n 'F': 9,\r\n 'Ne': 10,\r\n 'Na': 11,\r\n 'Mg': 12,\r\n 'Al': 13,\r\n 'Si': 14,\r\n 'P': 15,\r\n 'p': 15,\r\n 'S': 16,\r\n 's': 16,\r\n 'Cl': 17,\r\n 'Ar': 18,\r\n 'K': 19,\r\n 'Ca': 20,\r\n 'Sc': 21,\r\n 'Ti': 22,\r\n 'V': 23,\r\n 'Cr': 24,\r\n 'Mn': 25,\r\n 'Fe': 26,\r\n 'Co': 27,\r\n 'Ni': 28,\r\n 'Cu': 29,\r\n 'Zn': 30,\r\n 'Ga': 31,\r\n 'Ge': 32,\r\n 'As': 33,\r\n 'Se': 34,\r\n 'Br': 35,\r\n 'Kr': 36,\r\n 'Rb': 37,\r\n 'Sr': 38,\r\n 'Y': 39,\r\n 'Zr': 40,\r\n 'Nb': 41,\r\n 'Mo': 42,\r\n 'Tc': 43,\r\n 'Ru': 44,\r\n 'Rh': 45,\r\n 'Pd': 46,\r\n 'Ag': 47,\r\n 'Cd': 48,\r\n 'In': 49,\r\n 'Sn': 50,\r\n 'Sb': 51,\r\n 'Te': 52,\r\n 'I': 53,\r\n 'Xe': 54,\r\n 'Cs': 55,\r\n 'Ba': 56,\r\n 'La': 57,\r\n 'Ce': 58,\r\n 'Pr': 59,\r\n 'Nd': 60,\r\n 'Pm': 61,\r\n 'Sm': 62,\r\n 'Eu': 63,\r\n 'Gd': 64,\r\n 'Tb': 65,\r\n 'Dy': 66,\r\n 'Ho': 67,\r\n 'Er': 68,\r\n 'Tm': 69,\r\n 'Yb': 70,\r\n 'Lu': 71,\r\n 'Hf': 72,\r\n 'Ta': 73,\r\n 'W': 74,\r\n 'Re': 75,\r\n 'Os': 76,\r\n 'Ir': 77,\r\n 'Pt': 78,\r\n 'Au': 79,\r\n 'Hg': 80,\r\n 'Tl': 81,\r\n 'Pb': 82,\r\n 'Bi': 83,\r\n 'Po': 84,\r\n 'At': 85,\r\n 'Rn': 86,\r\n 'Fr': 87,\r\n 'Ra': 88,\r\n 'Ac': 89,\r\n 'Th': 90,\r\n 'Pa': 91,\r\n 'U': 92,\r\n 'Np': 93,\r\n 'Pu': 94,\r\n 'Am': 95,\r\n 'Cm': 96,\r\n 'Bk': 97,\r\n 'Cf': 98,\r\n 'Es': 99,\r\n 'Fm': 100,\r\n 'Md': 101,\r\n 'No': 102,\r\n 'Lr': 103,\r\n 'Rf': 104,\r\n 'Db': 105,\r\n 'Sg': 106,\r\n 'Bh': 107,\r\n 'Hs': 108,\r\n 'Mt': 109,\r\n 'Ds': 110,\r\n 'Rg': 111,\r\n 'Cn': 112,\r\n 'Uut': 113,\r\n 'Uuq': 114,\r\n 'Uup': 115,\r\n 'Uuh': 116,\r\n 'Uus': 117,\r\n 'Uuo': 118\r\n };\r\n }\r\n}\r\n\r\nmodule.exports = Atom;","//@ts-check\r\nconst MathHelper = require('./MathHelper')\r\nconst Vector2 = require('./Vector2')\r\nconst Line = require('./Line')\r\nconst Vertex = require('./Vertex')\r\nconst Ring = require('./Ring')\r\n\r\n/** \r\n * A class wrapping a canvas element.\r\n * \r\n * @property {HTMLElement} canvas The HTML element for the canvas associated with this CanvasWrapper instance.\r\n * @property {CanvasRenderingContext2D} ctx The CanvasRenderingContext2D of the canvas associated with this CanvasWrapper instance.\r\n * @property {Object} colors The colors object as defined in the SmilesDrawer options.\r\n * @property {Object} opts The SmilesDrawer options.\r\n * @property {Number} drawingWidth The width of the canvas.\r\n * @property {Number} drawingHeight The height of the canvas.\r\n * @property {Number} offsetX The horizontal offset required for centering the drawing.\r\n * @property {Number} offsetY The vertical offset required for centering the drawing.\r\n * @property {Number} fontLarge The large font size in pt.\r\n * @property {Number} fontSmall The small font size in pt.\r\n */\r\nclass CanvasWrapper {\r\n /**\r\n * The constructor for the class CanvasWrapper.\r\n *\r\n * @param {(String|HTMLElement)} target The canvas id or the canvas HTMLElement.\r\n * @param {Object} theme A theme from the smiles drawer options.\r\n * @param {Object} options The smiles drawer options object.\r\n */\r\n constructor(target, theme, options) {\r\n if (typeof target === 'string' || target instanceof String) {\r\n this.canvas = document.getElementById(target);\r\n } else {\r\n this.canvas = target;\r\n }\r\n\r\n this.ctx = this.canvas.getContext('2d');\r\n this.colors = theme;\r\n this.opts = options;\r\n this.drawingWidth = 0.0;\r\n this.drawingHeight = 0.0;\r\n this.offsetX = 0.0;\r\n this.offsetY = 0.0;\r\n\r\n this.fontLarge = this.opts.fontSizeLarge + 'pt Helvetica, Arial, sans-serif';\r\n this.fontSmall = this.opts.fontSizeSmall + 'pt Helvetica, Arial, sans-serif';\r\n\r\n this.updateSize(this.opts.width, this.opts.height);\r\n\r\n this.ctx.font = this.fontLarge;\r\n this.hydrogenWidth = this.ctx.measureText('H').width;\r\n this.halfHydrogenWidth = this.hydrogenWidth / 2.0;\r\n this.halfBondThickness = this.opts.bondThickness / 2.0;\r\n\r\n // TODO: Find out why clear was here.\r\n // this.clear();\r\n }\r\n\r\n /**\r\n * Update the width and height of the canvas\r\n * \r\n * @param {Number} width \r\n * @param {Number} height \r\n */\r\n updateSize(width, height) {\r\n this.devicePixelRatio = window.devicePixelRatio || 1;\r\n this.backingStoreRatio = this.ctx.webkitBackingStorePixelRatio || this.ctx.mozBackingStorePixelRatio ||\r\n this.ctx.msBackingStorePixelRatio || this.ctx.oBackingStorePixelRatio ||\r\n this.ctx.backingStorePixelRatio || 1;\r\n this.ratio = this.devicePixelRatio / this.backingStoreRatio;\r\n\r\n if (this.ratio !== 1) {\r\n this.canvas.width = width * this.ratio;\r\n this.canvas.height = height * this.ratio;\r\n this.canvas.style.width = width + 'px';\r\n this.canvas.style.height = height + 'px';\r\n this.ctx.setTransform(this.ratio, 0, 0, this.ratio, 0, 0);\r\n } else {\r\n this.canvas.width = width * this.ratio;\r\n this.canvas.height = height * this.ratio;\r\n }\r\n }\r\n\r\n /**\r\n * Sets a provided theme.\r\n *\r\n * @param {Object} theme A theme from the smiles drawer options.\r\n */\r\n setTheme(theme) {\r\n this.colors = theme;\r\n }\r\n\r\n /**\r\n * Scale the canvas based on vertex positions.\r\n *\r\n * @param {Vertex[]} vertices An array of vertices containing the vertices associated with the current molecule.\r\n */\r\n scale(vertices) {\r\n // Figure out the final size of the image\r\n let maxX = -Number.MAX_VALUE;\r\n let maxY = -Number.MAX_VALUE;\r\n let minX = Number.MAX_VALUE;\r\n let minY = Number.MAX_VALUE;\r\n\r\n for (var i = 0; i < vertices.length; i++) {\r\n if (!vertices[i].value.isDrawn) {\r\n continue;\r\n }\r\n\r\n let p = vertices[i].position;\r\n\r\n if (maxX < p.x) maxX = p.x;\r\n if (maxY < p.y) maxY = p.y;\r\n if (minX > p.x) minX = p.x;\r\n if (minY > p.y) minY = p.y;\r\n }\r\n\r\n // Add padding\r\n var padding = this.opts.padding;\r\n maxX += padding;\r\n maxY += padding;\r\n minX -= padding;\r\n minY -= padding;\r\n\r\n this.drawingWidth = maxX - minX;\r\n this.drawingHeight = maxY - minY;\r\n\r\n var scaleX = this.canvas.offsetWidth / this.drawingWidth;\r\n var scaleY = this.canvas.offsetHeight / this.drawingHeight;\r\n\r\n var scale = (scaleX < scaleY) ? scaleX : scaleY;\r\n\r\n this.ctx.scale(scale, scale);\r\n\r\n this.offsetX = -minX;\r\n this.offsetY = -minY;\r\n\r\n // Center\r\n if (scaleX < scaleY) {\r\n this.offsetY += this.canvas.offsetHeight / (2.0 * scale) - this.drawingHeight / 2.0;\r\n } else {\r\n this.offsetX += this.canvas.offsetWidth / (2.0 * scale) - this.drawingWidth / 2.0;\r\n }\r\n }\r\n\r\n /**\r\n * Resets the transform of the canvas.\r\n */\r\n reset() {\r\n this.ctx.setTransform(1, 0, 0, 1, 0, 0);\r\n }\r\n\r\n /**\r\n * Returns the hex code of a color associated with a key from the current theme.\r\n *\r\n * @param {String} key The color key in the theme (e.g. C, N, BACKGROUND, ...).\r\n * @returns {String} A color hex value.\r\n */\r\n getColor(key) {\r\n key = key.toUpperCase();\r\n\r\n if (key in this.colors) {\r\n return this.colors[key];\r\n }\r\n\r\n return this.colors['C'];\r\n }\r\n\r\n /**\r\n * Draws a circle to a canvas context.\r\n * @param {Number} x The x coordinate of the circles center.\r\n * @param {Number} y The y coordinate of the circles center.\r\n * @param {Number} radius The radius of the circle\r\n * @param {String} color A hex encoded color.\r\n * @param {Boolean} [fill=true] Whether to fill or stroke the circle.\r\n * @param {Boolean} [debug=false] Draw in debug mode.\r\n * @param {String} [debugText=''] A debug message.\r\n */\r\n drawCircle(x, y, radius, color, fill = true, debug = false, debugText = '') {\r\n let ctx = this.ctx;\r\n let offsetX = this.offsetX;\r\n let offsetY = this.offsetY;\r\n\r\n ctx.save();\r\n ctx.lineWidth = 1.5;\r\n ctx.beginPath();\r\n ctx.arc(x + offsetX, y + offsetY, radius, 0, MathHelper.twoPI, true);\r\n ctx.closePath();\r\n\r\n if (debug) {\r\n if (fill) {\r\n ctx.fillStyle = '#f00';\r\n ctx.fill();\r\n } else {\r\n ctx.strokeStyle = '#f00';\r\n ctx.stroke();\r\n }\r\n\r\n this.drawDebugText(x, y, debugText);\r\n } else {\r\n if (fill) {\r\n ctx.fillStyle = color;\r\n ctx.fill();\r\n } else {\r\n ctx.strokeStyle = color;\r\n ctx.stroke();\r\n }\r\n }\r\n\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draw a line to a canvas.\r\n *\r\n * @param {Line} line A line.\r\n * @param {Boolean} [dashed=false] Whether or not the line is dashed.\r\n * @param {Number} [alpha=1.0] The alpha value of the color.\r\n */\r\n drawLine(line, dashed = false, alpha = 1.0) {\r\n let ctx = this.ctx;\r\n let offsetX = this.offsetX;\r\n let offsetY = this.offsetY;\r\n\r\n // Add a shadow behind the line\r\n let shortLine = line.clone().shorten(4.0);\r\n\r\n let l = shortLine.getLeftVector().clone();\r\n let r = shortLine.getRightVector().clone();\r\n\r\n l.x += offsetX;\r\n l.y += offsetY;\r\n\r\n r.x += offsetX;\r\n r.y += offsetY;\r\n\r\n // Draw the \"shadow\"\r\n if (!dashed) {\r\n ctx.save();\r\n ctx.globalCompositeOperation = 'destination-out';\r\n ctx.beginPath();\r\n ctx.moveTo(l.x, l.y);\r\n ctx.lineTo(r.x, r.y);\r\n ctx.lineCap = 'round';\r\n ctx.lineWidth = this.opts.bondThickness + 1.2;\r\n ctx.strokeStyle = this.getColor('BACKGROUND');\r\n ctx.stroke();\r\n ctx.globalCompositeOperation = 'source-over';\r\n ctx.restore();\r\n }\r\n\r\n l = line.getLeftVector().clone();\r\n r = line.getRightVector().clone();\r\n\r\n l.x += offsetX;\r\n l.y += offsetY;\r\n\r\n r.x += offsetX;\r\n r.y += offsetY;\r\n\r\n ctx.save();\r\n ctx.beginPath();\r\n ctx.moveTo(l.x, l.y);\r\n ctx.lineTo(r.x, r.y);\r\n ctx.lineCap = 'round';\r\n ctx.lineWidth = this.opts.bondThickness;\r\n\r\n let gradient = this.ctx.createLinearGradient(l.x, l.y, r.x, r.y);\r\n gradient.addColorStop(0.4, this.getColor(line.getLeftElement()) ||\r\n this.getColor('C'));\r\n gradient.addColorStop(0.6, this.getColor(line.getRightElement()) ||\r\n this.getColor('C'));\r\n\r\n if (dashed) {\r\n ctx.setLineDash([1, 1.5]);\r\n ctx.lineWidth = this.opts.bondThickness / 1.5;\r\n }\r\n\r\n if (alpha < 1.0) {\r\n ctx.globalAlpha = alpha;\r\n }\r\n\r\n ctx.strokeStyle = gradient;\r\n\r\n ctx.stroke();\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draw a wedge on the canvas.\r\n *\r\n * @param {Line} line A line.\r\n * @param {Number} width The wedge width.\r\n */\r\n drawWedge(line, width = 1.0) {\r\n if (isNaN(line.from.x) || isNaN(line.from.y) ||\r\n isNaN(line.to.x) || isNaN(line.to.y)) {\r\n return;\r\n }\r\n\r\n let ctx = this.ctx;\r\n let offsetX = this.offsetX;\r\n let offsetY = this.offsetY;\r\n\r\n // Add a shadow behind the line\r\n let shortLine = line.clone().shorten(5.0);\r\n\r\n let l = shortLine.getLeftVector().clone();\r\n let r = shortLine.getRightVector().clone();\r\n\r\n l.x += offsetX;\r\n l.y += offsetY;\r\n\r\n r.x += offsetX;\r\n r.y += offsetY;\r\n\r\n l = line.getLeftVector().clone();\r\n r = line.getRightVector().clone();\r\n\r\n l.x += offsetX;\r\n l.y += offsetY;\r\n\r\n r.x += offsetX;\r\n r.y += offsetY;\r\n\r\n ctx.save();\r\n\r\n let normals = Vector2.normals(l, r);\r\n\r\n normals[0].normalize();\r\n normals[1].normalize();\r\n\r\n let isRightChiralCenter = line.getRightChiral();\r\n\r\n let start = l;\r\n let end = r;\r\n\r\n if (isRightChiralCenter) {\r\n start = r;\r\n end = l;\r\n }\r\n\r\n let t = Vector2.add(start, Vector2.multiplyScalar(normals[0], this.halfBondThickness));\r\n let u = Vector2.add(end, Vector2.multiplyScalar(normals[0], 1.5 + this.halfBondThickness));\r\n let v = Vector2.add(end, Vector2.multiplyScalar(normals[1], 1.5 + this.halfBondThickness));\r\n let w = Vector2.add(start, Vector2.multiplyScalar(normals[1], this.halfBondThickness));\r\n\r\n ctx.beginPath();\r\n ctx.moveTo(t.x, t.y);\r\n ctx.lineTo(u.x, u.y);\r\n ctx.lineTo(v.x, v.y);\r\n ctx.lineTo(w.x, w.y);\r\n\r\n let gradient = this.ctx.createRadialGradient(r.x, r.y, this.opts.bondLength, r.x, r.y, 0);\r\n gradient.addColorStop(0.4, this.getColor(line.getLeftElement()) ||\r\n this.getColor('C'));\r\n gradient.addColorStop(0.6, this.getColor(line.getRightElement()) ||\r\n this.getColor('C'));\r\n\r\n ctx.fillStyle = gradient;\r\n\r\n ctx.fill();\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draw a dashed wedge on the canvas.\r\n *\r\n * @param {Line} line A line.\r\n */\r\n drawDashedWedge(line) {\r\n if (isNaN(line.from.x) || isNaN(line.from.y) ||\r\n isNaN(line.to.x) || isNaN(line.to.y)) {\r\n return;\r\n }\r\n\r\n let ctx = this.ctx;\r\n let offsetX = this.offsetX;\r\n let offsetY = this.offsetY;\r\n\r\n let l = line.getLeftVector().clone();\r\n let r = line.getRightVector().clone();\r\n\r\n l.x += offsetX;\r\n l.y += offsetY;\r\n\r\n r.x += offsetX;\r\n r.y += offsetY;\r\n\r\n ctx.save();\r\n\r\n let normals = Vector2.normals(l, r);\r\n\r\n normals[0].normalize();\r\n normals[1].normalize();\r\n\r\n\r\n let isRightChiralCenter = line.getRightChiral();\r\n\r\n let start;\r\n let end;\r\n let sStart;\r\n let sEnd;\r\n\r\n let shortLine = line.clone();\r\n\r\n if (isRightChiralCenter) {\r\n start = r;\r\n end = l;\r\n\r\n shortLine.shortenRight(1.0);\r\n\r\n sStart = shortLine.getRightVector().clone();\r\n sEnd = shortLine.getLeftVector().clone();\r\n } else {\r\n start = l;\r\n end = r;\r\n\r\n shortLine.shortenLeft(1.0);\r\n\r\n sStart = shortLine.getLeftVector().clone();\r\n sEnd = shortLine.getRightVector().clone();\r\n }\r\n\r\n sStart.x += offsetX;\r\n sStart.y += offsetY;\r\n sEnd.x += offsetX;\r\n sEnd.y += offsetY;\r\n\r\n let dir = Vector2.subtract(end, start).normalize();\r\n ctx.strokeStyle = this.getColor('C');\r\n ctx.lineCap = 'round';\r\n ctx.lineWidth = this.opts.bondThickness;\r\n ctx.beginPath();\r\n let length = line.getLength();\r\n let step = 1.25 / (length / (this.opts.bondThickness * 3.0));\r\n\r\n let changed = false;\r\n for (var t = 0.0; t < 1.0; t += step) {\r\n let to = Vector2.multiplyScalar(dir, t * length);\r\n let startDash = Vector2.add(start, to);\r\n let width = 1.5 * t;\r\n let dashOffset = Vector2.multiplyScalar(normals[0], width);\r\n\r\n if (!changed && t > 0.5) {\r\n ctx.stroke();\r\n ctx.beginPath();\r\n ctx.strokeStyle = this.getColor(line.getRightElement()) || this.getColor('C');\r\n changed = true;\r\n }\r\n \r\n startDash.subtract(dashOffset);\r\n ctx.moveTo(startDash.x, startDash.y);\r\n startDash.add(Vector2.multiplyScalar(dashOffset, 2.0));\r\n ctx.lineTo(startDash.x, startDash.y);\r\n }\r\n\r\n ctx.stroke();\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draws a debug text message at a given position\r\n *\r\n * @param {Number} x The x coordinate.\r\n * @param {Number} y The y coordinate.\r\n * @param {String} text The debug text.\r\n */\r\n drawDebugText(x, y, text) {\r\n let ctx = this.ctx;\r\n\r\n ctx.save();\r\n ctx.font = '5px Droid Sans, sans-serif';\r\n ctx.textAlign = 'start';\r\n ctx.textBaseline = 'top';\r\n ctx.fillStyle = '#ff0000';\r\n ctx.fillText(text, x + this.offsetX, y + this.offsetY);\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draw a ball to the canvas.\r\n *\r\n * @param {Number} x The x position of the text.\r\n * @param {Number} y The y position of the text.\r\n * @param {String} elementName The name of the element (single-letter).\r\n */\r\n drawBall(x, y, elementName) {\r\n let ctx = this.ctx;\r\n\r\n ctx.save();\r\n ctx.beginPath();\r\n ctx.arc(x + this.offsetX, y + this.offsetY, this.opts.bondLength / 4.5, 0, MathHelper.twoPI, false);\r\n ctx.fillStyle = this.getColor(elementName);\r\n ctx.fill();\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draw a point to the canvas.\r\n *\r\n * @param {Number} x The x position of the point.\r\n * @param {Number} y The y position of the point.\r\n * @param {String} elementName The name of the element (single-letter).\r\n */\r\n drawPoint(x, y, elementName) {\r\n let ctx = this.ctx;\r\n let offsetX = this.offsetX;\r\n let offsetY = this.offsetY;\r\n\r\n ctx.save();\r\n ctx.globalCompositeOperation = 'destination-out';\r\n ctx.beginPath();\r\n ctx.arc(x + offsetX, y + offsetY, 1.5, 0, MathHelper.twoPI, true);\r\n ctx.closePath();\r\n ctx.fill();\r\n ctx.globalCompositeOperation = 'source-over';\r\n\r\n ctx.beginPath();\r\n ctx.arc(x + this.offsetX, y + this.offsetY, 0.75, 0, MathHelper.twoPI, false);\r\n ctx.fillStyle = this.getColor(elementName);\r\n ctx.fill();\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draw a text to the canvas.\r\n *\r\n * @param {Number} x The x position of the text.\r\n * @param {Number} y The y position of the text.\r\n * @param {String} elementName The name of the element (single-letter).\r\n * @param {Number} hydrogens The number of hydrogen atoms.\r\n * @param {String} direction The direction of the text in relation to the associated vertex.\r\n * @param {Boolean} isTerminal A boolean indicating whether or not the vertex is terminal.\r\n * @param {Number} charge The charge of the atom.\r\n * @param {Number} isotope The isotope number.\r\n * @param {Object} attachedPseudoElement A map with containing information for pseudo elements or concatinated elements. The key is comprised of the element symbol and the hydrogen count.\r\n * @param {String} attachedPseudoElement.element The element symbol.\r\n * @param {Number} attachedPseudoElement.count The number of occurences that match the key.\r\n * @param {Number} attachedPseudoElement.hyrogenCount The number of hydrogens attached to each atom matching the key.\r\n */\r\n drawText(x, y, elementName, hydrogens, direction, isTerminal, charge, isotope, attachedPseudoElement = {}) {\r\n let ctx = this.ctx;\r\n let offsetX = this.offsetX;\r\n let offsetY = this.offsetY;\r\n\r\n ctx.save();\r\n\r\n ctx.textAlign = 'start';\r\n ctx.textBaseline = 'alphabetic';\r\n\r\n let pseudoElementHandled = false;\r\n\r\n // Charge\r\n let chargeText = ''\r\n let chargeWidth = 0;\r\n\r\n if (charge) {\r\n chargeText = this.getChargeText(charge);\r\n\r\n ctx.font = this.fontSmall;\r\n chargeWidth = ctx.measureText(chargeText).width;\r\n }\r\n\r\n let isotopeText = '0';\r\n let isotopeWidth = 0;\r\n\r\n if (isotope > 0) {\r\n isotopeText = isotope.toString();\r\n ctx.font = this.fontSmall;\r\n isotopeWidth = ctx.measureText(isotopeText).width;\r\n }\r\n\r\n\r\n // TODO: Better handle exceptions\r\n // Exception for nitro (draw nitro as NO2 instead of N+O-O)\r\n if (charge === 1 && elementName === 'N' && attachedPseudoElement.hasOwnProperty('0O') && \r\n attachedPseudoElement.hasOwnProperty('0O-1')) {\r\n attachedPseudoElement = { '0O': { element: 'O', count: 2, hydrogenCount: 0, previousElement: 'C', charge: '' } }\r\n charge = 0;\r\n }\r\n\r\n\r\n ctx.font = this.fontLarge;\r\n ctx.fillStyle = this.getColor('BACKGROUND');\r\n\r\n let dim = ctx.measureText(elementName);\r\n\r\n dim.totalWidth = dim.width + chargeWidth;\r\n dim.height = parseInt(this.fontLarge, 10);\r\n\r\n let r = (dim.width > this.opts.fontSizeLarge) ? dim.width : this.opts.fontSizeLarge;\r\n r /= 1.5;\r\n\r\n ctx.globalCompositeOperation = 'destination-out';\r\n ctx.beginPath();\r\n ctx.arc(x + offsetX, y + offsetY, r, 0, MathHelper.twoPI, true);\r\n ctx.closePath();\r\n ctx.fill();\r\n ctx.globalCompositeOperation = 'source-over';\r\n\r\n let cursorPos = -dim.width / 2.0;\r\n let cursorPosLeft = -dim.width / 2.0;\r\n\r\n ctx.fillStyle = this.getColor(elementName);\r\n ctx.fillText(elementName, x + offsetX + cursorPos, y + this.opts.halfFontSizeLarge + offsetY);\r\n cursorPos += dim.width;\r\n\r\n if (charge) {\r\n ctx.font = this.fontSmall;\r\n ctx.fillText(chargeText, x + offsetX + cursorPos, y - this.opts.fifthFontSizeSmall + offsetY);\r\n cursorPos += chargeWidth;\r\n }\r\n\r\n if (isotope > 0) {\r\n ctx.font = this.fontSmall;\r\n ctx.fillText(isotopeText, x + offsetX + cursorPosLeft - isotopeWidth, y - this.opts.fifthFontSizeSmall + offsetY);\r\n cursorPosLeft -= isotopeWidth;\r\n }\r\n\r\n ctx.font = this.fontLarge;\r\n\r\n let hydrogenWidth = 0;\r\n let hydrogenCountWidth = 0;\r\n\r\n if (hydrogens === 1) {\r\n let hx = x + offsetX;\r\n let hy = y + offsetY + this.opts.halfFontSizeLarge;\r\n\r\n hydrogenWidth = this.hydrogenWidth;\r\n cursorPosLeft -= hydrogenWidth;\r\n\r\n if (direction === 'left') {\r\n hx += cursorPosLeft;\r\n } else if (direction === 'right') {\r\n hx += cursorPos;\r\n } else if (direction === 'up' && isTerminal) {\r\n hx += cursorPos;\r\n } else if (direction === 'down' && isTerminal) {\r\n hx += cursorPos;\r\n } else if (direction === 'up' && !isTerminal) {\r\n hy -= this.opts.fontSizeLarge + this.opts.quarterFontSizeLarge;\r\n hx -= this.halfHydrogenWidth;\r\n } else if (direction === 'down' && !isTerminal) {\r\n hy += this.opts.fontSizeLarge + this.opts.quarterFontSizeLarge;\r\n hx -= this.halfHydrogenWidth;\r\n }\r\n\r\n ctx.fillText('H', hx, hy);\r\n\r\n cursorPos += hydrogenWidth;\r\n } else if (hydrogens > 1) {\r\n let hx = x + offsetX;\r\n let hy = y + offsetY + this.opts.halfFontSizeLarge;\r\n\r\n hydrogenWidth = this.hydrogenWidth;\r\n ctx.font = this.fontSmall;\r\n hydrogenCountWidth = ctx.measureText(hydrogens).width;\r\n cursorPosLeft -= hydrogenWidth + hydrogenCountWidth;\r\n\r\n if (direction === 'left') {\r\n hx += cursorPosLeft;\r\n } else if (direction === 'right') {\r\n hx += cursorPos;\r\n } else if (direction === 'up' && isTerminal) {\r\n hx += cursorPos;\r\n } else if (direction === 'down' && isTerminal) {\r\n hx += cursorPos;\r\n } else if (direction === 'up' && !isTerminal) {\r\n hy -= this.opts.fontSizeLarge + this.opts.quarterFontSizeLarge;\r\n hx -= this.halfHydrogenWidth;\r\n } else if (direction === 'down' && !isTerminal) {\r\n hy += this.opts.fontSizeLarge + this.opts.quarterFontSizeLarge;\r\n hx -= this.halfHydrogenWidth;\r\n }\r\n\r\n ctx.font = this.fontLarge;\r\n ctx.fillText('H', hx, hy)\r\n\r\n ctx.font = this.fontSmall;\r\n ctx.fillText(hydrogens, hx + this.halfHydrogenWidth + hydrogenCountWidth, hy + this.opts.fifthFontSizeSmall);\r\n\r\n cursorPos += hydrogenWidth + this.halfHydrogenWidth + hydrogenCountWidth;\r\n }\r\n\r\n if (pseudoElementHandled) {\r\n ctx.restore();\r\n return;\r\n }\r\n\r\n for (let key in attachedPseudoElement) {\r\n if (!attachedPseudoElement.hasOwnProperty(key)) {\r\n continue;\r\n }\r\n\r\n let openParenthesisWidth = 0;\r\n let closeParenthesisWidth = 0;\r\n\r\n let element = attachedPseudoElement[key].element;\r\n let elementCount = attachedPseudoElement[key].count;\r\n let hydrogenCount = attachedPseudoElement[key].hydrogenCount;\r\n let elementCharge = attachedPseudoElement[key].charge;\r\n\r\n ctx.font = this.fontLarge;\r\n\r\n if (elementCount > 1 && hydrogenCount > 0) {\r\n openParenthesisWidth = ctx.measureText('(').width;\r\n closeParenthesisWidth = ctx.measureText(')').width;\r\n }\r\n\r\n let elementWidth = ctx.measureText(element).width;\r\n let elementCountWidth = 0;\r\n\r\n let elementChargeText = '';\r\n let elementChargeWidth = 0;\r\n\r\n hydrogenWidth = 0;\r\n\r\n if (hydrogenCount > 0) {\r\n hydrogenWidth = this.hydrogenWidth;\r\n }\r\n\r\n ctx.font = this.fontSmall;\r\n\r\n if (elementCount > 1) {\r\n elementCountWidth = ctx.measureText(elementCount).width;\r\n }\r\n\r\n if (elementCharge !== 0) {\r\n elementChargeText = this.getChargeText(elementCharge);\r\n elementChargeWidth = ctx.measureText(elementChargeText).width;\r\n }\r\n\r\n hydrogenCountWidth = 0;\r\n\r\n if (hydrogenCount > 1) {\r\n hydrogenCountWidth = ctx.measureText(hydrogenCount).width;\r\n }\r\n\r\n ctx.font = this.fontLarge;\r\n\r\n let hx = x + offsetX;\r\n let hy = y + offsetY + this.opts.halfFontSizeLarge;\r\n\r\n ctx.fillStyle = this.getColor(element);\r\n\r\n if (elementCount > 0) {\r\n cursorPosLeft -= elementCountWidth;\r\n }\r\n\r\n if (elementCount > 1 && hydrogenCount > 0) {\r\n if (direction === 'left') {\r\n cursorPosLeft -= closeParenthesisWidth;\r\n ctx.fillText(')', hx + cursorPosLeft, hy);\r\n } else {\r\n ctx.fillText('(', hx + cursorPos, hy);\r\n cursorPos += openParenthesisWidth;\r\n }\r\n }\r\n\r\n if (direction === 'left') {\r\n cursorPosLeft -= elementWidth;\r\n ctx.fillText(element, hx + cursorPosLeft, hy)\r\n } else {\r\n ctx.fillText(element, hx + cursorPos, hy)\r\n cursorPos += elementWidth;\r\n }\r\n\r\n if (hydrogenCount > 0) {\r\n if (direction === 'left') {\r\n cursorPosLeft -= hydrogenWidth + hydrogenCountWidth;\r\n ctx.fillText('H', hx + cursorPosLeft, hy)\r\n\r\n if (hydrogenCount > 1) {\r\n ctx.font = this.fontSmall;\r\n ctx.fillText(hydrogenCount, hx + cursorPosLeft + hydrogenWidth, hy + this.opts.fifthFontSizeSmall);\r\n }\r\n } else {\r\n ctx.fillText('H', hx + cursorPos, hy)\r\n cursorPos += hydrogenWidth;\r\n\r\n if (hydrogenCount > 1) {\r\n ctx.font = this.fontSmall;\r\n ctx.fillText(hydrogenCount, hx + cursorPos, hy + this.opts.fifthFontSizeSmall);\r\n cursorPos += hydrogenCountWidth;\r\n }\r\n }\r\n }\r\n\r\n ctx.font = this.fontLarge;\r\n\r\n if (elementCount > 1 && hydrogenCount > 0) {\r\n if (direction === 'left') {\r\n cursorPosLeft -= openParenthesisWidth;\r\n ctx.fillText('(', hx + cursorPosLeft, hy);\r\n } else {\r\n ctx.fillText(')', hx + cursorPos, hy);\r\n cursorPos += closeParenthesisWidth;\r\n }\r\n }\r\n\r\n ctx.font = this.fontSmall;\r\n\r\n if (elementCount > 1) {\r\n if (direction === 'left') {\r\n ctx.fillText(elementCount, hx + cursorPosLeft +\r\n openParenthesisWidth + closeParenthesisWidth + hydrogenWidth +\r\n hydrogenCountWidth + elementWidth, hy + this.opts.fifthFontSizeSmall);\r\n } else {\r\n ctx.fillText(elementCount, hx + cursorPos, hy + this.opts.fifthFontSizeSmall);\r\n cursorPos += elementCountWidth;\r\n }\r\n }\r\n\r\n if (elementCharge !== 0) {\r\n if (direction === 'left') {\r\n ctx.fillText(elementChargeText, hx + cursorPosLeft +\r\n openParenthesisWidth + closeParenthesisWidth + hydrogenWidth +\r\n hydrogenCountWidth + elementWidth, y - this.opts.fifthFontSizeSmall + offsetY);\r\n } else {\r\n ctx.fillText(elementChargeText, hx + cursorPos, y - this.opts.fifthFontSizeSmall + offsetY);\r\n cursorPos += elementChargeWidth;\r\n }\r\n }\r\n }\r\n\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Translate the integer indicating the charge to the appropriate text.\r\n * @param {Number} charge The integer indicating the charge.\r\n * @returns {String} A string representing a charge.\r\n */\r\n getChargeText(charge) {\r\n if (charge === 1) {\r\n return '+'\r\n } else if (charge === 2) {\r\n return '2+';\r\n } else if (charge === -1) {\r\n return '-';\r\n } else if (charge === -2) {\r\n return '2-';\r\n } else {\r\n return '';\r\n }\r\n }\r\n\r\n /**\r\n * Draws a dubug dot at a given coordinate and adds text.\r\n *\r\n * @param {Number} x The x coordinate.\r\n * @param {Number} y The y coordindate.\r\n * @param {String} [debugText=''] A string.\r\n * @param {String} [color='#f00'] A color in hex form.\r\n */\r\n drawDebugPoint(x, y, debugText = '', color = '#f00') {\r\n this.drawCircle(x, y, 2, color, true, true, debugText);\r\n }\r\n\r\n /**\r\n * Draws a ring inside a provided ring, indicating aromaticity.\r\n *\r\n * @param {Ring} ring A ring.\r\n */\r\n drawAromaticityRing(ring) {\r\n let ctx = this.ctx;\r\n let radius = MathHelper.apothemFromSideLength(this.opts.bondLength, ring.getSize());\r\n\r\n ctx.save();\r\n ctx.strokeStyle = this.getColor('C');\r\n ctx.lineWidth = this.opts.bondThickness;\r\n ctx.beginPath();\r\n ctx.arc(ring.center.x + this.offsetX, ring.center.y + this.offsetY,\r\n radius - this.opts.bondSpacing, 0, Math.PI * 2, true);\r\n ctx.closePath();\r\n ctx.stroke();\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Clear the canvas.\r\n *\r\n */\r\n clear() {\r\n this.ctx.clearRect(0, 0, this.canvas.offsetWidth, this.canvas.offsetHeight);\r\n }\r\n\r\n}\r\n\r\nmodule.exports = CanvasWrapper;","//@ts-check\r\nconst MathHelper = require('./MathHelper')\r\nconst ArrayHelper = require('./ArrayHelper')\r\nconst Vector2 = require('./Vector2')\r\nconst Line = require('./Line')\r\nconst Vertex = require('./Vertex')\r\nconst Edge = require('./Edge')\r\nconst Atom = require('./Atom')\r\nconst Ring = require('./Ring')\r\nconst RingConnection = require('./RingConnection')\r\nconst CanvasWrapper = require('./CanvasWrapper')\r\nconst Graph = require('./Graph')\r\nconst SSSR = require('./SSSR')\r\n\r\n/** \r\n * The main class of the application representing the smiles drawer \r\n * \r\n * @property {Graph} graph The graph associated with this SmilesDrawer.Drawer instance.\r\n * @property {Number} ringIdCounter An internal counter to keep track of ring ids.\r\n * @property {Number} ringConnectionIdCounter An internal counter to keep track of ring connection ids.\r\n * @property {CanvasWrapper} canvasWrapper The CanvasWrapper associated with this SmilesDrawer.Drawer instance.\r\n * @property {Number} totalOverlapScore The current internal total overlap score.\r\n * @property {Object} defaultOptions The default options.\r\n * @property {Object} opts The merged options.\r\n * @property {Object} theme The current theme.\r\n */\r\nclass Drawer {\r\n /**\r\n * The constructor for the class SmilesDrawer.\r\n *\r\n * @param {Object} options An object containing custom values for different options. It is merged with the default options.\r\n */\r\n constructor(options) {\r\n this.graph = null;\r\n this.doubleBondConfigCount = 0;\r\n this.doubleBondConfig = null;\r\n this.ringIdCounter = 0;\r\n this.ringConnectionIdCounter = 0;\r\n this.canvasWrapper = null;\r\n this.totalOverlapScore = 0;\r\n\r\n this.defaultOptions = {\r\n width: 500,\r\n height: 500,\r\n bondThickness: 0.6,\r\n bondLength: 15,\r\n shortBondLength: 0.85,\r\n bondSpacing: 0.18 * 15,\r\n atomVisualization: 'default',\r\n isomeric: true,\r\n debug: false,\r\n terminalCarbons: false,\r\n explicitHydrogens: false,\r\n overlapSensitivity: 0.42,\r\n overlapResolutionIterations: 1,\r\n compactDrawing: true,\r\n fontSizeLarge: 5,\r\n fontSizeSmall: 3,\r\n padding: 20.0,\r\n themes: {\r\n dark: {\r\n C: '#fff',\r\n O: '#e74c3c',\r\n N: '#3498db',\r\n F: '#27ae60',\r\n CL: '#16a085',\r\n BR: '#d35400',\r\n I: '#8e44ad',\r\n P: '#d35400',\r\n S: '#f1c40f',\r\n B: '#e67e22',\r\n SI: '#e67e22',\r\n H: '#fff',\r\n BACKGROUND: '#141414'\r\n },\r\n light: {\r\n C: '#222',\r\n O: '#e74c3c',\r\n N: '#3498db',\r\n F: '#27ae60',\r\n CL: '#16a085',\r\n BR: '#d35400',\r\n I: '#8e44ad',\r\n P: '#d35400',\r\n S: '#f1c40f',\r\n B: '#e67e22',\r\n SI: '#e67e22',\r\n H: '#222',\r\n BACKGROUND: '#fff'\r\n }\r\n }\r\n };\r\n\r\n this.opts = this.extend(true, this.defaultOptions, options);\r\n this.opts.halfBondSpacing = this.opts.bondSpacing / 2.0;\r\n this.opts.bondLengthSq = this.opts.bondLength * this.opts.bondLength;\r\n this.opts.halfFontSizeLarge = this.opts.fontSizeLarge / 2.0;\r\n this.opts.quarterFontSizeLarge = this.opts.fontSizeLarge / 4.0;\r\n this.opts.fifthFontSizeSmall = this.opts.fontSizeSmall / 5.0;\r\n\r\n // Set the default theme.\r\n this.theme = this.opts.themes.dark;\r\n }\r\n\r\n /**\r\n * A helper method to extend the default options with user supplied ones.\r\n */\r\n extend() {\r\n let that = this;\r\n let extended = {};\r\n let deep = false;\r\n let i = 0;\r\n let length = arguments.length;\r\n\r\n if (Object.prototype.toString.call(arguments[0]) === '[object Boolean]') {\r\n deep = arguments[0];\r\n i++;\r\n }\r\n\r\n let merge = function (obj) {\r\n for (var prop in obj) {\r\n if (Object.prototype.hasOwnProperty.call(obj, prop)) {\r\n if (deep && Object.prototype.toString.call(obj[prop]) === '[object Object]') {\r\n extended[prop] = that.extend(true, extended[prop], obj[prop]);\r\n } else {\r\n extended[prop] = obj[prop];\r\n }\r\n }\r\n }\r\n };\r\n\r\n for (; i < length; i++) {\r\n let obj = arguments[i];\r\n merge(obj);\r\n }\r\n\r\n return extended;\r\n };\r\n\r\n\r\n /**\r\n * Draws the parsed smiles data to a canvas element.\r\n *\r\n * @param {Object} data The tree returned by the smiles parser.\r\n * @param {(String|HTMLElement)} target The id of the HTML canvas element the structure is drawn to - or the element itself.\r\n * @param {String} themeName='dark' The name of the theme to use. Built-in themes are 'light' and 'dark'.\r\n * @param {Boolean} infoOnly=false Only output info on the molecule without drawing anything to the canvas.\r\n */\r\n draw(data, target, themeName = 'light', infoOnly = false) {\r\n this.data = data;\r\n this.infoOnly = infoOnly;\r\n\r\n if (!this.infoOnly) {\r\n this.canvasWrapper = new CanvasWrapper(target, this.opts.themes[themeName], this.opts);\r\n }\r\n\r\n this.ringIdCounter = 0;\r\n this.ringConnectionIdCounter = 0;\r\n\r\n this.graph = new Graph(data, this.opts.isomeric);\r\n this.rings = Array();\r\n this.ringConnections = Array();\r\n\r\n this.originalRings = Array();\r\n this.originalRingConnections = Array();\r\n\r\n this.bridgedRing = false;\r\n\r\n // Reset those, in case the previous drawn SMILES had a dangling \\ or /\r\n this.doubleBondConfigCount = null;\r\n this.doubleBondConfig = null;\r\n\r\n this.initRings();\r\n this.initHydrogens();\r\n\r\n if (!this.infoOnly) {\r\n this.position();\r\n\r\n // Restore the ring information (removes bridged rings and replaces them with the original, multiple, rings)\r\n this.restoreRingInformation();\r\n\r\n // Atoms bonded to the same ring atom\r\n this.resolvePrimaryOverlaps();\r\n\r\n let overlapScore = this.getOverlapScore();\r\n\r\n this.totalOverlapScore = this.getOverlapScore().total;\r\n\r\n for (var o = 0; o < this.opts.overlapResolutionIterations; o++) {\r\n for (var i = 0; i < this.graph.edges.length; i++) {\r\n let edge = this.graph.edges[i];\r\n if (this.isEdgeRotatable(edge)) {\r\n let subTreeDepthA = this.graph.getTreeDepth(edge.sourceId, edge.targetId);\r\n let subTreeDepthB = this.graph.getTreeDepth(edge.targetId, edge.sourceId);\r\n\r\n // Only rotate the shorter subtree\r\n let a = edge.targetId;\r\n let b = edge.sourceId;\r\n\r\n if (subTreeDepthA > subTreeDepthB) {\r\n a = edge.sourceId;\r\n b = edge.targetId;\r\n }\r\n\r\n let subTreeOverlap = this.getSubtreeOverlapScore(b, a, overlapScore.vertexScores);\r\n if (subTreeOverlap.value > this.opts.overlapSensitivity) {\r\n let vertexA = this.graph.vertices[a];\r\n let vertexB = this.graph.vertices[b];\r\n let neighboursB = vertexB.getNeighbours(a);\r\n\r\n if (neighboursB.length === 1) {\r\n let neighbour = this.graph.vertices[neighboursB[0]];\r\n let angle = neighbour.position.getRotateAwayFromAngle(vertexA.position, vertexB.position, MathHelper.toRad(120));\r\n\r\n this.rotateSubtree(neighbour.id, vertexB.id, angle, vertexB.position);\r\n // If the new overlap is bigger, undo change\r\n let newTotalOverlapScore = this.getOverlapScore().total;\r\n\r\n if (newTotalOverlapScore > this.totalOverlapScore) {\r\n this.rotateSubtree(neighbour.id, vertexB.id, -angle, vertexB.position);\r\n } else {\r\n this.totalOverlapScore = newTotalOverlapScore;\r\n }\r\n } else if (neighboursB.length === 2) {\r\n // Switch places / sides\r\n // If vertex a is in a ring, do nothing\r\n if (vertexB.value.rings.length !== 0 && vertexA.value.rings.length !== 0) {\r\n continue;\r\n }\r\n\r\n let neighbourA = this.graph.vertices[neighboursB[0]];\r\n let neighbourB = this.graph.vertices[neighboursB[1]];\r\n \r\n if (neighbourA.value.rings.length === 1 && neighbourB.value.rings.length === 1) {\r\n // Both neighbours in same ring. TODO: does this create problems with wedges? (up = down and vice versa?)\r\n if (neighbourA.value.rings[0] !== neighbourB.value.rings[0]) {\r\n continue;\r\n }\r\n // TODO: Rotate circle\r\n } else if (neighbourA.value.rings.length !== 0 || neighbourB.value.rings.length !== 0) {\r\n continue;\r\n } else {\r\n let angleA = neighbourA.position.getRotateAwayFromAngle(vertexA.position, vertexB.position, MathHelper.toRad(120));\r\n let angleB = neighbourB.position.getRotateAwayFromAngle(vertexA.position, vertexB.position, MathHelper.toRad(120));\r\n\r\n this.rotateSubtree(neighbourA.id, vertexB.id, angleA, vertexB.position);\r\n this.rotateSubtree(neighbourB.id, vertexB.id, angleB, vertexB.position);\r\n\r\n let newTotalOverlapScore = this.getOverlapScore().total;\r\n\r\n if (newTotalOverlapScore > this.totalOverlapScore) {\r\n this.rotateSubtree(neighbourA.id, vertexB.id, -angleA, vertexB.position);\r\n this.rotateSubtree(neighbourB.id, vertexB.id, -angleB, vertexB.position);\r\n } else {\r\n this.totalOverlapScore = newTotalOverlapScore;\r\n }\r\n }\r\n }\r\n\r\n overlapScore = this.getOverlapScore();\r\n }\r\n }\r\n }\r\n }\r\n\r\n this.resolveSecondaryOverlaps(overlapScore.scores);\r\n\r\n if (this.opts.isomeric) {\r\n this.annotateStereochemistry();\r\n }\r\n\r\n // Initialize pseudo elements or shortcuts\r\n if (this.opts.compactDrawing && this.opts.atomVisualization === 'default') {\r\n this.initPseudoElements();\r\n }\r\n\r\n this.rotateDrawing();\r\n\r\n // Set the canvas to the appropriate size\r\n this.canvasWrapper.scale(this.graph.vertices);\r\n\r\n // Do the actual drawing\r\n this.drawEdges(this.opts.debug);\r\n this.drawVertices(this.opts.debug);\r\n this.canvasWrapper.reset();\r\n }\r\n }\r\n\r\n /**\r\n * Returns the number of rings this edge is a part of.\r\n *\r\n * @param {Number} edgeId The id of an edge.\r\n * @returns {Number} The number of rings the provided edge is part of.\r\n */\r\n edgeRingCount(edgeId) {\r\n let edge = this.graph.edges[edgeId];\r\n let a = this.graph.vertices[edge.sourceId];\r\n let b = this.graph.vertices[edge.targetId];\r\n\r\n return Math.min(a.value.rings.length, b.value.rings.length);\r\n }\r\n\r\n /**\r\n * Returns an array containing the bridged rings associated with this molecule.\r\n *\r\n * @returns {Ring[]} An array containing all bridged rings associated with this molecule.\r\n */\r\n getBridgedRings() {\r\n let bridgedRings = Array();\r\n\r\n for (var i = 0; i < this.rings.length; i++) {\r\n if (this.rings[i].isBridged) {\r\n bridgedRings.push(this.rings[i]);\r\n }\r\n }\r\n\r\n return bridgedRings;\r\n }\r\n\r\n /**\r\n * Returns an array containing all fused rings associated with this molecule.\r\n *\r\n * @returns {Ring[]} An array containing all fused rings associated with this molecule.\r\n */\r\n getFusedRings() {\r\n let fusedRings = Array();\r\n\r\n for (var i = 0; i < this.rings.length; i++) {\r\n if (this.rings[i].isFused) {\r\n fusedRings.push(this.rings[i]);\r\n }\r\n }\r\n\r\n return fusedRings;\r\n }\r\n\r\n /**\r\n * Returns an array containing all spiros associated with this molecule.\r\n *\r\n * @returns {Ring[]} An array containing all spiros associated with this molecule.\r\n */\r\n getSpiros() {\r\n let spiros = Array();\r\n\r\n for (var i = 0; i < this.rings.length; i++) {\r\n if (this.rings[i].isSpiro) {\r\n spiros.push(this.rings[i]);\r\n }\r\n }\r\n\r\n return spiros;\r\n }\r\n\r\n /**\r\n * Returns a string containing a semicolon and new-line separated list of ring properties: Id; Members Count; Neighbours Count; IsSpiro; IsFused; IsBridged; Ring Count (subrings of bridged rings)\r\n *\r\n * @returns {String} A string as described in the method description.\r\n */\r\n printRingInfo() {\r\n let result = '';\r\n for (var i = 0; i < this.rings.length; i++) {\r\n const ring = this.rings[i];\r\n\r\n result += ring.id + ';';\r\n result += ring.members.length + ';';\r\n result += ring.neighbours.length + ';';\r\n result += ring.isSpiro ? 'true;' : 'false;'\r\n result += ring.isFused ? 'true;' : 'false;'\r\n result += ring.isBridged ? 'true;' : 'false;'\r\n result += ring.rings.length + ';';\r\n result += '\\n';\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Rotates the drawing to make the widest dimension horizontal.\r\n */\r\n rotateDrawing() {\r\n // Rotate the vertices to make the molecule align horizontally\r\n // Find the longest distance\r\n let a = 0;\r\n let b = 0;\r\n let maxDist = 0;\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertexA = this.graph.vertices[i];\r\n\r\n if (!vertexA.value.isDrawn) {\r\n continue;\r\n }\r\n\r\n for (var j = i + 1; j < this.graph.vertices.length; j++) {\r\n let vertexB = this.graph.vertices[j];\r\n\r\n if (!vertexB.value.isDrawn) {\r\n continue;\r\n }\r\n\r\n let dist = vertexA.position.distanceSq(vertexB.position);\r\n\r\n if (dist > maxDist) {\r\n maxDist = dist;\r\n a = i;\r\n b = j;\r\n }\r\n }\r\n }\r\n\r\n let angle = -Vector2.subtract(this.graph.vertices[a].position, this.graph.vertices[b].position).angle();\r\n\r\n if (!isNaN(angle)) {\r\n // Round to 30 degrees\r\n let remainder = angle % 0.523599;\r\n\r\n // Round either up or down in 30 degree steps\r\n if (remainder < 0.2617995) {\r\n angle = angle - remainder;\r\n } else {\r\n angle += 0.523599 - remainder;\r\n }\r\n\r\n // Finally, rotate everything\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n if (i === b) {\r\n continue;\r\n }\r\n\r\n this.graph.vertices[i].position.rotateAround(angle, this.graph.vertices[b].position);\r\n }\r\n\r\n for (var i = 0; i < this.rings.length; i++) {\r\n this.rings[i].center.rotateAround(angle, this.graph.vertices[b].position);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Returns the total overlap score of the current molecule.\r\n *\r\n * @returns {Number} The overlap score.\r\n */\r\n getTotalOverlapScore() {\r\n return this.totalOverlapScore;\r\n }\r\n\r\n /**\r\n * Returns the ring count of the current molecule.\r\n *\r\n * @returns {Number} The ring count.\r\n */\r\n getRingCount() {\r\n return this.rings.length;\r\n }\r\n\r\n /**\r\n * Checks whether or not the current molecule a bridged ring.\r\n *\r\n * @returns {Boolean} A boolean indicating whether or not the current molecule a bridged ring.\r\n */\r\n hasBridgedRing() {\r\n return this.bridgedRing;\r\n }\r\n\r\n /**\r\n * Returns the number of heavy atoms (non-hydrogen) in the current molecule.\r\n *\r\n * @returns {Number} The heavy atom count.\r\n */\r\n getHeavyAtomCount() {\r\n let hac = 0;\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n if (this.graph.vertices[i].value.element !== 'H') {\r\n hac++;\r\n }\r\n }\r\n\r\n return hac;\r\n }\r\n\r\n /**\r\n * Returns the molecular formula of the loaded molecule as a string.\r\n * \r\n * @returns {String} The molecular formula.\r\n */\r\n getMolecularFormula() {\r\n let molecularFormula = '';\r\n let counts = new Map();\r\n \r\n // Initialize element count\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let atom = this.graph.vertices[i].value;\r\n\r\n if (counts.has(atom.element)) {\r\n counts.set(atom.element, counts.get(atom.element) + 1);\r\n } else {\r\n counts.set(atom.element, 1);\r\n }\r\n\r\n // Hydrogens attached to a chiral center were added as vertices,\r\n // those in non chiral brackets are added here\r\n if (atom.bracket && !atom.bracket.chirality) {\r\n if (counts.has('H')) {\r\n counts.set('H', counts.get('H') + atom.bracket.hcount);\r\n } else {\r\n counts.set('H', atom.bracket.hcount);\r\n }\r\n }\r\n\r\n // Add the implicit hydrogens according to valency, exclude\r\n // bracket atoms as they were handled and always have the number\r\n // of hydrogens specified explicitly\r\n if (!atom.bracket) {\r\n let nHydrogens = Atom.maxBonds[atom.element] - atom.bondCount;\r\n\r\n if (atom.isPartOfAromaticRing) {\r\n nHydrogens--;\r\n }\r\n\r\n if (counts.has('H')) {\r\n counts.set('H', counts.get('H') + nHydrogens);\r\n } else {\r\n counts.set('H', nHydrogens);\r\n }\r\n }\r\n }\r\n\r\n if (counts.has('C')) {\r\n let count = counts.get('C');\r\n molecularFormula += 'C' + (count > 1 ? count : '');\r\n counts.delete('C');\r\n }\r\n\r\n if (counts.has('H')) {\r\n let count = counts.get('H');\r\n molecularFormula += 'H' + (count > 1 ? count : '');\r\n counts.delete('H');\r\n }\r\n\r\n let elements = Object.keys(Atom.atomicNumbers).sort();\r\n\r\n elements.map(e => {\r\n if (counts.has(e)) {\r\n let count = counts.get(e);\r\n molecularFormula += e + (count > 1 ? count : '');\r\n }\r\n });\r\n\r\n return molecularFormula;\r\n }\r\n\r\n /**\r\n * Returns the type of the ringbond (e.g. '=' for a double bond). The ringbond represents the break in a ring introduced when creating the MST. If the two vertices supplied as arguments are not part of a common ringbond, the method returns null.\r\n *\r\n * @param {Vertex} vertexA A vertex.\r\n * @param {Vertex} vertexB A vertex.\r\n * @returns {(String|null)} Returns the ringbond type or null, if the two supplied vertices are not connected by a ringbond.\r\n */\r\n getRingbondType(vertexA, vertexB) {\r\n // Checks whether the two vertices are the ones connecting the ring\r\n // and what the bond type should be.\r\n if (vertexA.value.getRingbondCount() < 1 || vertexB.value.getRingbondCount() < 1) {\r\n return null;\r\n }\r\n\r\n for (var i = 0; i < vertexA.value.ringbonds.length; i++) {\r\n for (var j = 0; j < vertexB.value.ringbonds.length; j++) {\r\n // if(i != j) continue;\r\n if (vertexA.value.ringbonds[i].id === vertexB.value.ringbonds[j].id) {\r\n // If the bonds are equal, it doesn't matter which bond is returned.\r\n // if they are not equal, return the one that is not the default (\"-\")\r\n if (vertexA.value.ringbonds[i].bondType === '-') {\r\n return vertexB.value.ringbonds[j].bond;\r\n } else {\r\n return vertexA.value.ringbonds[i].bond;\r\n }\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Initializes rings and ringbonds for the current molecule.\r\n */\r\n initRings() {\r\n let openBonds = new Map();\r\n\r\n // Close the open ring bonds (spanning tree -> graph)\r\n for (var i = this.graph.vertices.length - 1; i >= 0; i--) {\r\n let vertex = this.graph.vertices[i];\r\n\r\n if (vertex.value.ringbonds.length === 0) {\r\n continue;\r\n }\r\n\r\n for (var j = 0; j < vertex.value.ringbonds.length; j++) {\r\n let ringbondId = vertex.value.ringbonds[j].id;\r\n let ringbondBond = vertex.value.ringbonds[j].bond;\r\n\r\n // If the other ringbond id has not been discovered,\r\n // add it to the open bonds map and continue.\r\n // if the other ringbond id has already been discovered,\r\n // create a bond between the two atoms.\r\n if (!openBonds.has(ringbondId)) {\r\n openBonds.set(ringbondId, [vertex.id, ringbondBond]);\r\n } else {\r\n let sourceVertexId = vertex.id;\r\n let targetVertexId = openBonds.get(ringbondId)[0];\r\n let targetRingbondBond = openBonds.get(ringbondId)[1];\r\n let edge = new Edge(sourceVertexId, targetVertexId, 1);\r\n edge.setBondType(targetRingbondBond || ringbondBond || '-');\r\n let edgeId = this.graph.addEdge(edge);\r\n let targetVertex = this.graph.vertices[targetVertexId];\r\n\r\n vertex.addRingbondChild(targetVertexId, j);\r\n vertex.value.addNeighbouringElement(targetVertex.value.element);\r\n targetVertex.addRingbondChild(sourceVertexId, j);\r\n targetVertex.value.addNeighbouringElement(vertex.value.element);\r\n vertex.edges.push(edgeId);\r\n targetVertex.edges.push(edgeId);\r\n\r\n openBonds.delete(ringbondId);\r\n }\r\n }\r\n }\r\n\r\n // Get the rings in the graph (the SSSR)\r\n let rings = SSSR.getRings(this.graph);\r\n\r\n if (rings === null) {\r\n return;\r\n }\r\n\r\n for (var i = 0; i < rings.length; i++) {\r\n let ringVertices = [...rings[i]];\r\n let ringId = this.addRing(new Ring(ringVertices));\r\n\r\n // Add the ring to the atoms\r\n for (var j = 0; j < ringVertices.length; j++) {\r\n this.graph.vertices[ringVertices[j]].value.rings.push(ringId);\r\n }\r\n }\r\n\r\n // Find connection between rings\r\n // Check for common vertices and create ring connections. This is a bit\r\n // ugly, but the ringcount is always fairly low (< 100)\r\n for (var i = 0; i < this.rings.length - 1; i++) {\r\n for (var j = i + 1; j < this.rings.length; j++) {\r\n let a = this.rings[i];\r\n let b = this.rings[j];\r\n let ringConnection = new RingConnection(a, b);\r\n\r\n // If there are no vertices in the ring connection, then there\r\n // is no ring connection\r\n if (ringConnection.vertices.size > 0) {\r\n this.addRingConnection(ringConnection);\r\n }\r\n }\r\n }\r\n\r\n // Add neighbours to the rings\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let ring = this.rings[i];\r\n ring.neighbours = RingConnection.getNeighbours(this.ringConnections, ring.id);\r\n }\r\n\r\n // Anchor the ring to one of it's members, so that the ring center will always\r\n // be tied to a single vertex when doing repositionings\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let ring = this.rings[i];\r\n this.graph.vertices[ring.members[0]].value.addAnchoredRing(ring.id);\r\n }\r\n\r\n // Backup the ring information to restore after placing the bridged ring.\r\n // This is needed in order to identify aromatic rings and stuff like this in\r\n // rings that are member of the superring.\r\n this.backupRingInformation();\r\n\r\n\r\n // Replace rings contained by a larger bridged ring with a bridged ring\r\n while (this.rings.length > 0) {\r\n let id = -1;\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let ring = this.rings[i];\r\n\r\n if (this.isPartOfBridgedRing(ring.id) && !ring.isBridged) {\r\n id = ring.id;\r\n }\r\n }\r\n\r\n if (id === -1) {\r\n break;\r\n }\r\n\r\n let ring = this.getRing(id);\r\n\r\n let involvedRings = this.getBridgedRingRings(ring.id);\r\n\r\n this.bridgedRing = true;\r\n this.createBridgedRing(involvedRings, ring.members[0]);\r\n\r\n // Remove the rings\r\n for (var i = 0; i < involvedRings.length; i++) {\r\n this.removeRing(involvedRings[i]);\r\n }\r\n }\r\n }\r\n\r\n initHydrogens() {\r\n // Do not draw hydrogens except when they are connected to a stereocenter connected to two or more rings.\r\n if (!this.opts.explicitHydrogens) {\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertex = this.graph.vertices[i];\r\n\r\n if (vertex.value.element !== 'H') {\r\n continue;\r\n }\r\n\r\n // Hydrogens should have only one neighbour, so just take the first\r\n // Also set hasHydrogen true on connected atom\r\n let neighbour = this.graph.vertices[vertex.neighbours[0]];\r\n neighbour.value.hasHydrogen = true;\r\n\r\n if (!neighbour.value.isStereoCenter || neighbour.value.rings.length < 2 && !neighbour.value.bridgedRing ||\r\n neighbour.value.bridgedRing && neighbour.value.originalRings.length < 2) {\r\n vertex.value.isDrawn = false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Returns all rings connected by bridged bonds starting from the ring with the supplied ring id.\r\n *\r\n * @param {Number} ringId A ring id.\r\n * @returns {Number[]} An array containing all ring ids of rings part of a bridged ring system.\r\n */\r\n getBridgedRingRings(ringId) {\r\n let involvedRings = Array();\r\n let that = this;\r\n\r\n let recurse = function (r) {\r\n let ring = that.getRing(r);\r\n\r\n involvedRings.push(r);\r\n\r\n for (var i = 0; i < ring.neighbours.length; i++) {\r\n let n = ring.neighbours[i];\r\n\r\n if (involvedRings.indexOf(n) === -1 &&\r\n n !== r &&\r\n RingConnection.isBridge(that.ringConnections, that.graph.vertices, r, n)) {\r\n recurse(n);\r\n }\r\n }\r\n };\r\n\r\n recurse(ringId);\r\n\r\n return ArrayHelper.unique(involvedRings);\r\n }\r\n\r\n /**\r\n * Checks whether or not a ring is part of a bridged ring.\r\n *\r\n * @param {Number} ringId A ring id.\r\n * @returns {Boolean} A boolean indicating whether or not the supplied ring (by id) is part of a bridged ring system.\r\n */\r\n isPartOfBridgedRing(ringId) {\r\n for (var i = 0; i < this.ringConnections.length; i++) {\r\n if (this.ringConnections[i].containsRing(ringId) &&\r\n this.ringConnections[i].isBridge(this.graph.vertices)) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Creates a bridged ring.\r\n *\r\n * @param {Number[]} ringIds An array of ids of rings involved in the bridged ring.\r\n * @param {Number} sourceVertexId The vertex id to start the bridged ring discovery from.\r\n * @returns {Ring} The bridged ring.\r\n */\r\n createBridgedRing(ringIds, sourceVertexId) {\r\n let ringMembers = new Set();\r\n let vertices = new Set();\r\n let neighbours = new Set();\r\n\r\n for (var i = 0; i < ringIds.length; i++) {\r\n let ring = this.getRing(ringIds[i]);\r\n ring.isPartOfBridged = true;\r\n\r\n for (var j = 0; j < ring.members.length; j++) {\r\n vertices.add(ring.members[j]);\r\n }\r\n\r\n for (var j = 0; j < ring.neighbours.length; j++) {\r\n let id = ring.neighbours[j];\r\n\r\n if (ringIds.indexOf(id) === -1) {\r\n neighbours.add(ring.neighbours[j]);\r\n }\r\n }\r\n }\r\n\r\n // A vertex is part of the bridged ring if it only belongs to\r\n // one of the rings (or to another ring\r\n // which is not part of the bridged ring).\r\n let leftovers = new Set();\r\n\r\n for (let id of vertices) {\r\n let vertex = this.graph.vertices[id];\r\n let intersection = ArrayHelper.intersection(ringIds, vertex.value.rings);\r\n\r\n if (vertex.value.rings.length === 1 || intersection.length === 1) {\r\n ringMembers.add(vertex.id);\r\n } else {\r\n leftovers.add(vertex.id);\r\n }\r\n }\r\n\r\n // Vertices can also be part of multiple rings and lay on the bridged ring,\r\n // however, they have to have at least two neighbours that are not part of\r\n // two rings\r\n let tmp = Array();\r\n let insideRing = Array();\r\n\r\n for (let id of leftovers) {\r\n let vertex = this.graph.vertices[id];\r\n let onRing = false;\r\n\r\n for (let j = 0; j < vertex.edges.length; j++) {\r\n if (this.edgeRingCount(vertex.edges[j]) === 1) {\r\n onRing = true;\r\n }\r\n }\r\n\r\n if (onRing) {\r\n vertex.value.isBridgeNode = true;\r\n ringMembers.add(vertex.id);\r\n } else {\r\n vertex.value.isBridge = true;\r\n ringMembers.add(vertex.id);\r\n }\r\n }\r\n\r\n // Create the ring\r\n let ring = new Ring([...ringMembers]);\r\n\r\n ring.isBridged = true;\r\n ring.neighbours = [...neighbours];\r\n\r\n for (var i = 0; i < ringIds.length; i++) {\r\n ring.rings.push(this.getRing(ringIds[i]).clone());\r\n }\r\n\r\n this.addRing(ring);\r\n\r\n for (var i = 0; i < ring.members.length; i++) {\r\n this.graph.vertices[ring.members[i]].value.bridgedRing = ring.id;\r\n }\r\n\r\n // Atoms inside the ring are no longer part of a ring but are now\r\n // associated with the bridged ring\r\n for (var i = 0; i < insideRing.length; i++) {\r\n let vertex = this.graph.vertices[insideRing[i]];\r\n vertex.value.rings = Array();\r\n }\r\n\r\n // Remove former rings from members of the bridged ring and add the bridged ring\r\n for (let id of ringMembers) {\r\n let vertex = this.graph.vertices[id];\r\n vertex.value.rings = ArrayHelper.removeAll(vertex.value.rings, ringIds);\r\n vertex.value.rings.push(ring.id);\r\n }\r\n\r\n // Remove all the ring connections no longer used\r\n for (var i = 0; i < ringIds.length; i++) {\r\n for (var j = i + 1; j < ringIds.length; j++) {\r\n this.removeRingConnectionsBetween(ringIds[i], ringIds[j]);\r\n }\r\n }\r\n\r\n // Update the ring connections and add this ring to the neighbours neighbours\r\n for (let id of neighbours) {\r\n let connections = this.getRingConnections(id, ringIds);\r\n\r\n for (var j = 0; j < connections.length; j++) {\r\n this.getRingConnection(connections[j]).updateOther(ring.id, id);\r\n }\r\n\r\n this.getRing(id).neighbours.push(ring.id);\r\n }\r\n\r\n return ring;\r\n }\r\n\r\n /**\r\n * Checks whether or not two vertices are in the same ring.\r\n *\r\n * @param {Vertex} vertexA A vertex.\r\n * @param {Vertex} vertexB A vertex.\r\n * @returns {Boolean} A boolean indicating whether or not the two vertices are in the same ring.\r\n */\r\n areVerticesInSameRing(vertexA, vertexB) {\r\n // This is a little bit lighter (without the array and push) than\r\n // getCommonRings().length > 0\r\n for (var i = 0; i < vertexA.value.rings.length; i++) {\r\n for (var j = 0; j < vertexB.value.rings.length; j++) {\r\n if (vertexA.value.rings[i] === vertexB.value.rings[j]) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Returns an array of ring ids shared by both vertices.\r\n *\r\n * @param {Vertex} vertexA A vertex.\r\n * @param {Vertex} vertexB A vertex.\r\n * @returns {Number[]} An array of ids of rings shared by the two vertices.\r\n */\r\n getCommonRings(vertexA, vertexB) {\r\n let commonRings = Array();\r\n\r\n for (var i = 0; i < vertexA.value.rings.length; i++) {\r\n for (var j = 0; j < vertexB.value.rings.length; j++) {\r\n if (vertexA.value.rings[i] == vertexB.value.rings[j]) {\r\n commonRings.push(vertexA.value.rings[i]);\r\n }\r\n }\r\n }\r\n\r\n return commonRings;\r\n }\r\n\r\n /**\r\n * Returns the aromatic or largest ring shared by the two vertices.\r\n *\r\n * @param {Vertex} vertexA A vertex.\r\n * @param {Vertex} vertexB A vertex.\r\n * @returns {(Ring|null)} If an aromatic common ring exists, that ring, else the largest (non-aromatic) ring, else null.\r\n */\r\n getLargestOrAromaticCommonRing(vertexA, vertexB) {\r\n let commonRings = this.getCommonRings(vertexA, vertexB);\r\n let maxSize = 0;\r\n let largestCommonRing = null;\r\n\r\n for (var i = 0; i < commonRings.length; i++) {\r\n let ring = this.getRing(commonRings[i]);\r\n let size = ring.getSize();\r\n\r\n if (ring.isBenzeneLike(this.graph.vertices)) {\r\n return ring;\r\n } else if (size > maxSize) {\r\n maxSize = size;\r\n largestCommonRing = ring;\r\n }\r\n }\r\n\r\n return largestCommonRing;\r\n }\r\n\r\n /**\r\n * Returns an array of vertices positioned at a specified location.\r\n *\r\n * @param {Vector2} position The position to search for vertices.\r\n * @param {Number} radius The radius within to search.\r\n * @param {Number} excludeVertexId A vertex id to be excluded from the search results.\r\n * @returns {Number[]} An array containing vertex ids in a given location.\r\n */\r\n getVerticesAt(position, radius, excludeVertexId) {\r\n let locals = Array();\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertex = this.graph.vertices[i];\r\n\r\n if (vertex.id === excludeVertexId || !vertex.positioned) {\r\n continue;\r\n }\r\n\r\n let distance = position.distanceSq(vertex.position);\r\n\r\n if (distance <= radius * radius) {\r\n locals.push(vertex.id);\r\n }\r\n }\r\n\r\n return locals;\r\n }\r\n\r\n /**\r\n * Returns the closest vertex (connected as well as unconnected).\r\n *\r\n * @param {Vertex} vertex The vertex of which to find the closest other vertex.\r\n * @returns {Vertex} The closest vertex.\r\n */\r\n getClosestVertex(vertex) {\r\n let minDist = 99999;\r\n let minVertex = null;\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let v = this.graph.vertices[i];\r\n\r\n if (v.id === vertex.id) {\r\n continue;\r\n }\r\n\r\n let distSq = vertex.position.distanceSq(v.position);\r\n\r\n if (distSq < minDist) {\r\n minDist = distSq;\r\n minVertex = v;\r\n }\r\n }\r\n\r\n return minVertex;\r\n }\r\n\r\n /**\r\n * Add a ring to this representation of a molecule.\r\n *\r\n * @param {Ring} ring A new ring.\r\n * @returns {Number} The ring id of the new ring.\r\n */\r\n addRing(ring) {\r\n ring.id = this.ringIdCounter++;\r\n this.rings.push(ring);\r\n\r\n return ring.id;\r\n }\r\n\r\n /**\r\n * Removes a ring from the array of rings associated with the current molecule.\r\n *\r\n * @param {Number} ringId A ring id.\r\n */\r\n removeRing(ringId) {\r\n this.rings = this.rings.filter(function (item) {\r\n return item.id !== ringId;\r\n });\r\n\r\n // Also remove ring connections involving this ring\r\n this.ringConnections = this.ringConnections.filter(function (item) {\r\n return item.firstRingId !== ringId && item.secondRingId !== ringId;\r\n });\r\n\r\n // Remove the ring as neighbour of other rings\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let r = this.rings[i];\r\n r.neighbours = r.neighbours.filter(function (item) {\r\n return item !== ringId;\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Gets a ring object from the array of rings associated with the current molecule by its id. The ring id is not equal to the index, since rings can be added and removed when processing bridged rings.\r\n *\r\n * @param {Number} ringId A ring id.\r\n * @returns {Ring} A ring associated with the current molecule.\r\n */\r\n getRing(ringId) {\r\n for (var i = 0; i < this.rings.length; i++) {\r\n if (this.rings[i].id == ringId) {\r\n return this.rings[i];\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Add a ring connection to this representation of a molecule.\r\n *\r\n * @param {RingConnection} ringConnection A new ringConnection.\r\n * @returns {Number} The ring connection id of the new ring connection.\r\n */\r\n addRingConnection(ringConnection) {\r\n ringConnection.id = this.ringConnectionIdCounter++;\r\n this.ringConnections.push(ringConnection);\r\n\r\n return ringConnection.id;\r\n }\r\n\r\n /**\r\n * Removes a ring connection from the array of rings connections associated with the current molecule.\r\n *\r\n * @param {Number} ringConnectionId A ring connection id.\r\n */\r\n removeRingConnection(ringConnectionId) {\r\n this.ringConnections = this.ringConnections.filter(function (item) {\r\n return item.id !== ringConnectionId;\r\n });\r\n }\r\n\r\n /**\r\n * Removes all ring connections between two vertices.\r\n *\r\n * @param {Number} vertexIdA A vertex id.\r\n * @param {Number} vertexIdB A vertex id.\r\n */\r\n removeRingConnectionsBetween(vertexIdA, vertexIdB) {\r\n let toRemove = Array();\r\n for (var i = 0; i < this.ringConnections.length; i++) {\r\n let ringConnection = this.ringConnections[i];\r\n\r\n if (ringConnection.firstRingId === vertexIdA && ringConnection.secondRingId === vertexIdB ||\r\n ringConnection.firstRingId === vertexIdB && ringConnection.secondRingId === vertexIdA) {\r\n toRemove.push(ringConnection.id);\r\n }\r\n }\r\n\r\n for (var i = 0; i < toRemove.length; i++) {\r\n this.removeRingConnection(toRemove[i]);\r\n }\r\n }\r\n\r\n /**\r\n * Get a ring connection with a given id.\r\n * \r\n * @param {Number} id \r\n * @returns {RingConnection} The ring connection with the specified id.\r\n */\r\n getRingConnection(id) {\r\n for (var i = 0; i < this.ringConnections.length; i++) {\r\n if (this.ringConnections[i].id == id) {\r\n return this.ringConnections[i];\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Get the ring connections between a ring and a set of rings.\r\n *\r\n * @param {Number} ringId A ring id.\r\n * @param {Number[]} ringIds An array of ring ids.\r\n * @returns {Number[]} An array of ring connection ids.\r\n */\r\n getRingConnections(ringId, ringIds) {\r\n let ringConnections = Array();\r\n\r\n for (var i = 0; i < this.ringConnections.length; i++) {\r\n let rc = this.ringConnections[i];\r\n\r\n for (var j = 0; j < ringIds.length; j++) {\r\n let id = ringIds[j];\r\n\r\n if (rc.firstRingId === ringId && rc.secondRingId === id ||\r\n rc.firstRingId === id && rc.secondRingId === ringId) {\r\n ringConnections.push(rc.id);\r\n }\r\n }\r\n }\r\n\r\n return ringConnections;\r\n }\r\n\r\n /**\r\n * Returns the overlap score of the current molecule based on its positioned vertices. The higher the score, the more overlaps occur in the structure drawing.\r\n *\r\n * @returns {Object} Returns the total overlap score and the overlap score of each vertex sorted by score (higher to lower). Example: { total: 99, scores: [ { id: 0, score: 22 }, ... ] }\r\n */\r\n getOverlapScore() {\r\n let total = 0.0;\r\n let overlapScores = new Float32Array(this.graph.vertices.length);\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n overlapScores[i] = 0;\r\n }\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n var j = this.graph.vertices.length;\r\n while (--j > i) {\r\n let a = this.graph.vertices[i];\r\n let b = this.graph.vertices[j];\r\n\r\n if (!a.value.isDrawn || !b.value.isDrawn) {\r\n continue;\r\n }\r\n\r\n let dist = Vector2.subtract(a.position, b.position).lengthSq();\r\n\r\n if (dist < this.opts.bondLengthSq) {\r\n let weighted = (this.opts.bondLength - Math.sqrt(dist)) / this.opts.bondLength;\r\n total += weighted;\r\n overlapScores[i] += weighted;\r\n overlapScores[j] += weighted;\r\n }\r\n }\r\n }\r\n\r\n let sortable = Array();\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n sortable.push({\r\n id: i,\r\n score: overlapScores[i]\r\n });\r\n }\r\n\r\n sortable.sort(function (a, b) {\r\n return b.score - a.score;\r\n });\r\n\r\n return {\r\n total: total,\r\n scores: sortable,\r\n vertexScores: overlapScores\r\n };\r\n }\r\n\r\n /**\r\n * When drawing a double bond, choose the side to place the double bond. E.g. a double bond should always been drawn inside a ring.\r\n *\r\n * @param {Vertex} vertexA A vertex.\r\n * @param {Vertex} vertexB A vertex.\r\n * @param {Vector2[]} sides An array containing the two normals of the line spanned by the two provided vertices.\r\n * @returns {Object} Returns an object containing the following information: {\r\n totalSideCount: Counts the sides of each vertex in the molecule, is an array [ a, b ],\r\n totalPosition: Same as position, but based on entire molecule,\r\n sideCount: Counts the sides of each neighbour, is an array [ a, b ],\r\n position: which side to position the second bond, is 0 or 1, represents the index in the normal array. This is based on only the neighbours\r\n anCount: the number of neighbours of vertexA,\r\n bnCount: the number of neighbours of vertexB\r\n }\r\n */\r\n chooseSide(vertexA, vertexB, sides) {\r\n // Check which side has more vertices\r\n // Get all the vertices connected to the both ends\r\n let an = vertexA.getNeighbours(vertexB.id);\r\n let bn = vertexB.getNeighbours(vertexA.id);\r\n let anCount = an.length;\r\n let bnCount = bn.length;\r\n\r\n // All vertices connected to the edge vertexA to vertexB\r\n let tn = ArrayHelper.merge(an, bn);\r\n\r\n // Only considering the connected vertices\r\n let sideCount = [0, 0];\r\n\r\n for (var i = 0; i < tn.length; i++) {\r\n let v = this.graph.vertices[tn[i]].position;\r\n\r\n if (v.sameSideAs(vertexA.position, vertexB.position, sides[0])) {\r\n sideCount[0]++;\r\n } else {\r\n sideCount[1]++;\r\n }\r\n }\r\n\r\n // Considering all vertices in the graph, this is to resolve ties\r\n // from the above side counts\r\n let totalSideCount = [0, 0];\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let v = this.graph.vertices[i].position;\r\n\r\n if (v.sameSideAs(vertexA.position, vertexB.position, sides[0])) {\r\n totalSideCount[0]++;\r\n } else {\r\n totalSideCount[1]++;\r\n }\r\n }\r\n\r\n return {\r\n totalSideCount: totalSideCount,\r\n totalPosition: totalSideCount[0] > totalSideCount[1] ? 0 : 1,\r\n sideCount: sideCount,\r\n position: sideCount[0] > sideCount[1] ? 0 : 1,\r\n anCount: anCount,\r\n bnCount: bnCount\r\n };\r\n }\r\n\r\n /**\r\n * Sets the center for a ring.\r\n *\r\n * @param {Ring} ring A ring.\r\n */\r\n setRingCenter(ring) {\r\n let ringSize = ring.getSize();\r\n let total = new Vector2(0, 0);\r\n\r\n for (var i = 0; i < ringSize; i++) {\r\n total.add(this.graph.vertices[ring.members[i]].position);\r\n }\r\n\r\n ring.center = total.divide(ringSize);\r\n }\r\n\r\n /**\r\n * Gets the center of a ring contained within a bridged ring and containing a given vertex.\r\n *\r\n * @param {Ring} ring A bridged ring.\r\n * @param {Vertex} vertex A vertex.\r\n * @returns {Vector2} The center of the subring that containing the vertex.\r\n */\r\n getSubringCenter(ring, vertex) {\r\n let rings = vertex.value.originalRings;\r\n let center = ring.center;\r\n let smallest = Number.MAX_VALUE;\r\n\r\n // Always get the smallest ring.\r\n for (var i = 0; i < rings.length; i++) {\r\n for (var j = 0; j < ring.rings.length; j++) {\r\n if (rings[i] === ring.rings[j].id) {\r\n if (ring.rings[j].getSize() < smallest) {\r\n center = ring.rings[j].center;\r\n smallest = ring.rings[j].getSize();\r\n }\r\n }\r\n }\r\n }\r\n\r\n return center;\r\n }\r\n\r\n /**\r\n * Draw the actual edges as bonds to the canvas.\r\n *\r\n * @param {Boolean} debug A boolean indicating whether or not to draw debug helpers.\r\n */\r\n drawEdges(debug) {\r\n let that = this;\r\n let drawn = Array(this.graph.edges.length);\r\n drawn.fill(false);\r\n\r\n this.graph.traverseBF(0, function (vertex) {\r\n let edges = that.graph.getEdges(vertex.id);\r\n for (var i = 0; i < edges.length; i++) {\r\n let edgeId = edges[i];\r\n if (!drawn[edgeId]) {\r\n drawn[edgeId] = true;\r\n that.drawEdge(edgeId, debug);\r\n }\r\n }\r\n });\r\n\r\n // Draw ring for implicitly defined aromatic rings\r\n if (!this.bridgedRing) {\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let ring = this.rings[i];\r\n\r\n if (this.isRingAromatic(ring)) {\r\n this.canvasWrapper.drawAromaticityRing(ring);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Draw the an edge as a bonds to the canvas.\r\n *\r\n * @param {Number} edgeId An edge id.\r\n * @param {Boolean} debug A boolean indicating whether or not to draw debug helpers.\r\n */\r\n drawEdge(edgeId, debug) {\r\n let that = this;\r\n let edge = this.graph.edges[edgeId];\r\n let vertexA = this.graph.vertices[edge.sourceId];\r\n let vertexB = this.graph.vertices[edge.targetId];\r\n let elementA = vertexA.value.element;\r\n let elementB = vertexB.value.element;\r\n\r\n if ((!vertexA.value.isDrawn || !vertexB.value.isDrawn) && this.opts.atomVisualization === 'default') {\r\n return;\r\n }\r\n\r\n let a = vertexA.position;\r\n let b = vertexB.position;\r\n let normals = this.getEdgeNormals(edge);\r\n\r\n // Create a point on each side of the line\r\n let sides = ArrayHelper.clone(normals);\r\n\r\n sides[0].multiplyScalar(10).add(a);\r\n sides[1].multiplyScalar(10).add(a);\r\n\r\n if (edge.bondType === '=' || this.getRingbondType(vertexA, vertexB) === '=' ||\r\n (edge.isPartOfAromaticRing && this.bridgedRing)) {\r\n // Always draw double bonds inside the ring\r\n let inRing = this.areVerticesInSameRing(vertexA, vertexB);\r\n let s = this.chooseSide(vertexA, vertexB, sides);\r\n\r\n if (inRing) {\r\n // Always draw double bonds inside a ring\r\n // if the bond is shared by two rings, it is drawn in the larger\r\n // problem: smaller ring is aromatic, bond is still drawn in larger -> fix this\r\n let lcr = this.getLargestOrAromaticCommonRing(vertexA, vertexB);\r\n let center = lcr.center;\r\n\r\n normals[0].multiplyScalar(that.opts.bondSpacing);\r\n normals[1].multiplyScalar(that.opts.bondSpacing);\r\n\r\n // Choose the normal that is on the same side as the center\r\n let line = null;\r\n\r\n if (center.sameSideAs(vertexA.position, vertexB.position, Vector2.add(a, normals[0]))) {\r\n line = new Line(Vector2.add(a, normals[0]), Vector2.add(b, normals[0]), elementA, elementB);\r\n } else {\r\n line = new Line(Vector2.add(a, normals[1]), Vector2.add(b, normals[1]), elementA, elementB);\r\n }\r\n\r\n line.shorten(this.opts.bondLength - this.opts.shortBondLength * this.opts.bondLength);\r\n\r\n // The shortened edge\r\n if (edge.isPartOfAromaticRing) {\r\n this.canvasWrapper.drawLine(line, true);\r\n } else {\r\n this.canvasWrapper.drawLine(line);\r\n }\r\n\r\n // The normal edge\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB));\r\n } else if (edge.center || vertexA.isTerminal() && vertexB.isTerminal()) {\r\n normals[0].multiplyScalar(that.opts.halfBondSpacing);\r\n normals[1].multiplyScalar(that.opts.halfBondSpacing);\r\n\r\n let lineA = new Line(Vector2.add(a, normals[0]), Vector2.add(b, normals[0]), elementA, elementB);\r\n let lineB = new Line(Vector2.add(a, normals[1]), Vector2.add(b, normals[1]), elementA, elementB);\r\n\r\n this.canvasWrapper.drawLine(lineA);\r\n this.canvasWrapper.drawLine(lineB);\r\n } else if (s.anCount == 0 && s.bnCount > 1 || s.bnCount == 0 && s.anCount > 1) {\r\n // Both lines are the same length here\r\n // Add the spacing to the edges (which are of unit length)\r\n normals[0].multiplyScalar(that.opts.halfBondSpacing);\r\n normals[1].multiplyScalar(that.opts.halfBondSpacing);\r\n\r\n let lineA = new Line(Vector2.add(a, normals[0]), Vector2.add(b, normals[0]), elementA, elementB);\r\n let lineB = new Line(Vector2.add(a, normals[1]), Vector2.add(b, normals[1]), elementA, elementB);\r\n\r\n this.canvasWrapper.drawLine(lineA);\r\n this.canvasWrapper.drawLine(lineB);\r\n } else if (s.sideCount[0] > s.sideCount[1]) {\r\n normals[0].multiplyScalar(that.opts.bondSpacing);\r\n normals[1].multiplyScalar(that.opts.bondSpacing);\r\n\r\n let line = new Line(Vector2.add(a, normals[0]), Vector2.add(b, normals[0]), elementA, elementB);\r\n\r\n line.shorten(this.opts.bondLength - this.opts.shortBondLength * this.opts.bondLength);\r\n this.canvasWrapper.drawLine(line);\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB));\r\n } else if (s.sideCount[0] < s.sideCount[1]) {\r\n normals[0].multiplyScalar(that.opts.bondSpacing);\r\n normals[1].multiplyScalar(that.opts.bondSpacing);\r\n\r\n let line = new Line(Vector2.add(a, normals[1]), Vector2.add(b, normals[1]), elementA, elementB);\r\n\r\n line.shorten(this.opts.bondLength - this.opts.shortBondLength * this.opts.bondLength);\r\n this.canvasWrapper.drawLine(line);\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB));\r\n } else if (s.totalSideCount[0] > s.totalSideCount[1]) {\r\n normals[0].multiplyScalar(that.opts.bondSpacing);\r\n normals[1].multiplyScalar(that.opts.bondSpacing);\r\n\r\n let line = new Line(Vector2.add(a, normals[0]), Vector2.add(b, normals[0]), elementA, elementB);\r\n\r\n line.shorten(this.opts.bondLength - this.opts.shortBondLength * this.opts.bondLength);\r\n this.canvasWrapper.drawLine(line);\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB));\r\n } else if (s.totalSideCount[0] <= s.totalSideCount[1]) {\r\n normals[0].multiplyScalar(that.opts.bondSpacing);\r\n normals[1].multiplyScalar(that.opts.bondSpacing);\r\n\r\n let line = new Line(Vector2.add(a, normals[1]), Vector2.add(b, normals[1]), elementA, elementB);\r\n\r\n line.shorten(this.opts.bondLength - this.opts.shortBondLength * this.opts.bondLength);\r\n this.canvasWrapper.drawLine(line);\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB));\r\n } else {\r\n\r\n }\r\n } else if (edge.bondType === '#') {\r\n normals[0].multiplyScalar(that.opts.bondSpacing / 1.5);\r\n normals[1].multiplyScalar(that.opts.bondSpacing / 1.5);\r\n\r\n let lineA = new Line(Vector2.add(a, normals[0]), Vector2.add(b, normals[0]), elementA, elementB);\r\n let lineB = new Line(Vector2.add(a, normals[1]), Vector2.add(b, normals[1]), elementA, elementB);\r\n\r\n this.canvasWrapper.drawLine(lineA);\r\n this.canvasWrapper.drawLine(lineB);\r\n\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB));\r\n } else if (edge.bondType === '.') {\r\n // TODO: Something... maybe... version 2?\r\n } else {\r\n let isChiralCenterA = vertexA.value.isStereoCenter;\r\n let isChiralCenterB = vertexB.value.isStereoCenter;\r\n\r\n if (edge.wedge === 'up') {\r\n this.canvasWrapper.drawWedge(new Line(a, b, elementA, elementB, isChiralCenterA, isChiralCenterB));\r\n } else if (edge.wedge === 'down') {\r\n this.canvasWrapper.drawDashedWedge(new Line(a, b, elementA, elementB, isChiralCenterA, isChiralCenterB));\r\n } else {\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB, isChiralCenterA, isChiralCenterB));\r\n }\r\n }\r\n\r\n if (debug) {\r\n let midpoint = Vector2.midpoint(a, b);\r\n this.canvasWrapper.drawDebugText(midpoint.x, midpoint.y, 'e: ' + edgeId);\r\n }\r\n }\r\n\r\n /**\r\n * Draws the vertices representing atoms to the canvas.\r\n *\r\n * @param {Boolean} debug A boolean indicating whether or not to draw debug messages to the canvas.\r\n */\r\n drawVertices(debug) {\r\n var i = this.graph.vertices.length;\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertex = this.graph.vertices[i];\r\n let atom = vertex.value;\r\n let charge = 0;\r\n let isotope = 0;\r\n let bondCount = vertex.value.bondCount;\r\n let element = atom.element;\r\n let hydrogens = Atom.maxBonds[element] - bondCount;\r\n let dir = vertex.getTextDirection(this.graph.vertices);\r\n let isTerminal = this.opts.terminalCarbons || element !== 'C' || atom.hasAttachedPseudoElements ? vertex.isTerminal() : false;\r\n let isCarbon = atom.element === 'C';\r\n\r\n // This is a HACK to remove all hydrogens from nitrogens in aromatic rings, as this\r\n // should be the most common state. This has to be fixed by kekulization\r\n if (atom.element === 'N' && atom.isPartOfAromaticRing) {\r\n hydrogens = 0;\r\n }\r\n\r\n if (atom.bracket) {\r\n hydrogens = atom.bracket.hcount;\r\n charge = atom.bracket.charge;\r\n isotope = atom.bracket.isotope;\r\n }\r\n\r\n if (this.opts.atomVisualization === 'allballs') {\r\n this.canvasWrapper.drawBall(vertex.position.x, vertex.position.y, element);\r\n } else if ((atom.isDrawn && (!isCarbon || atom.drawExplicit || isTerminal || atom.hasAttachedPseudoElements)) || this.graph.vertices.length === 1) {\r\n if (this.opts.atomVisualization === 'default') {\r\n this.canvasWrapper.drawText(vertex.position.x, vertex.position.y,\r\n element, hydrogens, dir, isTerminal, charge, isotope, atom.getAttachedPseudoElements());\r\n } else if (this.opts.atomVisualization === 'balls') {\r\n this.canvasWrapper.drawBall(vertex.position.x, vertex.position.y, element);\r\n }\r\n } else if (vertex.getNeighbourCount() === 2 && vertex.forcePositioned == true) {\r\n // If there is a carbon which bonds are in a straight line, draw a dot\r\n let a = this.graph.vertices[vertex.neighbours[0]].position;\r\n let b = this.graph.vertices[vertex.neighbours[1]].position;\r\n let angle = Vector2.threePointangle(vertex.position, a, b);\r\n\r\n if (Math.abs(Math.PI - angle) < 0.1) {\r\n this.canvasWrapper.drawPoint(vertex.position.x, vertex.position.y, element);\r\n }\r\n }\r\n\r\n if (debug) {\r\n let value = 'v: ' + vertex.id + ' ' + ArrayHelper.print(atom.ringbonds);\r\n this.canvasWrapper.drawDebugText(vertex.position.x, vertex.position.y, value);\r\n } else {\r\n // this.canvasWrapper.drawDebugText(vertex.position.x, vertex.position.y, vertex.value.chirality);\r\n }\r\n }\r\n\r\n // Draw the ring centers for debug purposes\r\n if (this.opts.debug) {\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let center = this.rings[i].center;\r\n this.canvasWrapper.drawDebugPoint(center.x, center.y, 'r: ' + this.rings[i].id);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Position the vertices according to their bonds and properties.\r\n */\r\n position() {\r\n let startVertex = null;\r\n\r\n // Always start drawing at a bridged ring if there is one\r\n // If not, start with a ring\r\n // else, start with 0\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n if (this.graph.vertices[i].value.bridgedRing !== null) {\r\n startVertex = this.graph.vertices[i];\r\n break;\r\n }\r\n }\r\n\r\n for (var i = 0; i < this.rings.length; i++) {\r\n if (this.rings[i].isBridged) {\r\n startVertex = this.graph.vertices[this.rings[i].members[0]];\r\n }\r\n }\r\n\r\n if (this.rings.length > 0 && startVertex === null) {\r\n startVertex = this.graph.vertices[this.rings[0].members[0]];\r\n }\r\n\r\n if (startVertex === null) {\r\n startVertex = this.graph.vertices[0];\r\n }\r\n\r\n this.createNextBond(startVertex, null, 0.0);\r\n }\r\n\r\n /**\r\n * Stores the current information associated with rings.\r\n */\r\n backupRingInformation() {\r\n this.originalRings = Array();\r\n this.originalRingConnections = Array();\r\n\r\n for (var i = 0; i < this.rings.length; i++) {\r\n this.originalRings.push(this.rings[i]);\r\n }\r\n\r\n for (var i = 0; i < this.ringConnections.length; i++) {\r\n this.originalRingConnections.push(this.ringConnections[i]);\r\n }\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n this.graph.vertices[i].value.backupRings();\r\n }\r\n }\r\n\r\n /**\r\n * Restores the most recently backed up information associated with rings.\r\n */\r\n restoreRingInformation() {\r\n // Get the subring centers from the bridged rings\r\n let bridgedRings = this.getBridgedRings();\r\n\r\n this.rings = Array();\r\n this.ringConnections = Array();\r\n\r\n for (var i = 0; i < bridgedRings.length; i++) {\r\n let bridgedRing = bridgedRings[i];\r\n\r\n for (var j = 0; j < bridgedRing.rings.length; j++) {\r\n let ring = bridgedRing.rings[j];\r\n this.originalRings[ring.id].center = ring.center;\r\n }\r\n }\r\n\r\n for (var i = 0; i < this.originalRings.length; i++) {\r\n this.rings.push(this.originalRings[i]);\r\n }\r\n\r\n for (var i = 0; i < this.originalRingConnections.length; i++) {\r\n this.ringConnections.push(this.originalRingConnections[i]);\r\n }\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n this.graph.vertices[i].value.restoreRings();\r\n }\r\n }\r\n\r\n // TODO: This needs some cleaning up\r\n\r\n /**\r\n * Creates a new ring, that is, positiones all the vertices inside a ring.\r\n *\r\n * @param {Ring} ring The ring to position.\r\n * @param {(Vector2|null)} [center=null] The center of the ring to be created.\r\n * @param {(Vertex|null)} [startVertex=null] The first vertex to be positioned inside the ring.\r\n * @param {(Vertex|null)} [previousVertex=null] The last vertex that was positioned.\r\n * @param {Boolean} [previousVertex=false] A boolean indicating whether or not this ring was force positioned already - this is needed after force layouting a ring, in order to draw rings connected to it.\r\n */\r\n createRing(ring, center = null, startVertex = null, previousVertex = null) {\r\n if (ring.positioned) {\r\n return;\r\n }\r\n\r\n center = center ? center : new Vector2(0, 0);\r\n\r\n let orderedNeighbours = ring.getOrderedNeighbours(this.ringConnections);\r\n let startingAngle = startVertex ? Vector2.subtract(startVertex.position, center).angle() : 0;\r\n\r\n let radius = MathHelper.polyCircumradius(this.opts.bondLength, ring.getSize());\r\n let angle = MathHelper.centralAngle(ring.getSize());\r\n\r\n ring.centralAngle = angle;\r\n\r\n let a = startingAngle;\r\n let that = this;\r\n let startVertexId = (startVertex) ? startVertex.id : null;\r\n\r\n if (ring.members.indexOf(startVertexId) === -1) {\r\n if (startVertex) {\r\n startVertex.positioned = false;\r\n }\r\n\r\n startVertexId = ring.members[0];\r\n }\r\n\r\n // If the ring is bridged, then draw the vertices inside the ring\r\n // using a force based approach\r\n if (ring.isBridged) {\r\n this.graph.kkLayout(ring.members.slice(), center, startVertex.id, ring, this.opts.bondLength);\r\n ring.positioned = true;\r\n\r\n // Update the center of the bridged ring\r\n this.setRingCenter(ring);\r\n center = ring.center;\r\n\r\n // Setting the centers for the subrings\r\n for (var i = 0; i < ring.rings.length; i++) {\r\n this.setRingCenter(ring.rings[i]);\r\n }\r\n } else {\r\n ring.eachMember(this.graph.vertices, function (v) {\r\n let vertex = that.graph.vertices[v];\r\n\r\n if (!vertex.positioned) {\r\n vertex.setPosition(center.x + Math.cos(a) * radius, center.y + Math.sin(a) * radius);\r\n }\r\n\r\n a += angle;\r\n\r\n if (!ring.isBridged || ring.rings.length < 3) {\r\n vertex.angle = a;\r\n vertex.positioned = true;\r\n }\r\n }, startVertexId, (previousVertex) ? previousVertex.id : null);\r\n }\r\n\r\n ring.positioned = true;\r\n ring.center = center;\r\n\r\n // Draw neighbours in decreasing order of connectivity\r\n for (var i = 0; i < orderedNeighbours.length; i++) {\r\n let neighbour = this.getRing(orderedNeighbours[i].neighbour);\r\n\r\n if (neighbour.positioned) {\r\n continue;\r\n }\r\n\r\n let vertices = RingConnection.getVertices(this.ringConnections, ring.id, neighbour.id);\r\n\r\n if (vertices.length === 2) {\r\n // This ring is a fused ring\r\n ring.isFused = true;\r\n neighbour.isFused = true;\r\n\r\n let vertexA = this.graph.vertices[vertices[0]];\r\n let vertexB = this.graph.vertices[vertices[1]];\r\n\r\n // Get middle between vertex A and B\r\n let midpoint = Vector2.midpoint(vertexA.position, vertexB.position);\r\n\r\n // Get the normals to the line between A and B\r\n let normals = Vector2.normals(vertexA.position, vertexB.position);\r\n\r\n // Normalize the normals\r\n normals[0].normalize();\r\n normals[1].normalize();\r\n\r\n // Set length from middle of side to center (the apothem)\r\n let r = MathHelper.polyCircumradius(this.opts.bondLength, neighbour.getSize());\r\n let apothem = MathHelper.apothem(r, neighbour.getSize());\r\n\r\n normals[0].multiplyScalar(apothem).add(midpoint);\r\n normals[1].multiplyScalar(apothem).add(midpoint);\r\n\r\n // Pick the normal which results in a larger distance to the previous center\r\n // Also check whether it's inside another ring\r\n let nextCenter = normals[0];\r\n if (Vector2.subtract(center, normals[1]).lengthSq() > Vector2.subtract(center, normals[0]).lengthSq()) {\r\n nextCenter = normals[1];\r\n }\r\n\r\n // Get the vertex (A or B) which is in clock-wise direction of the other\r\n let posA = Vector2.subtract(vertexA.position, nextCenter);\r\n let posB = Vector2.subtract(vertexB.position, nextCenter);\r\n\r\n if (posA.clockwise(posB) === -1) {\r\n if (!neighbour.positioned) {\r\n this.createRing(neighbour, nextCenter, vertexA, vertexB);\r\n }\r\n } else {\r\n if (!neighbour.positioned) {\r\n this.createRing(neighbour, nextCenter, vertexB, vertexA);\r\n }\r\n }\r\n } else if (vertices.length === 1) {\r\n // This ring is a spiro\r\n ring.isSpiro = true;\r\n neighbour.isSpiro = true;\r\n\r\n let vertexA = this.graph.vertices[vertices[0]];\r\n\r\n // Get the vector pointing from the shared vertex to the new centpositioner\r\n let nextCenter = Vector2.subtract(center, vertexA.position);\r\n\r\n nextCenter.invert();\r\n nextCenter.normalize();\r\n\r\n // Get the distance from the vertex to the center\r\n let r = MathHelper.polyCircumradius(this.opts.bondLength, neighbour.getSize());\r\n\r\n nextCenter.multiplyScalar(r);\r\n nextCenter.add(vertexA.position);\r\n\r\n if (!neighbour.positioned) {\r\n this.createRing(neighbour, nextCenter, vertexA);\r\n }\r\n }\r\n }\r\n\r\n // Next, draw atoms that are not part of a ring that are directly attached to this ring\r\n for (var i = 0; i < ring.members.length; i++) {\r\n let ringMember = this.graph.vertices[ring.members[i]];\r\n let ringMemberNeighbours = ringMember.neighbours;\r\n\r\n // If there are multiple, the ovlerap will be resolved in the appropriate step\r\n for (var j = 0; j < ringMemberNeighbours.length; j++) {\r\n let v = this.graph.vertices[ringMemberNeighbours[j]];\r\n\r\n if (v.positioned) {\r\n continue;\r\n }\r\n\r\n v.value.isConnectedToRing = true;\r\n this.createNextBond(v, ringMember, 0.0);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Rotate an entire subtree by an angle around a center.\r\n *\r\n * @param {Number} vertexId A vertex id (the root of the sub-tree).\r\n * @param {Number} parentVertexId A vertex id in the previous direction of the subtree that is to rotate.\r\n * @param {Number} angle An angle in randians.\r\n * @param {Vector2} center The rotational center.\r\n */\r\n rotateSubtree(vertexId, parentVertexId, angle, center) {\r\n let that = this;\r\n\r\n this.graph.traverseTree(vertexId, parentVertexId, function (vertex) {\r\n vertex.position.rotateAround(angle, center);\r\n\r\n for (var i = 0; i < vertex.value.anchoredRings.length; i++) {\r\n let ring = that.rings[vertex.value.anchoredRings[i]];\r\n\r\n if (ring) {\r\n ring.center.rotateAround(angle, center);\r\n }\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Gets the overlap score of a subtree.\r\n *\r\n * @param {Number} vertexId A vertex id (the root of the sub-tree).\r\n * @param {Number} parentVertexId A vertex id in the previous direction of the subtree.\r\n * @param {Number[]} vertexOverlapScores An array containing the vertex overlap scores indexed by vertex id.\r\n * @returns {Object} An object containing the total overlap score and the center of mass of the subtree weighted by overlap score { value: 0.2, center: new Vector2() }.\r\n */\r\n getSubtreeOverlapScore(vertexId, parentVertexId, vertexOverlapScores) {\r\n let that = this;\r\n let score = 0;\r\n let center = new Vector2(0, 0);\r\n let count = 0;\r\n\r\n this.graph.traverseTree(vertexId, parentVertexId, function (vertex) {\r\n if (!vertex.value.isDrawn) {\r\n return;\r\n }\r\n\r\n let s = vertexOverlapScores[vertex.id];\r\n if (s > that.opts.overlapSensitivity) {\r\n score += s;\r\n count++;\r\n }\r\n\r\n let position = that.graph.vertices[vertex.id].position.clone();\r\n position.multiplyScalar(s)\r\n center.add(position);\r\n });\r\n\r\n center.divide(score);\r\n\r\n return {\r\n value: score / count,\r\n center: center\r\n };\r\n }\r\n\r\n /**\r\n * Returns the current (positioned vertices so far) center of mass.\r\n * \r\n * @returns {Vector2} The current center of mass.\r\n */\r\n getCurrentCenterOfMass() {\r\n let total = new Vector2(0, 0);\r\n let count = 0;\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertex = this.graph.vertices[i];\r\n\r\n if (vertex.positioned) {\r\n total.add(vertex.position);\r\n count++;\r\n }\r\n }\r\n\r\n return total.divide(count);\r\n }\r\n\r\n /**\r\n * Returns the current (positioned vertices so far) center of mass in the neighbourhood of a given position.\r\n *\r\n * @param {Vector2} vec The point at which to look for neighbours.\r\n * @param {Number} [r=currentBondLength*2.0] The radius of vertices to include.\r\n * @returns {Vector2} The current center of mass.\r\n */\r\n getCurrentCenterOfMassInNeigbourhood(vec, r = this.opts.bondLength * 2.0) {\r\n let total = new Vector2(0, 0);\r\n let count = 0;\r\n let rSq = r * r;\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertex = this.graph.vertices[i];\r\n\r\n if (vertex.positioned && vec.distanceSq(vertex.position) < rSq) {\r\n total.add(vertex.position);\r\n count++;\r\n }\r\n }\r\n\r\n return total.divide(count);\r\n }\r\n\r\n /**\r\n * Resolve primary (exact) overlaps, such as two vertices that are connected to the same ring vertex.\r\n */\r\n resolvePrimaryOverlaps() {\r\n let overlaps = Array();\r\n let done = Array(this.graph.vertices.length);\r\n\r\n // Looking for overlaps created by two bonds coming out of a ring atom, which both point straight\r\n // away from the ring and are thus perfectly overlapping.\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let ring = this.rings[i];\r\n\r\n for (var j = 0; j < ring.members.length; j++) {\r\n let vertex = this.graph.vertices[ring.members[j]];\r\n\r\n if (done[vertex.id]) {\r\n continue;\r\n }\r\n\r\n done[vertex.id] = true;\r\n\r\n let nonRingNeighbours = this.getNonRingNeighbours(vertex.id);\r\n\r\n if (nonRingNeighbours.length > 1) {\r\n // Look for rings where there are atoms with two bonds outside the ring (overlaps)\r\n let rings = Array();\r\n\r\n for (var k = 0; k < vertex.value.rings.length; k++) {\r\n rings.push(vertex.value.rings[k]);\r\n }\r\n\r\n overlaps.push({\r\n common: vertex,\r\n rings: rings,\r\n vertices: nonRingNeighbours\r\n });\r\n } else if (nonRingNeighbours.length === 1 && vertex.value.rings.length === 2) {\r\n // Look for bonds coming out of joined rings to adjust the angle, an example is: C1=CC(=CC=C1)[C@]12SCCN1CC1=CC=CC=C21\r\n // where the angle has to be adjusted to account for fused ring\r\n let rings = Array();\r\n \r\n for (var k = 0; k < vertex.value.rings.length; k++) {\r\n rings.push(vertex.value.rings[k]);\r\n }\r\n\r\n overlaps.push({\r\n common: vertex,\r\n rings: rings,\r\n vertices: nonRingNeighbours\r\n });\r\n }\r\n }\r\n }\r\n\r\n for (var i = 0; i < overlaps.length; i++) {\r\n let overlap = overlaps[i];\r\n\r\n if (overlap.vertices.length === 2) {\r\n let a = overlap.vertices[0];\r\n let b = overlap.vertices[1];\r\n\r\n if (!a.value.isDrawn || !b.value.isDrawn) {\r\n continue;\r\n }\r\n\r\n let angle = (2 * Math.PI - this.getRing(overlap.rings[0]).getAngle()) / 6.0;\r\n\r\n this.rotateSubtree(a.id, overlap.common.id, angle, overlap.common.position);\r\n this.rotateSubtree(b.id, overlap.common.id, -angle, overlap.common.position);\r\n\r\n // Decide which way to rotate the vertices depending on the effect it has on the overlap score\r\n let overlapScore = this.getOverlapScore();\r\n let subTreeOverlapA = this.getSubtreeOverlapScore(a.id, overlap.common.id, overlapScore.vertexScores);\r\n let subTreeOverlapB = this.getSubtreeOverlapScore(b.id, overlap.common.id, overlapScore.vertexScores);\r\n let total = subTreeOverlapA.value + subTreeOverlapB.value;\r\n\r\n this.rotateSubtree(a.id, overlap.common.id, -2.0 * angle, overlap.common.position);\r\n this.rotateSubtree(b.id, overlap.common.id, 2.0 * angle, overlap.common.position);\r\n\r\n overlapScore = this.getOverlapScore();\r\n subTreeOverlapA = this.getSubtreeOverlapScore(a.id, overlap.common.id, overlapScore.vertexScores);\r\n subTreeOverlapB = this.getSubtreeOverlapScore(b.id, overlap.common.id, overlapScore.vertexScores);\r\n\r\n if (subTreeOverlapA.value + subTreeOverlapB.value > total) {\r\n this.rotateSubtree(a.id, overlap.common.id, 2.0 * angle, overlap.common.position);\r\n this.rotateSubtree(b.id, overlap.common.id, -2.0 * angle, overlap.common.position);\r\n }\r\n } else if (overlap.vertices.length === 1) {\r\n if (overlap.rings.length === 2) {\r\n // TODO: Implement for more overlap resolution\r\n // console.log(overlap);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Resolve secondary overlaps. Those overlaps are due to the structure turning back on itself.\r\n *\r\n * @param {Object[]} scores An array of objects sorted descending by score.\r\n * @param {Number} scores[].id A vertex id.\r\n * @param {Number} scores[].score The overlap score associated with the vertex id.\r\n */\r\n resolveSecondaryOverlaps(scores) {\r\n for (var i = 0; i < scores.length; i++) {\r\n if (scores[i].score > this.opts.overlapSensitivity) {\r\n let vertex = this.graph.vertices[scores[i].id];\r\n\r\n if (vertex.isTerminal()) {\r\n let closest = this.getClosestVertex(vertex);\r\n\r\n if (closest) {\r\n // If one of the vertices is the first one, the previous vertex is not the central vertex but the dummy\r\n // so take the next rather than the previous, which is vertex 1\r\n let closestPosition = null;\r\n\r\n if (closest.isTerminal()) {\r\n closestPosition = closest.id === 0 ? this.graph.vertices[1].position : closest.previousPosition\r\n } else {\r\n closestPosition = closest.id === 0 ? this.graph.vertices[1].position : closest.position\r\n }\r\n\r\n let vertexPreviousPosition = vertex.id === 0 ? this.graph.vertices[1].position : vertex.previousPosition;\r\n\r\n vertex.position.rotateAwayFrom(closestPosition, vertexPreviousPosition, MathHelper.toRad(20));\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Get the last non-null or 0 angle vertex.\r\n * @param {Number} vertexId A vertex id.\r\n * @returns {Vertex} The last vertex with an angle that was not 0 or null.\r\n */\r\n getLastVertexWithAngle(vertexId) {\r\n let angle = 0;\r\n let vertex = null;\r\n\r\n while (!angle && vertexId) {\r\n vertex = this.graph.vertices[vertexId];\r\n angle = vertex.angle;\r\n vertexId = vertex.parentVertexId;\r\n }\r\n\r\n return vertex;\r\n }\r\n\r\n /**\r\n * Positiones the next vertex thus creating a bond.\r\n *\r\n * @param {Vertex} vertex A vertex.\r\n * @param {Vertex} [previousVertex=null] The previous vertex which has been positioned.\r\n * @param {Number} [angle=0.0] The (global) angle of the vertex.\r\n * @param {Boolean} [originShortest=false] Whether the origin is the shortest subtree in the branch.\r\n * @param {Boolean} [skipPositioning=false] Whether or not to skip positioning and just check the neighbours.\r\n */\r\n createNextBond(vertex, previousVertex = null, angle = 0.0, originShortest = false, skipPositioning = false) {\r\n if (vertex.positioned && !skipPositioning) {\r\n return;\r\n }\r\n\r\n // If the double bond config was set on this vertex, do not check later\r\n let doubleBondConfigSet = false;\r\n\r\n // Keeping track of configurations around double bonds\r\n if (previousVertex) {\r\n let edge = this.graph.getEdge(vertex.id, previousVertex.id);\r\n\r\n if ((edge.bondType === '/' || edge.bondType === '\\\\') && ++this.doubleBondConfigCount % 2 === 1) {\r\n if (this.doubleBondConfig === null) {\r\n this.doubleBondConfig = edge.bondType;\r\n doubleBondConfigSet = true;\r\n\r\n // Switch if the bond is a branch bond and previous vertex is the first\r\n // TODO: Why is it different with the first vertex?\r\n if (previousVertex.parentVertexId === null && vertex.value.branchBond) {\r\n if (this.doubleBondConfig === '/') {\r\n this.doubleBondConfig = '\\\\';\r\n } else if (this.doubleBondConfig === '\\\\') {\r\n this.doubleBondConfig = '/';\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n // If the current node is the member of one ring, then point straight away\r\n // from the center of the ring. However, if the current node is a member of\r\n // two rings, point away from the middle of the centers of the two rings\r\n if (!skipPositioning) {\r\n if (!previousVertex) {\r\n // Add a (dummy) previous position if there is no previous vertex defined\r\n // Since the first vertex is at (0, 0), create a vector at (bondLength, 0)\r\n // and rotate it by 90°\r\n\r\n let dummy = new Vector2(this.opts.bondLength, 0);\r\n dummy.rotate(MathHelper.toRad(-60));\r\n\r\n vertex.previousPosition = dummy;\r\n vertex.setPosition(this.opts.bondLength, 0);\r\n vertex.angle = MathHelper.toRad(-60);\r\n\r\n // Do not position the vertex if it belongs to a bridged ring that is positioned using a layout algorithm.\r\n if (vertex.value.bridgedRing === null) {\r\n vertex.positioned = true;\r\n }\r\n } else if (previousVertex.value.rings.length > 0) {\r\n let neighbours = previousVertex.neighbours;\r\n let joinedVertex = null;\r\n let pos = new Vector2(0.0, 0.0);\r\n\r\n if (previousVertex.value.bridgedRing === null && previousVertex.value.rings.length > 1) {\r\n for (var i = 0; i < neighbours.length; i++) {\r\n let neighbour = this.graph.vertices[neighbours[i]];\r\n if (ArrayHelper.containsAll(neighbour.value.rings, previousVertex.value.rings)) {\r\n joinedVertex = neighbour;\r\n break;\r\n }\r\n }\r\n }\r\n\r\n if (joinedVertex === null) {\r\n for (var i = 0; i < neighbours.length; i++) {\r\n let v = this.graph.vertices[neighbours[i]];\r\n\r\n if (v.positioned && this.areVerticesInSameRing(v, previousVertex)) {\r\n pos.add(Vector2.subtract(v.position, previousVertex.position));\r\n }\r\n }\r\n\r\n pos.invert().normalize().multiplyScalar(this.opts.bondLength).add(previousVertex.position);\r\n } else {\r\n pos = joinedVertex.position.clone().rotateAround(Math.PI, previousVertex.position);\r\n }\r\n\r\n vertex.previousPosition = previousVertex.position;\r\n vertex.setPositionFromVector(pos);\r\n vertex.positioned = true;\r\n } else {\r\n // If the previous vertex was not part of a ring, draw a bond based\r\n // on the global angle of the previous bond\r\n let v = new Vector2(this.opts.bondLength, 0);\r\n\r\n v.rotate(angle);\r\n v.add(previousVertex.position);\r\n\r\n vertex.setPositionFromVector(v);\r\n vertex.previousPosition = previousVertex.position;\r\n vertex.positioned = true;\r\n }\r\n }\r\n\r\n // Go to next vertex\r\n // If two rings are connected by a bond ...\r\n if (vertex.value.bridgedRing !== null) {\r\n let nextRing = this.getRing(vertex.value.bridgedRing);\r\n\r\n if (!nextRing.positioned) {\r\n let nextCenter = Vector2.subtract(vertex.previousPosition, vertex.position);\r\n\r\n nextCenter.invert();\r\n nextCenter.normalize();\r\n\r\n let r = MathHelper.polyCircumradius(this.opts.bondLength, nextRing.members.length);\r\n nextCenter.multiplyScalar(r);\r\n nextCenter.add(vertex.position);\r\n \r\n this.createRing(nextRing, nextCenter, vertex);\r\n }\r\n } else if (vertex.value.rings.length > 0) {\r\n let nextRing = this.getRing(vertex.value.rings[0]);\r\n \r\n if (!nextRing.positioned) {\r\n let nextCenter = Vector2.subtract(vertex.previousPosition, vertex.position);\r\n\r\n nextCenter.invert();\r\n nextCenter.normalize();\r\n\r\n let r = MathHelper.polyCircumradius(this.opts.bondLength, nextRing.getSize());\r\n\r\n nextCenter.multiplyScalar(r);\r\n nextCenter.add(vertex.position);\r\n \r\n this.createRing(nextRing, nextCenter, vertex);\r\n }\r\n } else {\r\n // Draw the non-ring vertices connected to this one \r\n let isStereoCenter = vertex.value.isStereoCenter;\r\n let tmpNeighbours = vertex.getNeighbours();\r\n let neighbours = Array();\r\n\r\n // Remove neighbours that are not drawn\r\n for (var i = 0; i < tmpNeighbours.length; i++) {\r\n if (this.graph.vertices[tmpNeighbours[i]].value.isDrawn) {\r\n neighbours.push(tmpNeighbours[i]);\r\n }\r\n }\r\n\r\n // Remove the previous vertex (which has already been drawn)\r\n if (previousVertex) {\r\n neighbours = ArrayHelper.remove(neighbours, previousVertex.id);\r\n }\r\n\r\n let previousAngle = vertex.getAngle();\r\n\r\n if (neighbours.length === 1) {\r\n let nextVertex = this.graph.vertices[neighbours[0]];\r\n\r\n // Make a single chain always cis except when there's a tribble (yes, this is a Star Trek reference) bond\r\n // or if there are successive double bonds. Added a ring check because if there is an aromatic ring the ring bond inside the ring counts as a double bond and leads to =-= being straight.\r\n if ((vertex.value.bondType === '#' || (previousVertex && previousVertex.value.bondType === '#')) ||\r\n vertex.value.bondType === '=' && previousVertex && previousVertex.value.rings.length === 0 &&\r\n previousVertex.value.bondType === '=' && vertex.value.branchBond !== '-') {\r\n vertex.value.drawExplicit = false;\r\n\r\n if (previousVertex) {\r\n let straightEdge1 = this.graph.getEdge(vertex.id, previousVertex.id);\r\n straightEdge1.center = true;\r\n }\r\n\r\n let straightEdge2 = this.graph.getEdge(vertex.id, nextVertex.id);\r\n straightEdge2.center = true;\r\n\r\n if (vertex.value.bondType === '#' || previousVertex && previousVertex.value.bondType === '#') {\r\n nextVertex.angle = 0.0;\r\n }\r\n\r\n nextVertex.drawExplicit = true;\r\n\r\n this.createNextBond(nextVertex, vertex, previousAngle + nextVertex.angle);\r\n } else if (previousVertex && previousVertex.value.rings.length > 0) {\r\n // If coming out of a ring, always draw away from the center of mass\r\n let proposedAngleA = MathHelper.toRad(60);\r\n let proposedAngleB = -proposedAngleA;\r\n\r\n let proposedVectorA = new Vector2(this.opts.bondLength, 0);\r\n let proposedVectorB = new Vector2(this.opts.bondLength, 0);\r\n\r\n proposedVectorA.rotate(proposedAngleA).add(vertex.position);\r\n proposedVectorB.rotate(proposedAngleB).add(vertex.position);\r\n\r\n // let centerOfMass = this.getCurrentCenterOfMassInNeigbourhood(vertex.position, 100);\r\n let centerOfMass = this.getCurrentCenterOfMass();\r\n let distanceA = proposedVectorA.distanceSq(centerOfMass);\r\n let distanceB = proposedVectorB.distanceSq(centerOfMass);\r\n\r\n nextVertex.angle = distanceA < distanceB ? proposedAngleB : proposedAngleA;\r\n\r\n this.createNextBond(nextVertex, vertex, previousAngle + nextVertex.angle);\r\n } else {\r\n let a = vertex.angle;\r\n // Take the min and max if the previous angle was in a 4-neighbourhood (90° angles)\r\n // TODO: If a is null or zero, it should be checked whether or not this one should go cis or trans, that is,\r\n // it should go into the oposite direction of the last non-null or 0 previous vertex / angle.\r\n if (previousVertex && previousVertex.neighbours.length > 3) {\r\n if (a > 0) {\r\n a = Math.min(1.0472, a);\r\n } else if (a < 0) {\r\n a = Math.max(-1.0472, a);\r\n } else {\r\n a = 1.0472;\r\n }\r\n } else if (!a) {\r\n let v = this.getLastVertexWithAngle(vertex.id);\r\n a = v.angle;\r\n\r\n if (!a) {\r\n a = 1.0472;\r\n }\r\n }\r\n\r\n // Handle configuration around double bonds\r\n if (previousVertex && !doubleBondConfigSet) {\r\n let bondType = this.graph.getEdge(vertex.id, nextVertex.id).bondType;\r\n\r\n if (bondType === '/') {\r\n if (this.doubleBondConfig === '/') {\r\n // Nothing to do since it will be trans per default\r\n } else if (this.doubleBondConfig === '\\\\') {\r\n a = -a;\r\n }\r\n this.doubleBondConfig = null;\r\n } else if (bondType === '\\\\') {\r\n if (this.doubleBondConfig === '/') {\r\n a = -a;\r\n } else if (this.doubleBondConfig === '\\\\') {\r\n // Nothing to do since it will be trans per default\r\n }\r\n this.doubleBondConfig = null;\r\n }\r\n }\r\n\r\n if (originShortest) {\r\n nextVertex.angle = a;\r\n } else {\r\n nextVertex.angle = -a;\r\n }\r\n\r\n this.createNextBond(nextVertex, vertex, previousAngle + nextVertex.angle);\r\n }\r\n } else if (neighbours.length === 2) {\r\n // If the previous vertex comes out of a ring, it doesn't have an angle set\r\n let a = vertex.angle;\r\n\r\n if (!a) {\r\n a = 1.0472;\r\n }\r\n\r\n // Check for the longer subtree - always go with cis for the longer subtree\r\n let subTreeDepthA = this.graph.getTreeDepth(neighbours[0], vertex.id);\r\n let subTreeDepthB = this.graph.getTreeDepth(neighbours[1], vertex.id);\r\n\r\n let l = this.graph.vertices[neighbours[0]];\r\n let r = this.graph.vertices[neighbours[1]];\r\n\r\n l.value.subtreeDepth = subTreeDepthA;\r\n r.value.subtreeDepth = subTreeDepthB;\r\n\r\n // Also get the subtree for the previous direction (this is important when\r\n // the previous vertex is the shortest path)\r\n let subTreeDepthC = this.graph.getTreeDepth(previousVertex ? previousVertex.id : null, vertex.id);\r\n if (previousVertex) {\r\n previousVertex.value.subtreeDepth = subTreeDepthC;\r\n }\r\n\r\n let cis = 0;\r\n let trans = 1;\r\n\r\n // Carbons go always cis\r\n if (r.value.element === 'C' && l.value.element !== 'C' && subTreeDepthB > 1 && subTreeDepthA < 5) {\r\n cis = 1;\r\n trans = 0;\r\n } else if (r.value.element !== 'C' && l.value.element === 'C' && subTreeDepthA > 1 && subTreeDepthB < 5) {\r\n cis = 0;\r\n trans = 1;\r\n } else if (subTreeDepthB > subTreeDepthA) {\r\n cis = 1;\r\n trans = 0;\r\n }\r\n\r\n let cisVertex = this.graph.vertices[neighbours[cis]];\r\n let transVertex = this.graph.vertices[neighbours[trans]];\r\n\r\n let edgeCis = this.graph.getEdge(vertex.id, cisVertex.id);\r\n let edgeTrans = this.graph.getEdge(vertex.id, transVertex.id);\r\n\r\n // If the origin tree is the shortest, make them the main chain\r\n let originShortest = false;\r\n if (subTreeDepthC < subTreeDepthA && subTreeDepthC < subTreeDepthB) {\r\n originShortest = true;\r\n }\r\n\r\n transVertex.angle = a;\r\n cisVertex.angle = -a;\r\n\r\n if (this.doubleBondConfig === '\\\\') {\r\n if (transVertex.value.branchBond === '\\\\') {\r\n transVertex.angle = -a;\r\n cisVertex.angle = a;\r\n }\r\n } else if (this.doubleBondConfig === '/') {\r\n if (transVertex.value.branchBond === '/') {\r\n transVertex.angle = -a;\r\n cisVertex.angle = a;\r\n }\r\n }\r\n\r\n this.createNextBond(transVertex, vertex, previousAngle + transVertex.angle, originShortest);\r\n this.createNextBond(cisVertex, vertex, previousAngle + cisVertex.angle, originShortest);\r\n } else if (neighbours.length === 3) {\r\n // The vertex with the longest sub-tree should always go straight\r\n let d1 = this.graph.getTreeDepth(neighbours[0], vertex.id);\r\n let d2 = this.graph.getTreeDepth(neighbours[1], vertex.id);\r\n let d3 = this.graph.getTreeDepth(neighbours[2], vertex.id);\r\n\r\n let s = this.graph.vertices[neighbours[0]];\r\n let l = this.graph.vertices[neighbours[1]];\r\n let r = this.graph.vertices[neighbours[2]];\r\n\r\n s.value.subtreeDepth = d1;\r\n l.value.subtreeDepth = d2;\r\n r.value.subtreeDepth = d3;\r\n\r\n if (d2 > d1 && d2 > d3) {\r\n s = this.graph.vertices[neighbours[1]];\r\n l = this.graph.vertices[neighbours[0]];\r\n r = this.graph.vertices[neighbours[2]];\r\n } else if (d3 > d1 && d3 > d2) {\r\n s = this.graph.vertices[neighbours[2]];\r\n l = this.graph.vertices[neighbours[0]];\r\n r = this.graph.vertices[neighbours[1]];\r\n }\r\n\r\n // Create a cross if more than one subtree is of length > 1\r\n // or the vertex is connected to a ring\r\n if (previousVertex &&\r\n previousVertex.value.rings.length < 1 &&\r\n s.value.rings.length < 1 &&\r\n l.value.rings.length < 1 &&\r\n r.value.rings.length < 1 &&\r\n this.graph.getTreeDepth(l.id, vertex.id) === 1 &&\r\n this.graph.getTreeDepth(r.id, vertex.id) === 1 &&\r\n this.graph.getTreeDepth(s.id, vertex.id) > 1) {\r\n\r\n s.angle = -vertex.angle;\r\n if (vertex.angle >= 0) {\r\n l.angle = MathHelper.toRad(30);\r\n r.angle = MathHelper.toRad(90);\r\n } else {\r\n l.angle = -MathHelper.toRad(30);\r\n r.angle = -MathHelper.toRad(90);\r\n }\r\n\r\n this.createNextBond(s, vertex, previousAngle + s.angle);\r\n this.createNextBond(l, vertex, previousAngle + l.angle);\r\n this.createNextBond(r, vertex, previousAngle + r.angle);\r\n } else {\r\n s.angle = 0.0;\r\n l.angle = MathHelper.toRad(90);\r\n r.angle = -MathHelper.toRad(90);\r\n\r\n this.createNextBond(s, vertex, previousAngle + s.angle);\r\n this.createNextBond(l, vertex, previousAngle + l.angle);\r\n this.createNextBond(r, vertex, previousAngle + r.angle);\r\n }\r\n } else if (neighbours.length === 4) {\r\n // The vertex with the longest sub-tree should always go to the reflected opposide direction\r\n let d1 = this.graph.getTreeDepth(neighbours[0], vertex.id);\r\n let d2 = this.graph.getTreeDepth(neighbours[1], vertex.id);\r\n let d3 = this.graph.getTreeDepth(neighbours[2], vertex.id);\r\n let d4 = this.graph.getTreeDepth(neighbours[3], vertex.id);\r\n\r\n let w = this.graph.vertices[neighbours[0]];\r\n let x = this.graph.vertices[neighbours[1]];\r\n let y = this.graph.vertices[neighbours[2]];\r\n let z = this.graph.vertices[neighbours[3]];\r\n\r\n w.value.subtreeDepth = d1;\r\n x.value.subtreeDepth = d2;\r\n y.value.subtreeDepth = d3;\r\n z.value.subtreeDepth = d4;\r\n\r\n if (d2 > d1 && d2 > d3 && d2 > d4) {\r\n w = this.graph.vertices[neighbours[1]];\r\n x = this.graph.vertices[neighbours[0]];\r\n y = this.graph.vertices[neighbours[2]];\r\n z = this.graph.vertices[neighbours[3]];\r\n } else if (d3 > d1 && d3 > d2 && d3 > d4) {\r\n w = this.graph.vertices[neighbours[2]];\r\n x = this.graph.vertices[neighbours[0]];\r\n y = this.graph.vertices[neighbours[1]];\r\n z = this.graph.vertices[neighbours[3]];\r\n } else if (d4 > d1 && d4 > d2 && d4 > d3) {\r\n w = this.graph.vertices[neighbours[3]];\r\n x = this.graph.vertices[neighbours[0]];\r\n y = this.graph.vertices[neighbours[1]];\r\n z = this.graph.vertices[neighbours[2]];\r\n }\r\n\r\n w.angle = -MathHelper.toRad(36);\r\n x.angle = MathHelper.toRad(36);\r\n y.angle = -MathHelper.toRad(108);\r\n z.angle = MathHelper.toRad(108);\r\n\r\n this.createNextBond(w, vertex, previousAngle + w.angle);\r\n this.createNextBond(x, vertex, previousAngle + x.angle);\r\n this.createNextBond(y, vertex, previousAngle + y.angle);\r\n this.createNextBond(z, vertex, previousAngle + z.angle);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Gets the vetex sharing the edge that is the common bond of two rings.\r\n *\r\n * @param {Vertex} vertex A vertex.\r\n * @returns {(Number|null)} The id of a vertex sharing the edge that is the common bond of two rings with the vertex provided or null, if none.\r\n */\r\n getCommonRingbondNeighbour(vertex) {\r\n let neighbours = vertex.neighbours;\r\n\r\n for (var i = 0; i < neighbours.length; i++) {\r\n let neighbour = this.graph.vertices[neighbours[i]];\r\n\r\n if (ArrayHelper.containsAll(neighbour.value.rings, vertex.value.rings)) {\r\n return neighbour;\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Check if a vector is inside any ring.\r\n *\r\n * @param {Vector2} vec A vector.\r\n * @returns {Boolean} A boolean indicating whether or not the point (vector) is inside any of the rings associated with the current molecule.\r\n */\r\n isPointInRing(vec) {\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let ring = this.rings[i];\r\n\r\n if (!ring.positioned) {\r\n continue;\r\n }\r\n\r\n let radius = MathHelper.polyCircumradius(this.opts.bondLength, ring.getSize());\r\n let radiusSq = radius * radius;\r\n\r\n if (vec.distanceSq(ring.center) < radiusSq) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Check whether or not an edge is part of a ring.\r\n *\r\n * @param {Edge} edge An edge.\r\n * @returns {Boolean} A boolean indicating whether or not the edge is part of a ring.\r\n */\r\n isEdgeInRing(edge) {\r\n let source = this.graph.vertices[edge.sourceId];\r\n let target = this.graph.vertices[edge.targetId];\r\n\r\n return this.areVerticesInSameRing(source, target);\r\n }\r\n\r\n /**\r\n * Check whether or not an edge is rotatable.\r\n *\r\n * @param {Edge} edge An edge.\r\n * @returns {Boolean} A boolean indicating whether or not the edge is rotatable.\r\n */\r\n isEdgeRotatable(edge) {\r\n let vertexA = this.graph.vertices[edge.sourceId];\r\n let vertexB = this.graph.vertices[edge.targetId];\r\n\r\n // Only single bonds are rotatable\r\n if (edge.bondType !== '-') {\r\n return false;\r\n }\r\n\r\n // Do not rotate edges that have a further single bond to each side - do that!\r\n // If the bond is terminal, it doesn't make sense to rotate it\r\n // if (vertexA.getNeighbourCount() + vertexB.getNeighbourCount() < 5) {\r\n // return false;\r\n // }\r\n\r\n if (vertexA.isTerminal() || vertexB.isTerminal()) {\r\n return false;\r\n }\r\n\r\n // Ringbonds are not rotatable\r\n if (vertexA.value.rings.length > 0 && vertexB.value.rings.length > 0 &&\r\n this.areVerticesInSameRing(vertexA, vertexB)) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Check whether or not a ring is an implicitly defined aromatic ring (lower case smiles).\r\n *\r\n * @param {Ring} ring A ring.\r\n * @returns {Boolean} A boolean indicating whether or not a ring is implicitly defined as aromatic.\r\n */\r\n isRingAromatic(ring) {\r\n for (var i = 0; i < ring.members.length; i++) {\r\n let vertex = this.graph.vertices[ring.members[i]];\r\n\r\n if (!vertex.value.isPartOfAromaticRing) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Get the normals of an edge.\r\n *\r\n * @param {Edge} edge An edge.\r\n * @returns {Vector2[]} An array containing two vectors, representing the normals.\r\n */\r\n getEdgeNormals(edge) {\r\n let v1 = this.graph.vertices[edge.sourceId].position;\r\n let v2 = this.graph.vertices[edge.targetId].position;\r\n\r\n // Get the normalized normals for the edge\r\n let normals = Vector2.units(v1, v2);\r\n\r\n return normals;\r\n }\r\n\r\n /**\r\n * Returns an array of vertices that are neighbouring a vertix but are not members of a ring (including bridges).\r\n *\r\n * @param {Number} vertexId A vertex id.\r\n * @returns {Vertex[]} An array of vertices.\r\n */\r\n getNonRingNeighbours(vertexId) {\r\n let nrneighbours = Array();\r\n let vertex = this.graph.vertices[vertexId];\r\n let neighbours = vertex.neighbours;\r\n\r\n for (var i = 0; i < neighbours.length; i++) {\r\n let neighbour = this.graph.vertices[neighbours[i]];\r\n let nIntersections = ArrayHelper.intersection(vertex.value.rings, neighbour.value.rings).length;\r\n\r\n if (nIntersections === 0 && neighbour.value.isBridge == false) {\r\n nrneighbours.push(neighbour);\r\n }\r\n }\r\n\r\n return nrneighbours;\r\n }\r\n\r\n /**\r\n * Annotaed stereochemistry information for visualization.\r\n */\r\n annotateStereochemistry() {\r\n let maxDepth = 10;\r\n\r\n // For each stereo-center\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertex = this.graph.vertices[i];\r\n\r\n if (!vertex.value.isStereoCenter) {\r\n continue;\r\n }\r\n\r\n let neighbours = vertex.getNeighbours();\r\n let nNeighbours = neighbours.length;\r\n let priorities = Array(nNeighbours);\r\n\r\n for (var j = 0; j < nNeighbours; j++) {\r\n let visited = new Uint8Array(this.graph.vertices.length);\r\n let priority = Array(Array());\r\n visited[vertex.id] = 1;\r\n\r\n this.visitStereochemistry(neighbours[j], vertex.id, visited, priority, maxDepth, 0);\r\n\r\n // Sort each level according to atomic number\r\n for (var k = 0; k < priority.length; k++) {\r\n priority[k].sort(function (a, b) {\r\n return b - a\r\n });\r\n }\r\n\r\n priorities[j] = [j, priority];\r\n }\r\n\r\n let maxLevels = 0;\r\n let maxEntries = 0;\r\n for (var j = 0; j < priorities.length; j++) {\r\n if (priorities[j][1].length > maxLevels) {\r\n maxLevels = priorities[j][1].length;\r\n }\r\n\r\n for (var k = 0; k < priorities[j][1].length; k++) {\r\n if (priorities[j][1][k].length > maxEntries) {\r\n maxEntries = priorities[j][1][k].length;\r\n }\r\n }\r\n }\r\n\r\n for (var j = 0; j < priorities.length; j++) {\r\n let diff = maxLevels - priorities[j][1].length;\r\n for (var k = 0; k < diff; k++) {\r\n priorities[j][1].push([]);\r\n }\r\n\r\n // Break ties by the position in the SMILES string as per specification\r\n priorities[j][1].push([neighbours[j]]);\r\n\r\n // Make all same length. Fill with zeroes.\r\n for (var k = 0; k < priorities[j][1].length; k++) {\r\n let diff = maxEntries - priorities[j][1][k].length;\r\n\r\n for (var l = 0; l < diff; l++) {\r\n priorities[j][1][k].push(0);\r\n }\r\n }\r\n }\r\n\r\n priorities.sort(function (a, b) {\r\n for (var j = 0; j < a[1].length; j++) {\r\n for (var k = 0; k < a[1][j].length; k++) {\r\n if (a[1][j][k] > b[1][j][k]) {\r\n return -1;\r\n } else if (a[1][j][k] < b[1][j][k]) {\r\n return 1;\r\n }\r\n }\r\n }\r\n\r\n return 0;\r\n });\r\n\r\n let order = new Uint8Array(nNeighbours);\r\n for (var j = 0; j < nNeighbours; j++) {\r\n order[j] = priorities[j][0];\r\n vertex.value.priority = j;\r\n }\r\n\r\n // Check the angles between elements 0 and 1, and 0 and 2 to determine whether they are\r\n // drawn cw or ccw\r\n // TODO: OC(Cl)=[C@]=C(C)F currently fails here, however this is, IMHO, not a valid SMILES.\r\n let posA = this.graph.vertices[neighbours[order[0]]].position;\r\n let posB = this.graph.vertices[neighbours[order[1]]].position;\r\n let posC = this.graph.vertices[neighbours[order[2]]].position;\r\n\r\n let cwA = posA.relativeClockwise(posB, vertex.position);\r\n let cwB = posA.relativeClockwise(posC, vertex.position);\r\n\r\n // If the second priority is clockwise from the first, the ligands are drawn clockwise, since\r\n // The hydrogen can be drawn on either side\r\n let isCw = cwA === -1;\r\n\r\n let rotation = vertex.value.bracket.chirality === '@' ? -1 : 1;\r\n let rs = MathHelper.parityOfPermutation(order) * rotation === 1 ? 'R' : 'S';\r\n\r\n // Flip the hydrogen direction when the drawing doesn't match the chirality.\r\n let wedgeA = 'down';\r\n let wedgeB = 'up';\r\n if (isCw && rs !== 'R' || !isCw && rs !== 'S') {\r\n vertex.value.hydrogenDirection = 'up';\r\n wedgeA = 'up';\r\n wedgeB = 'down';\r\n }\r\n\r\n if (vertex.value.hasHydrogen) {\r\n this.graph.getEdge(vertex.id, neighbours[order[order.length - 1]]).wedge = wedgeA;\r\n }\r\n\r\n // Get the shortest subtree to flip up / down. Ignore lowest priority\r\n // The rules are following:\r\n // 1. Do not draw wedge between two stereocenters\r\n // 2. Heteroatoms\r\n // 3. Draw outside ring\r\n // 4. Shortest subtree\r\n\r\n let wedgeOrder = new Array(neighbours.length - 1);\r\n let showHydrogen = vertex.value.rings.length > 1 && vertex.value.hasHydrogen;\r\n let offset = vertex.value.hasHydrogen ? 1 : 0;\r\n\r\n for (var j = 0; j < order.length - offset; j++) {\r\n wedgeOrder[j] = new Uint32Array(2);\r\n let neighbour = this.graph.vertices[neighbours[order[j]]];\r\n wedgeOrder[j][0] += neighbour.value.isStereoCenter ? 0 : 100000;\r\n // wedgeOrder[j][0] += neighbour.value.rings.length > 0 ? 0 : 10000;\r\n // Only add if in same ring, unlike above\r\n wedgeOrder[j][0] += this.areVerticesInSameRing(neighbour, vertex) ? 0 : 10000;\r\n wedgeOrder[j][0] += neighbour.value.isHeteroAtom() ? 1000 : 0;\r\n wedgeOrder[j][0] -= neighbour.value.subtreeDepth === 0 ? 1000 : 0;\r\n wedgeOrder[j][0] += 1000 - neighbour.value.subtreeDepth;\r\n wedgeOrder[j][1] = neighbours[order[j]];\r\n }\r\n \r\n\r\n wedgeOrder.sort(function (a, b) {\r\n if (a[0] > b[0]) {\r\n return -1;\r\n } else if (a[0] < b[0]) {\r\n return 1;\r\n }\r\n return 0;\r\n });\r\n\r\n // If all neighbours are in a ring, do not draw wedge, the hydrogen will be drawn.\r\n if (!showHydrogen) {\r\n let wedgeId = wedgeOrder[0][1];\r\n\r\n if (vertex.value.hasHydrogen) {\r\n this.graph.getEdge(vertex.id, wedgeId).wedge = wedgeB; \r\n } else {\r\n let wedge = wedgeB; \r\n\r\n for (var j = order.length - 1; j >= 0; j--) {\r\n if (wedge === wedgeA) {\r\n wedge = wedgeB;\r\n } else {\r\n wedge = wedgeA;\r\n }\r\n if (neighbours[order[j]] === wedgeId) {\r\n break;\r\n }\r\n }\r\n\r\n this.graph.getEdge(vertex.id, wedgeId).wedge = wedge;\r\n }\r\n }\r\n\r\n vertex.value.chirality = rs;\r\n }\r\n }\r\n\r\n /**\r\n * \r\n * \r\n * @param {Number} vertexId The id of a vertex.\r\n * @param {(Number|null)} previousVertexId The id of the parent vertex of the vertex.\r\n * @param {Uint8Array} visited An array containing the visited flag for all vertices in the graph.\r\n * @param {Array} priority An array of arrays storing the atomic numbers for each level.\r\n * @param {Number} maxDepth The maximum depth.\r\n * @param {Number} depth The current depth.\r\n */\r\n visitStereochemistry(vertexId, previousVertexId, visited, priority, maxDepth, depth, parentAtomicNumber = 0) {\r\n visited[vertexId] = 1;\r\n let vertex = this.graph.vertices[vertexId];\r\n let atomicNumber = vertex.value.getAtomicNumber();\r\n\r\n if (priority.length <= depth) {\r\n priority.push(Array());\r\n }\r\n\r\n for (var i = 0; i < this.graph.getEdge(vertexId, previousVertexId).weight; i++) {\r\n priority[depth].push(parentAtomicNumber * 1000 + atomicNumber);\r\n }\r\n\r\n let neighbours = this.graph.vertices[vertexId].neighbours;\r\n\r\n for (var i = 0; i < neighbours.length; i++) {\r\n if (visited[neighbours[i]] !== 1 && depth < maxDepth - 1) {\r\n this.visitStereochemistry(neighbours[i], vertexId, visited.slice(), priority, maxDepth, depth + 1, atomicNumber);\r\n }\r\n }\r\n\r\n // Valences are filled with hydrogens and passed to the next level.\r\n if (depth < maxDepth - 1) {\r\n let bonds = 0;\r\n\r\n for (var i = 0; i < neighbours.length; i++) {\r\n bonds += this.graph.getEdge(vertexId, neighbours[i]).weight;\r\n }\r\n\r\n for (var i = 0; i < vertex.value.getMaxBonds() - bonds; i++) {\r\n if (priority.length <= depth + 1) {\r\n priority.push(Array());\r\n }\r\n\r\n priority[depth + 1].push(atomicNumber * 1000 + 1);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Creates pseudo-elements (such as Et, Me, Ac, Bz, ...) at the position of the carbon sets\r\n * the involved atoms not to be displayed.\r\n */\r\n initPseudoElements() {\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n const vertex = this.graph.vertices[i];\r\n const neighbourIds = vertex.neighbours;\r\n let neighbours = Array(neighbourIds.length);\r\n\r\n for (var j = 0; j < neighbourIds.length; j++) {\r\n neighbours[j] = this.graph.vertices[neighbourIds[j]];\r\n }\r\n\r\n // Ignore atoms that have less than 3 neighbours, except if\r\n // the vertex is connected to a ring and has two neighbours\r\n if (vertex.getNeighbourCount() < 3 || vertex.value.rings.length > 0) {\r\n continue;\r\n }\r\n\r\n // TODO: This exceptions should be handled more elegantly (via config file?)\r\n\r\n // Ignore phosphates (especially for triphosphates)\r\n if (vertex.value.element === 'P') {\r\n continue;\r\n }\r\n\r\n // Ignore also guanidine\r\n if (vertex.value.element === 'C' && neighbours.length === 3 &&\r\n neighbours[0].value.element === 'N' && neighbours[1].value.element === 'N' && neighbours[2].value.element === 'N') {\r\n continue;\r\n }\r\n\r\n // Continue if there are less than two heteroatoms\r\n // or if a neighbour has more than 1 neighbour\r\n let heteroAtomCount = 0;\r\n let ctn = 0;\r\n\r\n for (var j = 0; j < neighbours.length; j++) {\r\n let neighbour = neighbours[j];\r\n let neighbouringElement = neighbour.value.element;\r\n let neighbourCount = neighbour.getNeighbourCount();\r\n\r\n if (neighbouringElement !== 'C' && neighbouringElement !== 'H' &&\r\n neighbourCount === 1) {\r\n heteroAtomCount++;\r\n }\r\n\r\n if (neighbourCount > 1) {\r\n ctn++;\r\n }\r\n }\r\n\r\n if (ctn > 1 || heteroAtomCount < 2) {\r\n continue;\r\n }\r\n\r\n // Get the previous atom (the one which is not terminal)\r\n let previous = null;\r\n\r\n for (var j = 0; j < neighbours.length; j++) {\r\n let neighbour = neighbours[j];\r\n\r\n if (neighbour.getNeighbourCount() > 1) {\r\n previous = neighbour;\r\n }\r\n }\r\n\r\n for (var j = 0; j < neighbours.length; j++) {\r\n let neighbour = neighbours[j];\r\n\r\n if (neighbour.getNeighbourCount() > 1) {\r\n continue;\r\n }\r\n\r\n neighbour.value.isDrawn = false;\r\n\r\n let hydrogens = Atom.maxBonds[neighbour.value.element] - neighbour.value.bondCount;\r\n let charge = '';\r\n\r\n if (neighbour.value.bracket) {\r\n hydrogens = neighbour.value.bracket.hcount;\r\n charge = neighbour.value.bracket.charge || 0;\r\n }\r\n\r\n vertex.value.attachPseudoElement(neighbour.value.element, previous ? previous.value.element : null, hydrogens, charge);\r\n }\r\n }\r\n\r\n // The second pass\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n const vertex = this.graph.vertices[i];\r\n const atom = vertex.value;\r\n const element = atom.element;\r\n\r\n if (element === 'C' || element === 'H' || !atom.isDrawn) {\r\n continue;\r\n }\r\n\r\n const neighbourIds = vertex.neighbours;\r\n let neighbours = Array(neighbourIds.length);\r\n\r\n for (var j = 0; j < neighbourIds.length; j++) {\r\n neighbours[j] = this.graph.vertices[neighbourIds[j]];\r\n }\r\n\r\n for (var j = 0; j < neighbours.length; j++) {\r\n let neighbour = neighbours[j].value;\r\n\r\n if (!neighbour.hasAttachedPseudoElements || neighbour.getAttachedPseudoElementsCount() !== 2) {\r\n continue;\r\n }\r\n\r\n const pseudoElements = neighbour.getAttachedPseudoElements();\r\n\r\n if (pseudoElements.hasOwnProperty('0O') && pseudoElements.hasOwnProperty('3C')) {\r\n neighbour.isDrawn = false;\r\n vertex.value.attachPseudoElement('Ac', '', 0);\r\n }\r\n }\r\n }\r\n }\r\n}\r\n\r\nmodule.exports = Drawer;","//@ts-check\r\n\r\n/** \r\n * A class representing an edge. \r\n * \r\n * @property {Number} id The id of this edge.\r\n * @property {Number} sourceId The id of the source vertex.\r\n * @property {Number} targetId The id of the target vertex.\r\n * @property {Number} weight The weight of this edge. That is, the degree of the bond (single bond = 1, double bond = 2, etc).\r\n * @property {String} [bondType='-'] The bond type of this edge.\r\n * @property {Boolean} [isPartOfAromaticRing=false] Whether or not this edge is part of an aromatic ring.\r\n * @property {Boolean} [center=false] Wheter or not the bond is centered. For example, this affects straight double bonds.\r\n * @property {String} [wedge=''] Wedge direction. Either '', 'up' or 'down'\r\n */\r\nclass Edge {\r\n /**\r\n * The constructor for the class Edge.\r\n *\r\n * @param {Number} sourceId A vertex id.\r\n * @param {Number} targetId A vertex id.\r\n * @param {Number} [weight=1] The weight of the edge.\r\n */\r\n constructor(sourceId, targetId, weight = 1) {\r\n this.id = null;\r\n this.sourceId = sourceId;\r\n this.targetId = targetId;\r\n this.weight = weight;\r\n this.bondType = '-';\r\n this.isPartOfAromaticRing = false;\r\n this.center = false;\r\n this.wedge = '';\r\n }\r\n\r\n /**\r\n * Set the bond type of this edge. This also sets the edge weight.\r\n * @param {String} bondType \r\n */\r\n setBondType(bondType) {\r\n this.bondType = bondType;\r\n this.weight = Edge.bonds[bondType];\r\n }\r\n\r\n /**\r\n * An object mapping the bond type to the number of bonds.\r\n *\r\n * @returns {Object} The object containing the map.\r\n */\r\n static get bonds() {\r\n return {\r\n '-': 1,\r\n '/': 1,\r\n '\\\\': 1,\r\n '=': 2,\r\n '#': 3,\r\n '$': 4\r\n }\r\n }\r\n}\r\n\r\nmodule.exports = Edge","//@ts-check\r\nconst MathHelper = require('./MathHelper')\r\nconst Vector2 = require('./Vector2')\r\nconst Vertex = require('./Vertex')\r\nconst Edge = require('./Edge')\r\nconst Ring = require('./Ring')\r\nconst Atom = require('./Atom')\r\n\r\n/** \r\n * A class representing the molecular graph. \r\n * \r\n * @property {Vertex[]} vertices The vertices of the graph.\r\n * @property {Edge[]} edges The edges of this graph.\r\n * @property {Object} vertexIdsToEdgeId A map mapping vertex ids to the edge between the two vertices. The key is defined as vertexAId + '_' + vertexBId.\r\n * @property {Boolean} isometric A boolean indicating whether or not the SMILES associated with this graph is isometric.\r\n */\r\nclass Graph {\r\n /**\r\n * The constructor of the class Graph.\r\n * \r\n * @param {Object} parseTree A SMILES parse tree.\r\n * @param {Boolean} [isomeric=false] A boolean specifying whether or not the SMILES is isomeric.\r\n */\r\n constructor(parseTree, isomeric = false) {\r\n this.vertices = Array();\r\n this.edges = Array();\r\n this.vertexIdsToEdgeId = {};\r\n this.isomeric = isomeric;\r\n\r\n // Used for the bridge detection algorithm\r\n this._time = 0;\r\n this._init(parseTree);\r\n }\r\n\r\n /**\r\n * PRIVATE FUNCTION. Initializing the graph from the parse tree.\r\n *\r\n * @param {Object} node The current node in the parse tree.\r\n * @param {Number} parentVertexId=null The id of the previous vertex.\r\n * @param {Boolean} isBranch=false Whether or not the bond leading to this vertex is a branch bond. Branches are represented by parentheses in smiles (e.g. CC(O)C).\r\n */\r\n _init(node, order = 0, parentVertexId = null, isBranch = false) {\r\n // Create a new vertex object\r\n let atom = new Atom(node.atom.element ? node.atom.element : node.atom, node.bond);\r\n\r\n atom.branchBond = node.branchBond;\r\n atom.ringbonds = node.ringbonds;\r\n atom.bracket = node.atom.element ? node.atom : null;\r\n\r\n let vertex = new Vertex(atom);\r\n let parentVertex = this.vertices[parentVertexId];\r\n\r\n this.addVertex(vertex);\r\n\r\n // Add the id of this node to the parent as child\r\n if (parentVertexId !== null) {\r\n vertex.setParentVertexId(parentVertexId);\r\n vertex.value.addNeighbouringElement(parentVertex.value.element);\r\n parentVertex.addChild(vertex.id);\r\n parentVertex.value.addNeighbouringElement(atom.element);\r\n\r\n // In addition create a spanningTreeChildren property, which later will\r\n // not contain the children added through ringbonds\r\n parentVertex.spanningTreeChildren.push(vertex.id);\r\n\r\n // Add edge between this node and its parent\r\n let edge = new Edge(parentVertexId, vertex.id, 1);\r\n let vertexId = null;\r\n\r\n if (isBranch) {\r\n edge.setBondType(vertex.value.branchBond || '-');\r\n vertexId = vertex.id;\r\n edge.setBondType(vertex.value.branchBond || '-');\r\n vertexId = vertex.id;\r\n } else {\r\n edge.setBondType(parentVertex.value.bondType || '-');\r\n vertexId = parentVertex.id;\r\n }\r\n\r\n let edgeId = this.addEdge(edge);\r\n }\r\n\r\n let offset = node.ringbondCount + 1;\r\n\r\n if (atom.bracket) {\r\n offset += atom.bracket.hcount;\r\n }\r\n\r\n let stereoHydrogens = 0;\r\n if (atom.bracket && atom.bracket.chirality) {\r\n atom.isStereoCenter = true;\r\n stereoHydrogens = atom.bracket.hcount;\r\n for (var i = 0; i < stereoHydrogens; i++) {\r\n this._init({\r\n atom: 'H',\r\n isBracket: 'false',\r\n branches: Array(),\r\n branchCount: 0,\r\n ringbonds: Array(),\r\n ringbondCount: false,\r\n next: null,\r\n hasNext: false,\r\n bond: '-'\r\n }, i, vertex.id, true);\r\n }\r\n }\r\n\r\n for (var i = 0; i < node.branchCount; i++) {\r\n this._init(node.branches[i], i + offset, vertex.id, true);\r\n }\r\n\r\n if (node.hasNext) {\r\n this._init(node.next, node.branchCount + offset, vertex.id);\r\n }\r\n }\r\n\r\n /**\r\n * Clears all the elements in this graph (edges and vertices).\r\n */\r\n clear() {\r\n this.vertices = Array();\r\n this.edges = Array();\r\n this.vertexIdsToEdgeId = {};\r\n }\r\n\r\n /**\r\n * Add a vertex to the graph.\r\n *\r\n * @param {Vertex} vertex A new vertex.\r\n * @returns {Number} The vertex id of the new vertex.\r\n */\r\n addVertex(vertex) {\r\n vertex.id = this.vertices.length;\r\n this.vertices.push(vertex);\r\n\r\n return vertex.id;\r\n }\r\n\r\n /**\r\n * Add an edge to the graph.\r\n *\r\n * @param {Edge} edge A new edge.\r\n * @returns {Number} The edge id of the new edge.\r\n */\r\n addEdge(edge) {\r\n let source = this.vertices[edge.sourceId];\r\n let target = this.vertices[edge.targetId];\r\n\r\n edge.id = this.edges.length;\r\n this.edges.push(edge);\r\n\r\n this.vertexIdsToEdgeId[edge.sourceId + '_' + edge.targetId] = edge.id;\r\n this.vertexIdsToEdgeId[edge.targetId + '_' + edge.sourceId] = edge.id;\r\n edge.isPartOfAromaticRing = source.value.isPartOfAromaticRing && target.value.isPartOfAromaticRing;\r\n\r\n source.value.bondCount += edge.weight;\r\n target.value.bondCount += edge.weight;\r\n\r\n source.edges.push(edge.id);\r\n target.edges.push(edge.id);\r\n\r\n return edge.id;\r\n }\r\n\r\n /**\r\n * Returns the edge between two given vertices.\r\n *\r\n * @param {Number} vertexIdA A vertex id.\r\n * @param {Number} vertexIdB A vertex id.\r\n * @returns {(Edge|null)} The edge or, if no edge can be found, null.\r\n */\r\n getEdge(vertexIdA, vertexIdB) {\r\n let edgeId = this.vertexIdsToEdgeId[vertexIdA + '_' + vertexIdB];\r\n\r\n return edgeId === undefined ? null : this.edges[edgeId];\r\n }\r\n\r\n /**\r\n * Returns the ids of edges connected to a vertex.\r\n *\r\n * @param {Number} vertexId A vertex id.\r\n * @returns {Number[]} An array containing the ids of edges connected to the vertex.\r\n */\r\n getEdges(vertexId) {\r\n let edgeIds = Array();\r\n let vertex = this.vertices[vertexId];\r\n\r\n for (var i = 0; i < vertex.neighbours.length; i++) {\r\n edgeIds.push(this.vertexIdsToEdgeId[vertexId + '_' + vertex.neighbours[i]]);\r\n }\r\n\r\n return edgeIds;\r\n }\r\n\r\n\r\n /**\r\n * Check whether or not two vertices are connected by an edge.\r\n *\r\n * @param {Number} vertexIdA A vertex id.\r\n * @param {Number} vertexIdB A vertex id.\r\n * @returns {Boolean} A boolean indicating whether or not two vertices are connected by an edge.\r\n */\r\n hasEdge(vertexIdA, vertexIdB) {\r\n return this.vertexIdsToEdgeId[vertexIdA + '_' + vertexIdB] !== undefined\r\n }\r\n\r\n /**\r\n * Returns an array containing the vertex ids of this graph.\r\n * \r\n * @returns {Number[]} An array containing all vertex ids of this graph.\r\n */\r\n getVertexList() {\r\n let arr = [this.vertices.length];\r\n\r\n for (var i = 0; i < this.vertices.length; i++) {\r\n arr[i] = this.vertices[i].id;\r\n }\r\n\r\n return arr;\r\n }\r\n\r\n /**\r\n * Returns an array containing source, target arrays of this graphs edges.\r\n * \r\n * @returns {Array[]} An array containing source, target arrays of this graphs edges. Example: [ [ 2, 5 ], [ 6, 9 ] ].\r\n */\r\n getEdgeList() {\r\n let arr = Array(this.edges.length);\r\n\r\n for (var i = 0; i < this.edges.length; i++) {\r\n arr[i] = [this.edges[i].sourceId, this.edges[i].targetId];\r\n }\r\n\r\n return arr;\r\n }\r\n\r\n /**\r\n * Get the adjacency matrix of the graph.\r\n * \r\n * @returns {Array[]} The adjancency matrix of the molecular graph.\r\n */\r\n getAdjacencyMatrix() {\r\n let length = this.vertices.length;\r\n let adjacencyMatrix = Array(length);\r\n\r\n for (var i = 0; i < length; i++) {\r\n adjacencyMatrix[i] = new Array(length);\r\n adjacencyMatrix[i].fill(0);\r\n }\r\n\r\n for (var i = 0; i < this.edges.length; i++) {\r\n let edge = this.edges[i];\r\n\r\n adjacencyMatrix[edge.sourceId][edge.targetId] = 1;\r\n adjacencyMatrix[edge.targetId][edge.sourceId] = 1;\r\n }\r\n\r\n return adjacencyMatrix;\r\n }\r\n\r\n /**\r\n * Get the adjacency matrix of the graph with all bridges removed (thus the components). Thus the remaining vertices are all part of ring systems.\r\n * \r\n * @returns {Array[]} The adjancency matrix of the molecular graph with all bridges removed.\r\n */\r\n getComponentsAdjacencyMatrix() {\r\n let length = this.vertices.length;\r\n let adjacencyMatrix = Array(length);\r\n let bridges = this.getBridges();\r\n\r\n for (var i = 0; i < length; i++) {\r\n adjacencyMatrix[i] = new Array(length);\r\n adjacencyMatrix[i].fill(0);\r\n }\r\n\r\n for (var i = 0; i < this.edges.length; i++) {\r\n let edge = this.edges[i];\r\n\r\n adjacencyMatrix[edge.sourceId][edge.targetId] = 1;\r\n adjacencyMatrix[edge.targetId][edge.sourceId] = 1;\r\n }\r\n\r\n for (var i = 0; i < bridges.length; i++) {\r\n adjacencyMatrix[bridges[i][0]][bridges[i][1]] = 0;\r\n adjacencyMatrix[bridges[i][1]][bridges[i][0]] = 0;\r\n }\r\n\r\n return adjacencyMatrix;\r\n }\r\n\r\n /**\r\n * Get the adjacency matrix of a subgraph.\r\n * \r\n * @param {Number[]} vertexIds An array containing the vertex ids contained within the subgraph.\r\n * @returns {Array[]} The adjancency matrix of the subgraph.\r\n */\r\n getSubgraphAdjacencyMatrix(vertexIds) {\r\n let length = vertexIds.length;\r\n let adjacencyMatrix = Array(length);\r\n\r\n for (var i = 0; i < length; i++) {\r\n adjacencyMatrix[i] = new Array(length);\r\n adjacencyMatrix[i].fill(0);\r\n\r\n for (var j = 0; j < length; j++) {\r\n if (i === j) {\r\n continue;\r\n }\r\n\r\n if (this.hasEdge(vertexIds[i], vertexIds[j])) {\r\n adjacencyMatrix[i][j] = 1;\r\n }\r\n }\r\n }\r\n\r\n return adjacencyMatrix;\r\n }\r\n\r\n /**\r\n * Get the distance matrix of the graph.\r\n * \r\n * @returns {Array[]} The distance matrix of the graph.\r\n */\r\n getDistanceMatrix() {\r\n let length = this.vertices.length;\r\n let adja = this.getAdjacencyMatrix();\r\n let dist = Array(length);\r\n\r\n for (var i = 0; i < length; i++) {\r\n dist[i] = Array(length);\r\n dist[i].fill(Infinity);\r\n }\r\n\r\n for (var i = 0; i < length; i++) {\r\n for (var j = 0; j < length; j++) {\r\n if (adja[i][j] === 1) {\r\n dist[i][j] = 1;\r\n }\r\n }\r\n }\r\n\r\n for (var k = 0; k < length; k++) {\r\n for (var i = 0; i < length; i++) {\r\n for (var j = 0; j < length; j++) {\r\n if (dist[i][j] > dist[i][k] + dist[k][j]) {\r\n dist[i][j] = dist[i][k] + dist[k][j]\r\n }\r\n }\r\n }\r\n }\r\n\r\n return dist;\r\n }\r\n\r\n /**\r\n * Get the distance matrix of a subgraph.\r\n * \r\n * @param {Number[]} vertexIds An array containing the vertex ids contained within the subgraph.\r\n * @returns {Array[]} The distance matrix of the subgraph.\r\n */\r\n getSubgraphDistanceMatrix(vertexIds) {\r\n let length = vertexIds.length;\r\n let adja = this.getSubgraphAdjacencyMatrix(vertexIds);\r\n let dist = Array(length);\r\n\r\n for (var i = 0; i < length; i++) {\r\n dist[i] = Array(length);\r\n dist[i].fill(Infinity);\r\n }\r\n\r\n for (var i = 0; i < length; i++) {\r\n for (var j = 0; j < length; j++) {\r\n if (adja[i][j] === 1) {\r\n dist[i][j] = 1;\r\n }\r\n }\r\n }\r\n\r\n for (var k = 0; k < length; k++) {\r\n for (var i = 0; i < length; i++) {\r\n for (var j = 0; j < length; j++) {\r\n if (dist[i][j] > dist[i][k] + dist[k][j]) {\r\n dist[i][j] = dist[i][k] + dist[k][j]\r\n }\r\n }\r\n }\r\n }\r\n\r\n return dist;\r\n }\r\n\r\n /**\r\n * Get the adjacency list of the graph.\r\n * \r\n * @returns {Array[]} The adjancency list of the graph.\r\n */\r\n getAdjacencyList() {\r\n let length = this.vertices.length;\r\n let adjacencyList = Array(length);\r\n\r\n for (var i = 0; i < length; i++) {\r\n adjacencyList[i] = [];\r\n\r\n for (var j = 0; j < length; j++) {\r\n if (i === j) {\r\n continue;\r\n }\r\n\r\n if (this.hasEdge(this.vertices[i].id, this.vertices[j].id)) {\r\n adjacencyList[i].push(j);\r\n }\r\n }\r\n }\r\n\r\n return adjacencyList;\r\n }\r\n\r\n /**\r\n * Get the adjacency list of a subgraph.\r\n * \r\n * @param {Number[]} vertexIds An array containing the vertex ids contained within the subgraph.\r\n * @returns {Array[]} The adjancency list of the subgraph.\r\n */\r\n getSubgraphAdjacencyList(vertexIds) {\r\n let length = vertexIds.length;\r\n let adjacencyList = Array(length);\r\n\r\n for (var i = 0; i < length; i++) {\r\n adjacencyList[i] = Array();\r\n\r\n for (var j = 0; j < length; j++) {\r\n if (i === j) {\r\n continue;\r\n }\r\n\r\n if (this.hasEdge(vertexIds[i], vertexIds[j])) {\r\n adjacencyList[i].push(j);\r\n }\r\n }\r\n }\r\n\r\n return adjacencyList;\r\n }\r\n\r\n /**\r\n * Returns an array containing the edge ids of bridges. A bridge splits the graph into multiple components when removed.\r\n * \r\n * @returns {Number[]} An array containing the edge ids of the bridges.\r\n */\r\n getBridges() {\r\n let length = this.vertices.length;\r\n let visited = new Array(length);\r\n let disc = new Array(length);\r\n let low = new Array(length);\r\n let parent = new Array(length);\r\n let adj = this.getAdjacencyList();\r\n let outBridges = Array();\r\n\r\n visited.fill(false);\r\n parent.fill(null);\r\n this._time = 0;\r\n\r\n for (var i = 0; i < length; i++) {\r\n if (!visited[i]) {\r\n this._bridgeDfs(i, visited, disc, low, parent, adj, outBridges);\r\n }\r\n }\r\n\r\n return outBridges;\r\n }\r\n\r\n /**\r\n * Traverses the graph in breadth-first order.\r\n * \r\n * @param {Number} startVertexId The id of the starting vertex.\r\n * @param {Function} callback The callback function to be called on every vertex.\r\n */\r\n traverseBF(startVertexId, callback) {\r\n let length = this.vertices.length;\r\n let visited = new Array(length);\r\n\r\n visited.fill(false);\r\n\r\n var queue = [startVertexId];\r\n\r\n while (queue.length > 0) {\r\n // JavaScripts shift() is O(n) ... bad JavaScript, bad!\r\n let u = queue.shift();\r\n let vertex = this.vertices[u];\r\n\r\n callback(vertex);\r\n\r\n for (var i = 0; i < vertex.neighbours.length; i++) {\r\n let v = vertex.neighbours[i]\r\n if (!visited[v]) {\r\n visited[v] = true;\r\n queue.push(v);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Get the depth of a subtree in the direction opposite to the vertex specified as the parent vertex.\r\n *\r\n * @param {Number} vertexId A vertex id.\r\n * @param {Number} parentVertexId The id of a neighbouring vertex.\r\n * @returns {Number} The depth of the sub-tree.\r\n */\r\n getTreeDepth(vertexId, parentVertexId) {\r\n if (vertexId === null || parentVertexId === null) {\r\n return 0;\r\n }\r\n\r\n let neighbours = this.vertices[vertexId].getSpanningTreeNeighbours(parentVertexId);\r\n let max = 0;\r\n\r\n for (var i = 0; i < neighbours.length; i++) {\r\n let childId = neighbours[i];\r\n let d = this.getTreeDepth(childId, vertexId);\r\n\r\n if (d > max) {\r\n max = d;\r\n }\r\n }\r\n\r\n return max + 1;\r\n }\r\n\r\n /**\r\n * Traverse a sub-tree in the graph.\r\n *\r\n * @param {Number} vertexId A vertex id.\r\n * @param {Number} parentVertexId A neighbouring vertex.\r\n * @param {Function} callback The callback function that is called with each visited as an argument.\r\n * @param {Number} [maxDepth=Number.MAX_SAFE_INTEGER] The maximum depth of the recursion.\r\n * @param {Boolean} [ignoreFirst=false] Whether or not to ignore the starting vertex supplied as vertexId in the callback.\r\n * @param {Number} [depth=1] The current depth in the tree.\r\n * @param {Uint8Array} [visited=null] An array holding a flag on whether or not a node has been visited.\r\n */\r\n traverseTree(vertexId, parentVertexId, callback, maxDepth = Number.MAX_SAFE_INTEGER, ignoreFirst = false, depth = 1, visited = null) {\r\n if (visited === null) {\r\n visited = new Uint8Array(this.vertices.length);\r\n }\r\n\r\n if (depth > maxDepth + 1 || visited[vertexId] === 1) {\r\n return;\r\n }\r\n\r\n visited[vertexId] = 1;\r\n\r\n let vertex = this.vertices[vertexId];\r\n let neighbours = vertex.getNeighbours(parentVertexId);\r\n\r\n if (!ignoreFirst || depth > 1) {\r\n callback(vertex);\r\n }\r\n\r\n for (var i = 0; i < neighbours.length; i++) {\r\n this.traverseTree(neighbours[i], vertexId, callback, maxDepth, ignoreFirst, depth + 1, visited);\r\n }\r\n }\r\n\r\n /**\r\n * Positiones the (sub)graph using Kamada and Kawais algorithm for drawing general undirected graphs. https://pdfs.semanticscholar.org/b8d3/bca50ccc573c5cb99f7d201e8acce6618f04.pdf\r\n * \r\n * @param {Number[]} vertexIds An array containing vertexIds to be placed using the force based layout.\r\n * @param {Vector2} center The center of the layout.\r\n * @param {Number} startVertexId A vertex id. Should be the starting vertex - e.g. the first to be positioned and connected to a previously place vertex.\r\n * @param {Ring} ring The bridged ring associated with this force-based layout.\r\n */\r\n kkLayout(vertexIds, center, startVertexId, ring, bondLength) {\r\n let edgeStrength = bondLength;\r\n\r\n // Add vertices that are directly connected to the ring\r\n var i = vertexIds.length;\r\n while (i--) {\r\n let vertex = this.vertices[vertexIds[i]];\r\n var j = vertex.neighbours.length;\r\n }\r\n\r\n let matDist = this.getSubgraphDistanceMatrix(vertexIds);\r\n let length = vertexIds.length;\r\n\r\n // Initialize the positions. Place all vertices on a ring around the center\r\n let radius = MathHelper.polyCircumradius(500, length);\r\n let angle = MathHelper.centralAngle(length);\r\n let a = 0.0;\r\n let arrPositionX = new Float32Array(length);\r\n let arrPositionY = new Float32Array(length);\r\n let arrPositioned = Array(length);\r\n\r\n i = length;\r\n while (i--) {\r\n let vertex = this.vertices[vertexIds[i]];\r\n if (!vertex.positioned) {\r\n arrPositionX[i] = center.x + Math.cos(a) * radius;\r\n arrPositionY[i] = center.y + Math.sin(a) * radius;\r\n } else {\r\n arrPositionX[i] = vertex.position.x;\r\n arrPositionY[i] = vertex.position.y;\r\n }\r\n arrPositioned[i] = vertex.positioned;\r\n a += angle;\r\n }\r\n\r\n // Create the matrix containing the lengths\r\n let matLength = Array(length);\r\n i = length;\r\n while (i--) {\r\n matLength[i] = new Array(length);\r\n var j = length;\r\n while (j--) {\r\n matLength[i][j] = bondLength * matDist[i][j];\r\n }\r\n }\r\n\r\n // Create the matrix containing the spring strenghts\r\n let matStrength = Array(length);\r\n i = length;\r\n while (i--) {\r\n matStrength[i] = Array(length);\r\n var j = length;\r\n while (j--) {\r\n matStrength[i][j] = edgeStrength * Math.pow(matDist[i][j], -2.0);\r\n }\r\n }\r\n\r\n // Create the matrix containing the energies\r\n let matEnergy = Array(length);\r\n let arrEnergySumX = new Float32Array(length);\r\n let arrEnergySumY = new Float32Array(length);\r\n i = length;\r\n while (i--) {\r\n matEnergy[i] = Array(length);\r\n }\r\n\r\n i = length;\r\n let ux, uy, dEx, dEy, vx, vy, denom;\r\n\r\n while (i--) {\r\n ux = arrPositionX[i];\r\n uy = arrPositionY[i];\r\n dEx = 0.0;\r\n dEy = 0.0;\r\n let j = length;\r\n while (j--) {\r\n if (i === j) {\r\n continue;\r\n }\r\n vx = arrPositionX[j];\r\n vy = arrPositionY[j];\r\n denom = 1.0 / Math.sqrt((ux - vx) * (ux - vx) + (uy - vy) * (uy - vy));\r\n matEnergy[i][j] = [\r\n matStrength[i][j] * ((ux - vx) - matLength[i][j] * (ux - vx) * denom),\r\n matStrength[i][j] * ((uy - vy) - matLength[i][j] * (uy - vy) * denom)\r\n ]\r\n matEnergy[j][i] = matEnergy[i][j];\r\n dEx += matEnergy[i][j][0];\r\n dEy += matEnergy[i][j][1];\r\n }\r\n arrEnergySumX[i] = dEx;\r\n arrEnergySumY[i] = dEy;\r\n }\r\n\r\n // Utility functions, maybe inline them later\r\n let energy = function (index) {\r\n return [arrEnergySumX[index] * arrEnergySumX[index] + arrEnergySumY[index] * arrEnergySumY[index], arrEnergySumX[index], arrEnergySumY[index]];\r\n }\r\n\r\n let highestEnergy = function () {\r\n let maxEnergy = 0.0;\r\n let maxEnergyId = 0;\r\n let maxDEX = 0.0;\r\n let maxDEY = 0.0\r\n\r\n i = length;\r\n while (i--) {\r\n let [delta, dEX, dEY] = energy(i);\r\n\r\n if (delta > maxEnergy && arrPositioned[i] === false) {\r\n maxEnergy = delta;\r\n maxEnergyId = i;\r\n maxDEX = dEX;\r\n maxDEY = dEY;\r\n }\r\n }\r\n\r\n return [maxEnergyId, maxEnergy, maxDEX, maxDEY];\r\n }\r\n\r\n let update = function (index, dEX, dEY) {\r\n let dxx = 0.0;\r\n let dyy = 0.0;\r\n let dxy = 0.0;\r\n let ux = arrPositionX[index];\r\n let uy = arrPositionY[index];\r\n let arrL = matLength[index];\r\n let arrK = matStrength[index];\r\n\r\n i = length;\r\n while (i--) {\r\n if (i === index) {\r\n continue;\r\n }\r\n\r\n let vx = arrPositionX[i];\r\n let vy = arrPositionY[i];\r\n let l = arrL[i];\r\n let k = arrK[i];\r\n let m = (ux - vx) * (ux - vx);\r\n let denom = 1.0 / Math.pow(m + (uy - vy) * (uy - vy), 1.5);\r\n \r\n dxx += k * (1 - l * (uy - vy) * (uy - vy) * denom);\r\n dyy += k * (1 - l * m * denom);\r\n dxy += k * (l * (ux - vx) * (uy - vy) * denom);\r\n }\r\n\r\n // Prevent division by zero\r\n if (dxx === 0) {\r\n dxx = 0.1;\r\n }\r\n\r\n if (dyy === 0) {\r\n dyy = 0.1;\r\n }\r\n\r\n if (dxy === 0) {\r\n dxy = 0.1;\r\n }\r\n\r\n let dy = (dEX / dxx + dEY / dxy);\r\n dy /= (dxy / dxx - dyy / dxy); // had to split this onto two lines because the syntax highlighter went crazy.\r\n let dx = -(dxy * dy + dEX) / dxx;\r\n\r\n arrPositionX[index] += dx;\r\n arrPositionY[index] += dy;\r\n\r\n // Update the energies\r\n let arrE = matEnergy[index];\r\n dEX = 0.0;\r\n dEY = 0.0;\r\n\r\n ux = arrPositionX[index];\r\n uy = arrPositionY[index];\r\n\r\n let vx, vy, prevEx, prevEy, denom;\r\n\r\n i = length;\r\n while (i--) {\r\n if (index === i) {\r\n continue;\r\n }\r\n vx = arrPositionX[i];\r\n vy = arrPositionY[i];\r\n // Store old energies\r\n prevEx = arrE[i][0];\r\n prevEy = arrE[i][1];\r\n denom = 1.0 / Math.sqrt((ux - vx) * (ux - vx) + (uy - vy) * (uy - vy));\r\n dx = arrK[i] * ((ux - vx) - arrL[i] * (ux - vx) * denom);\r\n dy = arrK[i] * ((uy - vy) - arrL[i] * (uy - vy) * denom);\r\n\r\n arrE[i] = [dx, dy];\r\n dEX += dx;\r\n dEY += dy;\r\n arrEnergySumX[i] += dx - prevEx;\r\n arrEnergySumY[i] += dy - prevEy;\r\n }\r\n arrEnergySumX[index] = dEX;\r\n arrEnergySumY[index] = dEY;\r\n }\r\n\r\n // Setting parameters\r\n let threshold = 0.1;\r\n let innerThreshold = 0.1;\r\n let maxIteration = 2000;\r\n let maxInnerIteration = 50;\r\n let maxEnergy = 1e9;\r\n\r\n // Setting up variables for the while loops\r\n let maxEnergyId = 0;\r\n let dEX = 0.0;\r\n let dEY = 0.0;\r\n let delta = 0.0;\r\n let iteration = 0;\r\n let innerIteration = 0;\r\n\r\n while (maxEnergy > threshold && maxIteration > iteration) {\r\n iteration++;\r\n [maxEnergyId, maxEnergy, dEX, dEY] = highestEnergy();\r\n delta = maxEnergy;\r\n innerIteration = 0;\r\n while (delta > innerThreshold && maxInnerIteration > innerIteration) {\r\n innerIteration++;\r\n update(maxEnergyId, dEX, dEY);\r\n [delta, dEX, dEY] = energy(maxEnergyId);\r\n }\r\n }\r\n\r\n i = length;\r\n while (i--) {\r\n let index = vertexIds[i];\r\n let vertex = this.vertices[index];\r\n vertex.position.x = arrPositionX[i];\r\n vertex.position.y = arrPositionY[i];\r\n vertex.positioned = true;\r\n vertex.forcePositioned = true;\r\n }\r\n }\r\n\r\n /**\r\n * PRIVATE FUNCTION used by getBridges().\r\n */\r\n _bridgeDfs(u, visited, disc, low, parent, adj, outBridges) {\r\n visited[u] = true;\r\n disc[u] = low[u] = ++this._time;\r\n\r\n for (var i = 0; i < adj[u].length; i++) {\r\n let v = adj[u][i];\r\n\r\n if (!visited[v]) {\r\n parent[v] = u;\r\n\r\n this._bridgeDfs(v, visited, disc, low, parent, adj, outBridges);\r\n\r\n low[u] = Math.min(low[u], low[v]);\r\n\r\n // If low > disc, we have a bridge\r\n if (low[v] > disc[u]) {\r\n outBridges.push([u, v]);\r\n }\r\n } else if (v !== parent[u]) {\r\n low[u] = Math.min(low[u], disc[v]);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Returns the connected components of the graph.\r\n * \r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @returns {Set[]} Connected components as sets.\r\n */\r\n static getConnectedComponents(adjacencyMatrix) {\r\n let length = adjacencyMatrix.length;\r\n let visited = new Array(length);\r\n let components = new Array();\r\n let count = 0;\r\n\r\n visited.fill(false);\r\n\r\n for (var u = 0; u < length; u++) {\r\n if (!visited[u]) {\r\n let component = Array();\r\n visited[u] = true;\r\n component.push(u);\r\n count++;\r\n Graph._ccGetDfs(u, visited, adjacencyMatrix, component);\r\n if (component.length > 1) {\r\n components.push(component);\r\n }\r\n }\r\n }\r\n\r\n return components;\r\n }\r\n\r\n /**\r\n * Returns the number of connected components for the graph. \r\n * \r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @returns {Number} The number of connected components of the supplied graph.\r\n */\r\n static getConnectedComponentCount(adjacencyMatrix) {\r\n let length = adjacencyMatrix.length;\r\n let visited = new Array(length);\r\n let count = 0;\r\n\r\n visited.fill(false);\r\n\r\n for (var u = 0; u < length; u++) {\r\n if (!visited[u]) {\r\n visited[u] = true;\r\n count++;\r\n Graph._ccCountDfs(u, visited, adjacencyMatrix);\r\n }\r\n }\r\n\r\n return count;\r\n }\r\n\r\n /**\r\n * PRIVATE FUNCTION used by getConnectedComponentCount().\r\n */\r\n static _ccCountDfs(u, visited, adjacencyMatrix) {\r\n for (var v = 0; v < adjacencyMatrix[u].length; v++) {\r\n let c = adjacencyMatrix[u][v];\r\n\r\n if (!c || visited[v] || u === v) {\r\n continue;\r\n }\r\n\r\n visited[v] = true;\r\n Graph._ccCountDfs(v, visited, adjacencyMatrix);\r\n }\r\n }\r\n\r\n /**\r\n * PRIVATE FUNCTION used by getConnectedComponents().\r\n */\r\n static _ccGetDfs(u, visited, adjacencyMatrix, component) {\r\n for (var v = 0; v < adjacencyMatrix[u].length; v++) {\r\n let c = adjacencyMatrix[u][v];\r\n\r\n if (!c || visited[v] || u === v) {\r\n continue;\r\n }\r\n\r\n visited[v] = true;\r\n component.push(v);\r\n Graph._ccGetDfs(v, visited, adjacencyMatrix, component);\r\n }\r\n }\r\n}\r\n\r\nmodule.exports = Graph","//@ts-check\r\nconst Vector2 = require('./Vector2')\r\n\r\n/** \r\n * A class representing a line.\r\n * \r\n * @property {Vector2} from The Vector2 defining the start of the line.\r\n * @property {Vector2} to The Vector2 defining the end of the line.\r\n * @property {String} elementFrom The element symbol associated with the start of the line.\r\n * @property {String} elementTo The element symbol associated with the end of the line.\r\n * @property {Boolean} chiralFrom A boolean indicating whether or not the source atom is a chiral center.\r\n * @property {Boolean} chiralTo A boolean indicating whether or tno the target atom is a chiral center.\r\n */\r\nclass Line {\r\n /**\r\n * The constructor for the class Line.\r\n *\r\n * @param {Vector2} [from=new Vector2(0, 0)] A vector marking the beginning of the line.\r\n * @param {Vector2} [to=new Vector2(0, 0)] A vector marking the end of the line.\r\n * @param {string} [elementFrom=null] A one-letter representation of the element associated with the vector marking the beginning of the line.\r\n * @param {string} [elementTo=null] A one-letter representation of the element associated with the vector marking the end of the line.\r\n * @param {Boolean} [chiralFrom=false] Whether or not the from atom is a chiral center.\r\n * @param {Boolean} [chiralTo=false] Whether or not the to atom is a chiral center.\r\n */\r\n constructor(from = new Vector2(0,0), to = new Vector2(0, 0), elementFrom = null, elementTo = null, chiralFrom = false, chiralTo = false) {\r\n this.from = from;\r\n this.to = to;\r\n this.elementFrom = elementFrom;\r\n this.elementTo = elementTo;\r\n this.chiralFrom = chiralFrom;\r\n this.chiralTo = chiralTo;\r\n }\r\n\r\n /**\r\n * Clones this line and returns the clone.\r\n *\r\n * @returns {Line} A clone of this line.\r\n */\r\n clone() {\r\n return new Line(this.from.clone(), this.to.clone(), this.elementFrom, this.elementTo);\r\n }\r\n\r\n /**\r\n * Returns the length of this line.\r\n *\r\n * @returns {Number} The length of this line.\r\n */\r\n getLength() {\r\n return Math.sqrt(Math.pow(this.to.x - this.from.x, 2) + \r\n Math.pow(this.to.y - this.from.y, 2));\r\n }\r\n\r\n\r\n /**\r\n * Returns the angle of the line in relation to the coordinate system (the x-axis).\r\n *\r\n * @returns {Number} The angle in radians.\r\n */\r\n getAngle() {\r\n // Get the angle between the line and the x-axis\r\n let diff = Vector2.subtract(this.getRightVector(), this.getLeftVector());\r\n return diff.angle();\r\n }\r\n\r\n /**\r\n * Returns the right vector (the vector with the larger x value).\r\n *\r\n * @returns {Vector2} The right vector.\r\n */\r\n getRightVector() {\r\n // Return the vector with the larger x value (the right one)\r\n if (this.from.x < this.to.x) {\r\n return this.to;\r\n } else {\r\n return this.from;\r\n }\r\n }\r\n \r\n /**\r\n * Returns the left vector (the vector with the smaller x value).\r\n *\r\n * @returns {Vector2} The left vector.\r\n */\r\n getLeftVector() {\r\n // Return the vector with the smaller x value (the left one)\r\n if (this.from.x < this.to.x) {\r\n return this.from;\r\n } else {\r\n return this.to;\r\n }\r\n }\r\n\r\n /**\r\n * Returns the element associated with the right vector (the vector with the larger x value).\r\n *\r\n * @returns {String} The element associated with the right vector.\r\n */\r\n getRightElement() {\r\n if (this.from.x < this.to.x) {\r\n return this.elementTo;\r\n } else {\r\n return this.elementFrom;\r\n }\r\n }\r\n\r\n /**\r\n * Returns the element associated with the left vector (the vector with the smaller x value).\r\n *\r\n * @returns {String} The element associated with the left vector.\r\n */\r\n getLeftElement() {\r\n if (this.from.x < this.to.x) {\r\n return this.elementFrom;\r\n } else {\r\n return this.elementTo;\r\n }\r\n }\r\n\r\n /**\r\n * Returns whether or not the atom associated with the right vector (the vector with the larger x value) is a chiral center.\r\n *\r\n * @returns {Boolean} Whether or not the atom associated with the right vector is a chiral center.\r\n */\r\n getRightChiral() {\r\n if (this.from.x < this.to.x) {\r\n return this.chiralTo;\r\n } else {\r\n return this.chiralFrom;\r\n }\r\n }\r\n\r\n /**\r\n * Returns whether or not the atom associated with the left vector (the vector with the smaller x value) is a chiral center.\r\n *\r\n * @returns {Boolean} Whether or not the atom associated with the left vector is a chiral center.\r\n */\r\n getLeftChiral() {\r\n if (this.from.x < this.to.x) {\r\n return this.chiralFrom;\r\n } else {\r\n return this.chiralTo;\r\n }\r\n }\r\n\r\n /**\r\n * Set the value of the right vector.\r\n *\r\n * @param {Number} x The x value.\r\n * @param {Number} y The y value.\r\n * @returns {Line} This line.\r\n */\r\n setRightVector(x, y) {\r\n if (this.from.x < this.to.x) {\r\n this.to.x = x;\r\n this.to.y = y;\r\n } else {\r\n this.from.x = x;\r\n this.from.y = y;\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set the value of the left vector.\r\n *\r\n * @param {Number} x The x value.\r\n * @param {Number} y The y value.\r\n * @returns {Line} This line.\r\n */\r\n setLeftVector(x, y) {\r\n if (this.from.x < this.to.x) {\r\n this.from.x = x;\r\n this.from.y = y;\r\n } else {\r\n this.to.x = x;\r\n this.to.y = y;\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Rotates this line to be aligned with the x-axis. The center of rotation is the left vector.\r\n *\r\n * @returns {Line} This line.\r\n */\r\n rotateToXAxis() {\r\n let left = this.getLeftVector();\r\n \r\n this.setRightVector(left.x + this.getLength(), left.y);\r\n \r\n return this;\r\n }\r\n\r\n /**\r\n * Rotate the line by a given value (in radians). The center of rotation is the left vector.\r\n *\r\n * @param {Number} theta The angle (in radians) to rotate the line.\r\n * @returns {Line} This line.\r\n */\r\n rotate(theta) {\r\n let l = this.getLeftVector();\r\n let r = this.getRightVector();\r\n let sinTheta = Math.sin(theta);\r\n let cosTheta = Math.cos(theta);\r\n\r\n let x = cosTheta * (r.x - l.x) - sinTheta * (r.y - l.y) + l.x;\r\n let y = sinTheta * (r.x - l.x) - cosTheta * (r.y - l.y) + l.y;\r\n \r\n this.setRightVector(x, y);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Shortens this line from the \"from\" direction by a given value (in pixels).\r\n *\r\n * @param {Number} by The length in pixels to shorten the vector by.\r\n * @returns {Line} This line.\r\n */\r\n shortenFrom(by) {\r\n let f = Vector2.subtract(this.to, this.from);\r\n \r\n f.normalize();\r\n f.multiplyScalar(by);\r\n \r\n this.from.add(f);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Shortens this line from the \"to\" direction by a given value (in pixels).\r\n *\r\n * @param {Number} by The length in pixels to shorten the vector by.\r\n * @returns {Line} This line.\r\n */\r\n shortenTo(by) {\r\n let f = Vector2.subtract(this.from, this.to);\r\n \r\n f.normalize();\r\n f.multiplyScalar(by);\r\n \r\n this.to.add(f);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Shorten the right side.\r\n *\r\n * @param {Number} by The length in pixels to shorten the vector by.\r\n * @returns {Line} Returns itself.\r\n */\r\n shortenRight(by) {\r\n if (this.from.x < this.to.x) {\r\n this.shortenTo(by);\r\n } else {\r\n this.shortenFrom(by);\r\n }\r\n\r\n return this;\r\n }\r\n \r\n /**\r\n * Shorten the left side.\r\n * \r\n * @param {Number} by The length in pixels to shorten the vector by.\r\n * @returns {Line} Returns itself.\r\n */\r\n shortenLeft(by) {\r\n if (this.from.x < this.to.x) {\r\n this.shortenFrom(by);\r\n } else {\r\n this.shortenTo(by);\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Shortens this line from both directions by a given value (in pixels).\r\n *\r\n * @param {Number} by The length in pixels to shorten the vector by.\r\n * @returns {Line} This line.\r\n */\r\n shorten(by) {\r\n let f = Vector2.subtract(this.from, this.to);\r\n \r\n f.normalize();\r\n f.multiplyScalar(by / 2.0);\r\n \r\n this.to.add(f);\r\n this.from.subtract(f);\r\n\r\n return this;\r\n }\r\n}\r\n\r\nmodule.exports = Line;","/** \r\n * A static class containing helper functions for math-related tasks. \r\n */\r\nclass MathHelper {\r\n /**\r\n * Rounds a value to a given number of decimals.\r\n *\r\n * @static\r\n * @param {Number} value A number.\r\n * @param {Number} decimals The number of decimals.\r\n * @returns {Number} A number rounded to a given number of decimals.\r\n */\r\n static round(value, decimals) {\r\n decimals = decimals ? decimals : 1;\r\n return Number(Math.round(value + 'e' + decimals) + 'e-' + decimals);\r\n }\r\n\r\n /**\r\n * Returns the means of the angles contained in an array. In radians.\r\n *\r\n * @static\r\n * @param {Number[]} arr An array containing angles (in radians).\r\n * @returns {Number} The mean angle in radians.\r\n */\r\n static meanAngle(arr) {\r\n let sin = 0.0;\r\n let cos = 0.0;\r\n \r\n for (var i = 0; i < arr.length; i++) {\r\n sin += Math.sin(arr[i]);\r\n cos += Math.cos(arr[i]);\r\n }\r\n\r\n return Math.atan2(sin / arr.length, cos / arr.length);\r\n }\r\n\r\n /**\r\n * Returns the inner angle of a n-sided regular polygon.\r\n *\r\n * @static\r\n * @param {Number} n Number of sides of a regular polygon.\r\n * @returns {Number} The inner angle of a given regular polygon.\r\n */\r\n static innerAngle(n) {\r\n return MathHelper.toRad((n - 2) * 180 / n);\r\n }\r\n\r\n /**\r\n * Returns the circumradius of a n-sided regular polygon with a given side-length.\r\n *\r\n * @static\r\n * @param {Number} s The side length of the regular polygon.\r\n * @param {Number} n The number of sides.\r\n * @returns {Number} The circumradius of the regular polygon.\r\n */\r\n static polyCircumradius(s, n) {\r\n return s / (2 * Math.sin(Math.PI / n));\r\n }\r\n\r\n /**\r\n * Returns the apothem of a regular n-sided polygon based on its radius.\r\n *\r\n * @static\r\n * @param {Number} r The radius.\r\n * @param {Number} n The number of edges of the regular polygon.\r\n * @returns {Number} The apothem of a n-sided polygon based on its radius.\r\n */\r\n static apothem(r, n) {\r\n return r * Math.cos(Math.PI / n);\r\n }\r\n\r\n static apothemFromSideLength(s, n) {\r\n let r = MathHelper.polyCircumradius(s, n);\r\n \r\n return MathHelper.apothem(r, n);\r\n }\r\n\r\n /**\r\n * The central angle of a n-sided regular polygon. In radians.\r\n *\r\n * @static\r\n * @param {Number} n The number of sides of the regular polygon.\r\n * @returns {Number} The central angle of the n-sided polygon in radians.\r\n */\r\n static centralAngle(n) {\r\n return MathHelper.toRad(360 / n);\r\n }\r\n\r\n /**\r\n * Convertes radians to degrees.\r\n *\r\n * @static\r\n * @param {Number} rad An angle in radians.\r\n * @returns {Number} The angle in degrees.\r\n */\r\n static toDeg(rad) {\r\n return rad * MathHelper.degFactor;\r\n }\r\n\r\n /**\r\n * Converts degrees to radians.\r\n *\r\n * @static\r\n * @param {Number} deg An angle in degrees.\r\n * @returns {Number} The angle in radians.\r\n */\r\n static toRad(deg) {\r\n return deg * MathHelper.radFactor;\r\n }\r\n\r\n /**\r\n * Returns the parity of the permutation (1 or -1)\r\n * @param {(Array|Uint8Array)} arr An array containing the permutation.\r\n * @returns {Number} The parity of the permutation (1 or -1), where 1 means even and -1 means odd.\r\n */\r\n static parityOfPermutation(arr) {\r\n let visited = new Uint8Array(arr.length);\r\n let evenLengthCycleCount = 0;\r\n\r\n let traverseCycle = function(i, cycleLength = 0) {\r\n if (visited[i] === 1) {\r\n return cycleLength;\r\n }\r\n\r\n cycleLength++;\r\n\r\n visited[i] = 1;\r\n return traverseCycle(arr[i], cycleLength);\r\n }\r\n\r\n for (var i = 0; i < arr.length; i++) {\r\n if (visited[i] === 1) {\r\n continue;\r\n }\r\n\r\n let cycleLength = traverseCycle(i);\r\n evenLengthCycleCount += (1 - cycleLength % 2);\r\n }\r\n\r\n return evenLengthCycleCount % 2 ? -1 : 1;\r\n }\r\n\r\n /** The factor to convert degrees to radians. */\r\n static get radFactor() {\r\n return Math.PI / 180.0;\r\n }\r\n\r\n /** The factor to convert radians to degrees. */\r\n static get degFactor() {\r\n return 180.0 / Math.PI;\r\n }\r\n\r\n /** Two times PI. */\r\n static get twoPI() {\r\n return 2.0 * Math.PI;\r\n }\r\n}\r\n\r\nmodule.exports = MathHelper;","// WHEN REPLACING, CHECK FOR:\n// KEEP THIS WHEN REGENERATING THE PARSER !!\n\nmodule.exports = (function () {\n \"use strict\";\n\n /*\n * Generated by PEG.js 0.10.0.\n *\n * http://pegjs.org/\n */\n\n function peg$subclass(child, parent) {\n function ctor() {\n this.constructor = child;\n }\n ctor.prototype = parent.prototype;\n child.prototype = new ctor();\n }\n\n function peg$SyntaxError(message, expected, found, location) {\n this.message = message;\n this.expected = expected;\n this.found = found;\n this.location = location;\n this.name = \"SyntaxError\";\n\n if (typeof Error.captureStackTrace === \"function\") {\n Error.captureStackTrace(this, peg$SyntaxError);\n }\n }\n\n peg$subclass(peg$SyntaxError, Error);\n\n peg$SyntaxError.buildMessage = function (expected, found) {\n var DESCRIBE_EXPECTATION_FNS = {\n literal: function (expectation) {\n return \"\\\"\" + literalEscape(expectation.text) + \"\\\"\";\n },\n\n \"class\": function (expectation) {\n var escapedParts = \"\",\n i;\n\n for (i = 0; i < expectation.parts.length; i++) {\n escapedParts += expectation.parts[i] instanceof Array ?\n classEscape(expectation.parts[i][0]) + \"-\" + classEscape(expectation.parts[i][1]) :\n classEscape(expectation.parts[i]);\n }\n\n return \"[\" + (expectation.inverted ? \"^\" : \"\") + escapedParts + \"]\";\n },\n\n any: function (expectation) {\n return \"any character\";\n },\n\n end: function (expectation) {\n return \"end of input\";\n },\n\n other: function (expectation) {\n return expectation.description;\n }\n };\n\n function hex(ch) {\n return ch.charCodeAt(0).toString(16).toUpperCase();\n }\n\n function literalEscape(s) {\n return s\n .replace(/\\\\/g, '\\\\\\\\')\n .replace(/\"/g, '\\\\\"')\n .replace(/\\0/g, '\\\\0')\n .replace(/\\t/g, '\\\\t')\n .replace(/\\n/g, '\\\\n')\n .replace(/\\r/g, '\\\\r')\n .replace(/[\\x00-\\x0F]/g, function (ch) {\n return '\\\\x0' + hex(ch);\n })\n .replace(/[\\x10-\\x1F\\x7F-\\x9F]/g, function (ch) {\n return '\\\\x' + hex(ch);\n });\n }\n\n function classEscape(s) {\n return s\n .replace(/\\\\/g, '\\\\\\\\')\n .replace(/\\]/g, '\\\\]')\n .replace(/\\^/g, '\\\\^')\n .replace(/-/g, '\\\\-')\n .replace(/\\0/g, '\\\\0')\n .replace(/\\t/g, '\\\\t')\n .replace(/\\n/g, '\\\\n')\n .replace(/\\r/g, '\\\\r')\n .replace(/[\\x00-\\x0F]/g, function (ch) {\n return '\\\\x0' + hex(ch);\n })\n .replace(/[\\x10-\\x1F\\x7F-\\x9F]/g, function (ch) {\n return '\\\\x' + hex(ch);\n });\n }\n\n function describeExpectation(expectation) {\n return DESCRIBE_EXPECTATION_FNS[expectation.type](expectation);\n }\n\n function describeExpected(expected) {\n var descriptions = new Array(expected.length),\n i, j;\n\n for (i = 0; i < expected.length; i++) {\n descriptions[i] = describeExpectation(expected[i]);\n }\n\n descriptions.sort();\n\n if (descriptions.length > 0) {\n for (i = 1, j = 1; i < descriptions.length; i++) {\n if (descriptions[i - 1] !== descriptions[i]) {\n descriptions[j] = descriptions[i];\n j++;\n }\n }\n descriptions.length = j;\n }\n\n switch (descriptions.length) {\n case 1:\n return descriptions[0];\n\n case 2:\n return descriptions[0] + \" or \" + descriptions[1];\n\n default:\n return descriptions.slice(0, -1).join(\", \") +\n \", or \" +\n descriptions[descriptions.length - 1];\n }\n }\n\n function describeFound(found) {\n return found ? \"\\\"\" + literalEscape(found) + \"\\\"\" : \"end of input\";\n }\n\n return \"Expected \" + describeExpected(expected) + \" but \" + describeFound(found) + \" found.\";\n };\n\n function peg$parse(input, options) {\n options = options !== void 0 ? options : {};\n\n // KEEP THIS WHEN REGENERATING THE PARSER !!\n var nOpenParentheses = input.split('(').length - 1;\n var nCloseParentheses = input.split(')').length - 1;\n\n if (nOpenParentheses !== nCloseParentheses) {\n throw peg$buildSimpleError('The number of opening parentheses does not match the number of closing parentheses.', 0);\n }\n // KEEP THIS WHEN REGENERATING THE PARSER !!\n\n var peg$FAILED = {},\n\n peg$startRuleFunctions = {\n chain: peg$parsechain\n },\n peg$startRuleFunction = peg$parsechain,\n\n peg$c0 = function (s) {\n var branches = [];\n var rings = [];\n\n for (var i = 0; i < s[1].length; i++) {\n branches.push(s[1][i]);\n }\n\n\n for (var i = 0; i < s[2].length; i++) {\n var bond = (s[2][i][0]) ? s[2][i][0] : '-';\n rings.push({\n 'bond': bond,\n 'id': s[2][i][1]\n });\n }\n\n for (var i = 0; i < s[3].length; i++) {\n branches.push(s[3][i]);\n }\n\n for (var i = 0; i < s[6].length; i++) {\n branches.push(s[6][i]);\n }\n\n return {\n 'atom': s[0],\n 'isBracket': s[0].element ? true : false,\n 'branches': branches,\n 'branchCount': branches.length,\n 'ringbonds': rings,\n 'ringbondCount': rings.length,\n 'bond': s[4] ? s[4] : '-',\n 'next': s[5],\n 'hasNext': s[5] ? true : false\n }\n\n return s;\n },\n peg$c1 = \"(\",\n peg$c2 = peg$literalExpectation(\"(\", false),\n peg$c3 = \")\",\n peg$c4 = peg$literalExpectation(\")\", false),\n peg$c5 = function (b) {\n var bond = (b[1]) ? b[1] : '-';\n b[2].branchBond = bond;\n return b[2]\n },\n peg$c6 = function (a) {\n return a;\n },\n peg$c7 = /^[\\-=#$:\\/\\\\.]/,\n peg$c8 = peg$classExpectation([\"-\", \"=\", \"#\", \"$\", \":\", \"/\", \"\\\\\", \".\"], false, false),\n peg$c9 = function (b) {\n return b;\n },\n peg$c10 = \"[\",\n peg$c11 = peg$literalExpectation(\"[\", false),\n peg$c12 = \"se\",\n peg$c13 = peg$literalExpectation(\"se\", false),\n peg$c14 = \"as\",\n peg$c15 = peg$literalExpectation(\"as\", false),\n peg$c16 = \"]\",\n peg$c17 = peg$literalExpectation(\"]\", false),\n peg$c18 = function (b) {\n return {\n 'isotope': b[1],\n 'element': b[2],\n 'chirality': b[3],\n 'hcount': b[4],\n 'charge': b[5],\n 'class': b[6]\n }\n },\n peg$c19 = \"B\",\n peg$c20 = peg$literalExpectation(\"B\", false),\n peg$c21 = \"r\",\n peg$c22 = peg$literalExpectation(\"r\", false),\n peg$c23 = \"C\",\n peg$c24 = peg$literalExpectation(\"C\", false),\n peg$c25 = \"l\",\n peg$c26 = peg$literalExpectation(\"l\", false),\n peg$c27 = /^[NOPSFI]/,\n peg$c28 = peg$classExpectation([\"N\", \"O\", \"P\", \"S\", \"F\", \"I\"], false, false),\n peg$c29 = function (o) {\n if (o.length > 1) return o.join('');\n return o;\n },\n peg$c30 = /^[bcnops]/,\n peg$c31 = peg$classExpectation([\"b\", \"c\", \"n\", \"o\", \"p\", \"s\"], false, false),\n peg$c32 = \"*\",\n peg$c33 = peg$literalExpectation(\"*\", false),\n peg$c34 = function (w) {\n return w;\n },\n peg$c35 = /^[A-Z]/,\n peg$c36 = peg$classExpectation([\n [\"A\", \"Z\"]\n ], false, false),\n peg$c37 = /^[a-z]/,\n peg$c38 = peg$classExpectation([\n [\"a\", \"z\"]\n ], false, false),\n peg$c39 = function (e) {\n return e.join('');\n },\n peg$c40 = \"%\",\n peg$c41 = peg$literalExpectation(\"%\", false),\n peg$c42 = /^[1-9]/,\n peg$c43 = peg$classExpectation([\n [\"1\", \"9\"]\n ], false, false),\n peg$c44 = /^[0-9]/,\n peg$c45 = peg$classExpectation([\n [\"0\", \"9\"]\n ], false, false),\n peg$c46 = function (r) {\n if (r.length == 1) return Number(r);\n return Number(r.join('').replace('%', ''));\n },\n peg$c47 = \"@\",\n peg$c48 = peg$literalExpectation(\"@\", false),\n peg$c49 = \"TH\",\n peg$c50 = peg$literalExpectation(\"TH\", false),\n peg$c51 = /^[12]/,\n peg$c52 = peg$classExpectation([\"1\", \"2\"], false, false),\n peg$c53 = \"AL\",\n peg$c54 = peg$literalExpectation(\"AL\", false),\n peg$c55 = \"SP\",\n peg$c56 = peg$literalExpectation(\"SP\", false),\n peg$c57 = /^[1-3]/,\n peg$c58 = peg$classExpectation([\n [\"1\", \"3\"]\n ], false, false),\n peg$c59 = \"TB\",\n peg$c60 = peg$literalExpectation(\"TB\", false),\n peg$c61 = \"OH\",\n peg$c62 = peg$literalExpectation(\"OH\", false),\n peg$c63 = function (c) {\n if (!c[1]) return '@';\n if (c[1] == '@') return '@@';\n\n return c[1].join('').replace(',', '');\n },\n peg$c64 = function (c) {\n return c;\n },\n peg$c65 = \"+\",\n peg$c66 = peg$literalExpectation(\"+\", false),\n peg$c67 = function (c) {\n if (!c[1]) return 1;\n if (c[1] != '+') return Number(c[1].join(''));\n return 2;\n },\n peg$c68 = \"-\",\n peg$c69 = peg$literalExpectation(\"-\", false),\n peg$c70 = function (c) {\n if (!c[1]) return -1;\n if (c[1] != '-') return -Number(c[1].join(''));\n return -2;\n },\n peg$c71 = \"H\",\n peg$c72 = peg$literalExpectation(\"H\", false),\n peg$c73 = function (h) {\n if (h[1]) return Number(h[1]);\n return 1;\n },\n peg$c74 = \":\",\n peg$c75 = peg$literalExpectation(\":\", false),\n peg$c76 = /^[0]/,\n peg$c77 = peg$classExpectation([\"0\"], false, false),\n peg$c78 = function (c) {\n return Number(c[1][0] + c[1][1].join(''));\n },\n peg$c79 = function (i) {\n return Number(i.join(''));\n },\n\n peg$currPos = 0,\n peg$savedPos = 0,\n peg$posDetailsCache = [{\n line: 1,\n column: 1\n }],\n peg$maxFailPos = 0,\n peg$maxFailExpected = [],\n peg$silentFails = 0,\n\n peg$result;\n\n if (\"startRule\" in options) {\n if (!(options.startRule in peg$startRuleFunctions)) {\n throw new Error(\"Can't start parsing from rule \\\"\" + options.startRule + \"\\\".\");\n }\n\n peg$startRuleFunction = peg$startRuleFunctions[options.startRule];\n }\n\n function text() {\n return input.substring(peg$savedPos, peg$currPos);\n }\n\n function location() {\n return peg$computeLocation(peg$savedPos, peg$currPos);\n }\n\n function expected(description, location) {\n location = location !== void 0 ? location : peg$computeLocation(peg$savedPos, peg$currPos)\n\n throw peg$buildStructuredError(\n [peg$otherExpectation(description)],\n input.substring(peg$savedPos, peg$currPos),\n location\n );\n }\n\n function error(message, location) {\n location = location !== void 0 ? location : peg$computeLocation(peg$savedPos, peg$currPos)\n\n throw peg$buildSimpleError(message, location);\n }\n\n function peg$literalExpectation(text, ignoreCase) {\n return {\n type: \"literal\",\n text: text,\n ignoreCase: ignoreCase\n };\n }\n\n function peg$classExpectation(parts, inverted, ignoreCase) {\n return {\n type: \"class\",\n parts: parts,\n inverted: inverted,\n ignoreCase: ignoreCase\n };\n }\n\n function peg$anyExpectation() {\n return {\n type: \"any\"\n };\n }\n\n function peg$endExpectation() {\n return {\n type: \"end\"\n };\n }\n\n function peg$otherExpectation(description) {\n return {\n type: \"other\",\n description: description\n };\n }\n\n function peg$computePosDetails(pos) {\n var details = peg$posDetailsCache[pos],\n p;\n\n if (details) {\n return details;\n } else {\n p = pos - 1;\n while (!peg$posDetailsCache[p]) {\n p--;\n }\n\n details = peg$posDetailsCache[p];\n details = {\n line: details.line,\n column: details.column\n };\n\n while (p < pos) {\n if (input.charCodeAt(p) === 10) {\n details.line++;\n details.column = 1;\n } else {\n details.column++;\n }\n\n p++;\n }\n\n peg$posDetailsCache[pos] = details;\n return details;\n }\n }\n\n function peg$computeLocation(startPos, endPos) {\n var startPosDetails = peg$computePosDetails(startPos),\n endPosDetails = peg$computePosDetails(endPos);\n\n return {\n start: {\n offset: startPos,\n line: startPosDetails.line,\n column: startPosDetails.column\n },\n end: {\n offset: endPos,\n line: endPosDetails.line,\n column: endPosDetails.column\n }\n };\n }\n\n function peg$fail(expected) {\n if (peg$currPos < peg$maxFailPos) {\n return;\n }\n\n if (peg$currPos > peg$maxFailPos) {\n peg$maxFailPos = peg$currPos;\n peg$maxFailExpected = [];\n }\n\n peg$maxFailExpected.push(expected);\n }\n\n function peg$buildSimpleError(message, location) {\n return new peg$SyntaxError(message, null, null, location);\n }\n\n function peg$buildStructuredError(expected, found, location) {\n return new peg$SyntaxError(\n peg$SyntaxError.buildMessage(expected, found),\n expected,\n found,\n location\n );\n }\n\n function peg$parsechain() {\n var s0, s1, s2, s3, s4, s5, s6, s7, s8, s9;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n s2 = peg$parseatom();\n if (s2 !== peg$FAILED) {\n s3 = [];\n s4 = peg$parsebranch();\n while (s4 !== peg$FAILED) {\n s3.push(s4);\n s4 = peg$parsebranch();\n }\n if (s3 !== peg$FAILED) {\n s4 = [];\n s5 = peg$currPos;\n s6 = peg$parsebond();\n if (s6 === peg$FAILED) {\n s6 = null;\n }\n if (s6 !== peg$FAILED) {\n s7 = peg$parsering();\n if (s7 !== peg$FAILED) {\n s6 = [s6, s7];\n s5 = s6;\n } else {\n peg$currPos = s5;\n s5 = peg$FAILED;\n }\n } else {\n peg$currPos = s5;\n s5 = peg$FAILED;\n }\n while (s5 !== peg$FAILED) {\n s4.push(s5);\n s5 = peg$currPos;\n s6 = peg$parsebond();\n if (s6 === peg$FAILED) {\n s6 = null;\n }\n if (s6 !== peg$FAILED) {\n s7 = peg$parsering();\n if (s7 !== peg$FAILED) {\n s6 = [s6, s7];\n s5 = s6;\n } else {\n peg$currPos = s5;\n s5 = peg$FAILED;\n }\n } else {\n peg$currPos = s5;\n s5 = peg$FAILED;\n }\n }\n if (s4 !== peg$FAILED) {\n s5 = [];\n s6 = peg$parsebranch();\n while (s6 !== peg$FAILED) {\n s5.push(s6);\n s6 = peg$parsebranch();\n }\n if (s5 !== peg$FAILED) {\n s6 = peg$parsebond();\n if (s6 === peg$FAILED) {\n s6 = null;\n }\n if (s6 !== peg$FAILED) {\n s7 = peg$parsechain();\n if (s7 === peg$FAILED) {\n s7 = null;\n }\n if (s7 !== peg$FAILED) {\n s8 = [];\n s9 = peg$parsebranch();\n while (s9 !== peg$FAILED) {\n s8.push(s9);\n s9 = peg$parsebranch();\n }\n if (s8 !== peg$FAILED) {\n s2 = [s2, s3, s4, s5, s6, s7, s8];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c0(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsebranch() {\n var s0, s1, s2, s3, s4, s5;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 40) {\n s2 = peg$c1;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c2);\n }\n }\n if (s2 !== peg$FAILED) {\n s3 = peg$parsebond();\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s4 = peg$parsechain();\n if (s4 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 41) {\n s5 = peg$c3;\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c4);\n }\n }\n if (s5 !== peg$FAILED) {\n s2 = [s2, s3, s4, s5];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c5(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parseatom() {\n var s0, s1;\n\n s0 = peg$currPos;\n s1 = peg$parseorganicsymbol();\n if (s1 === peg$FAILED) {\n s1 = peg$parsearomaticsymbol();\n if (s1 === peg$FAILED) {\n s1 = peg$parsebracketatom();\n if (s1 === peg$FAILED) {\n s1 = peg$parsewildcard();\n }\n }\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c6(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsebond() {\n var s0, s1;\n\n s0 = peg$currPos;\n if (peg$c7.test(input.charAt(peg$currPos))) {\n s1 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s1 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c8);\n }\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c9(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsebracketatom() {\n var s0, s1, s2, s3, s4, s5, s6, s7, s8, s9;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 91) {\n s2 = peg$c10;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c11);\n }\n }\n if (s2 !== peg$FAILED) {\n s3 = peg$parseisotope();\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n if (input.substr(peg$currPos, 2) === peg$c12) {\n s4 = peg$c12;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c13);\n }\n }\n if (s4 === peg$FAILED) {\n if (input.substr(peg$currPos, 2) === peg$c14) {\n s4 = peg$c14;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c15);\n }\n }\n if (s4 === peg$FAILED) {\n s4 = peg$parsearomaticsymbol();\n if (s4 === peg$FAILED) {\n s4 = peg$parseelementsymbol();\n if (s4 === peg$FAILED) {\n s4 = peg$parsewildcard();\n }\n }\n }\n }\n if (s4 !== peg$FAILED) {\n s5 = peg$parsechiral();\n if (s5 === peg$FAILED) {\n s5 = null;\n }\n if (s5 !== peg$FAILED) {\n s6 = peg$parsehcount();\n if (s6 === peg$FAILED) {\n s6 = null;\n }\n if (s6 !== peg$FAILED) {\n s7 = peg$parsecharge();\n if (s7 === peg$FAILED) {\n s7 = null;\n }\n if (s7 !== peg$FAILED) {\n s8 = peg$parseclass();\n if (s8 === peg$FAILED) {\n s8 = null;\n }\n if (s8 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 93) {\n s9 = peg$c16;\n peg$currPos++;\n } else {\n s9 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c17);\n }\n }\n if (s9 !== peg$FAILED) {\n s2 = [s2, s3, s4, s5, s6, s7, s8, s9];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c18(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parseorganicsymbol() {\n var s0, s1, s2, s3;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 66) {\n s2 = peg$c19;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c20);\n }\n }\n if (s2 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 114) {\n s3 = peg$c21;\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c22);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 === peg$FAILED) {\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 67) {\n s2 = peg$c23;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c24);\n }\n }\n if (s2 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 108) {\n s3 = peg$c25;\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c26);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 === peg$FAILED) {\n if (peg$c27.test(input.charAt(peg$currPos))) {\n s1 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s1 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c28);\n }\n }\n }\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c29(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsearomaticsymbol() {\n var s0, s1;\n\n s0 = peg$currPos;\n if (peg$c30.test(input.charAt(peg$currPos))) {\n s1 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s1 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c31);\n }\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c6(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsewildcard() {\n var s0, s1;\n\n s0 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 42) {\n s1 = peg$c32;\n peg$currPos++;\n } else {\n s1 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c33);\n }\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c34(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parseelementsymbol() {\n var s0, s1, s2, s3;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (peg$c35.test(input.charAt(peg$currPos))) {\n s2 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c36);\n }\n }\n if (s2 !== peg$FAILED) {\n if (peg$c37.test(input.charAt(peg$currPos))) {\n s3 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c38);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c39(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsering() {\n var s0, s1, s2, s3, s4;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 37) {\n s2 = peg$c40;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c41);\n }\n }\n if (s2 !== peg$FAILED) {\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s3 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s3 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s4 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s4 !== peg$FAILED) {\n s2 = [s2, s3, s4];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 === peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s1 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s1 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c46(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsechiral() {\n var s0, s1, s2, s3, s4, s5, s6;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 64) {\n s2 = peg$c47;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c48);\n }\n }\n if (s2 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 64) {\n s3 = peg$c47;\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c48);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (input.substr(peg$currPos, 2) === peg$c49) {\n s4 = peg$c49;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c50);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c51.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c52);\n }\n }\n if (s5 !== peg$FAILED) {\n s4 = [s4, s5];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (input.substr(peg$currPos, 2) === peg$c53) {\n s4 = peg$c53;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c54);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c51.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c52);\n }\n }\n if (s5 !== peg$FAILED) {\n s4 = [s4, s5];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (input.substr(peg$currPos, 2) === peg$c55) {\n s4 = peg$c55;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c56);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c57.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c58);\n }\n }\n if (s5 !== peg$FAILED) {\n s4 = [s4, s5];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (input.substr(peg$currPos, 2) === peg$c59) {\n s4 = peg$c59;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c60);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s5 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s6 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s6 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s6 === peg$FAILED) {\n s6 = null;\n }\n if (s6 !== peg$FAILED) {\n s4 = [s4, s5, s6];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (input.substr(peg$currPos, 2) === peg$c61) {\n s4 = peg$c61;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c62);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s5 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s6 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s6 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s6 === peg$FAILED) {\n s6 = null;\n }\n if (s6 !== peg$FAILED) {\n s4 = [s4, s5, s6];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n }\n }\n }\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c63(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsecharge() {\n var s0, s1;\n\n s0 = peg$currPos;\n s1 = peg$parseposcharge();\n if (s1 === peg$FAILED) {\n s1 = peg$parsenegcharge();\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c64(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parseposcharge() {\n var s0, s1, s2, s3, s4, s5;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 43) {\n s2 = peg$c65;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c66);\n }\n }\n if (s2 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 43) {\n s3 = peg$c65;\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c66);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s4 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s5 === peg$FAILED) {\n s5 = null;\n }\n if (s5 !== peg$FAILED) {\n s4 = [s4, s5];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c67(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsenegcharge() {\n var s0, s1, s2, s3, s4, s5;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 45) {\n s2 = peg$c68;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c69);\n }\n }\n if (s2 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 45) {\n s3 = peg$c68;\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c69);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s4 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s5 === peg$FAILED) {\n s5 = null;\n }\n if (s5 !== peg$FAILED) {\n s4 = [s4, s5];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c70(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsehcount() {\n var s0, s1, s2, s3;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 72) {\n s2 = peg$c71;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c72);\n }\n }\n if (s2 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s3 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c73(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parseclass() {\n var s0, s1, s2, s3, s4, s5, s6;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 58) {\n s2 = peg$c74;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c75);\n }\n }\n if (s2 !== peg$FAILED) {\n s3 = peg$currPos;\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s4 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s4 !== peg$FAILED) {\n s5 = [];\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s6 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s6 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n while (s6 !== peg$FAILED) {\n s5.push(s6);\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s6 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s6 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n }\n if (s5 !== peg$FAILED) {\n s4 = [s4, s5];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n if (s3 === peg$FAILED) {\n if (peg$c76.test(input.charAt(peg$currPos))) {\n s3 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c77);\n }\n }\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c78(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parseisotope() {\n var s0, s1, s2, s3, s4;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s2 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s2 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s3 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s4 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s4 === peg$FAILED) {\n s4 = null;\n }\n if (s4 !== peg$FAILED) {\n s2 = [s2, s3, s4];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c79(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n peg$result = peg$startRuleFunction();\n\n if (peg$result !== peg$FAILED && peg$currPos === input.length) {\n return peg$result;\n } else {\n if (peg$result !== peg$FAILED && peg$currPos < input.length) {\n peg$fail(peg$endExpectation());\n }\n\n throw peg$buildStructuredError(\n peg$maxFailExpected,\n peg$maxFailPos < input.length ? input.charAt(peg$maxFailPos) : null,\n peg$maxFailPos < input.length ?\n peg$computeLocation(peg$maxFailPos, peg$maxFailPos + 1) :\n peg$computeLocation(peg$maxFailPos, peg$maxFailPos)\n );\n }\n }\n\n return {\n SyntaxError: peg$SyntaxError,\n parse: peg$parse\n };\n})();","//@ts-check\r\nconst ArrayHelper = require('./ArrayHelper')\r\nconst Vector2 = require('./Vector2')\r\nconst Vertex = require('./Vertex')\r\nconst RingConnection = require('./RingConnection')\r\n\r\n/** \r\n * A class representing a ring.\r\n * \r\n * @property {Number} id The id of this ring.\r\n * @property {Number[]} members An array containing the vertex ids of the ring members.\r\n * @property {Number[]} edges An array containing the edge ids of the edges between the ring members.\r\n * @property {Number[]} insiders An array containing the vertex ids of the vertices contained within the ring if it is a bridged ring.\r\n * @property {Number[]} neighbours An array containing the ids of neighbouring rings.\r\n * @property {Boolean} positioned A boolean indicating whether or not this ring has been positioned.\r\n * @property {Vector2} center The center of this ring.\r\n * @property {Ring[]} rings The rings contained within this ring if this ring is bridged.\r\n * @property {Boolean} isBridged A boolean whether or not this ring is bridged.\r\n * @property {Boolean} isPartOfBridged A boolean whether or not this ring is part of a bridge ring.\r\n * @property {Boolean} isSpiro A boolean whether or not this ring is part of a spiro.\r\n * @property {Boolean} isFused A boolean whether or not this ring is part of a fused ring.\r\n * @property {Number} centralAngle The central angle of this ring.\r\n * @property {Boolean} canFlip A boolean indicating whether or not this ring allows flipping of attached vertices to the inside of the ring.\r\n */\r\nclass Ring {\r\n /**\r\n * The constructor for the class Ring.\r\n *\r\n * @param {Number[]} members An array containing the vertex ids of the members of the ring to be created.\r\n */\r\n constructor(members) {\r\n this.id = null;\r\n this.members = members;\r\n this.edges = [];\r\n this.insiders = [];\r\n this.neighbours = [];\r\n this.positioned = false;\r\n this.center = new Vector2(0, 0);\r\n this.rings = [];\r\n this.isBridged = false;\r\n this.isPartOfBridged = false;\r\n this.isSpiro = false;\r\n this.isFused = false;\r\n this.centralAngle = 0.0;\r\n this.canFlip = true;\r\n }\r\n \r\n /**\r\n * Clones this ring and returns the clone.\r\n *\r\n * @returns {Ring} A clone of this ring.\r\n */\r\n clone() {\r\n let clone = new Ring(this.members);\r\n\r\n clone.id = this.id;\r\n clone.insiders = ArrayHelper.clone(this.insiders);\r\n clone.neighbours = ArrayHelper.clone(this.neighbours);\r\n clone.positioned = this.positioned;\r\n clone.center = this.center.clone();\r\n clone.rings = ArrayHelper.clone(this.rings);\r\n clone.isBridged = this.isBridged;\r\n clone.isPartOfBridged = this.isPartOfBridged;\r\n clone.isSpiro = this.isSpiro;\r\n clone.isFused = this.isFused;\r\n clone.centralAngle = this.centralAngle;\r\n clone.canFlip = this.canFlip;\r\n\r\n return clone;\r\n }\r\n\r\n /**\r\n * Returns the size (number of members) of this ring.\r\n *\r\n * @returns {Number} The size (number of members) of this ring.\r\n */\r\n getSize() {\r\n return this.members.length;\r\n }\r\n\r\n /**\r\n * Gets the polygon representation (an array of the ring-members positional vectors) of this ring.\r\n *\r\n * @param {Vertex[]} vertices An array of vertices representing the current molecule.\r\n * @returns {Vector2[]} An array of the positional vectors of the ring members.\r\n */\r\n getPolygon(vertices) {\r\n let polygon = [];\r\n\r\n for (let i = 0; i < this.members.length; i++) {\r\n polygon.push(vertices[this.members[i]].position);\r\n }\r\n\r\n return polygon;\r\n }\r\n\r\n /**\r\n * Returns the angle of this ring in relation to the coordinate system.\r\n *\r\n * @returns {Number} The angle in radians.\r\n */\r\n getAngle() {\r\n return Math.PI - this.centralAngle;\r\n }\r\n\r\n /**\r\n * Loops over the members of this ring from a given start position in a direction opposite to the vertex id passed as the previousId.\r\n *\r\n * @param {Vertex[]} vertices The vertices associated with the current molecule.\r\n * @param {Function} callback A callback with the current vertex id as a parameter.\r\n * @param {Number} startVertexId The vertex id of the start vertex.\r\n * @param {Number} previousVertexId The vertex id of the previous vertex (the loop calling the callback function will run in the opposite direction of this vertex).\r\n */\r\n eachMember(vertices, callback, startVertexId, previousVertexId) {\r\n startVertexId = startVertexId || startVertexId === 0 ? startVertexId : this.members[0];\r\n let current = startVertexId;\r\n let max = 0;\r\n\r\n while (current != null && max < 100) {\r\n let prev = current;\r\n \r\n callback(prev);\r\n current = vertices[current].getNextInRing(vertices, this.id, previousVertexId);\r\n previousVertexId = prev;\r\n \r\n // Stop while loop when arriving back at the start vertex\r\n if (current == startVertexId) {\r\n current = null;\r\n }\r\n\r\n max++;\r\n }\r\n }\r\n\r\n /**\r\n * Returns an array containing the neighbouring rings of this ring ordered by ring size.\r\n *\r\n * @param {RingConnection[]} ringConnections An array of ring connections associated with the current molecule.\r\n * @returns {Object[]} An array of neighbouring rings sorted by ring size. Example: { n: 5, neighbour: 1 }.\r\n */\r\n getOrderedNeighbours(ringConnections) {\r\n let orderedNeighbours = Array(this.neighbours.length);\r\n \r\n for (let i = 0; i < this.neighbours.length; i++) {\r\n let vertices = RingConnection.getVertices(ringConnections, this.id, this.neighbours[i]);\r\n \r\n orderedNeighbours[i] = {\r\n n: vertices.length,\r\n neighbour: this.neighbours[i]\r\n };\r\n }\r\n\r\n orderedNeighbours.sort(function (a, b) {\r\n // Sort highest to lowest\r\n return b.n - a.n;\r\n });\r\n\r\n return orderedNeighbours;\r\n }\r\n\r\n /**\r\n * Check whether this ring is an implicitly defined benzene-like (e.g. C1=CC=CC=C1) with 6 members and 3 double bonds.\r\n *\r\n * @param {Vertex[]} vertices An array of vertices associated with the current molecule.\r\n * @returns {Boolean} A boolean indicating whether or not this ring is an implicitly defined benzene-like.\r\n */\r\n isBenzeneLike(vertices) {\r\n let db = this.getDoubleBondCount(vertices);\r\n let length = this.members.length;\r\n\r\n return db === 3 && length === 6 ||\r\n db === 2 && length === 5 ;\r\n }\r\n\r\n /**\r\n * Get the number of double bonds inside this ring.\r\n *\r\n * @param {Vertex[]} vertices An array of vertices associated with the current molecule.\r\n * @returns {Number} The number of double bonds inside this ring.\r\n */\r\n getDoubleBondCount(vertices) {\r\n let doubleBondCount = 0;\r\n\r\n for (let i = 0; i < this.members.length; i++) {\r\n let atom = vertices[this.members[i]].value;\r\n\r\n if (atom.bondType === '=' || atom.branchBond === '=') {\r\n doubleBondCount++;\r\n }\r\n }\r\n\r\n return doubleBondCount;\r\n }\r\n\r\n /**\r\n * Checks whether or not this ring contains a member with a given vertex id.\r\n *\r\n * @param {Number} vertexId A vertex id.\r\n * @returns {Boolean} A boolean indicating whether or not this ring contains a member with the given vertex id.\r\n */\r\n contains(vertexId) {\r\n for (let i = 0; i < this.members.length; i++) {\r\n if (this.members[i] == vertexId) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n}\r\n\r\nmodule.exports = Ring;","//@ts-check\r\nconst Vertex = require('./Vertex')\r\nconst Ring = require('./Ring')\r\n\r\n/** \r\n * A class representing a ring connection.\r\n * \r\n * @property {Number} id The id of this ring connection.\r\n * @property {Number} firstRingId A ring id.\r\n * @property {Number} secondRingId A ring id.\r\n * @property {Set} vertices A set containing the vertex ids participating in the ring connection.\r\n */\r\nclass RingConnection {\r\n /**\r\n * The constructor for the class RingConnection.\r\n *\r\n * @param {Ring} firstRing A ring.\r\n * @param {Ring} secondRing A ring.\r\n */\r\n constructor(firstRing, secondRing) {\r\n this.id = null;\r\n this.firstRingId = firstRing.id;\r\n this.secondRingId = secondRing.id;\r\n this.vertices = new Set();\r\n\r\n for (var m = 0; m < firstRing.members.length; m++) {\r\n let c = firstRing.members[m];\r\n\r\n for (let n = 0; n < secondRing.members.length; n++) {\r\n let d = secondRing.members[n];\r\n\r\n if (c === d) {\r\n this.addVertex(c);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Adding a vertex to the ring connection.\r\n *\r\n * @param {Number} vertexId A vertex id.\r\n */\r\n addVertex(vertexId) {\r\n this.vertices.add(vertexId);\r\n }\r\n\r\n /**\r\n * Update the ring id of this ring connection that is not the ring id supplied as the second argument.\r\n *\r\n * @param {Number} ringId A ring id. The new ring id to be set.\r\n * @param {Number} otherRingId A ring id. The id that is NOT to be updated.\r\n */\r\n updateOther(ringId, otherRingId) {\r\n if (this.firstRingId === otherRingId) {\r\n this.secondRingId = ringId;\r\n } else {\r\n this.firstRingId = ringId;\r\n }\r\n }\r\n\r\n /**\r\n * Returns a boolean indicating whether or not a ring with a given id is participating in this ring connection.\r\n * \r\n * @param {Number} ringId A ring id.\r\n * @returns {Boolean} A boolean indicating whether or not a ring with a given id participates in this ring connection.\r\n */\r\n containsRing(ringId) {\r\n return this.firstRingId === ringId || this.secondRingId === ringId;\r\n }\r\n\r\n /**\r\n * Checks whether or not this ring connection is a bridge in a bridged ring.\r\n *\r\n * @param {Vertex[]} vertices The array of vertices associated with the current molecule.\r\n * @returns {Boolean} A boolean indicating whether or not this ring connection is a bridge.\r\n */\r\n isBridge(vertices) {\r\n if (this.vertices.size > 2) {\r\n return true;\r\n }\r\n\r\n for (let vertexId of this.vertices) {\r\n if(vertices[vertexId].value.rings.length > 2) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Checks whether or not two rings are connected by a bridged bond.\r\n *\r\n * @static\r\n * @param {RingConnection[]} ringConnections An array of ring connections containing the ring connections associated with the current molecule.\r\n * @param {Vertex[]} vertices An array of vertices containing the vertices associated with the current molecule.\r\n * @param {Number} firstRingId A ring id.\r\n * @param {Number} secondRingId A ring id.\r\n * @returns {Boolean} A boolean indicating whether or not two rings ar connected by a bridged bond.\r\n */\r\n static isBridge(ringConnections, vertices, firstRingId, secondRingId) {\r\n let ringConnection = null;\r\n \r\n for (let i = 0; i < ringConnections.length; i++) {\r\n ringConnection = ringConnections[i];\r\n\r\n if (ringConnection.firstRingId === firstRingId && ringConnection.secondRingId === secondRingId ||\r\n ringConnection.firstRingId === secondRingId && ringConnection.secondRingId === firstRingId) {\r\n return ringConnection.isBridge(vertices);\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Retruns the neighbouring rings of a given ring.\r\n *\r\n * @static\r\n * @param {RingConnection[]} ringConnections An array of ring connections containing ring connections associated with the current molecule.\r\n * @param {Number} ringId A ring id.\r\n * @returns {Number[]} An array of ring ids of neighbouring rings.\r\n */\r\n static getNeighbours(ringConnections, ringId) {\r\n let neighbours = [];\r\n\r\n for (let i = 0; i < ringConnections.length; i++) {\r\n let ringConnection = ringConnections[i];\r\n \r\n if (ringConnection.firstRingId === ringId) {\r\n neighbours.push(ringConnection.secondRingId);\r\n } else if (ringConnection.secondRingId === ringId) {\r\n neighbours.push(ringConnection.firstRingId);\r\n }\r\n }\r\n\r\n return neighbours;\r\n }\r\n\r\n /**\r\n * Returns an array of vertex ids associated with a given ring connection.\r\n *\r\n * @static\r\n * @param {RingConnection[]} ringConnections An array of ring connections containing ring connections associated with the current molecule.\r\n * @param {Number} firstRingId A ring id.\r\n * @param {Number} secondRingId A ring id.\r\n * @returns {Number[]} An array of vertex ids associated with the ring connection.\r\n */\r\n static getVertices(ringConnections, firstRingId, secondRingId) {\r\n for (let i = 0; i < ringConnections.length; i++) {\r\n let ringConnection = ringConnections[i];\r\n if (ringConnection.firstRingId === firstRingId && ringConnection.secondRingId === secondRingId ||\r\n ringConnection.firstRingId === secondRingId && ringConnection.secondRingId === firstRingId) {\r\n return [...ringConnection.vertices];\r\n }\r\n }\r\n }\r\n}\r\n\r\nmodule.exports = RingConnection","//@ts-check\r\nconst Graph = require('./Graph')\r\n\r\n/** A class encapsulating the functionality to find the smallest set of smallest rings in a graph. */\r\nclass SSSR {\r\n /**\r\n * Returns an array containing arrays, each representing a ring from the smallest set of smallest rings in the graph.\r\n * \r\n * @param {Graph} graph A Graph object.\r\n * @returns {Array[]} An array containing arrays, each representing a ring from the smallest set of smallest rings in the group.\r\n */\r\n static getRings(graph) {\r\n let adjacencyMatrix = graph.getComponentsAdjacencyMatrix();\r\n if (adjacencyMatrix.length === 0) {\r\n return null;\r\n }\r\n\r\n let connectedComponents = Graph.getConnectedComponents(adjacencyMatrix);\r\n let rings = Array();\r\n\r\n for (var i = 0; i < connectedComponents.length; i++) {\r\n let connectedComponent = connectedComponents[i];\r\n let ccAdjacencyMatrix = graph.getSubgraphAdjacencyMatrix([...connectedComponent]);\r\n\r\n let arrBondCount = new Uint16Array(ccAdjacencyMatrix.length);\r\n let arrRingCount = new Uint16Array(ccAdjacencyMatrix.length);\r\n\r\n for (var j = 0; j < ccAdjacencyMatrix.length; j++) {\r\n arrRingCount[j] = 0;\r\n arrBondCount[j] = 0;\r\n\r\n for (var k = 0; k < ccAdjacencyMatrix[j].length; k++) {\r\n arrBondCount[j] += ccAdjacencyMatrix[j][k];\r\n }\r\n }\r\n\r\n // Get the edge number and the theoretical number of rings in SSSR\r\n let nEdges = 0;\r\n\r\n for (var j = 0; j < ccAdjacencyMatrix.length; j++) {\r\n for (var k = j + 1; k < ccAdjacencyMatrix.length; k++) {\r\n nEdges += ccAdjacencyMatrix[j][k];\r\n }\r\n }\r\n\r\n let nSssr = nEdges - ccAdjacencyMatrix.length + 1;\r\n\r\n // console.log(nEdges, ccAdjacencyMatrix.length, nSssr);\r\n // console.log(SSSR.getEdgeList(ccAdjacencyMatrix));\r\n // console.log(ccAdjacencyMatrix);\r\n\r\n // If all vertices have 3 incident edges, calculate with different formula (see Euler)\r\n let allThree = true;\r\n for (var j = 0; j < arrBondCount.length; j++) {\r\n if (arrBondCount[j] !== 3) {\r\n allThree = false;\r\n }\r\n }\r\n\r\n if (allThree) {\r\n nSssr = 2.0 + nEdges - ccAdjacencyMatrix.length;\r\n }\r\n\r\n // All vertices are part of one ring if theres only one ring.\r\n if (nSssr === 1) {\r\n rings.push([...connectedComponent]);\r\n continue;\r\n }\r\n\r\n let { d, pe, pe_prime } = SSSR.getPathIncludedDistanceMatrices(ccAdjacencyMatrix);\r\n let c = SSSR.getRingCandidates(d, pe, pe_prime);\r\n let sssr = SSSR.getSSSR(c, d, ccAdjacencyMatrix, pe, pe_prime, arrBondCount, arrRingCount, nSssr);\r\n\r\n for (var j = 0; j < sssr.length; j++) {\r\n let ring = Array(sssr[j].size);\r\n let index = 0;\r\n\r\n for (let val of sssr[j]) {\r\n // Get the original id of the vertex back\r\n ring[index++] = connectedComponent[val];\r\n }\r\n\r\n rings.push(ring);\r\n }\r\n }\r\n\r\n return rings;\r\n }\r\n\r\n /**\r\n * Creates a printable string from a matrix (2D array).\r\n * \r\n * @param {Array[]} matrix A 2D array.\r\n * @returns {String} A string representing the matrix.\r\n */\r\n static matrixToString(matrix) {\r\n let str = '';\r\n\r\n for (var i = 0; i < matrix.length; i++) {\r\n for (var j = 0; j < matrix[i].length; j++) {\r\n str += matrix[i][j] + ' ';\r\n }\r\n\r\n str += '\\n';\r\n }\r\n\r\n return str;\r\n }\r\n\r\n /**\r\n * Returnes the two path-included distance matrices used to find the sssr.\r\n * \r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @returns {Object} The path-included distance matrices. { p1, p2 }\r\n */\r\n static getPathIncludedDistanceMatrices(adjacencyMatrix) {\r\n let length = adjacencyMatrix.length;\r\n let d = Array(length);\r\n let pe = Array(length);\r\n let pe_prime = Array(length);\r\n var l = 0;\r\n var m = 0;\r\n var n = 0;\r\n\r\n var i = length;\r\n while (i--) {\r\n d[i] = Array(length);\r\n pe[i] = Array(length);\r\n pe_prime[i] = Array(length);\r\n\r\n var j = length;\r\n while (j--) {\r\n d[i][j] = (i === j || adjacencyMatrix[i][j] === 1) ? adjacencyMatrix[i][j] : Number.POSITIVE_INFINITY;\r\n\r\n if (d[i][j] === 1) {\r\n pe[i][j] = [[[i, j]]];\r\n } else {\r\n pe[i][j] = Array();\r\n }\r\n\r\n pe_prime[i][j] = Array();\r\n }\r\n }\r\n\r\n var k = length;\r\n var j;\r\n while (k--) {\r\n i = length;\r\n while (i--) {\r\n j = length;\r\n while (j--) {\r\n const previousPathLength = d[i][j];\r\n const newPathLength = d[i][k] + d[k][j];\r\n\r\n if (previousPathLength > newPathLength) {\r\n var l, m, n;\r\n if (previousPathLength === newPathLength + 1) {\r\n pe_prime[i][j] = [pe[i][j].length];\r\n l = pe[i][j].length\r\n while (l--) {\r\n pe_prime[i][j][l] = [pe[i][j][l].length];\r\n m = pe[i][j][l].length\r\n while (m--) {\r\n pe_prime[i][j][l][m] = [pe[i][j][l][m].length];\r\n n = pe[i][j][l][m].length;\r\n while (n--) {\r\n pe_prime[i][j][l][m][n] = [pe[i][j][l][m][0], pe[i][j][l][m][1]];\r\n }\r\n }\r\n }\r\n } else {\r\n pe_prime[i][j] = Array();\r\n }\r\n\r\n d[i][j] = newPathLength;\r\n\r\n pe[i][j] = [[]];\r\n\r\n l = pe[i][k][0].length;\r\n while (l--) {\r\n pe[i][j][0].push(pe[i][k][0][l]);\r\n }\r\n\r\n l = pe[k][j][0].length;\r\n while (l--) {\r\n pe[i][j][0].push(pe[k][j][0][l]);\r\n }\r\n } else if (previousPathLength === newPathLength) {\r\n if (pe[i][k].length && pe[k][j].length) {\r\n var l;\r\n if (pe[i][j].length) {\r\n let tmp = Array();\r\n\r\n l = pe[i][k][0].length;\r\n while (l--) {\r\n tmp.push(pe[i][k][0][l]);\r\n }\r\n\r\n l = pe[k][j][0].length;\r\n while (l--) {\r\n tmp.push(pe[k][j][0][l]);\r\n }\r\n\r\n pe[i][j].push(tmp);\r\n } else {\r\n let tmp = Array();\r\n l = pe[i][k][0].length;\r\n while (l--) {\r\n tmp.push(pe[i][k][0][l]);\r\n }\r\n\r\n l = pe[k][j][0].length;\r\n while (l--) {\r\n tmp.push(pe[k][j][0][l]);\r\n }\r\n\r\n pe[i][j][0] = tmp\r\n }\r\n }\r\n } else if (previousPathLength === newPathLength - 1) {\r\n var l;\r\n if (pe_prime[i][j].length) {\r\n let tmp = Array();\r\n\r\n l = pe[i][k][0].length;\r\n while (l--) {\r\n tmp.push(pe[i][k][0][l]);\r\n }\r\n\r\n l = pe[k][j][0].length;\r\n while (l--) {\r\n tmp.push(pe[k][j][0][l]);\r\n }\r\n\r\n pe_prime[i][j].push(tmp);\r\n } else {\r\n let tmp = Array();\r\n\r\n l = pe[i][k][0].length;\r\n while (l--) {\r\n tmp.push(pe[i][k][0][l]);\r\n }\r\n\r\n l = pe[k][j][0].length;\r\n while (l--) {\r\n tmp.push(pe[k][j][0][l]);\r\n }\r\n\r\n pe_prime[i][j][0] = tmp;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n return {\r\n d: d,\r\n pe: pe,\r\n pe_prime: pe_prime\r\n };\r\n }\r\n\r\n /**\r\n * Get the ring candidates from the path-included distance matrices.\r\n * \r\n * @param {Array[]} d The distance matrix.\r\n * @param {Array[]} pe A matrix containing the shortest paths.\r\n * @param {Array[]} pe_prime A matrix containing the shortest paths + one vertex.\r\n * @returns {Array[]} The ring candidates.\r\n */\r\n static getRingCandidates(d, pe, pe_prime) {\r\n let length = d.length;\r\n let candidates = Array();\r\n let c = 0;\r\n\r\n for (let i = 0; i < length; i++) {\r\n for (let j = 0; j < length; j++) {\r\n if (d[i][j] === 0 || (pe[i][j].length === 1 && pe_prime[i][j] === 0)) {\r\n continue;\r\n } else {\r\n // c is the number of vertices in the cycle.\r\n if (pe_prime[i][j].length !== 0) {\r\n c = 2 * (d[i][j] + 0.5);\r\n } else {\r\n c = 2 * d[i][j];\r\n }\r\n\r\n if (c !== Infinity) {\r\n candidates.push([c, pe[i][j], pe_prime[i][j]]);\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Candidates have to be sorted by c\r\n candidates.sort(function (a, b) {\r\n return a[0] - b[0];\r\n });\r\n\r\n return candidates;\r\n }\r\n\r\n /**\r\n * Searches the candidates for the smallest set of smallest rings.\r\n * \r\n * @param {Array[]} c The candidates.\r\n * @param {Array[]} d The distance matrix.\r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @param {Array[]} pe A matrix containing the shortest paths.\r\n * @param {Array[]} pe_prime A matrix containing the shortest paths + one vertex.\r\n * @param {Uint16Array} arrBondCount A matrix containing the bond count of each vertex.\r\n * @param {Uint16Array} arrRingCount A matrix containing the number of rings associated with each vertex.\r\n * @param {Number} nsssr The theoretical number of rings in the graph.\r\n * @returns {Set[]} The smallest set of smallest rings.\r\n */\r\n static getSSSR(c, d, adjacencyMatrix, pe, pe_prime, arrBondCount, arrRingCount, nsssr) {\r\n let cSssr = Array();\r\n let allBonds = Array();\r\n\r\n for (let i = 0; i < c.length; i++) {\r\n if (c[i][0] % 2 !== 0) {\r\n for (let j = 0; j < c[i][2].length; j++) {\r\n let bonds = c[i][1][0].concat(c[i][2][j]);\r\n // Some bonds are added twice, resulting in [[u, v], [u, v]] instead of [u, v].\r\n // TODO: This is a workaround, fix later. Probably should be a set rather than an array, however the computational overhead\r\n // is probably bigger compared to leaving it like this.\r\n for (var k = 0; k < bonds.length; k++) {\r\n if (bonds[k][0].constructor === Array) bonds[k] = bonds[k][0];\r\n }\r\n\r\n let atoms = SSSR.bondsToAtoms(bonds);\r\n\r\n if (SSSR.getBondCount(atoms, adjacencyMatrix) === atoms.size && !SSSR.pathSetsContain(cSssr, atoms, bonds, allBonds, arrBondCount, arrRingCount)) {\r\n cSssr.push(atoms);\r\n allBonds = allBonds.concat(bonds);\r\n }\r\n\r\n if (cSssr.length > nsssr) {\r\n return cSssr;\r\n }\r\n }\r\n } else {\r\n for (let j = 0; j < c[i][1].length - 1; j++) {\r\n let bonds = c[i][1][j].concat(c[i][1][j + 1]);\r\n // Some bonds are added twice, resulting in [[u, v], [u, v]] instead of [u, v].\r\n // TODO: This is a workaround, fix later. Probably should be a set rather than an array, however the computational overhead\r\n // is probably bigger compared to leaving it like this.\r\n for (var k = 0; k < bonds.length; k++) {\r\n if (bonds[k][0].constructor === Array) bonds[k] = bonds[k][0];\r\n }\r\n\r\n let atoms = SSSR.bondsToAtoms(bonds);\r\n\r\n if (SSSR.getBondCount(atoms, adjacencyMatrix) === atoms.size && !SSSR.pathSetsContain(cSssr, atoms, bonds, allBonds, arrBondCount, arrRingCount)) {\r\n cSssr.push(atoms);\r\n allBonds = allBonds.concat(bonds);\r\n }\r\n\r\n if (cSssr.length > nsssr) {\r\n return cSssr;\r\n }\r\n }\r\n }\r\n }\r\n\r\n return cSssr;\r\n }\r\n\r\n /**\r\n * Returns the number of edges in a graph defined by an adjacency matrix.\r\n * \r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @returns {Number} The number of edges in the graph defined by the adjacency matrix.\r\n */\r\n static getEdgeCount(adjacencyMatrix) {\r\n let edgeCount = 0;\r\n let length = adjacencyMatrix.length;\r\n\r\n var i = length - 1;\r\n while (i--) {\r\n var j = length;\r\n while (j--) {\r\n if (adjacencyMatrix[i][j] === 1) {\r\n edgeCount++;\r\n }\r\n }\r\n }\r\n\r\n return edgeCount;\r\n }\r\n\r\n /**\r\n * Returns an edge list constructed form an adjacency matrix.\r\n * \r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @returns {Array[]} An edge list. E.g. [ [ 0, 1 ], ..., [ 16, 2 ] ]\r\n */\r\n static getEdgeList(adjacencyMatrix) {\r\n let length = adjacencyMatrix.length;\r\n let edgeList = Array();\r\n\r\n var i = length - 1;\r\n while (i--) {\r\n var j = length;\r\n while (j--) {\r\n if (adjacencyMatrix[i][j] === 1) {\r\n edgeList.push([i, j]);\r\n }\r\n }\r\n }\r\n\r\n return edgeList;\r\n }\r\n\r\n /**\r\n * Return a set of vertex indices contained in an array of bonds.\r\n * \r\n * @param {Array} bonds An array of bonds. A bond is defined as [ sourceVertexId, targetVertexId ].\r\n * @returns {Set} An array of vertices.\r\n */\r\n static bondsToAtoms(bonds) {\r\n let atoms = new Set();\r\n\r\n var i = bonds.length;\r\n while (i--) {\r\n atoms.add(bonds[i][0]);\r\n atoms.add(bonds[i][1]);\r\n }\r\n return atoms;\r\n }\r\n\r\n /**\r\n * Returns the number of bonds within a set of atoms.\r\n * \r\n * @param {Set} atoms An array of atom ids.\r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @returns {Number} The number of bonds in a set of atoms.\r\n */\r\n static getBondCount(atoms, adjacencyMatrix) {\r\n let count = 0;\r\n for (let u of atoms) {\r\n for (let v of atoms) {\r\n if (u === v) {\r\n continue;\r\n }\r\n count += adjacencyMatrix[u][v]\r\n }\r\n }\r\n\r\n return count / 2;\r\n }\r\n\r\n /**\r\n * Checks whether or not a given path already exists in an array of paths.\r\n * \r\n * @param {Set[]} pathSets An array of sets each representing a path.\r\n * @param {Set} pathSet A set representing a path.\r\n * @param {Array[]} bonds The bonds associated with the current path.\r\n * @param {Array[]} allBonds All bonds currently associated with rings in the SSSR set.\r\n * @param {Uint16Array} arrBondCount A matrix containing the bond count of each vertex.\r\n * @param {Uint16Array} arrRingCount A matrix containing the number of rings associated with each vertex.\r\n * @returns {Boolean} A boolean indicating whether or not a give path is contained within a set.\r\n */\r\n static pathSetsContain(pathSets, pathSet, bonds, allBonds, arrBondCount, arrRingCount) {\r\n var i = pathSets.length;\r\n while (i--) {\r\n if (SSSR.isSupersetOf(pathSet, pathSets[i])) {\r\n return true;\r\n }\r\n\r\n if (pathSets[i].size !== pathSet.size) {\r\n continue;\r\n }\r\n\r\n if (SSSR.areSetsEqual(pathSets[i], pathSet)) {\r\n return true;\r\n }\r\n }\r\n\r\n // Check if the edges from the candidate are already all contained within the paths of the set of paths.\r\n // TODO: For some reason, this does not replace the isSupersetOf method above -> why?\r\n let count = 0;\r\n let allContained = false;\r\n i = bonds.length;\r\n while (i--) {\r\n var j = allBonds.length;\r\n while (j--) {\r\n if (bonds[i][0] === allBonds[j][0] && bonds[i][1] === allBonds[j][1] ||\r\n bonds[i][1] === allBonds[j][0] && bonds[i][0] === allBonds[j][1]) {\r\n count++;\r\n }\r\n\r\n if (count === bonds.length) {\r\n allContained = true;\r\n }\r\n }\r\n }\r\n\r\n // If all the bonds and thus vertices are already contained within other rings\r\n // check if there's one vertex with ringCount < bondCount\r\n let specialCase = false;\r\n if (allContained) {\r\n for (let element of pathSet) {\r\n if (arrRingCount[element] < arrBondCount[element]) {\r\n specialCase = true;\r\n break;\r\n }\r\n }\r\n }\r\n\r\n if (allContained && !specialCase) {\r\n return true;\r\n }\r\n\r\n // Update the ring counts for the vertices\r\n for (let element of pathSet) {\r\n arrRingCount[element]++;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Checks whether or not two sets are equal (contain the same elements).\r\n * \r\n * @param {Set} setA A set.\r\n * @param {Set} setB A set.\r\n * @returns {Boolean} A boolean indicating whether or not the two sets are equal.\r\n */\r\n static areSetsEqual(setA, setB) {\r\n if (setA.size !== setB.size) {\r\n return false;\r\n }\r\n\r\n for (let element of setA) {\r\n if (!setB.has(element)) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Checks whether or not a set (setA) is a superset of another set (setB).\r\n * \r\n * @param {Set} setA A set.\r\n * @param {Set} setB A set.\r\n * @returns {Boolean} A boolean indicating whether or not setB is a superset of setA.\r\n */\r\n static isSupersetOf(setA, setB) {\r\n for (var element of setB) {\r\n if (!setA.has(element)) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n}\r\n\r\nmodule.exports = SSSR;","//@ts-check\r\n\r\n/** \r\n * A class representing a 2D vector.\r\n * \r\n * @property {Number} x The x component of the vector.\r\n * @property {Number} y The y component of the vector.\r\n */\r\nclass Vector2 {\r\n /**\r\n * The constructor of the class Vector2.\r\n *\r\n * @param {(Number|Vector2)} x The initial x coordinate value or, if the single argument, a Vector2 object.\r\n * @param {Number} y The initial y coordinate value.\r\n */\r\n constructor(x, y) {\r\n if (arguments.length == 0) {\r\n this.x = 0;\r\n this.y = 0;\r\n } else if (arguments.length == 1) {\r\n this.x = x.x;\r\n this.y = x.y;\r\n } else {\r\n this.x = x;\r\n this.y = y;\r\n }\r\n }\r\n\r\n /**\r\n * Clones this vector and returns the clone.\r\n *\r\n * @returns {Vector2} The clone of this vector.\r\n */\r\n clone() {\r\n return new Vector2(this.x, this.y);\r\n }\r\n\r\n /**\r\n * Returns a string representation of this vector.\r\n *\r\n * @returns {String} A string representation of this vector.\r\n */\r\n toString() {\r\n return '(' + this.x + ',' + this.y + ')';\r\n }\r\n\r\n /**\r\n * Add the x and y coordinate values of a vector to the x and y coordinate values of this vector.\r\n *\r\n * @param {Vector2} vec Another vector.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n add(vec) {\r\n this.x += vec.x;\r\n this.y += vec.y;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Subtract the x and y coordinate values of a vector from the x and y coordinate values of this vector.\r\n *\r\n * @param {Vector2} vec Another vector.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n subtract(vec) {\r\n this.x -= vec.x;\r\n this.y -= vec.y;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Divide the x and y coordinate values of this vector by a scalar.\r\n *\r\n * @param {Number} scalar The scalar.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n divide(scalar) {\r\n this.x /= scalar;\r\n this.y /= scalar;\r\n\r\n return this;\r\n }\r\n \r\n /**\r\n * Multiply the x and y coordinate values of this vector by the values of another vector.\r\n *\r\n * @param {Vector2} v A vector.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n multiply(v) {\r\n this.x *= v.x;\r\n this.y *= v.y;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Multiply the x and y coordinate values of this vector by a scalar.\r\n *\r\n * @param {Number} scalar The scalar.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n multiplyScalar(scalar) {\r\n this.x *= scalar;\r\n this.y *= scalar;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Inverts this vector. Same as multiply(-1.0).\r\n *\r\n * @returns {Vector2} Returns itself.\r\n */\r\n invert() {\r\n this.x = -this.x;\r\n this.y = -this.y;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns the angle of this vector in relation to the coordinate system.\r\n *\r\n * @returns {Number} The angle in radians.\r\n */\r\n angle() {\r\n return Math.atan2(this.y, this.x);\r\n }\r\n\r\n /**\r\n * Returns the euclidean distance between this vector and another vector.\r\n *\r\n * @param {Vector2} vec A vector.\r\n * @returns {Number} The euclidean distance between the two vectors.\r\n */\r\n distance(vec) {\r\n return Math.sqrt((vec.x - this.x) * (vec.x - this.x) + (vec.y - this.y) * (vec.y - this.y));\r\n }\r\n\r\n /**\r\n * Returns the squared euclidean distance between this vector and another vector. When only the relative distances of a set of vectors are needed, this is is less expensive than using distance(vec).\r\n *\r\n * @param {Vector2} vec Another vector.\r\n * @returns {Number} The squared euclidean distance of the two vectors.\r\n */\r\n distanceSq(vec) {\r\n return (vec.x - this.x) * (vec.x - this.x) + (vec.y - this.y) * (vec.y - this.y);\r\n }\r\n\r\n /**\r\n * Checks whether or not this vector is in a clockwise or counter-clockwise rotational direction compared to another vector in relation to the coordinate system.\r\n *\r\n * @param {Vector2} vec Another vector.\r\n * @returns {Number} Returns -1, 0 or 1 if the vector supplied as an argument is clockwise, neutral or counter-clockwise respectively to this vector in relation to the coordinate system.\r\n */\r\n clockwise(vec) {\r\n let a = this.y * vec.x;\r\n let b = this.x * vec.y;\r\n \r\n if (a > b) {\r\n return -1;\r\n }\r\n else if (a === b) {\r\n return 0;\r\n }\r\n\r\n return 1;\r\n }\r\n\r\n /**\r\n * Checks whether or not this vector is in a clockwise or counter-clockwise rotational direction compared to another vector in relation to an arbitrary third vector.\r\n *\r\n * @param {Vector2} center The central vector.\r\n * @param {Vector2} vec Another vector.\r\n * @returns {Number} Returns -1, 0 or 1 if the vector supplied as an argument is clockwise, neutral or counter-clockwise respectively to this vector in relation to an arbitrary third vector.\r\n */\r\n relativeClockwise(center, vec) {\r\n let a = (this.y - center.y) * (vec.x - center.x);\r\n let b = (this.x - center.x) * (vec.y - center.y);\r\n \r\n if (a > b) {\r\n return -1;\r\n }\r\n else if (a === b) {\r\n return 0;\r\n }\r\n\r\n return 1;\r\n }\r\n\r\n /**\r\n * Rotates this vector by a given number of radians around the origin of the coordinate system.\r\n *\r\n * @param {Number} angle The angle in radians to rotate the vector.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n rotate(angle) {\r\n let tmp = new Vector2(0, 0);\r\n let cosAngle = Math.cos(angle);\r\n let sinAngle = Math.sin(angle);\r\n\r\n tmp.x = this.x * cosAngle - this.y * sinAngle;\r\n tmp.y = this.x * sinAngle + this.y * cosAngle;\r\n \r\n this.x = tmp.x;\r\n this.y = tmp.y;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Rotates this vector around another vector.\r\n *\r\n * @param {Number} angle The angle in radians to rotate the vector.\r\n * @param {Vector2} vec The vector which is used as the rotational center.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n rotateAround(angle, vec) {\r\n let s = Math.sin(angle);\r\n let c = Math.cos(angle);\r\n\r\n this.x -= vec.x;\r\n this.y -= vec.y;\r\n\r\n let x = this.x * c - this.y * s;\r\n let y = this.x * s + this.y * c;\r\n\r\n this.x = x + vec.x;\r\n this.y = y + vec.y;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Rotate a vector around a given center to the same angle as another vector (so that the two vectors and the center are in a line, with both vectors on one side of the center), keeps the distance from this vector to the center.\r\n *\r\n * @param {Vector2} vec The vector to rotate this vector to.\r\n * @param {Vector2} center The rotational center.\r\n * @param {Number} [offsetAngle=0.0] An additional amount of radians to rotate the vector.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n rotateTo(vec, center, offsetAngle = 0.0) {\r\n // Problem if this is first position\r\n this.x += 0.001;\r\n this.y -= 0.001;\r\n\r\n let a = Vector2.subtract(this, center);\r\n let b = Vector2.subtract(vec, center);\r\n let angle = Vector2.angle(b, a);\r\n\r\n this.rotateAround(angle + offsetAngle, center);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Rotates the vector away from a specified vector around a center.\r\n * \r\n * @param {Vector2} vec The vector this one is rotated away from.\r\n * @param {Vector2} center The rotational center.\r\n * @param {Number} angle The angle by which to rotate.\r\n */\r\n rotateAwayFrom(vec, center, angle) {\r\n this.rotateAround(angle, center);\r\n \r\n let distSqA = this.distanceSq(vec);\r\n \r\n this.rotateAround(-2.0 * angle, center);\r\n\r\n let distSqB = this.distanceSq(vec);\r\n\r\n // If it was rotated towards the other vertex, rotate in other direction by same amount.\r\n if (distSqB < distSqA) {\r\n this.rotateAround(2.0 * angle, center);\r\n }\r\n }\r\n\r\n /**\r\n * Returns the angle in radians used to rotate this vector away from a given vector.\r\n * \r\n * @param {Vector2} vec The vector this one is rotated away from.\r\n * @param {Vector2} center The rotational center.\r\n * @param {Number} angle The angle by which to rotate.\r\n * @returns {Number} The angle in radians.\r\n */\r\n getRotateAwayFromAngle(vec, center, angle) {\r\n let tmp = this.clone();\r\n\r\n tmp.rotateAround(angle, center);\r\n \r\n let distSqA = tmp.distanceSq(vec);\r\n \r\n tmp.rotateAround(-2.0 * angle, center);\r\n\r\n let distSqB = tmp.distanceSq(vec);\r\n\r\n if (distSqB < distSqA) {\r\n return angle;\r\n } else {\r\n return -angle;\r\n }\r\n }\r\n\r\n /**\r\n * Returns the angle in radians used to rotate this vector towards a given vector.\r\n * \r\n * @param {Vector2} vec The vector this one is rotated towards to.\r\n * @param {Vector2} center The rotational center.\r\n * @param {Number} angle The angle by which to rotate.\r\n * @returns {Number} The angle in radians.\r\n */\r\n getRotateTowardsAngle(vec, center, angle) {\r\n let tmp = this.clone();\r\n\r\n tmp.rotateAround(angle, center);\r\n \r\n let distSqA = tmp.distanceSq(vec);\r\n \r\n tmp.rotateAround(-2.0 * angle, center);\r\n\r\n let distSqB = tmp.distanceSq(vec);\r\n\r\n if (distSqB > distSqA) {\r\n return angle;\r\n } else {\r\n return -angle;\r\n }\r\n }\r\n\r\n /**\r\n * Gets the angles between this vector and another vector around a common center of rotation.\r\n *\r\n * @param {Vector2} vec Another vector.\r\n * @param {Vector2} center The center of rotation.\r\n * @returns {Number} The angle between this vector and another vector around a center of rotation in radians.\r\n */\r\n getRotateToAngle(vec, center) {\r\n let a = Vector2.subtract(this, center);\r\n let b = Vector2.subtract(vec, center);\r\n let angle = Vector2.angle(b, a);\r\n \r\n return Number.isNaN(angle) ? 0.0 : angle;\r\n }\r\n\r\n /**\r\n * Checks whether a vector lies within a polygon spanned by a set of vectors.\r\n *\r\n * @param {Vector2[]} polygon An array of vectors spanning the polygon.\r\n * @returns {Boolean} A boolean indicating whether or not this vector is within a polygon.\r\n */\r\n isInPolygon(polygon) {\r\n let inside = false;\r\n\r\n // Its not always a given, that the polygon is convex (-> sugars)\r\n for (let i = 0, j = polygon.length - 1; i < polygon.length; j = i++) {\r\n if (((polygon[i].y > this.y) != (polygon[j].y > this.y)) &&\r\n (this.x < (polygon[j].x - polygon[i].x) * (this.y - polygon[i].y) /\r\n (polygon[j].y - polygon[i].y) + polygon[i].x)) {\r\n inside = !inside;\r\n }\r\n }\r\n\r\n\r\n return inside;\r\n }\r\n\r\n /**\r\n * Returns the length of this vector.\r\n *\r\n * @returns {Number} The length of this vector.\r\n */\r\n length() {\r\n return Math.sqrt((this.x * this.x) + (this.y * this.y));\r\n }\r\n\r\n /**\r\n * Returns the square of the length of this vector.\r\n *\r\n * @returns {Number} The square of the length of this vector.\r\n */\r\n lengthSq() {\r\n return (this.x * this.x) + (this.y * this.y);\r\n }\r\n\r\n /**\r\n * Normalizes this vector.\r\n *\r\n * @returns {Vector2} Returns itself.\r\n */\r\n normalize() {\r\n this.divide(this.length());\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a normalized copy of this vector.\r\n *\r\n * @returns {Vector2} A normalized copy of this vector.\r\n */\r\n normalized() {\r\n return Vector2.divideScalar(this, this.length());\r\n }\r\n\r\n /**\r\n * Calculates which side of a line spanned by two vectors this vector is.\r\n *\r\n * @param {Vector2} vecA A vector.\r\n * @param {Vector2} vecB A vector.\r\n * @returns {Number} A number indicating the side of this vector, given a line spanned by two other vectors.\r\n */\r\n whichSide(vecA, vecB) {\r\n return (this.x - vecA.x) * (vecB.y - vecA.y) - (this.y - vecA.y) * (vecB.x - vecA.x);\r\n }\r\n\r\n /**\r\n * Checks whether or not this vector is on the same side of a line spanned by two vectors as another vector.\r\n *\r\n * @param {Vector2} vecA A vector spanning the line.\r\n * @param {Vector2} vecB A vector spanning the line.\r\n * @param {Vector2} vecC A vector to check whether or not it is on the same side as this vector.\r\n * @returns {Boolean} Returns a boolean indicating whether or not this vector is on the same side as another vector.\r\n */\r\n sameSideAs(vecA, vecB, vecC) {\r\n let d = this.whichSide(vecA, vecB);\r\n let dRef = vecC.whichSide(vecA, vecB);\r\n\r\n return d < 0 && dRef < 0 || d == 0 && dRef == 0 || d > 0 && dRef > 0;\r\n }\r\n\r\n /**\r\n * Adds two vectors and returns the result as a new vector.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A summand.\r\n * @param {Vector2} vecB A summand.\r\n * @returns {Vector2} Returns the sum of two vectors.\r\n */\r\n static add(vecA, vecB) {\r\n return new Vector2(vecA.x + vecB.x, vecA.y + vecB.y);\r\n }\r\n\r\n /**\r\n * Subtracts one vector from another and returns the result as a new vector.\r\n *\r\n * @static\r\n * @param {Vector2} vecA The minuend.\r\n * @param {Vector2} vecB The subtrahend.\r\n * @returns {Vector2} Returns the difference of two vectors.\r\n */\r\n static subtract(vecA, vecB) {\r\n return new Vector2(vecA.x - vecB.x, vecA.y - vecB.y);\r\n }\r\n\r\n /**\r\n * Multiplies two vectors (value by value) and returns the result.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector.\r\n * @param {Vector2} vecB A vector.\r\n * @returns {Vector2} Returns the product of two vectors.\r\n */\r\n static multiply(vecA, vecB) {\r\n return new Vector2(vecA.x * vecB.x, vecA.y * vecB.y);\r\n }\r\n\r\n /**\r\n * Multiplies two vectors (value by value) and returns the result.\r\n *\r\n * @static\r\n * @param {Vector2} vec A vector.\r\n * @param {Number} scalar A scalar.\r\n * @returns {Vector2} Returns the product of two vectors.\r\n */\r\n static multiplyScalar(vec, scalar) {\r\n return new Vector2(vec.x, vec.y).multiplyScalar(scalar);\r\n }\r\n\r\n /**\r\n * Returns the midpoint of a line spanned by two vectors.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector spanning the line.\r\n * @param {Vector2} vecB A vector spanning the line.\r\n * @returns {Vector2} The midpoint of the line spanned by two vectors.\r\n */\r\n static midpoint(vecA, vecB) {\r\n return new Vector2((vecA.x + vecB.x) / 2, (vecA.y + vecB.y) / 2);\r\n }\r\n\r\n /**\r\n * Returns the normals of a line spanned by two vectors.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector spanning the line.\r\n * @param {Vector2} vecB A vector spanning the line.\r\n * @returns {Vector2[]} An array containing the two normals, each represented by a vector.\r\n */\r\n static normals(vecA, vecB) {\r\n let delta = Vector2.subtract(vecB, vecA);\r\n\r\n return [\r\n new Vector2(-delta.y, delta.x),\r\n new Vector2(delta.y, -delta.x)\r\n ];\r\n }\r\n\r\n /**\r\n * Returns the unit (normalized normal) vectors of a line spanned by two vectors.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector spanning the line.\r\n * @param {Vector2} vecB A vector spanning the line.\r\n * @returns {Vector2[]} An array containing the two unit vectors.\r\n */\r\n static units(vecA, vecB) {\r\n let delta = Vector2.subtract(vecB, vecA);\r\n\r\n return [\r\n (new Vector2(-delta.y, delta.x)).normalize(),\r\n (new Vector2(delta.y, -delta.x)).normalize()\r\n ];\r\n }\r\n\r\n /**\r\n * Divides a vector by another vector and returns the result as new vector.\r\n *\r\n * @static\r\n * @param {Vector2} vecA The dividend.\r\n * @param {Vector2} vecB The divisor.\r\n * @returns {Vector2} The fraction of the two vectors.\r\n */\r\n static divide(vecA, vecB) {\r\n return new Vector2(vecA.x / vecB.x, vecA.y / vecB.y);\r\n }\r\n\r\n /**\r\n * Divides a vector by a scalar and returns the result as new vector.\r\n *\r\n * @static\r\n * @param {Vector2} vecA The dividend.\r\n * @param {Number} s The scalar.\r\n * @returns {Vector2} The fraction of the two vectors.\r\n */\r\n static divideScalar(vecA, s) {\r\n return new Vector2(vecA.x / s, vecA.y / s);\r\n }\r\n\r\n /**\r\n * Returns the dot product of two vectors.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector.\r\n * @param {Vector2} vecB A vector.\r\n * @returns {Number} The dot product of two vectors.\r\n */\r\n static dot(vecA, vecB) {\r\n return vecA.x * vecB.x + vecA.y * vecB.y;\r\n }\r\n\r\n /**\r\n * Returns the angle between two vectors.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector.\r\n * @param {Vector2} vecB A vector.\r\n * @returns {Number} The angle between two vectors in radians.\r\n */\r\n static angle(vecA, vecB) {\r\n let dot = Vector2.dot(vecA, vecB);\r\n\r\n return Math.acos(dot / (vecA.length() * vecB.length()));\r\n }\r\n\r\n /**\r\n * Returns the angle between two vectors based on a third vector in between.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector.\r\n * @param {Vector2} vecB A (central) vector.\r\n * @param {Vector2} vecC A vector.\r\n * @returns {Number} The angle in radians.\r\n */\r\n static threePointangle(vecA, vecB, vecC) {\r\n let ab = Vector2.subtract(vecB, vecA);\r\n let bc = Vector2.subtract(vecC, vecB);\r\n let abLength = vecA.distance(vecB);\r\n let bcLength = vecB.distance(vecC);\r\n\r\n return Math.acos(Vector2.dot(ab, bc) / (abLength * bcLength));\r\n }\r\n \r\n /**\r\n * Returns the scalar projection of a vector on another vector.\r\n *\r\n * @static\r\n * @param {Vector2} vecA The vector to be projected.\r\n * @param {Vector2} vecB The vector to be projection upon.\r\n * @returns {Number} The scalar component.\r\n */\r\n static scalarProjection(vecA, vecB) {\r\n let unit = vecB.normalized();\r\n \r\n return Vector2.dot(vecA, unit);\r\n }\r\n\r\n /**\r\n * Returns the average vector (normalized) of the input vectors.\r\n *\r\n * @static\r\n * @param {Array} vecs An array containing vectors.\r\n * @returns {Vector2} The resulting vector (normalized).\r\n */\r\n static averageDirection(vecs) {\r\n let avg = new Vector2(0.0, 0.0);\r\n\r\n for (var i = 0; i < vecs.length; i++) {\r\n let vec = vecs[i];\r\n avg.add(vec);\r\n }\r\n\r\n return avg.normalize();\r\n }\r\n}\r\n\r\nmodule.exports = Vector2;","//@ts-check\r\nconst MathHelper = require('./MathHelper')\r\nconst ArrayHelper = require('./ArrayHelper')\r\nconst Vector2 = require('./Vector2')\r\nconst Atom = require('./Atom')\r\n\r\n/** \r\n * A class representing a vertex.\r\n * \r\n * @property {Number} id The id of this vertex.\r\n * @property {Atom} value The atom associated with this vertex.\r\n * @property {Vector2} position The position of this vertex.\r\n * @property {Vector2} previousPosition The position of the previous vertex.\r\n * @property {Number|null} parentVertexId The id of the previous vertex.\r\n * @property {Number[]} children The ids of the children of this vertex.\r\n * @property {Number[]} spanningTreeChildren The ids of the children of this vertex as defined in the spanning tree defined by the SMILES.\r\n * @property {Number[]} edges The ids of edges associated with this vertex.\r\n * @property {Boolean} positioned A boolean indicating whether or not this vertex has been positioned.\r\n * @property {Number} angle The angle of this vertex.\r\n * @property {Number} dir The direction of this vertex.\r\n * @property {Number} neighbourCount The number of neighbouring vertices.\r\n * @property {Number[]} neighbours The vertex ids of neighbouring vertices.\r\n * @property {String[]} neighbouringElements The element symbols associated with neighbouring vertices.\r\n * @property {Boolean} forcePositioned A boolean indicating whether or not this vertex was positioned using a force-based approach.\r\n */\r\n\r\nclass Vertex {\r\n /**\r\n * The constructor for the class Vertex.\r\n *\r\n * @param {Atom} value The value associated with this vertex.\r\n * @param {Number} [x=0] The initial x coordinate of the positional vector of this vertex.\r\n * @param {Number} [y=0] The initial y coordinate of the positional vector of this vertex.\r\n */\r\n constructor(value, x = 0, y = 0) {\r\n this.id = null;\r\n this.value = value;\r\n this.position = new Vector2(x ? x : 0, y ? y : 0);\r\n this.previousPosition = new Vector2(0, 0);\r\n this.parentVertexId = null;\r\n this.children = Array();\r\n this.spanningTreeChildren = Array();\r\n this.edges = Array();\r\n this.positioned = false;\r\n this.angle = null;\r\n this.dir = 1.0;\r\n this.neighbourCount = 0;\r\n this.neighbours = Array();\r\n this.neighbouringElements = Array();\r\n this.forcePositioned = false;\r\n }\r\n\r\n /**\r\n * Set the 2D coordinates of the vertex.\r\n * \r\n * @param {Number} x The x component of the coordinates.\r\n * @param {Number} y The y component of the coordinates.\r\n * \r\n */\r\n setPosition(x, y) {\r\n this.position.x = x;\r\n this.position.y = y;\r\n }\r\n\r\n /**\r\n * Set the 2D coordinates of the vertex from a Vector2.\r\n * \r\n * @param {Vector2} v A 2D vector.\r\n * \r\n */\r\n setPositionFromVector(v) {\r\n this.position.x = v.x;\r\n this.position.y = v.y;\r\n }\r\n\r\n /**\r\n * Add a child vertex id to this vertex.\r\n * @param {Number} vertexId The id of a vertex to be added as a child to this vertex.\r\n */\r\n addChild(vertexId) {\r\n this.children.push(vertexId);\r\n this.neighbours.push(vertexId);\r\n\r\n this.neighbourCount++;\r\n }\r\n\r\n /**\r\n * Add a child vertex id to this vertex as the second child of the neighbours array,\r\n * except this vertex is the first vertex of the SMILE string, then it is added as the first.\r\n * This is used to get the correct ordering of neighbours for parity calculations.\r\n * If a hydrogen is implicitly attached to the chiral center, insert as the third child.\r\n * @param {Number} vertexId The id of a vertex to be added as a child to this vertex.\r\n * @param {Number} ringbondIndex The index of the ringbond.\r\n */\r\n addRingbondChild(vertexId, ringbondIndex) {\r\n this.children.push(vertexId);\r\n\r\n if (this.value.bracket) {\r\n let index = 1;\r\n\r\n if (this.id === 0 && this.value.bracket.hcount === 0) {\r\n index = 0;\r\n }\r\n \r\n if (this.value.bracket.hcount === 1 && ringbondIndex === 0) {\r\n index = 2;\r\n }\r\n\r\n if (this.value.bracket.hcount === 1 && ringbondIndex === 1) {\r\n if (this.neighbours.length < 3) {\r\n index = 2;\r\n } else {\r\n index = 3;\r\n }\r\n }\r\n\r\n if (this.value.bracket.hcount === null && ringbondIndex === 0) {\r\n index = 1;\r\n }\r\n\r\n if (this.value.bracket.hcount === null && ringbondIndex === 1) {\r\n if (this.neighbours.length < 3) {\r\n index = 1;\r\n } else {\r\n index = 2;\r\n }\r\n }\r\n \r\n this.neighbours.splice(index, 0, vertexId);\r\n } else {\r\n this.neighbours.push(vertexId);\r\n }\r\n\r\n this.neighbourCount++;\r\n }\r\n\r\n /**\r\n * Set the vertex id of the parent.\r\n * \r\n * @param {Number} parentVertexId The parents vertex id.\r\n */\r\n setParentVertexId(parentVertexId) {\r\n this.neighbourCount++;\r\n this.parentVertexId = parentVertexId;\r\n this.neighbours.push(parentVertexId);\r\n }\r\n\r\n /**\r\n * Returns true if this vertex is terminal (has no parent or child vertices), otherwise returns false. Always returns true if associated value has property hasAttachedPseudoElements set to true.\r\n *\r\n * @returns {Boolean} A boolean indicating whether or not this vertex is terminal.\r\n */\r\n isTerminal() {\r\n if (this.value.hasAttachedPseudoElements) {\r\n return true;\r\n }\r\n\r\n return (this.parentVertexId === null && this.children.length < 2) || this.children.length === 0;\r\n }\r\n\r\n /**\r\n * Clones this vertex and returns the clone.\r\n *\r\n * @returns {Vertex} A clone of this vertex.\r\n */\r\n clone() {\r\n let clone = new Vertex(this.value, this.position.x, this.position.y);\r\n clone.id = this.id;\r\n clone.previousPosition = new Vector2(this.previousPosition.x, this.previousPosition.y);\r\n clone.parentVertexId = this.parentVertexId;\r\n clone.children = ArrayHelper.clone(this.children);\r\n clone.spanningTreeChildren = ArrayHelper.clone(this.spanningTreeChildren);\r\n clone.edges = ArrayHelper.clone(this.edges);\r\n clone.positioned = this.positioned;\r\n clone.angle = this.angle;\r\n clone.forcePositioned = this.forcePositioned;\r\n return clone;\r\n }\r\n\r\n /**\r\n * Returns true if this vertex and the supplied vertex both have the same id, else returns false.\r\n *\r\n * @param {Vertex} vertex The vertex to check.\r\n * @returns {Boolean} A boolean indicating whether or not the two vertices have the same id.\r\n */\r\n equals(vertex) {\r\n return this.id === vertex.id;\r\n }\r\n\r\n /**\r\n * Returns the angle of this vertexes positional vector. If a reference vector is supplied in relations to this vector, else in relations to the coordinate system.\r\n *\r\n * @param {Vector2} [referenceVector=null] - The reference vector.\r\n * @param {Boolean} [returnAsDegrees=false] - If true, returns angle in degrees, else in radians.\r\n * @returns {Number} The angle of this vertex.\r\n */\r\n getAngle(referenceVector = null, returnAsDegrees = false) {\r\n let u = null;\r\n\r\n if (!referenceVector) {\r\n u = Vector2.subtract(this.position, this.previousPosition);\r\n } else {\r\n u = Vector2.subtract(this.position, referenceVector);\r\n }\r\n\r\n if (returnAsDegrees) {\r\n return MathHelper.toDeg(u.angle());\r\n }\r\n\r\n return u.angle();\r\n }\r\n\r\n /**\r\n * Returns the suggested text direction when text is added at the position of this vertex.\r\n *\r\n * @param {Vertex[]} vertices The array of vertices for the current molecule.\r\n * @returns {String} The suggested direction of the text.\r\n */\r\n getTextDirection(vertices) {\r\n let neighbours = this.getDrawnNeighbours(vertices);\r\n let angles = Array();\r\n\r\n for (let i = 0; i < neighbours.length; i++) {\r\n angles.push(this.getAngle(vertices[neighbours[i]].position));\r\n }\r\n\r\n let textAngle = MathHelper.meanAngle(angles);\r\n\r\n // Round to 0, 90, 180 or 270 degree\r\n let halfPi = Math.PI / 2.0;\r\n textAngle = Math.round(Math.round(textAngle / halfPi) * halfPi);\r\n\r\n if (textAngle === 2) {\r\n return 'down';\r\n } else if (textAngle === -2) {\r\n return 'up';\r\n } else if (textAngle === 0 || textAngle === -0) {\r\n return 'right'; // is checking for -0 necessary?\r\n } else if (textAngle === 3 || textAngle === -3) {\r\n return 'left';\r\n } else {\r\n return 'down'; // default to down\r\n }\r\n }\r\n\r\n /**\r\n * Returns an array of ids of neighbouring vertices.\r\n *\r\n * @param {Number} [vertexId=null] If a value is supplied, the vertex with this id is excluded from the returned indices.\r\n * @returns {Number[]} An array containing the ids of neighbouring vertices.\r\n */\r\n getNeighbours(vertexId = null) {\r\n if (vertexId === null) {\r\n return this.neighbours.slice();\r\n }\r\n\r\n let arr = Array();\r\n\r\n for (let i = 0; i < this.neighbours.length; i++) {\r\n if (this.neighbours[i] !== vertexId) {\r\n arr.push(this.neighbours[i]);\r\n }\r\n }\r\n\r\n return arr;\r\n }\r\n\r\n /**\r\n * Returns an array of ids of neighbouring vertices that will be drawn (vertex.value.isDrawn === true).\r\n * \r\n * @param {Vertex[]} vertices An array containing the vertices associated with the current molecule.\r\n * @returns {Number[]} An array containing the ids of neighbouring vertices that will be drawn.\r\n */\r\n getDrawnNeighbours(vertices) {\r\n let arr = Array();\r\n\r\n for (let i = 0; i < this.neighbours.length; i++) {\r\n if (vertices[this.neighbours[i]].value.isDrawn) {\r\n arr.push(this.neighbours[i]);\r\n }\r\n }\r\n\r\n return arr;\r\n }\r\n\r\n /**\r\n * Returns the number of neighbours of this vertex.\r\n *\r\n * @returns {Number} The number of neighbours.\r\n */\r\n getNeighbourCount() {\r\n return this.neighbourCount;\r\n }\r\n\r\n /**\r\n * Returns a list of ids of vertices neighbouring this one in the original spanning tree, excluding the ringbond connections.\r\n *\r\n * @param {Number} [vertexId=null] If supplied, the vertex with this id is excluded from the array returned.\r\n * @returns {Number[]} An array containing the ids of the neighbouring vertices.\r\n */\r\n getSpanningTreeNeighbours(vertexId = null) {\r\n let neighbours = Array();\r\n\r\n for (let i = 0; i < this.spanningTreeChildren.length; i++) {\r\n if (vertexId === undefined || vertexId != this.spanningTreeChildren[i]) {\r\n neighbours.push(this.spanningTreeChildren[i]);\r\n }\r\n }\r\n\r\n if (this.parentVertexId != null) {\r\n if (vertexId === undefined || vertexId != this.parentVertexId) {\r\n neighbours.push(this.parentVertexId);\r\n }\r\n }\r\n\r\n return neighbours;\r\n }\r\n\r\n /**\r\n * Gets the next vertex in the ring in opposide direction to the supplied vertex id.\r\n *\r\n * @param {Vertex[]} vertices The array of vertices for the current molecule.\r\n * @param {Number} ringId The id of the ring containing this vertex.\r\n * @param {Number} previousVertexId The id of the previous vertex. The next vertex will be opposite from the vertex with this id as seen from this vertex.\r\n * @returns {Number} The id of the next vertex in the ring.\r\n */\r\n getNextInRing(vertices, ringId, previousVertexId) {\r\n let neighbours = this.getNeighbours();\r\n\r\n for (let i = 0; i < neighbours.length; i++) {\r\n if (ArrayHelper.contains(vertices[neighbours[i]].value.rings, {\r\n value: ringId\r\n }) &&\r\n neighbours[i] != previousVertexId) {\r\n return neighbours[i];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n}\r\n\r\nmodule.exports = Vertex;"]} \ No newline at end of file +{"version":3,"sources":["node_modules/browser-pack/_prelude.js","app.js","src/ArrayHelper.js","src/Atom.js","src/CanvasWrapper.js","src/Drawer.js","src/Edge.js","src/Graph.js","src/Line.js","src/MathHelper.js","src/Parser.js","src/Ring.js","src/RingConnection.js","src/SSSR.js","src/Vector2.js","src/Vertex.js"],"names":[],"mappings":"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","file":"smiles-drawer.min.js","sourcesContent":["(function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c=\"function\"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error(\"Cannot find module '\"+i+\"'\");throw a.code=\"MODULE_NOT_FOUND\",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u=\"function\"==typeof require&&require,i=0;i>> 0;\r\n\r\n // Steps 6-7.\r\n var start = arguments[1];\r\n var relativeStart = start >> 0;\r\n\r\n // Step 8.\r\n var k = relativeStart < 0 ?\r\n Math.max(len + relativeStart, 0) :\r\n Math.min(relativeStart, len);\r\n\r\n // Steps 9-10.\r\n var end = arguments[2];\r\n var relativeEnd = end === undefined ?\r\n len : end >> 0;\r\n\r\n // Step 11.\r\n var final = relativeEnd < 0 ?\r\n Math.max(len + relativeEnd, 0) :\r\n Math.min(relativeEnd, len);\r\n\r\n // Step 12.\r\n while (k < final) {\r\n O[k] = value;\r\n k++;\r\n }\r\n\r\n // Step 13.\r\n return O;\r\n }\r\n});\r\n}\r\n\r\nmodule.exports = SmilesDrawer;","//@ts-check\r\n\r\n/** \r\n * A static class containing helper functions for array-related tasks. \r\n */\r\nclass ArrayHelper {\r\n /**\r\n * Clone an array or an object. If an object is passed, a shallow clone will be created.\r\n *\r\n * @static\r\n * @param {*} arr The array or object to be cloned.\r\n * @returns {*} A clone of the array or object.\r\n */\r\n static clone(arr) {\r\n let out = Array.isArray(arr) ? Array() : {};\r\n \r\n for (let key in arr) {\r\n let value = arr[key];\r\n \r\n if (typeof value.clone === 'function') {\r\n out[key] = value.clone();\r\n }\r\n else {\r\n out[key] = (typeof value === 'object') ? ArrayHelper.clone(value) : value;\r\n }\r\n }\r\n \r\n return out;\r\n }\r\n\r\n /**\r\n * Returns a boolean indicating whether or not the two arrays contain the same elements.\r\n * Only supports 1d, non-nested arrays.\r\n *\r\n * @static\r\n * @param {Array} arrA An array.\r\n * @param {Array} arrB An array.\r\n * @returns {Boolean} A boolean indicating whether or not the two arrays contain the same elements.\r\n */\r\n static equals(arrA, arrB) {\r\n if (arrA.length !== arrB.length) {\r\n return false;\r\n }\r\n\r\n let tmpA = arrA.slice().sort();\r\n let tmpB = arrB.slice().sort();\r\n\r\n for (var i = 0; i < tmpA.length; i++) {\r\n if (tmpA[i] !== tmpB[i]) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Returns a string representation of an array. If the array contains objects with an id property, the id property is printed for each of the elements.\r\n *\r\n * @static\r\n * @param {Object[]} arr An array.\r\n * @param {*} arr[].id If the array contains an object with the property 'id', the properties value is printed. Else, the array elements value is printend.\r\n * @returns {String} A string representation of the array.\r\n */\r\n static print(arr) {\r\n if (arr.length == 0) {\r\n return '';\r\n }\r\n\r\n let s = '(';\r\n\r\n for (let i = 0; i < arr.length; i++) {\r\n s += arr[i].id ? arr[i].id + ', ' : arr[i] + ', ';\r\n }\r\n\r\n s = s.substring(0, s.length - 2);\r\n\r\n return s + ')';\r\n }\r\n\r\n /**\r\n * Run a function for each element in the array. The element is supplied as an argument for the callback function\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {Function} callback The callback function that is called for each element.\r\n */\r\n static each(arr, callback) {\r\n for (let i = 0; i < arr.length; i++) {\r\n callback(arr[i]);\r\n }\r\n }\r\n\r\n /**\r\n * Return the array element from an array containing objects, where a property of the object is set to a given value.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {(String|Number)} property A property contained within an object in the array.\r\n * @param {(String|Number)} value The value of the property.\r\n * @returns {*} The array element matching the value.\r\n */\r\n static get(arr, property, value) {\r\n for (let i = 0; i < arr.length; i++) {\r\n if (arr[i][property] == value) {\r\n return arr[i];\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Checks whether or not an array contains a given value. the options object passed as a second argument can contain three properties. value: The value to be searched for. property: The property that is to be searched for a given value. func: A function that is used as a callback to return either true or false in order to do a custom comparison.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {Object} options See method description.\r\n * @param {*} options.value The value for which to check.\r\n * @param {String} [options.property=undefined] The property on which to check.\r\n * @param {Function} [options.func=undefined] A custom property function.\r\n * @returns {Boolean} A boolean whether or not the array contains a value.\r\n */\r\n static contains(arr, options) {\r\n if (!options.property && !options.func) {\r\n for (let i = 0; i < arr.length; i++) {\r\n if (arr[i] == options.value) {\r\n return true;\r\n }\r\n }\r\n } else if (options.func) {\r\n for (let i = 0; i < arr.length; i++) {\r\n if (options.func(arr[i])) {\r\n return true;\r\n }\r\n }\r\n } else {\r\n for (let i = 0; i < arr.length; i++) {\r\n if (arr[i][options.property] == options.value) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Returns an array containing the intersection between two arrays. That is, values that are common to both arrays.\r\n *\r\n * @static\r\n * @param {Array} arrA An array.\r\n * @param {Array} arrB An array.\r\n * @returns {Array} The intersecting vlaues.\r\n */\r\n static intersection(arrA, arrB) {\r\n let intersection = new Array();\r\n \r\n for (let i = 0; i < arrA.length; i++) {\r\n for (let j = 0; j < arrB.length; j++) {\r\n if (arrA[i] === arrB[j]) {\r\n intersection.push(arrA[i]);\r\n }\r\n }\r\n }\r\n\r\n return intersection;\r\n }\r\n\r\n /**\r\n * Returns an array of unique elements contained in an array.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @returns {Array} An array of unique elements contained within the array supplied as an argument.\r\n */\r\n static unique(arr) {\r\n let contains = {};\r\n return arr.filter(function (i) {\r\n // using !== instead of hasOwnProperty (http://andrew.hedges.name/experiments/in/)\r\n return contains[i] !== undefined ? false : (contains[i] = true);\r\n });\r\n }\r\n\r\n /**\r\n * Count the number of occurences of a value in an array.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {*} value A value to be counted.\r\n * @returns {Number} The number of occurences of a value in the array.\r\n */\r\n static count(arr, value) {\r\n let count = 0;\r\n\r\n for (let i = 0; i < arr.length; i++) {\r\n if (arr[i] === value) {\r\n count++;\r\n }\r\n }\r\n\r\n return count;\r\n }\r\n\r\n /**\r\n * Toggles the value of an array. If a value is not contained in an array, the array returned will contain all the values of the original array including the value. If a value is contained in an array, the array returned will contain all the values of the original array excluding the value.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {*} value A value to be toggled.\r\n * @returns {Array} The toggled array.\r\n */\r\n static toggle(arr, value) {\r\n let newArr = Array();\r\n\r\n let removed = false;\r\n for (let i = 0; i < arr.length; i++) {\r\n // Do not copy value if it exists\r\n if (arr[i] !== value) {\r\n newArr.push(arr[i]);\r\n } else {\r\n // The element was not copied to the new array, which\r\n // means it was removed\r\n removed = true;\r\n }\r\n }\r\n\r\n // If the element was not removed, then it was not in the array\r\n // so add it\r\n if (!removed) {\r\n newArr.push(value);\r\n }\r\n\r\n return newArr;\r\n }\r\n\r\n /**\r\n * Remove a value from an array.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {*} value A value to be removed.\r\n * @returns {Array} A new array with the element with a given value removed.\r\n */\r\n static remove(arr, value) {\r\n let tmp = Array();\r\n\r\n for (let i = 0; i < arr.length; i++) {\r\n if (arr[i] !== value) {\r\n tmp.push(arr[i]);\r\n }\r\n }\r\n\r\n return tmp;\r\n }\r\n\r\n /**\r\n * Remove a value from an array with unique values.\r\n *\r\n * @static\r\n * @param {Array} arr An array.\r\n * @param {*} value A value to be removed.\r\n * @returns {Array} An array with the element with a given value removed.\r\n */\r\n static removeUnique(arr, value) {\r\n let index = arr.indexOf(value);\r\n\r\n if (index > -1) {\r\n arr.splice(index, 1);\r\n }\r\n\r\n return arr;\r\n }\r\n\r\n /**\r\n * Remove all elements contained in one array from another array.\r\n *\r\n * @static\r\n * @param {Array} arrA The array to be filtered.\r\n * @param {Array} arrB The array containing elements that will be removed from the other array.\r\n * @returns {Array} The filtered array.\r\n */\r\n static removeAll(arrA, arrB) {\r\n return arrA.filter(function (item) {\r\n return arrB.indexOf(item) === -1;\r\n });\r\n }\r\n\r\n /**\r\n * Merges two arrays and returns the result. The first array will be appended to the second array.\r\n *\r\n * @static\r\n * @param {Array} arrA An array.\r\n * @param {Array} arrB An array.\r\n * @returns {Array} The merged array.\r\n */\r\n static merge(arrA, arrB) {\r\n let arr = new Array(arrA.length + arrB.length);\r\n\r\n for (let i = 0; i < arrA.length; i++) {\r\n arr[i] = arrA[i];\r\n }\r\n\r\n for (let i = 0; i < arrB.length; i++) {\r\n arr[arrA.length + i] = arrB[i];\r\n }\r\n\r\n return arr;\r\n }\r\n\r\n /**\r\n * Checks whether or not an array contains all the elements of another array, without regard to the order.\r\n *\r\n * @static\r\n * @param {Array} arrA An array.\r\n * @param {Array} arrB An array.\r\n * @returns {Boolean} A boolean indicating whether or not both array contain the same elements.\r\n */\r\n static containsAll(arrA, arrB) {\r\n let containing = 0;\r\n for (let i = 0; i < arrA.length; i++) {\r\n for (let j = 0; j < arrB.length; j++) {\r\n if (arrA[i] === arrB[j]) {\r\n containing++;\r\n }\r\n }\r\n }\r\n\r\n return containing === arrB.length;\r\n }\r\n \r\n /**\r\n * Sort an array of atomic number information. Where the number is indicated as x, x.y, x.y.z, ...\r\n *\r\n * @param {Object[]} arr An array of vertex ids with their associated atomic numbers.\r\n * @param {Number} arr[].vertexId A vertex id.\r\n * @param {String} arr[].atomicNumber The atomic number associated with the vertex id.\r\n * @returns {Object[]} The array sorted by atomic number. Example of an array entry: { atomicNumber: 2, vertexId: 5 }.\r\n */\r\n static sortByAtomicNumberDesc(arr) {\r\n let map = arr.map(function(e, i) {\r\n return { index: i, value: e.atomicNumber.split('.').map(Number) };\r\n });\r\n\r\n map.sort(function(a, b) {\r\n let min = Math.min(b.value.length, a.value.length);\r\n let i = 0;\r\n \r\n while(i < min && b.value[i] === a.value[i]) {\r\n i++;\r\n }\r\n\r\n return i === min ? b.value.length - a.value.length : b.value[i] - a.value[i];\r\n });\r\n\r\n return map.map(function(e) {\r\n return arr[e.index];\r\n });\r\n }\r\n\r\n /**\r\n * Copies a an n-dimensional array.\r\n * \r\n * @param {Array} arr The array to be copied.\r\n * @returns {Array} The copy.\r\n */\r\n static deepCopy(arr) {\r\n let newArr = Array();\r\n\r\n for (let i = 0; i < arr.length; i++) {\r\n let item = arr[i];\r\n\r\n if (item instanceof Array) {\r\n newArr[i] = ArrayHelper.deepCopy(item);\r\n } else {\r\n newArr[i] = item;\r\n }\r\n }\r\n\r\n return newArr;\r\n }\r\n\r\n}\r\n\r\nmodule.exports = ArrayHelper;","//@ts-check\r\nconst ArrayHelper = require('./ArrayHelper')\r\nconst Vertex = require('./Vertex')\r\nconst Ring = require('./Ring')\r\n\r\n/** \r\n * A class representing an atom.\r\n * \r\n * @property {String} element The element symbol of this atom. Single-letter symbols are always uppercase. Examples: H, C, F, Br, Si, ...\r\n * @property {Boolean} drawExplicit A boolean indicating whether or not this atom is drawn explicitly (for example, a carbon atom). This overrides the default behaviour.\r\n * @property {Object[]} ringbonds An array containing the ringbond ids and bond types as specified in the original SMILE.\r\n * @property {String} branchBond The branch bond as defined in the SMILES.\r\n * @property {Number} ringbonds[].id The ringbond id as defined in the SMILES.\r\n * @property {String} ringbonds[].bondType The bond type of the ringbond as defined in the SMILES.\r\n * @property {Number[]} rings The ids of rings which contain this atom.\r\n * @property {String} bondType The bond type associated with this array. Examples: -, =, #, ...\r\n * @property {Boolean} isBridge A boolean indicating whether or not this atom is part of a bridge in a bridged ring (contained by the largest ring).\r\n * @property {Boolean} isBridgeNode A boolean indicating whether or not this atom is a bridge node (a member of the largest ring in a bridged ring which is connected to a bridge-atom).\r\n * @property {Number[]} originalRings Used to back up rings when they are replaced by a bridged ring.\r\n * @property {Number} bridgedRing The id of the bridged ring if the atom is part of a bridged ring.\r\n * @property {Number[]} anchoredRings The ids of the rings that are anchored to this atom. The centers of anchored rings are translated when this atom is translated.\r\n * @property {Object} bracket If this atom is defined as a bracket atom in the original SMILES, this object contains all the bracket information. Example: { hcount: {Number}, charge: ['--', '-', '+', '++'], isotope: {Number} }.\r\n * @property {Number} plane Specifies on which \"plane\" the atoms is in stereochemical deptictions (-1 back, 0 middle, 1 front).\r\n * @property {Object[]} attachedPseudoElements A map with containing information for pseudo elements or concatinated elements. The key is comprised of the element symbol and the hydrogen count.\r\n * @property {String} attachedPseudoElement[].element The element symbol.\r\n * @property {Number} attachedPseudoElement[].count The number of occurences that match the key.\r\n * @property {Number} attachedPseudoElement[].hyrogenCount The number of hydrogens attached to each atom matching the key.\r\n * @property {Boolean} hasAttachedPseudoElements A boolean indicating whether or not this attom will be drawn with an attached pseudo element or concatinated elements.\r\n * @property {Boolean} isDrawn A boolean indicating whether or not this atom is drawn. In contrast to drawExplicit, the bond is drawn neither.\r\n * @property {Boolean} isConnectedToRing A boolean indicating whether or not this atom is directly connected (but not a member of) a ring.\r\n * @property {String[]} neighbouringElements An array containing the element symbols of neighbouring atoms.\r\n * @property {Boolean} isPartOfAromaticRing A boolean indicating whether or not this atom is part of an explicitly defined aromatic ring. Example: c1ccccc1.\r\n * @property {Number} bondCount The number of bonds in which this atom is participating.\r\n * @property {String} chirality The chirality of this atom if it is a stereocenter (R or S).\r\n * @property {Number} priority The priority of this atom acording to the CIP rules, where 0 is the highest priority.\r\n * @property {Boolean} mainChain A boolean indicating whether or not this atom is part of the main chain (used for chirality).\r\n * @property {String} hydrogenDirection The direction of the hydrogen, either up or down. Only for stereocenters with and explicit hydrogen.\r\n * @property {Number} subtreeDepth The depth of the subtree coming from a stereocenter.\r\n */\r\nclass Atom {\r\n /**\r\n * The constructor of the class Atom.\r\n *\r\n * @param {String} element The one-letter code of the element.\r\n * @param {String} [bondType='-'] The type of the bond associated with this atom.\r\n */\r\n constructor(element, bondType = '-') {\r\n this.element = element.length === 1 ? element.toUpperCase() : element;\r\n this.drawExplicit = false;\r\n this.ringbonds = Array();\r\n this.rings = Array();\r\n this.bondType = bondType;\r\n this.branchBond = null;\r\n this.isBridge = false;\r\n this.isBridgeNode = false;\r\n this.originalRings = Array();\r\n this.bridgedRing = null;\r\n this.anchoredRings = Array();\r\n this.bracket = null;\r\n this.plane = 0;\r\n this.attachedPseudoElements = {};\r\n this.hasAttachedPseudoElements = false;\r\n this.isDrawn = true;\r\n this.isConnectedToRing = false;\r\n this.neighbouringElements = Array();\r\n this.isPartOfAromaticRing = element !== this.element;\r\n this.bondCount = 0;\r\n this.chirality = '';\r\n this.isStereoCenter = false;\r\n this.priority = 0;\r\n this.mainChain = false;\r\n this.hydrogenDirection = 'down';\r\n this.subtreeDepth = 1;\r\n this.hasHydrogen = false;\r\n }\r\n\r\n /**\r\n * Adds a neighbouring element to this atom.\r\n * \r\n * @param {String} element A string representing an element.\r\n */\r\n addNeighbouringElement(element) {\r\n this.neighbouringElements.push(element);\r\n }\r\n\r\n /**\r\n * Attaches a pseudo element (e.g. Ac) to the atom.\r\n * @param {String} element The element identifier (e.g. Br, C, ...).\r\n * @param {String} previousElement The element that is part of the main chain (not the terminals that are converted to the pseudo element or concatinated).\r\n * @param {Number} [hydrogenCount=0] The number of hydrogens for the element.\r\n * @param {Number} [charge=0] The charge for the element.\r\n */\r\n attachPseudoElement(element, previousElement, hydrogenCount = 0, charge = 0) {\r\n if (hydrogenCount === null) {\r\n hydrogenCount = 0;\r\n }\r\n\r\n if (charge === null) {\r\n charge = 0;\r\n }\r\n\r\n let key = hydrogenCount + element + charge;\r\n\r\n if (this.attachedPseudoElements[key]) {\r\n this.attachedPseudoElements[key].count += 1;\r\n } else {\r\n this.attachedPseudoElements[key] = {\r\n element: element,\r\n count: 1,\r\n hydrogenCount: hydrogenCount,\r\n previousElement: previousElement,\r\n charge: charge\r\n };\r\n }\r\n\r\n this.hasAttachedPseudoElements = true;\r\n }\r\n\r\n /**\r\n * Returns the attached pseudo elements sorted by hydrogen count (ascending).\r\n *\r\n * @returns {Object} The sorted attached pseudo elements.\r\n */\r\n getAttachedPseudoElements() {\r\n let ordered = {};\r\n let that = this;\r\n\r\n Object.keys(this.attachedPseudoElements).sort().forEach(function (key) {\r\n ordered[key] = that.attachedPseudoElements[key];\r\n });\r\n\r\n return ordered;\r\n }\r\n\r\n /**\r\n * Returns the number of attached pseudo elements.\r\n *\r\n * @returns {Number} The number of attached pseudo elements.\r\n */\r\n getAttachedPseudoElementsCount() {\r\n return Object.keys(this.attachedPseudoElements).length;\r\n }\r\n\r\n /**\r\n * Returns whether this atom is a heteroatom (not C and not H).\r\n *\r\n * @returns {Boolean} A boolean indicating whether this atom is a heteroatom.\r\n */\r\n isHeteroAtom() {\r\n return this.element !== 'C' && this.element !== 'H';\r\n }\r\n\r\n /**\r\n * Defines this atom as the anchor for a ring. When doing repositionings of the vertices and the vertex associated with this atom is moved, the center of this ring is moved as well.\r\n *\r\n * @param {Number} ringId A ring id.\r\n */\r\n addAnchoredRing(ringId) {\r\n if (!ArrayHelper.contains(this.anchoredRings, {\r\n value: ringId\r\n })) {\r\n this.anchoredRings.push(ringId);\r\n }\r\n }\r\n\r\n /**\r\n * Returns the number of ringbonds (breaks in rings to generate the MST of the smiles) within this atom is connected to.\r\n *\r\n * @returns {Number} The number of ringbonds this atom is connected to.\r\n */\r\n getRingbondCount() {\r\n return this.ringbonds.length;\r\n }\r\n\r\n /**\r\n * Backs up the current rings.\r\n */\r\n backupRings() {\r\n this.originalRings = Array(this.rings.length);\r\n\r\n for (let i = 0; i < this.rings.length; i++) {\r\n this.originalRings[i] = this.rings[i];\r\n }\r\n }\r\n\r\n /**\r\n * Restores the most recent backed up rings.\r\n */\r\n restoreRings() {\r\n this.rings = Array(this.originalRings.length);\r\n\r\n for (let i = 0; i < this.originalRings.length; i++) {\r\n this.rings[i] = this.originalRings[i];\r\n }\r\n }\r\n\r\n /**\r\n * Checks whether or not two atoms share a common ringbond id. A ringbond is a break in a ring created when generating the spanning tree of a structure.\r\n *\r\n * @param {Atom} atomA An atom.\r\n * @param {Atom} atomB An atom.\r\n * @returns {Boolean} A boolean indicating whether or not two atoms share a common ringbond.\r\n */\r\n haveCommonRingbond(atomA, atomB) {\r\n for (let i = 0; i < atomA.ringbonds.length; i++) {\r\n for (let j = 0; j < atomB.ringbonds.length; j++) {\r\n if (atomA.ringbonds[i].id == atomB.ringbonds[j].id) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Check whether or not the neighbouring elements of this atom equal the supplied array.\r\n * \r\n * @param {String[]} arr An array containing all the elements that are neighbouring this atom. E.g. ['C', 'O', 'O', 'N']\r\n * @returns {Boolean} A boolean indicating whether or not the neighbours match the supplied array of elements.\r\n */\r\n neighbouringElementsEqual(arr) {\r\n if (arr.length !== this.neighbouringElements.length) {\r\n return false;\r\n }\r\n\r\n arr.sort();\r\n this.neighbouringElements.sort();\r\n\r\n for (var i = 0; i < this.neighbouringElements.length; i++) {\r\n if (arr[i] !== this.neighbouringElements[i]) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Get the atomic number of this atom.\r\n * \r\n * @returns {Number} The atomic number of this atom.\r\n */\r\n getAtomicNumber() {\r\n return Atom.atomicNumbers[this.element];\r\n }\r\n\r\n /**\r\n * Get the maximum number of bonds for this atom.\r\n * \r\n * @returns {Number} The maximum number of bonds of this atom.\r\n */\r\n getMaxBonds() {\r\n return Atom.maxBonds[this.element];\r\n }\r\n\r\n /**\r\n * A map mapping element symbols to their maximum bonds.\r\n */\r\n static get maxBonds() {\r\n return {\r\n 'H': 1,\r\n 'C': 4,\r\n 'N': 3,\r\n 'O': 2,\r\n 'P': 3,\r\n 'S': 2,\r\n 'B': 3,\r\n 'F': 1,\r\n 'I': 1,\r\n 'Cl': 1,\r\n 'Br': 1\r\n };\r\n }\r\n\r\n /**\r\n * A map mapping element symbols to the atomic number.\r\n */\r\n static get atomicNumbers() {\r\n return {\r\n 'H': 1,\r\n 'He': 2,\r\n 'Li': 3,\r\n 'Be': 4,\r\n 'B': 5,\r\n 'b': 5,\r\n 'C': 6,\r\n 'c': 6,\r\n 'N': 7,\r\n 'n': 7,\r\n 'O': 8,\r\n 'o': 8,\r\n 'F': 9,\r\n 'Ne': 10,\r\n 'Na': 11,\r\n 'Mg': 12,\r\n 'Al': 13,\r\n 'Si': 14,\r\n 'P': 15,\r\n 'p': 15,\r\n 'S': 16,\r\n 's': 16,\r\n 'Cl': 17,\r\n 'Ar': 18,\r\n 'K': 19,\r\n 'Ca': 20,\r\n 'Sc': 21,\r\n 'Ti': 22,\r\n 'V': 23,\r\n 'Cr': 24,\r\n 'Mn': 25,\r\n 'Fe': 26,\r\n 'Co': 27,\r\n 'Ni': 28,\r\n 'Cu': 29,\r\n 'Zn': 30,\r\n 'Ga': 31,\r\n 'Ge': 32,\r\n 'As': 33,\r\n 'Se': 34,\r\n 'Br': 35,\r\n 'Kr': 36,\r\n 'Rb': 37,\r\n 'Sr': 38,\r\n 'Y': 39,\r\n 'Zr': 40,\r\n 'Nb': 41,\r\n 'Mo': 42,\r\n 'Tc': 43,\r\n 'Ru': 44,\r\n 'Rh': 45,\r\n 'Pd': 46,\r\n 'Ag': 47,\r\n 'Cd': 48,\r\n 'In': 49,\r\n 'Sn': 50,\r\n 'Sb': 51,\r\n 'Te': 52,\r\n 'I': 53,\r\n 'Xe': 54,\r\n 'Cs': 55,\r\n 'Ba': 56,\r\n 'La': 57,\r\n 'Ce': 58,\r\n 'Pr': 59,\r\n 'Nd': 60,\r\n 'Pm': 61,\r\n 'Sm': 62,\r\n 'Eu': 63,\r\n 'Gd': 64,\r\n 'Tb': 65,\r\n 'Dy': 66,\r\n 'Ho': 67,\r\n 'Er': 68,\r\n 'Tm': 69,\r\n 'Yb': 70,\r\n 'Lu': 71,\r\n 'Hf': 72,\r\n 'Ta': 73,\r\n 'W': 74,\r\n 'Re': 75,\r\n 'Os': 76,\r\n 'Ir': 77,\r\n 'Pt': 78,\r\n 'Au': 79,\r\n 'Hg': 80,\r\n 'Tl': 81,\r\n 'Pb': 82,\r\n 'Bi': 83,\r\n 'Po': 84,\r\n 'At': 85,\r\n 'Rn': 86,\r\n 'Fr': 87,\r\n 'Ra': 88,\r\n 'Ac': 89,\r\n 'Th': 90,\r\n 'Pa': 91,\r\n 'U': 92,\r\n 'Np': 93,\r\n 'Pu': 94,\r\n 'Am': 95,\r\n 'Cm': 96,\r\n 'Bk': 97,\r\n 'Cf': 98,\r\n 'Es': 99,\r\n 'Fm': 100,\r\n 'Md': 101,\r\n 'No': 102,\r\n 'Lr': 103,\r\n 'Rf': 104,\r\n 'Db': 105,\r\n 'Sg': 106,\r\n 'Bh': 107,\r\n 'Hs': 108,\r\n 'Mt': 109,\r\n 'Ds': 110,\r\n 'Rg': 111,\r\n 'Cn': 112,\r\n 'Uut': 113,\r\n 'Uuq': 114,\r\n 'Uup': 115,\r\n 'Uuh': 116,\r\n 'Uus': 117,\r\n 'Uuo': 118\r\n };\r\n }\r\n\r\n /**\r\n * A map mapping element symbols to the atomic mass.\r\n */\r\n static get mass() {\r\n return {\r\n 'H': 1,\r\n 'He': 2,\r\n 'Li': 3,\r\n 'Be': 4,\r\n 'B': 5,\r\n 'b': 5,\r\n 'C': 6,\r\n 'c': 6,\r\n 'N': 7,\r\n 'n': 7,\r\n 'O': 8,\r\n 'o': 8,\r\n 'F': 9,\r\n 'Ne': 10,\r\n 'Na': 11,\r\n 'Mg': 12,\r\n 'Al': 13,\r\n 'Si': 14,\r\n 'P': 15,\r\n 'p': 15,\r\n 'S': 16,\r\n 's': 16,\r\n 'Cl': 17,\r\n 'Ar': 18,\r\n 'K': 19,\r\n 'Ca': 20,\r\n 'Sc': 21,\r\n 'Ti': 22,\r\n 'V': 23,\r\n 'Cr': 24,\r\n 'Mn': 25,\r\n 'Fe': 26,\r\n 'Co': 27,\r\n 'Ni': 28,\r\n 'Cu': 29,\r\n 'Zn': 30,\r\n 'Ga': 31,\r\n 'Ge': 32,\r\n 'As': 33,\r\n 'Se': 34,\r\n 'Br': 35,\r\n 'Kr': 36,\r\n 'Rb': 37,\r\n 'Sr': 38,\r\n 'Y': 39,\r\n 'Zr': 40,\r\n 'Nb': 41,\r\n 'Mo': 42,\r\n 'Tc': 43,\r\n 'Ru': 44,\r\n 'Rh': 45,\r\n 'Pd': 46,\r\n 'Ag': 47,\r\n 'Cd': 48,\r\n 'In': 49,\r\n 'Sn': 50,\r\n 'Sb': 51,\r\n 'Te': 52,\r\n 'I': 53,\r\n 'Xe': 54,\r\n 'Cs': 55,\r\n 'Ba': 56,\r\n 'La': 57,\r\n 'Ce': 58,\r\n 'Pr': 59,\r\n 'Nd': 60,\r\n 'Pm': 61,\r\n 'Sm': 62,\r\n 'Eu': 63,\r\n 'Gd': 64,\r\n 'Tb': 65,\r\n 'Dy': 66,\r\n 'Ho': 67,\r\n 'Er': 68,\r\n 'Tm': 69,\r\n 'Yb': 70,\r\n 'Lu': 71,\r\n 'Hf': 72,\r\n 'Ta': 73,\r\n 'W': 74,\r\n 'Re': 75,\r\n 'Os': 76,\r\n 'Ir': 77,\r\n 'Pt': 78,\r\n 'Au': 79,\r\n 'Hg': 80,\r\n 'Tl': 81,\r\n 'Pb': 82,\r\n 'Bi': 83,\r\n 'Po': 84,\r\n 'At': 85,\r\n 'Rn': 86,\r\n 'Fr': 87,\r\n 'Ra': 88,\r\n 'Ac': 89,\r\n 'Th': 90,\r\n 'Pa': 91,\r\n 'U': 92,\r\n 'Np': 93,\r\n 'Pu': 94,\r\n 'Am': 95,\r\n 'Cm': 96,\r\n 'Bk': 97,\r\n 'Cf': 98,\r\n 'Es': 99,\r\n 'Fm': 100,\r\n 'Md': 101,\r\n 'No': 102,\r\n 'Lr': 103,\r\n 'Rf': 104,\r\n 'Db': 105,\r\n 'Sg': 106,\r\n 'Bh': 107,\r\n 'Hs': 108,\r\n 'Mt': 109,\r\n 'Ds': 110,\r\n 'Rg': 111,\r\n 'Cn': 112,\r\n 'Uut': 113,\r\n 'Uuq': 114,\r\n 'Uup': 115,\r\n 'Uuh': 116,\r\n 'Uus': 117,\r\n 'Uuo': 118\r\n };\r\n }\r\n}\r\n\r\nmodule.exports = Atom;","//@ts-check\r\nconst MathHelper = require('./MathHelper')\r\nconst Vector2 = require('./Vector2')\r\nconst Line = require('./Line')\r\nconst Vertex = require('./Vertex')\r\nconst Ring = require('./Ring')\r\n\r\n/** \r\n * A class wrapping a canvas element.\r\n * \r\n * @property {HTMLElement} canvas The HTML element for the canvas associated with this CanvasWrapper instance.\r\n * @property {CanvasRenderingContext2D} ctx The CanvasRenderingContext2D of the canvas associated with this CanvasWrapper instance.\r\n * @property {Object} colors The colors object as defined in the SmilesDrawer options.\r\n * @property {Object} opts The SmilesDrawer options.\r\n * @property {Number} drawingWidth The width of the canvas.\r\n * @property {Number} drawingHeight The height of the canvas.\r\n * @property {Number} offsetX The horizontal offset required for centering the drawing.\r\n * @property {Number} offsetY The vertical offset required for centering the drawing.\r\n * @property {Number} fontLarge The large font size in pt.\r\n * @property {Number} fontSmall The small font size in pt.\r\n */\r\nclass CanvasWrapper {\r\n /**\r\n * The constructor for the class CanvasWrapper.\r\n *\r\n * @param {(String|HTMLElement)} target The canvas id or the canvas HTMLElement.\r\n * @param {Object} theme A theme from the smiles drawer options.\r\n * @param {Object} options The smiles drawer options object.\r\n */\r\n constructor(target, theme, options) {\r\n if (typeof target === 'string' || target instanceof String) {\r\n this.canvas = document.getElementById(target);\r\n } else {\r\n this.canvas = target;\r\n }\r\n\r\n this.ctx = this.canvas.getContext('2d');\r\n this.colors = theme;\r\n this.opts = options;\r\n this.drawingWidth = 0.0;\r\n this.drawingHeight = 0.0;\r\n this.offsetX = 0.0;\r\n this.offsetY = 0.0;\r\n\r\n this.fontLarge = this.opts.fontSizeLarge + 'pt Helvetica, Arial, sans-serif';\r\n this.fontSmall = this.opts.fontSizeSmall + 'pt Helvetica, Arial, sans-serif';\r\n\r\n this.updateSize(this.opts.width, this.opts.height);\r\n\r\n this.ctx.font = this.fontLarge;\r\n this.hydrogenWidth = this.ctx.measureText('H').width;\r\n this.halfHydrogenWidth = this.hydrogenWidth / 2.0;\r\n this.halfBondThickness = this.opts.bondThickness / 2.0;\r\n\r\n // TODO: Find out why clear was here.\r\n // this.clear();\r\n }\r\n\r\n /**\r\n * Update the width and height of the canvas\r\n * \r\n * @param {Number} width \r\n * @param {Number} height \r\n */\r\n updateSize(width, height) {\r\n this.devicePixelRatio = window.devicePixelRatio || 1;\r\n this.backingStoreRatio = this.ctx.webkitBackingStorePixelRatio || this.ctx.mozBackingStorePixelRatio ||\r\n this.ctx.msBackingStorePixelRatio || this.ctx.oBackingStorePixelRatio ||\r\n this.ctx.backingStorePixelRatio || 1;\r\n this.ratio = this.devicePixelRatio / this.backingStoreRatio;\r\n\r\n if (this.ratio !== 1) {\r\n this.canvas.width = width * this.ratio;\r\n this.canvas.height = height * this.ratio;\r\n this.canvas.style.width = width + 'px';\r\n this.canvas.style.height = height + 'px';\r\n this.ctx.setTransform(this.ratio, 0, 0, this.ratio, 0, 0);\r\n } else {\r\n this.canvas.width = width * this.ratio;\r\n this.canvas.height = height * this.ratio;\r\n }\r\n }\r\n\r\n /**\r\n * Sets a provided theme.\r\n *\r\n * @param {Object} theme A theme from the smiles drawer options.\r\n */\r\n setTheme(theme) {\r\n this.colors = theme;\r\n }\r\n\r\n /**\r\n * Scale the canvas based on vertex positions.\r\n *\r\n * @param {Vertex[]} vertices An array of vertices containing the vertices associated with the current molecule.\r\n */\r\n scale(vertices) {\r\n // Figure out the final size of the image\r\n let maxX = -Number.MAX_VALUE;\r\n let maxY = -Number.MAX_VALUE;\r\n let minX = Number.MAX_VALUE;\r\n let minY = Number.MAX_VALUE;\r\n\r\n for (var i = 0; i < vertices.length; i++) {\r\n if (!vertices[i].value.isDrawn) {\r\n continue;\r\n }\r\n\r\n let p = vertices[i].position;\r\n\r\n if (maxX < p.x) maxX = p.x;\r\n if (maxY < p.y) maxY = p.y;\r\n if (minX > p.x) minX = p.x;\r\n if (minY > p.y) minY = p.y;\r\n }\r\n\r\n // Add padding\r\n var padding = this.opts.padding;\r\n maxX += padding;\r\n maxY += padding;\r\n minX -= padding;\r\n minY -= padding;\r\n\r\n this.drawingWidth = maxX - minX;\r\n this.drawingHeight = maxY - minY;\r\n\r\n var scaleX = this.canvas.offsetWidth / this.drawingWidth;\r\n var scaleY = this.canvas.offsetHeight / this.drawingHeight;\r\n\r\n var scale = (scaleX < scaleY) ? scaleX : scaleY;\r\n\r\n this.ctx.scale(scale, scale);\r\n\r\n this.offsetX = -minX;\r\n this.offsetY = -minY;\r\n\r\n // Center\r\n if (scaleX < scaleY) {\r\n this.offsetY += this.canvas.offsetHeight / (2.0 * scale) - this.drawingHeight / 2.0;\r\n } else {\r\n this.offsetX += this.canvas.offsetWidth / (2.0 * scale) - this.drawingWidth / 2.0;\r\n }\r\n }\r\n\r\n /**\r\n * Resets the transform of the canvas.\r\n */\r\n reset() {\r\n this.ctx.setTransform(1, 0, 0, 1, 0, 0);\r\n }\r\n\r\n /**\r\n * Returns the hex code of a color associated with a key from the current theme.\r\n *\r\n * @param {String} key The color key in the theme (e.g. C, N, BACKGROUND, ...).\r\n * @returns {String} A color hex value.\r\n */\r\n getColor(key) {\r\n key = key.toUpperCase();\r\n\r\n if (key in this.colors) {\r\n return this.colors[key];\r\n }\r\n\r\n return this.colors['C'];\r\n }\r\n\r\n /**\r\n * Draws a circle to a canvas context.\r\n * @param {Number} x The x coordinate of the circles center.\r\n * @param {Number} y The y coordinate of the circles center.\r\n * @param {Number} radius The radius of the circle\r\n * @param {String} color A hex encoded color.\r\n * @param {Boolean} [fill=true] Whether to fill or stroke the circle.\r\n * @param {Boolean} [debug=false] Draw in debug mode.\r\n * @param {String} [debugText=''] A debug message.\r\n */\r\n drawCircle(x, y, radius, color, fill = true, debug = false, debugText = '') {\r\n let ctx = this.ctx;\r\n let offsetX = this.offsetX;\r\n let offsetY = this.offsetY;\r\n\r\n ctx.save();\r\n ctx.lineWidth = 1.5;\r\n ctx.beginPath();\r\n ctx.arc(x + offsetX, y + offsetY, radius, 0, MathHelper.twoPI, true);\r\n ctx.closePath();\r\n\r\n if (debug) {\r\n if (fill) {\r\n ctx.fillStyle = '#f00';\r\n ctx.fill();\r\n } else {\r\n ctx.strokeStyle = '#f00';\r\n ctx.stroke();\r\n }\r\n\r\n this.drawDebugText(x, y, debugText);\r\n } else {\r\n if (fill) {\r\n ctx.fillStyle = color;\r\n ctx.fill();\r\n } else {\r\n ctx.strokeStyle = color;\r\n ctx.stroke();\r\n }\r\n }\r\n\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draw a line to a canvas.\r\n *\r\n * @param {Line} line A line.\r\n * @param {Boolean} [dashed=false] Whether or not the line is dashed.\r\n * @param {Number} [alpha=1.0] The alpha value of the color.\r\n */\r\n drawLine(line, dashed = false, alpha = 1.0) {\r\n let ctx = this.ctx;\r\n let offsetX = this.offsetX;\r\n let offsetY = this.offsetY;\r\n\r\n // Add a shadow behind the line\r\n let shortLine = line.clone().shorten(4.0);\r\n\r\n let l = shortLine.getLeftVector().clone();\r\n let r = shortLine.getRightVector().clone();\r\n\r\n l.x += offsetX;\r\n l.y += offsetY;\r\n\r\n r.x += offsetX;\r\n r.y += offsetY;\r\n\r\n // Draw the \"shadow\"\r\n if (!dashed) {\r\n ctx.save();\r\n ctx.globalCompositeOperation = 'destination-out';\r\n ctx.beginPath();\r\n ctx.moveTo(l.x, l.y);\r\n ctx.lineTo(r.x, r.y);\r\n ctx.lineCap = 'round';\r\n ctx.lineWidth = this.opts.bondThickness + 1.2;\r\n ctx.strokeStyle = this.getColor('BACKGROUND');\r\n ctx.stroke();\r\n ctx.globalCompositeOperation = 'source-over';\r\n ctx.restore();\r\n }\r\n\r\n l = line.getLeftVector().clone();\r\n r = line.getRightVector().clone();\r\n\r\n l.x += offsetX;\r\n l.y += offsetY;\r\n\r\n r.x += offsetX;\r\n r.y += offsetY;\r\n\r\n ctx.save();\r\n ctx.beginPath();\r\n ctx.moveTo(l.x, l.y);\r\n ctx.lineTo(r.x, r.y);\r\n ctx.lineCap = 'round';\r\n ctx.lineWidth = this.opts.bondThickness;\r\n\r\n let gradient = this.ctx.createLinearGradient(l.x, l.y, r.x, r.y);\r\n gradient.addColorStop(0.4, this.getColor(line.getLeftElement()) ||\r\n this.getColor('C'));\r\n gradient.addColorStop(0.6, this.getColor(line.getRightElement()) ||\r\n this.getColor('C'));\r\n\r\n if (dashed) {\r\n ctx.setLineDash([1, 1.5]);\r\n ctx.lineWidth = this.opts.bondThickness / 1.5;\r\n }\r\n\r\n if (alpha < 1.0) {\r\n ctx.globalAlpha = alpha;\r\n }\r\n\r\n ctx.strokeStyle = gradient;\r\n\r\n ctx.stroke();\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draw a wedge on the canvas.\r\n *\r\n * @param {Line} line A line.\r\n * @param {Number} width The wedge width.\r\n */\r\n drawWedge(line, width = 1.0) {\r\n if (isNaN(line.from.x) || isNaN(line.from.y) ||\r\n isNaN(line.to.x) || isNaN(line.to.y)) {\r\n return;\r\n }\r\n\r\n let ctx = this.ctx;\r\n let offsetX = this.offsetX;\r\n let offsetY = this.offsetY;\r\n\r\n // Add a shadow behind the line\r\n let shortLine = line.clone().shorten(5.0);\r\n\r\n let l = shortLine.getLeftVector().clone();\r\n let r = shortLine.getRightVector().clone();\r\n\r\n l.x += offsetX;\r\n l.y += offsetY;\r\n\r\n r.x += offsetX;\r\n r.y += offsetY;\r\n\r\n l = line.getLeftVector().clone();\r\n r = line.getRightVector().clone();\r\n\r\n l.x += offsetX;\r\n l.y += offsetY;\r\n\r\n r.x += offsetX;\r\n r.y += offsetY;\r\n\r\n ctx.save();\r\n\r\n let normals = Vector2.normals(l, r);\r\n\r\n normals[0].normalize();\r\n normals[1].normalize();\r\n\r\n let isRightChiralCenter = line.getRightChiral();\r\n\r\n let start = l;\r\n let end = r;\r\n\r\n if (isRightChiralCenter) {\r\n start = r;\r\n end = l;\r\n }\r\n\r\n let t = Vector2.add(start, Vector2.multiplyScalar(normals[0], this.halfBondThickness));\r\n let u = Vector2.add(end, Vector2.multiplyScalar(normals[0], 1.5 + this.halfBondThickness));\r\n let v = Vector2.add(end, Vector2.multiplyScalar(normals[1], 1.5 + this.halfBondThickness));\r\n let w = Vector2.add(start, Vector2.multiplyScalar(normals[1], this.halfBondThickness));\r\n\r\n ctx.beginPath();\r\n ctx.moveTo(t.x, t.y);\r\n ctx.lineTo(u.x, u.y);\r\n ctx.lineTo(v.x, v.y);\r\n ctx.lineTo(w.x, w.y);\r\n\r\n let gradient = this.ctx.createRadialGradient(r.x, r.y, this.opts.bondLength, r.x, r.y, 0);\r\n gradient.addColorStop(0.4, this.getColor(line.getLeftElement()) ||\r\n this.getColor('C'));\r\n gradient.addColorStop(0.6, this.getColor(line.getRightElement()) ||\r\n this.getColor('C'));\r\n\r\n ctx.fillStyle = gradient;\r\n\r\n ctx.fill();\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draw a dashed wedge on the canvas.\r\n *\r\n * @param {Line} line A line.\r\n */\r\n drawDashedWedge(line) {\r\n if (isNaN(line.from.x) || isNaN(line.from.y) ||\r\n isNaN(line.to.x) || isNaN(line.to.y)) {\r\n return;\r\n }\r\n\r\n let ctx = this.ctx;\r\n let offsetX = this.offsetX;\r\n let offsetY = this.offsetY;\r\n\r\n let l = line.getLeftVector().clone();\r\n let r = line.getRightVector().clone();\r\n\r\n l.x += offsetX;\r\n l.y += offsetY;\r\n\r\n r.x += offsetX;\r\n r.y += offsetY;\r\n\r\n ctx.save();\r\n\r\n let normals = Vector2.normals(l, r);\r\n\r\n normals[0].normalize();\r\n normals[1].normalize();\r\n\r\n\r\n let isRightChiralCenter = line.getRightChiral();\r\n\r\n let start;\r\n let end;\r\n let sStart;\r\n let sEnd;\r\n\r\n let shortLine = line.clone();\r\n\r\n if (isRightChiralCenter) {\r\n start = r;\r\n end = l;\r\n\r\n shortLine.shortenRight(1.0);\r\n\r\n sStart = shortLine.getRightVector().clone();\r\n sEnd = shortLine.getLeftVector().clone();\r\n } else {\r\n start = l;\r\n end = r;\r\n\r\n shortLine.shortenLeft(1.0);\r\n\r\n sStart = shortLine.getLeftVector().clone();\r\n sEnd = shortLine.getRightVector().clone();\r\n }\r\n\r\n sStart.x += offsetX;\r\n sStart.y += offsetY;\r\n sEnd.x += offsetX;\r\n sEnd.y += offsetY;\r\n\r\n let dir = Vector2.subtract(end, start).normalize();\r\n ctx.strokeStyle = this.getColor('C');\r\n ctx.lineCap = 'round';\r\n ctx.lineWidth = this.opts.bondThickness;\r\n ctx.beginPath();\r\n let length = line.getLength();\r\n let step = 1.25 / (length / (this.opts.bondThickness * 3.0));\r\n\r\n let changed = false;\r\n for (var t = 0.0; t < 1.0; t += step) {\r\n let to = Vector2.multiplyScalar(dir, t * length);\r\n let startDash = Vector2.add(start, to);\r\n let width = 1.5 * t;\r\n let dashOffset = Vector2.multiplyScalar(normals[0], width);\r\n\r\n if (!changed && t > 0.5) {\r\n ctx.stroke();\r\n ctx.beginPath();\r\n ctx.strokeStyle = this.getColor(line.getRightElement()) || this.getColor('C');\r\n changed = true;\r\n }\r\n \r\n startDash.subtract(dashOffset);\r\n ctx.moveTo(startDash.x, startDash.y);\r\n startDash.add(Vector2.multiplyScalar(dashOffset, 2.0));\r\n ctx.lineTo(startDash.x, startDash.y);\r\n }\r\n\r\n ctx.stroke();\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draws a debug text message at a given position\r\n *\r\n * @param {Number} x The x coordinate.\r\n * @param {Number} y The y coordinate.\r\n * @param {String} text The debug text.\r\n */\r\n drawDebugText(x, y, text) {\r\n let ctx = this.ctx;\r\n\r\n ctx.save();\r\n ctx.font = '5px Droid Sans, sans-serif';\r\n ctx.textAlign = 'start';\r\n ctx.textBaseline = 'top';\r\n ctx.fillStyle = '#ff0000';\r\n ctx.fillText(text, x + this.offsetX, y + this.offsetY);\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draw a ball to the canvas.\r\n *\r\n * @param {Number} x The x position of the text.\r\n * @param {Number} y The y position of the text.\r\n * @param {String} elementName The name of the element (single-letter).\r\n */\r\n drawBall(x, y, elementName) {\r\n let ctx = this.ctx;\r\n\r\n ctx.save();\r\n ctx.beginPath();\r\n ctx.arc(x + this.offsetX, y + this.offsetY, this.opts.bondLength / 4.5, 0, MathHelper.twoPI, false);\r\n ctx.fillStyle = this.getColor(elementName);\r\n ctx.fill();\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draw a point to the canvas.\r\n *\r\n * @param {Number} x The x position of the point.\r\n * @param {Number} y The y position of the point.\r\n * @param {String} elementName The name of the element (single-letter).\r\n */\r\n drawPoint(x, y, elementName) {\r\n let ctx = this.ctx;\r\n let offsetX = this.offsetX;\r\n let offsetY = this.offsetY;\r\n\r\n ctx.save();\r\n ctx.globalCompositeOperation = 'destination-out';\r\n ctx.beginPath();\r\n ctx.arc(x + offsetX, y + offsetY, 1.5, 0, MathHelper.twoPI, true);\r\n ctx.closePath();\r\n ctx.fill();\r\n ctx.globalCompositeOperation = 'source-over';\r\n\r\n ctx.beginPath();\r\n ctx.arc(x + this.offsetX, y + this.offsetY, 0.75, 0, MathHelper.twoPI, false);\r\n ctx.fillStyle = this.getColor(elementName);\r\n ctx.fill();\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Draw a text to the canvas.\r\n *\r\n * @param {Number} x The x position of the text.\r\n * @param {Number} y The y position of the text.\r\n * @param {String} elementName The name of the element (single-letter).\r\n * @param {Number} hydrogens The number of hydrogen atoms.\r\n * @param {String} direction The direction of the text in relation to the associated vertex.\r\n * @param {Boolean} isTerminal A boolean indicating whether or not the vertex is terminal.\r\n * @param {Number} charge The charge of the atom.\r\n * @param {Number} isotope The isotope number.\r\n * @param {Object} attachedPseudoElement A map with containing information for pseudo elements or concatinated elements. The key is comprised of the element symbol and the hydrogen count.\r\n * @param {String} attachedPseudoElement.element The element symbol.\r\n * @param {Number} attachedPseudoElement.count The number of occurences that match the key.\r\n * @param {Number} attachedPseudoElement.hyrogenCount The number of hydrogens attached to each atom matching the key.\r\n */\r\n drawText(x, y, elementName, hydrogens, direction, isTerminal, charge, isotope, attachedPseudoElement = {}) {\r\n let ctx = this.ctx;\r\n let offsetX = this.offsetX;\r\n let offsetY = this.offsetY;\r\n\r\n ctx.save();\r\n\r\n ctx.textAlign = 'start';\r\n ctx.textBaseline = 'alphabetic';\r\n\r\n let pseudoElementHandled = false;\r\n\r\n // Charge\r\n let chargeText = ''\r\n let chargeWidth = 0;\r\n\r\n if (charge) {\r\n chargeText = this.getChargeText(charge);\r\n\r\n ctx.font = this.fontSmall;\r\n chargeWidth = ctx.measureText(chargeText).width;\r\n }\r\n\r\n let isotopeText = '0';\r\n let isotopeWidth = 0;\r\n\r\n if (isotope > 0) {\r\n isotopeText = isotope.toString();\r\n ctx.font = this.fontSmall;\r\n isotopeWidth = ctx.measureText(isotopeText).width;\r\n }\r\n\r\n\r\n // TODO: Better handle exceptions\r\n // Exception for nitro (draw nitro as NO2 instead of N+O-O)\r\n if (charge === 1 && elementName === 'N' && attachedPseudoElement.hasOwnProperty('0O') && \r\n attachedPseudoElement.hasOwnProperty('0O-1')) {\r\n attachedPseudoElement = { '0O': { element: 'O', count: 2, hydrogenCount: 0, previousElement: 'C', charge: '' } }\r\n charge = 0;\r\n }\r\n\r\n\r\n ctx.font = this.fontLarge;\r\n ctx.fillStyle = this.getColor('BACKGROUND');\r\n\r\n let dim = ctx.measureText(elementName);\r\n\r\n dim.totalWidth = dim.width + chargeWidth;\r\n dim.height = parseInt(this.fontLarge, 10);\r\n\r\n let r = (dim.width > this.opts.fontSizeLarge) ? dim.width : this.opts.fontSizeLarge;\r\n r /= 1.5;\r\n\r\n ctx.globalCompositeOperation = 'destination-out';\r\n ctx.beginPath();\r\n ctx.arc(x + offsetX, y + offsetY, r, 0, MathHelper.twoPI, true);\r\n ctx.closePath();\r\n ctx.fill();\r\n ctx.globalCompositeOperation = 'source-over';\r\n\r\n let cursorPos = -dim.width / 2.0;\r\n let cursorPosLeft = -dim.width / 2.0;\r\n\r\n ctx.fillStyle = this.getColor(elementName);\r\n ctx.fillText(elementName, x + offsetX + cursorPos, y + this.opts.halfFontSizeLarge + offsetY);\r\n cursorPos += dim.width;\r\n\r\n if (charge) {\r\n ctx.font = this.fontSmall;\r\n ctx.fillText(chargeText, x + offsetX + cursorPos, y - this.opts.fifthFontSizeSmall + offsetY);\r\n cursorPos += chargeWidth;\r\n }\r\n\r\n if (isotope > 0) {\r\n ctx.font = this.fontSmall;\r\n ctx.fillText(isotopeText, x + offsetX + cursorPosLeft - isotopeWidth, y - this.opts.fifthFontSizeSmall + offsetY);\r\n cursorPosLeft -= isotopeWidth;\r\n }\r\n\r\n ctx.font = this.fontLarge;\r\n\r\n let hydrogenWidth = 0;\r\n let hydrogenCountWidth = 0;\r\n\r\n if (hydrogens === 1) {\r\n let hx = x + offsetX;\r\n let hy = y + offsetY + this.opts.halfFontSizeLarge;\r\n\r\n hydrogenWidth = this.hydrogenWidth;\r\n cursorPosLeft -= hydrogenWidth;\r\n\r\n if (direction === 'left') {\r\n hx += cursorPosLeft;\r\n } else if (direction === 'right') {\r\n hx += cursorPos;\r\n } else if (direction === 'up' && isTerminal) {\r\n hx += cursorPos;\r\n } else if (direction === 'down' && isTerminal) {\r\n hx += cursorPos;\r\n } else if (direction === 'up' && !isTerminal) {\r\n hy -= this.opts.fontSizeLarge + this.opts.quarterFontSizeLarge;\r\n hx -= this.halfHydrogenWidth;\r\n } else if (direction === 'down' && !isTerminal) {\r\n hy += this.opts.fontSizeLarge + this.opts.quarterFontSizeLarge;\r\n hx -= this.halfHydrogenWidth;\r\n }\r\n\r\n ctx.fillText('H', hx, hy);\r\n\r\n cursorPos += hydrogenWidth;\r\n } else if (hydrogens > 1) {\r\n let hx = x + offsetX;\r\n let hy = y + offsetY + this.opts.halfFontSizeLarge;\r\n\r\n hydrogenWidth = this.hydrogenWidth;\r\n ctx.font = this.fontSmall;\r\n hydrogenCountWidth = ctx.measureText(hydrogens).width;\r\n cursorPosLeft -= hydrogenWidth + hydrogenCountWidth;\r\n\r\n if (direction === 'left') {\r\n hx += cursorPosLeft;\r\n } else if (direction === 'right') {\r\n hx += cursorPos;\r\n } else if (direction === 'up' && isTerminal) {\r\n hx += cursorPos;\r\n } else if (direction === 'down' && isTerminal) {\r\n hx += cursorPos;\r\n } else if (direction === 'up' && !isTerminal) {\r\n hy -= this.opts.fontSizeLarge + this.opts.quarterFontSizeLarge;\r\n hx -= this.halfHydrogenWidth;\r\n } else if (direction === 'down' && !isTerminal) {\r\n hy += this.opts.fontSizeLarge + this.opts.quarterFontSizeLarge;\r\n hx -= this.halfHydrogenWidth;\r\n }\r\n\r\n ctx.font = this.fontLarge;\r\n ctx.fillText('H', hx, hy)\r\n\r\n ctx.font = this.fontSmall;\r\n ctx.fillText(hydrogens, hx + this.halfHydrogenWidth + hydrogenCountWidth, hy + this.opts.fifthFontSizeSmall);\r\n\r\n cursorPos += hydrogenWidth + this.halfHydrogenWidth + hydrogenCountWidth;\r\n }\r\n\r\n if (pseudoElementHandled) {\r\n ctx.restore();\r\n return;\r\n }\r\n\r\n for (let key in attachedPseudoElement) {\r\n if (!attachedPseudoElement.hasOwnProperty(key)) {\r\n continue;\r\n }\r\n\r\n let openParenthesisWidth = 0;\r\n let closeParenthesisWidth = 0;\r\n\r\n let element = attachedPseudoElement[key].element;\r\n let elementCount = attachedPseudoElement[key].count;\r\n let hydrogenCount = attachedPseudoElement[key].hydrogenCount;\r\n let elementCharge = attachedPseudoElement[key].charge;\r\n\r\n ctx.font = this.fontLarge;\r\n\r\n if (elementCount > 1 && hydrogenCount > 0) {\r\n openParenthesisWidth = ctx.measureText('(').width;\r\n closeParenthesisWidth = ctx.measureText(')').width;\r\n }\r\n\r\n let elementWidth = ctx.measureText(element).width;\r\n let elementCountWidth = 0;\r\n\r\n let elementChargeText = '';\r\n let elementChargeWidth = 0;\r\n\r\n hydrogenWidth = 0;\r\n\r\n if (hydrogenCount > 0) {\r\n hydrogenWidth = this.hydrogenWidth;\r\n }\r\n\r\n ctx.font = this.fontSmall;\r\n\r\n if (elementCount > 1) {\r\n elementCountWidth = ctx.measureText(elementCount).width;\r\n }\r\n\r\n if (elementCharge !== 0) {\r\n elementChargeText = this.getChargeText(elementCharge);\r\n elementChargeWidth = ctx.measureText(elementChargeText).width;\r\n }\r\n\r\n hydrogenCountWidth = 0;\r\n\r\n if (hydrogenCount > 1) {\r\n hydrogenCountWidth = ctx.measureText(hydrogenCount).width;\r\n }\r\n\r\n ctx.font = this.fontLarge;\r\n\r\n let hx = x + offsetX;\r\n let hy = y + offsetY + this.opts.halfFontSizeLarge;\r\n\r\n ctx.fillStyle = this.getColor(element);\r\n\r\n if (elementCount > 0) {\r\n cursorPosLeft -= elementCountWidth;\r\n }\r\n\r\n if (elementCount > 1 && hydrogenCount > 0) {\r\n if (direction === 'left') {\r\n cursorPosLeft -= closeParenthesisWidth;\r\n ctx.fillText(')', hx + cursorPosLeft, hy);\r\n } else {\r\n ctx.fillText('(', hx + cursorPos, hy);\r\n cursorPos += openParenthesisWidth;\r\n }\r\n }\r\n\r\n if (direction === 'left') {\r\n cursorPosLeft -= elementWidth;\r\n ctx.fillText(element, hx + cursorPosLeft, hy)\r\n } else {\r\n ctx.fillText(element, hx + cursorPos, hy)\r\n cursorPos += elementWidth;\r\n }\r\n\r\n if (hydrogenCount > 0) {\r\n if (direction === 'left') {\r\n cursorPosLeft -= hydrogenWidth + hydrogenCountWidth;\r\n ctx.fillText('H', hx + cursorPosLeft, hy)\r\n\r\n if (hydrogenCount > 1) {\r\n ctx.font = this.fontSmall;\r\n ctx.fillText(hydrogenCount, hx + cursorPosLeft + hydrogenWidth, hy + this.opts.fifthFontSizeSmall);\r\n }\r\n } else {\r\n ctx.fillText('H', hx + cursorPos, hy)\r\n cursorPos += hydrogenWidth;\r\n\r\n if (hydrogenCount > 1) {\r\n ctx.font = this.fontSmall;\r\n ctx.fillText(hydrogenCount, hx + cursorPos, hy + this.opts.fifthFontSizeSmall);\r\n cursorPos += hydrogenCountWidth;\r\n }\r\n }\r\n }\r\n\r\n ctx.font = this.fontLarge;\r\n\r\n if (elementCount > 1 && hydrogenCount > 0) {\r\n if (direction === 'left') {\r\n cursorPosLeft -= openParenthesisWidth;\r\n ctx.fillText('(', hx + cursorPosLeft, hy);\r\n } else {\r\n ctx.fillText(')', hx + cursorPos, hy);\r\n cursorPos += closeParenthesisWidth;\r\n }\r\n }\r\n\r\n ctx.font = this.fontSmall;\r\n\r\n if (elementCount > 1) {\r\n if (direction === 'left') {\r\n ctx.fillText(elementCount, hx + cursorPosLeft +\r\n openParenthesisWidth + closeParenthesisWidth + hydrogenWidth +\r\n hydrogenCountWidth + elementWidth, hy + this.opts.fifthFontSizeSmall);\r\n } else {\r\n ctx.fillText(elementCount, hx + cursorPos, hy + this.opts.fifthFontSizeSmall);\r\n cursorPos += elementCountWidth;\r\n }\r\n }\r\n\r\n if (elementCharge !== 0) {\r\n if (direction === 'left') {\r\n ctx.fillText(elementChargeText, hx + cursorPosLeft +\r\n openParenthesisWidth + closeParenthesisWidth + hydrogenWidth +\r\n hydrogenCountWidth + elementWidth, y - this.opts.fifthFontSizeSmall + offsetY);\r\n } else {\r\n ctx.fillText(elementChargeText, hx + cursorPos, y - this.opts.fifthFontSizeSmall + offsetY);\r\n cursorPos += elementChargeWidth;\r\n }\r\n }\r\n }\r\n\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Translate the integer indicating the charge to the appropriate text.\r\n * @param {Number} charge The integer indicating the charge.\r\n * @returns {String} A string representing a charge.\r\n */\r\n getChargeText(charge) {\r\n if (charge === 1) {\r\n return '+'\r\n } else if (charge === 2) {\r\n return '2+';\r\n } else if (charge === -1) {\r\n return '-';\r\n } else if (charge === -2) {\r\n return '2-';\r\n } else {\r\n return '';\r\n }\r\n }\r\n\r\n /**\r\n * Draws a dubug dot at a given coordinate and adds text.\r\n *\r\n * @param {Number} x The x coordinate.\r\n * @param {Number} y The y coordindate.\r\n * @param {String} [debugText=''] A string.\r\n * @param {String} [color='#f00'] A color in hex form.\r\n */\r\n drawDebugPoint(x, y, debugText = '', color = '#f00') {\r\n this.drawCircle(x, y, 2, color, true, true, debugText);\r\n }\r\n\r\n /**\r\n * Draws a ring inside a provided ring, indicating aromaticity.\r\n *\r\n * @param {Ring} ring A ring.\r\n */\r\n drawAromaticityRing(ring) {\r\n let ctx = this.ctx;\r\n let radius = MathHelper.apothemFromSideLength(this.opts.bondLength, ring.getSize());\r\n\r\n ctx.save();\r\n ctx.strokeStyle = this.getColor('C');\r\n ctx.lineWidth = this.opts.bondThickness;\r\n ctx.beginPath();\r\n ctx.arc(ring.center.x + this.offsetX, ring.center.y + this.offsetY,\r\n radius - this.opts.bondSpacing, 0, Math.PI * 2, true);\r\n ctx.closePath();\r\n ctx.stroke();\r\n ctx.restore();\r\n }\r\n\r\n /**\r\n * Clear the canvas.\r\n *\r\n */\r\n clear() {\r\n this.ctx.clearRect(0, 0, this.canvas.offsetWidth, this.canvas.offsetHeight);\r\n }\r\n\r\n}\r\n\r\nmodule.exports = CanvasWrapper;","//@ts-check\r\nconst MathHelper = require('./MathHelper')\r\nconst ArrayHelper = require('./ArrayHelper')\r\nconst Vector2 = require('./Vector2')\r\nconst Line = require('./Line')\r\nconst Vertex = require('./Vertex')\r\nconst Edge = require('./Edge')\r\nconst Atom = require('./Atom')\r\nconst Ring = require('./Ring')\r\nconst RingConnection = require('./RingConnection')\r\nconst CanvasWrapper = require('./CanvasWrapper')\r\nconst Graph = require('./Graph')\r\nconst SSSR = require('./SSSR')\r\n\r\n/** \r\n * The main class of the application representing the smiles drawer \r\n * \r\n * @property {Graph} graph The graph associated with this SmilesDrawer.Drawer instance.\r\n * @property {Number} ringIdCounter An internal counter to keep track of ring ids.\r\n * @property {Number} ringConnectionIdCounter An internal counter to keep track of ring connection ids.\r\n * @property {CanvasWrapper} canvasWrapper The CanvasWrapper associated with this SmilesDrawer.Drawer instance.\r\n * @property {Number} totalOverlapScore The current internal total overlap score.\r\n * @property {Object} defaultOptions The default options.\r\n * @property {Object} opts The merged options.\r\n * @property {Object} theme The current theme.\r\n */\r\nclass Drawer {\r\n /**\r\n * The constructor for the class SmilesDrawer.\r\n *\r\n * @param {Object} options An object containing custom values for different options. It is merged with the default options.\r\n */\r\n constructor(options) {\r\n this.graph = null;\r\n this.doubleBondConfigCount = 0;\r\n this.doubleBondConfig = null;\r\n this.ringIdCounter = 0;\r\n this.ringConnectionIdCounter = 0;\r\n this.canvasWrapper = null;\r\n this.totalOverlapScore = 0;\r\n\r\n this.defaultOptions = {\r\n width: 500,\r\n height: 500,\r\n bondThickness: 0.6,\r\n bondLength: 15,\r\n shortBondLength: 0.85,\r\n bondSpacing: 0.18 * 15,\r\n atomVisualization: 'default',\r\n isomeric: true,\r\n debug: false,\r\n terminalCarbons: false,\r\n explicitHydrogens: false,\r\n overlapSensitivity: 0.42,\r\n overlapResolutionIterations: 1,\r\n compactDrawing: true,\r\n fontSizeLarge: 5,\r\n fontSizeSmall: 3,\r\n padding: 20.0,\r\n experimentalSSSR: false,\r\n kkThreshold: 0.1,\r\n kkInnerThreshold: 0.1,\r\n kkMaxIteration: 20000,\r\n kkMaxInnerIteration: 50,\r\n kkMaxEnergy: 1e9,\r\n themes: {\r\n dark: {\r\n C: '#fff',\r\n O: '#e74c3c',\r\n N: '#3498db',\r\n F: '#27ae60',\r\n CL: '#16a085',\r\n BR: '#d35400',\r\n I: '#8e44ad',\r\n P: '#d35400',\r\n S: '#f1c40f',\r\n B: '#e67e22',\r\n SI: '#e67e22',\r\n H: '#fff',\r\n BACKGROUND: '#141414'\r\n },\r\n light: {\r\n C: '#222',\r\n O: '#e74c3c',\r\n N: '#3498db',\r\n F: '#27ae60',\r\n CL: '#16a085',\r\n BR: '#d35400',\r\n I: '#8e44ad',\r\n P: '#d35400',\r\n S: '#f1c40f',\r\n B: '#e67e22',\r\n SI: '#e67e22',\r\n H: '#222',\r\n BACKGROUND: '#fff'\r\n }\r\n }\r\n };\r\n\r\n this.opts = this.extend(true, this.defaultOptions, options);\r\n this.opts.halfBondSpacing = this.opts.bondSpacing / 2.0;\r\n this.opts.bondLengthSq = this.opts.bondLength * this.opts.bondLength;\r\n this.opts.halfFontSizeLarge = this.opts.fontSizeLarge / 2.0;\r\n this.opts.quarterFontSizeLarge = this.opts.fontSizeLarge / 4.0;\r\n this.opts.fifthFontSizeSmall = this.opts.fontSizeSmall / 5.0;\r\n\r\n // Set the default theme.\r\n this.theme = this.opts.themes.dark;\r\n }\r\n\r\n /**\r\n * A helper method to extend the default options with user supplied ones.\r\n */\r\n extend() {\r\n let that = this;\r\n let extended = {};\r\n let deep = false;\r\n let i = 0;\r\n let length = arguments.length;\r\n\r\n if (Object.prototype.toString.call(arguments[0]) === '[object Boolean]') {\r\n deep = arguments[0];\r\n i++;\r\n }\r\n\r\n let merge = function (obj) {\r\n for (var prop in obj) {\r\n if (Object.prototype.hasOwnProperty.call(obj, prop)) {\r\n if (deep && Object.prototype.toString.call(obj[prop]) === '[object Object]') {\r\n extended[prop] = that.extend(true, extended[prop], obj[prop]);\r\n } else {\r\n extended[prop] = obj[prop];\r\n }\r\n }\r\n }\r\n };\r\n\r\n for (; i < length; i++) {\r\n let obj = arguments[i];\r\n merge(obj);\r\n }\r\n\r\n return extended;\r\n };\r\n\r\n\r\n /**\r\n * Draws the parsed smiles data to a canvas element.\r\n *\r\n * @param {Object} data The tree returned by the smiles parser.\r\n * @param {(String|HTMLElement)} target The id of the HTML canvas element the structure is drawn to - or the element itself.\r\n * @param {String} themeName='dark' The name of the theme to use. Built-in themes are 'light' and 'dark'.\r\n * @param {Boolean} infoOnly=false Only output info on the molecule without drawing anything to the canvas.\r\n */\r\n draw(data, target, themeName = 'light', infoOnly = false) {\r\n this.data = data;\r\n this.infoOnly = infoOnly;\r\n\r\n if (!this.infoOnly) {\r\n this.canvasWrapper = new CanvasWrapper(target, this.opts.themes[themeName], this.opts);\r\n }\r\n\r\n this.ringIdCounter = 0;\r\n this.ringConnectionIdCounter = 0;\r\n\r\n this.graph = new Graph(data, this.opts.isomeric);\r\n this.rings = Array();\r\n this.ringConnections = Array();\r\n\r\n this.originalRings = Array();\r\n this.originalRingConnections = Array();\r\n\r\n this.bridgedRing = false;\r\n\r\n // Reset those, in case the previous drawn SMILES had a dangling \\ or /\r\n this.doubleBondConfigCount = null;\r\n this.doubleBondConfig = null;\r\n\r\n this.initRings();\r\n this.initHydrogens();\r\n\r\n if (!this.infoOnly) {\r\n this.position();\r\n\r\n // Restore the ring information (removes bridged rings and replaces them with the original, multiple, rings)\r\n this.restoreRingInformation();\r\n\r\n // Atoms bonded to the same ring atom\r\n this.resolvePrimaryOverlaps();\r\n\r\n let overlapScore = this.getOverlapScore();\r\n\r\n this.totalOverlapScore = this.getOverlapScore().total;\r\n\r\n for (var o = 0; o < this.opts.overlapResolutionIterations; o++) {\r\n for (var i = 0; i < this.graph.edges.length; i++) {\r\n let edge = this.graph.edges[i];\r\n if (this.isEdgeRotatable(edge)) {\r\n let subTreeDepthA = this.graph.getTreeDepth(edge.sourceId, edge.targetId);\r\n let subTreeDepthB = this.graph.getTreeDepth(edge.targetId, edge.sourceId);\r\n\r\n // Only rotate the shorter subtree\r\n let a = edge.targetId;\r\n let b = edge.sourceId;\r\n\r\n if (subTreeDepthA > subTreeDepthB) {\r\n a = edge.sourceId;\r\n b = edge.targetId;\r\n }\r\n\r\n let subTreeOverlap = this.getSubtreeOverlapScore(b, a, overlapScore.vertexScores);\r\n if (subTreeOverlap.value > this.opts.overlapSensitivity) {\r\n let vertexA = this.graph.vertices[a];\r\n let vertexB = this.graph.vertices[b];\r\n let neighboursB = vertexB.getNeighbours(a);\r\n\r\n if (neighboursB.length === 1) {\r\n let neighbour = this.graph.vertices[neighboursB[0]];\r\n let angle = neighbour.position.getRotateAwayFromAngle(vertexA.position, vertexB.position, MathHelper.toRad(120));\r\n\r\n this.rotateSubtree(neighbour.id, vertexB.id, angle, vertexB.position);\r\n // If the new overlap is bigger, undo change\r\n let newTotalOverlapScore = this.getOverlapScore().total;\r\n\r\n if (newTotalOverlapScore > this.totalOverlapScore) {\r\n this.rotateSubtree(neighbour.id, vertexB.id, -angle, vertexB.position);\r\n } else {\r\n this.totalOverlapScore = newTotalOverlapScore;\r\n }\r\n } else if (neighboursB.length === 2) {\r\n // Switch places / sides\r\n // If vertex a is in a ring, do nothing\r\n if (vertexB.value.rings.length !== 0 && vertexA.value.rings.length !== 0) {\r\n continue;\r\n }\r\n\r\n let neighbourA = this.graph.vertices[neighboursB[0]];\r\n let neighbourB = this.graph.vertices[neighboursB[1]];\r\n \r\n if (neighbourA.value.rings.length === 1 && neighbourB.value.rings.length === 1) {\r\n // Both neighbours in same ring. TODO: does this create problems with wedges? (up = down and vice versa?)\r\n if (neighbourA.value.rings[0] !== neighbourB.value.rings[0]) {\r\n continue;\r\n }\r\n // TODO: Rotate circle\r\n } else if (neighbourA.value.rings.length !== 0 || neighbourB.value.rings.length !== 0) {\r\n continue;\r\n } else {\r\n let angleA = neighbourA.position.getRotateAwayFromAngle(vertexA.position, vertexB.position, MathHelper.toRad(120));\r\n let angleB = neighbourB.position.getRotateAwayFromAngle(vertexA.position, vertexB.position, MathHelper.toRad(120));\r\n\r\n this.rotateSubtree(neighbourA.id, vertexB.id, angleA, vertexB.position);\r\n this.rotateSubtree(neighbourB.id, vertexB.id, angleB, vertexB.position);\r\n\r\n let newTotalOverlapScore = this.getOverlapScore().total;\r\n\r\n if (newTotalOverlapScore > this.totalOverlapScore) {\r\n this.rotateSubtree(neighbourA.id, vertexB.id, -angleA, vertexB.position);\r\n this.rotateSubtree(neighbourB.id, vertexB.id, -angleB, vertexB.position);\r\n } else {\r\n this.totalOverlapScore = newTotalOverlapScore;\r\n }\r\n }\r\n }\r\n\r\n overlapScore = this.getOverlapScore();\r\n }\r\n }\r\n }\r\n }\r\n\r\n this.resolveSecondaryOverlaps(overlapScore.scores);\r\n\r\n if (this.opts.isomeric) {\r\n this.annotateStereochemistry();\r\n }\r\n\r\n // Initialize pseudo elements or shortcuts\r\n if (this.opts.compactDrawing && this.opts.atomVisualization === 'default') {\r\n this.initPseudoElements();\r\n }\r\n\r\n this.rotateDrawing();\r\n\r\n // Set the canvas to the appropriate size\r\n this.canvasWrapper.scale(this.graph.vertices);\r\n\r\n // Do the actual drawing\r\n this.drawEdges(this.opts.debug);\r\n this.drawVertices(this.opts.debug);\r\n this.canvasWrapper.reset();\r\n\r\n if (this.opts.debug) {\r\n console.log(this.graph);\r\n console.log(this.rings);\r\n console.log(this.ringConnections);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Returns the number of rings this edge is a part of.\r\n *\r\n * @param {Number} edgeId The id of an edge.\r\n * @returns {Number} The number of rings the provided edge is part of.\r\n */\r\n edgeRingCount(edgeId) {\r\n let edge = this.graph.edges[edgeId];\r\n let a = this.graph.vertices[edge.sourceId];\r\n let b = this.graph.vertices[edge.targetId];\r\n\r\n return Math.min(a.value.rings.length, b.value.rings.length);\r\n }\r\n\r\n /**\r\n * Returns an array containing the bridged rings associated with this molecule.\r\n *\r\n * @returns {Ring[]} An array containing all bridged rings associated with this molecule.\r\n */\r\n getBridgedRings() {\r\n let bridgedRings = Array();\r\n\r\n for (var i = 0; i < this.rings.length; i++) {\r\n if (this.rings[i].isBridged) {\r\n bridgedRings.push(this.rings[i]);\r\n }\r\n }\r\n\r\n return bridgedRings;\r\n }\r\n\r\n /**\r\n * Returns an array containing all fused rings associated with this molecule.\r\n *\r\n * @returns {Ring[]} An array containing all fused rings associated with this molecule.\r\n */\r\n getFusedRings() {\r\n let fusedRings = Array();\r\n\r\n for (var i = 0; i < this.rings.length; i++) {\r\n if (this.rings[i].isFused) {\r\n fusedRings.push(this.rings[i]);\r\n }\r\n }\r\n\r\n return fusedRings;\r\n }\r\n\r\n /**\r\n * Returns an array containing all spiros associated with this molecule.\r\n *\r\n * @returns {Ring[]} An array containing all spiros associated with this molecule.\r\n */\r\n getSpiros() {\r\n let spiros = Array();\r\n\r\n for (var i = 0; i < this.rings.length; i++) {\r\n if (this.rings[i].isSpiro) {\r\n spiros.push(this.rings[i]);\r\n }\r\n }\r\n\r\n return spiros;\r\n }\r\n\r\n /**\r\n * Returns a string containing a semicolon and new-line separated list of ring properties: Id; Members Count; Neighbours Count; IsSpiro; IsFused; IsBridged; Ring Count (subrings of bridged rings)\r\n *\r\n * @returns {String} A string as described in the method description.\r\n */\r\n printRingInfo() {\r\n let result = '';\r\n for (var i = 0; i < this.rings.length; i++) {\r\n const ring = this.rings[i];\r\n\r\n result += ring.id + ';';\r\n result += ring.members.length + ';';\r\n result += ring.neighbours.length + ';';\r\n result += ring.isSpiro ? 'true;' : 'false;'\r\n result += ring.isFused ? 'true;' : 'false;'\r\n result += ring.isBridged ? 'true;' : 'false;'\r\n result += ring.rings.length + ';';\r\n result += '\\n';\r\n }\r\n\r\n return result;\r\n }\r\n\r\n /**\r\n * Rotates the drawing to make the widest dimension horizontal.\r\n */\r\n rotateDrawing() {\r\n // Rotate the vertices to make the molecule align horizontally\r\n // Find the longest distance\r\n let a = 0;\r\n let b = 0;\r\n let maxDist = 0;\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertexA = this.graph.vertices[i];\r\n\r\n if (!vertexA.value.isDrawn) {\r\n continue;\r\n }\r\n\r\n for (var j = i + 1; j < this.graph.vertices.length; j++) {\r\n let vertexB = this.graph.vertices[j];\r\n\r\n if (!vertexB.value.isDrawn) {\r\n continue;\r\n }\r\n\r\n let dist = vertexA.position.distanceSq(vertexB.position);\r\n\r\n if (dist > maxDist) {\r\n maxDist = dist;\r\n a = i;\r\n b = j;\r\n }\r\n }\r\n }\r\n\r\n let angle = -Vector2.subtract(this.graph.vertices[a].position, this.graph.vertices[b].position).angle();\r\n\r\n if (!isNaN(angle)) {\r\n // Round to 30 degrees\r\n let remainder = angle % 0.523599;\r\n\r\n // Round either up or down in 30 degree steps\r\n if (remainder < 0.2617995) {\r\n angle = angle - remainder;\r\n } else {\r\n angle += 0.523599 - remainder;\r\n }\r\n\r\n // Finally, rotate everything\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n if (i === b) {\r\n continue;\r\n }\r\n\r\n this.graph.vertices[i].position.rotateAround(angle, this.graph.vertices[b].position);\r\n }\r\n\r\n for (var i = 0; i < this.rings.length; i++) {\r\n this.rings[i].center.rotateAround(angle, this.graph.vertices[b].position);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Returns the total overlap score of the current molecule.\r\n *\r\n * @returns {Number} The overlap score.\r\n */\r\n getTotalOverlapScore() {\r\n return this.totalOverlapScore;\r\n }\r\n\r\n /**\r\n * Returns the ring count of the current molecule.\r\n *\r\n * @returns {Number} The ring count.\r\n */\r\n getRingCount() {\r\n return this.rings.length;\r\n }\r\n\r\n /**\r\n * Checks whether or not the current molecule a bridged ring.\r\n *\r\n * @returns {Boolean} A boolean indicating whether or not the current molecule a bridged ring.\r\n */\r\n hasBridgedRing() {\r\n return this.bridgedRing;\r\n }\r\n\r\n /**\r\n * Returns the number of heavy atoms (non-hydrogen) in the current molecule.\r\n *\r\n * @returns {Number} The heavy atom count.\r\n */\r\n getHeavyAtomCount() {\r\n let hac = 0;\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n if (this.graph.vertices[i].value.element !== 'H') {\r\n hac++;\r\n }\r\n }\r\n\r\n return hac;\r\n }\r\n\r\n /**\r\n * Returns the molecular formula of the loaded molecule as a string.\r\n * \r\n * @returns {String} The molecular formula.\r\n */\r\n getMolecularFormula() {\r\n let molecularFormula = '';\r\n let counts = new Map();\r\n \r\n // Initialize element count\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let atom = this.graph.vertices[i].value;\r\n\r\n if (counts.has(atom.element)) {\r\n counts.set(atom.element, counts.get(atom.element) + 1);\r\n } else {\r\n counts.set(atom.element, 1);\r\n }\r\n\r\n // Hydrogens attached to a chiral center were added as vertices,\r\n // those in non chiral brackets are added here\r\n if (atom.bracket && !atom.bracket.chirality) {\r\n if (counts.has('H')) {\r\n counts.set('H', counts.get('H') + atom.bracket.hcount);\r\n } else {\r\n counts.set('H', atom.bracket.hcount);\r\n }\r\n }\r\n\r\n // Add the implicit hydrogens according to valency, exclude\r\n // bracket atoms as they were handled and always have the number\r\n // of hydrogens specified explicitly\r\n if (!atom.bracket) {\r\n let nHydrogens = Atom.maxBonds[atom.element] - atom.bondCount;\r\n\r\n if (atom.isPartOfAromaticRing) {\r\n nHydrogens--;\r\n }\r\n\r\n if (counts.has('H')) {\r\n counts.set('H', counts.get('H') + nHydrogens);\r\n } else {\r\n counts.set('H', nHydrogens);\r\n }\r\n }\r\n }\r\n\r\n if (counts.has('C')) {\r\n let count = counts.get('C');\r\n molecularFormula += 'C' + (count > 1 ? count : '');\r\n counts.delete('C');\r\n }\r\n\r\n if (counts.has('H')) {\r\n let count = counts.get('H');\r\n molecularFormula += 'H' + (count > 1 ? count : '');\r\n counts.delete('H');\r\n }\r\n\r\n let elements = Object.keys(Atom.atomicNumbers).sort();\r\n\r\n elements.map(e => {\r\n if (counts.has(e)) {\r\n let count = counts.get(e);\r\n molecularFormula += e + (count > 1 ? count : '');\r\n }\r\n });\r\n\r\n return molecularFormula;\r\n }\r\n\r\n /**\r\n * Returns the type of the ringbond (e.g. '=' for a double bond). The ringbond represents the break in a ring introduced when creating the MST. If the two vertices supplied as arguments are not part of a common ringbond, the method returns null.\r\n *\r\n * @param {Vertex} vertexA A vertex.\r\n * @param {Vertex} vertexB A vertex.\r\n * @returns {(String|null)} Returns the ringbond type or null, if the two supplied vertices are not connected by a ringbond.\r\n */\r\n getRingbondType(vertexA, vertexB) {\r\n // Checks whether the two vertices are the ones connecting the ring\r\n // and what the bond type should be.\r\n if (vertexA.value.getRingbondCount() < 1 || vertexB.value.getRingbondCount() < 1) {\r\n return null;\r\n }\r\n\r\n for (var i = 0; i < vertexA.value.ringbonds.length; i++) {\r\n for (var j = 0; j < vertexB.value.ringbonds.length; j++) {\r\n // if(i != j) continue;\r\n if (vertexA.value.ringbonds[i].id === vertexB.value.ringbonds[j].id) {\r\n // If the bonds are equal, it doesn't matter which bond is returned.\r\n // if they are not equal, return the one that is not the default (\"-\")\r\n if (vertexA.value.ringbonds[i].bondType === '-') {\r\n return vertexB.value.ringbonds[j].bond;\r\n } else {\r\n return vertexA.value.ringbonds[i].bond;\r\n }\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Initializes rings and ringbonds for the current molecule.\r\n */\r\n initRings() {\r\n let openBonds = new Map();\r\n\r\n // Close the open ring bonds (spanning tree -> graph)\r\n for (var i = this.graph.vertices.length - 1; i >= 0; i--) {\r\n let vertex = this.graph.vertices[i];\r\n\r\n if (vertex.value.ringbonds.length === 0) {\r\n continue;\r\n }\r\n\r\n for (var j = 0; j < vertex.value.ringbonds.length; j++) {\r\n let ringbondId = vertex.value.ringbonds[j].id;\r\n let ringbondBond = vertex.value.ringbonds[j].bond;\r\n\r\n // If the other ringbond id has not been discovered,\r\n // add it to the open bonds map and continue.\r\n // if the other ringbond id has already been discovered,\r\n // create a bond between the two atoms.\r\n if (!openBonds.has(ringbondId)) {\r\n openBonds.set(ringbondId, [vertex.id, ringbondBond]);\r\n } else {\r\n let sourceVertexId = vertex.id;\r\n let targetVertexId = openBonds.get(ringbondId)[0];\r\n let targetRingbondBond = openBonds.get(ringbondId)[1];\r\n let edge = new Edge(sourceVertexId, targetVertexId, 1);\r\n edge.setBondType(targetRingbondBond || ringbondBond || '-');\r\n let edgeId = this.graph.addEdge(edge);\r\n let targetVertex = this.graph.vertices[targetVertexId];\r\n\r\n vertex.addRingbondChild(targetVertexId, j);\r\n vertex.value.addNeighbouringElement(targetVertex.value.element);\r\n targetVertex.addRingbondChild(sourceVertexId, j);\r\n targetVertex.value.addNeighbouringElement(vertex.value.element);\r\n vertex.edges.push(edgeId);\r\n targetVertex.edges.push(edgeId);\r\n\r\n openBonds.delete(ringbondId);\r\n }\r\n }\r\n }\r\n\r\n // Get the rings in the graph (the SSSR)\r\n let rings = SSSR.getRings(this.graph, this.opts.experimentalSSSR);\r\n\r\n if (rings === null) {\r\n return;\r\n }\r\n\r\n for (var i = 0; i < rings.length; i++) {\r\n let ringVertices = [...rings[i]];\r\n let ringId = this.addRing(new Ring(ringVertices));\r\n\r\n // Add the ring to the atoms\r\n for (var j = 0; j < ringVertices.length; j++) {\r\n this.graph.vertices[ringVertices[j]].value.rings.push(ringId);\r\n }\r\n }\r\n\r\n // Find connection between rings\r\n // Check for common vertices and create ring connections. This is a bit\r\n // ugly, but the ringcount is always fairly low (< 100)\r\n for (var i = 0; i < this.rings.length - 1; i++) {\r\n for (var j = i + 1; j < this.rings.length; j++) {\r\n let a = this.rings[i];\r\n let b = this.rings[j];\r\n let ringConnection = new RingConnection(a, b);\r\n\r\n // If there are no vertices in the ring connection, then there\r\n // is no ring connection\r\n if (ringConnection.vertices.size > 0) {\r\n this.addRingConnection(ringConnection);\r\n }\r\n }\r\n }\r\n\r\n // Add neighbours to the rings\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let ring = this.rings[i];\r\n ring.neighbours = RingConnection.getNeighbours(this.ringConnections, ring.id);\r\n }\r\n\r\n // Anchor the ring to one of it's members, so that the ring center will always\r\n // be tied to a single vertex when doing repositionings\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let ring = this.rings[i];\r\n this.graph.vertices[ring.members[0]].value.addAnchoredRing(ring.id);\r\n }\r\n\r\n // Backup the ring information to restore after placing the bridged ring.\r\n // This is needed in order to identify aromatic rings and stuff like this in\r\n // rings that are member of the superring.\r\n this.backupRingInformation();\r\n\r\n\r\n // Replace rings contained by a larger bridged ring with a bridged ring\r\n while (this.rings.length > 0) {\r\n let id = -1;\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let ring = this.rings[i];\r\n\r\n if (this.isPartOfBridgedRing(ring.id) && !ring.isBridged) {\r\n id = ring.id;\r\n }\r\n }\r\n\r\n if (id === -1) {\r\n break;\r\n }\r\n\r\n let ring = this.getRing(id);\r\n\r\n let involvedRings = this.getBridgedRingRings(ring.id);\r\n\r\n this.bridgedRing = true;\r\n this.createBridgedRing(involvedRings, ring.members[0]);\r\n\r\n // Remove the rings\r\n for (var i = 0; i < involvedRings.length; i++) {\r\n this.removeRing(involvedRings[i]);\r\n }\r\n }\r\n }\r\n\r\n initHydrogens() {\r\n // Do not draw hydrogens except when they are connected to a stereocenter connected to two or more rings.\r\n if (!this.opts.explicitHydrogens) {\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertex = this.graph.vertices[i];\r\n\r\n if (vertex.value.element !== 'H') {\r\n continue;\r\n }\r\n\r\n // Hydrogens should have only one neighbour, so just take the first\r\n // Also set hasHydrogen true on connected atom\r\n let neighbour = this.graph.vertices[vertex.neighbours[0]];\r\n neighbour.value.hasHydrogen = true;\r\n\r\n if (!neighbour.value.isStereoCenter || neighbour.value.rings.length < 2 && !neighbour.value.bridgedRing ||\r\n neighbour.value.bridgedRing && neighbour.value.originalRings.length < 2) {\r\n vertex.value.isDrawn = false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Returns all rings connected by bridged bonds starting from the ring with the supplied ring id.\r\n *\r\n * @param {Number} ringId A ring id.\r\n * @returns {Number[]} An array containing all ring ids of rings part of a bridged ring system.\r\n */\r\n getBridgedRingRings(ringId) {\r\n let involvedRings = Array();\r\n let that = this;\r\n\r\n let recurse = function (r) {\r\n let ring = that.getRing(r);\r\n\r\n involvedRings.push(r);\r\n\r\n for (var i = 0; i < ring.neighbours.length; i++) {\r\n let n = ring.neighbours[i];\r\n\r\n if (involvedRings.indexOf(n) === -1 &&\r\n n !== r &&\r\n RingConnection.isBridge(that.ringConnections, that.graph.vertices, r, n)) {\r\n recurse(n);\r\n }\r\n }\r\n };\r\n\r\n recurse(ringId);\r\n\r\n return ArrayHelper.unique(involvedRings);\r\n }\r\n\r\n /**\r\n * Checks whether or not a ring is part of a bridged ring.\r\n *\r\n * @param {Number} ringId A ring id.\r\n * @returns {Boolean} A boolean indicating whether or not the supplied ring (by id) is part of a bridged ring system.\r\n */\r\n isPartOfBridgedRing(ringId) {\r\n for (var i = 0; i < this.ringConnections.length; i++) {\r\n if (this.ringConnections[i].containsRing(ringId) &&\r\n this.ringConnections[i].isBridge(this.graph.vertices)) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Creates a bridged ring.\r\n *\r\n * @param {Number[]} ringIds An array of ids of rings involved in the bridged ring.\r\n * @param {Number} sourceVertexId The vertex id to start the bridged ring discovery from.\r\n * @returns {Ring} The bridged ring.\r\n */\r\n createBridgedRing(ringIds, sourceVertexId) {\r\n let ringMembers = new Set();\r\n let vertices = new Set();\r\n let neighbours = new Set();\r\n\r\n for (var i = 0; i < ringIds.length; i++) {\r\n let ring = this.getRing(ringIds[i]);\r\n ring.isPartOfBridged = true;\r\n\r\n for (var j = 0; j < ring.members.length; j++) {\r\n vertices.add(ring.members[j]);\r\n }\r\n\r\n for (var j = 0; j < ring.neighbours.length; j++) {\r\n let id = ring.neighbours[j];\r\n\r\n if (ringIds.indexOf(id) === -1) {\r\n neighbours.add(ring.neighbours[j]);\r\n }\r\n }\r\n }\r\n\r\n // A vertex is part of the bridged ring if it only belongs to\r\n // one of the rings (or to another ring\r\n // which is not part of the bridged ring).\r\n let leftovers = new Set();\r\n\r\n for (let id of vertices) {\r\n let vertex = this.graph.vertices[id];\r\n let intersection = ArrayHelper.intersection(ringIds, vertex.value.rings);\r\n\r\n if (vertex.value.rings.length === 1 || intersection.length === 1) {\r\n ringMembers.add(vertex.id);\r\n } else {\r\n leftovers.add(vertex.id);\r\n }\r\n }\r\n\r\n // Vertices can also be part of multiple rings and lay on the bridged ring,\r\n // however, they have to have at least two neighbours that are not part of\r\n // two rings\r\n let tmp = Array();\r\n let insideRing = Array();\r\n\r\n for (let id of leftovers) {\r\n let vertex = this.graph.vertices[id];\r\n let onRing = false;\r\n\r\n for (let j = 0; j < vertex.edges.length; j++) {\r\n if (this.edgeRingCount(vertex.edges[j]) === 1) {\r\n onRing = true;\r\n }\r\n }\r\n\r\n if (onRing) {\r\n vertex.value.isBridgeNode = true;\r\n ringMembers.add(vertex.id);\r\n } else {\r\n vertex.value.isBridge = true;\r\n ringMembers.add(vertex.id);\r\n }\r\n }\r\n\r\n // Create the ring\r\n let ring = new Ring([...ringMembers]);\r\n this.addRing(ring);\r\n\r\n ring.isBridged = true;\r\n ring.neighbours = [...neighbours];\r\n\r\n for (var i = 0; i < ringIds.length; i++) {\r\n ring.rings.push(this.getRing(ringIds[i]).clone());\r\n }\r\n\r\n for (var i = 0; i < ring.members.length; i++) {\r\n this.graph.vertices[ring.members[i]].value.bridgedRing = ring.id;\r\n }\r\n\r\n // Atoms inside the ring are no longer part of a ring but are now\r\n // associated with the bridged ring\r\n for (var i = 0; i < insideRing.length; i++) {\r\n let vertex = this.graph.vertices[insideRing[i]];\r\n vertex.value.rings = Array();\r\n }\r\n\r\n // Remove former rings from members of the bridged ring and add the bridged ring\r\n for (let id of ringMembers) {\r\n let vertex = this.graph.vertices[id];\r\n vertex.value.rings = ArrayHelper.removeAll(vertex.value.rings, ringIds);\r\n vertex.value.rings.push(ring.id);\r\n }\r\n\r\n // Remove all the ring connections no longer used\r\n for (var i = 0; i < ringIds.length; i++) {\r\n for (var j = i + 1; j < ringIds.length; j++) {\r\n this.removeRingConnectionsBetween(ringIds[i], ringIds[j]);\r\n }\r\n }\r\n\r\n // Update the ring connections and add this ring to the neighbours neighbours\r\n for (let id of neighbours) {\r\n let connections = this.getRingConnections(id, ringIds);\r\n\r\n for (var j = 0; j < connections.length; j++) {\r\n this.getRingConnection(connections[j]).updateOther(ring.id, id);\r\n }\r\n\r\n this.getRing(id).neighbours.push(ring.id);\r\n }\r\n\r\n return ring;\r\n }\r\n\r\n /**\r\n * Checks whether or not two vertices are in the same ring.\r\n *\r\n * @param {Vertex} vertexA A vertex.\r\n * @param {Vertex} vertexB A vertex.\r\n * @returns {Boolean} A boolean indicating whether or not the two vertices are in the same ring.\r\n */\r\n areVerticesInSameRing(vertexA, vertexB) {\r\n // This is a little bit lighter (without the array and push) than\r\n // getCommonRings().length > 0\r\n for (var i = 0; i < vertexA.value.rings.length; i++) {\r\n for (var j = 0; j < vertexB.value.rings.length; j++) {\r\n if (vertexA.value.rings[i] === vertexB.value.rings[j]) {\r\n return true;\r\n }\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Returns an array of ring ids shared by both vertices.\r\n *\r\n * @param {Vertex} vertexA A vertex.\r\n * @param {Vertex} vertexB A vertex.\r\n * @returns {Number[]} An array of ids of rings shared by the two vertices.\r\n */\r\n getCommonRings(vertexA, vertexB) {\r\n let commonRings = Array();\r\n\r\n for (var i = 0; i < vertexA.value.rings.length; i++) {\r\n for (var j = 0; j < vertexB.value.rings.length; j++) {\r\n if (vertexA.value.rings[i] == vertexB.value.rings[j]) {\r\n commonRings.push(vertexA.value.rings[i]);\r\n }\r\n }\r\n }\r\n\r\n return commonRings;\r\n }\r\n\r\n /**\r\n * Returns the aromatic or largest ring shared by the two vertices.\r\n *\r\n * @param {Vertex} vertexA A vertex.\r\n * @param {Vertex} vertexB A vertex.\r\n * @returns {(Ring|null)} If an aromatic common ring exists, that ring, else the largest (non-aromatic) ring, else null.\r\n */\r\n getLargestOrAromaticCommonRing(vertexA, vertexB) {\r\n let commonRings = this.getCommonRings(vertexA, vertexB);\r\n let maxSize = 0;\r\n let largestCommonRing = null;\r\n\r\n for (var i = 0; i < commonRings.length; i++) {\r\n let ring = this.getRing(commonRings[i]);\r\n let size = ring.getSize();\r\n\r\n if (ring.isBenzeneLike(this.graph.vertices)) {\r\n return ring;\r\n } else if (size > maxSize) {\r\n maxSize = size;\r\n largestCommonRing = ring;\r\n }\r\n }\r\n\r\n return largestCommonRing;\r\n }\r\n\r\n /**\r\n * Returns an array of vertices positioned at a specified location.\r\n *\r\n * @param {Vector2} position The position to search for vertices.\r\n * @param {Number} radius The radius within to search.\r\n * @param {Number} excludeVertexId A vertex id to be excluded from the search results.\r\n * @returns {Number[]} An array containing vertex ids in a given location.\r\n */\r\n getVerticesAt(position, radius, excludeVertexId) {\r\n let locals = Array();\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertex = this.graph.vertices[i];\r\n\r\n if (vertex.id === excludeVertexId || !vertex.positioned) {\r\n continue;\r\n }\r\n\r\n let distance = position.distanceSq(vertex.position);\r\n\r\n if (distance <= radius * radius) {\r\n locals.push(vertex.id);\r\n }\r\n }\r\n\r\n return locals;\r\n }\r\n\r\n /**\r\n * Returns the closest vertex (connected as well as unconnected).\r\n *\r\n * @param {Vertex} vertex The vertex of which to find the closest other vertex.\r\n * @returns {Vertex} The closest vertex.\r\n */\r\n getClosestVertex(vertex) {\r\n let minDist = 99999;\r\n let minVertex = null;\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let v = this.graph.vertices[i];\r\n\r\n if (v.id === vertex.id) {\r\n continue;\r\n }\r\n\r\n let distSq = vertex.position.distanceSq(v.position);\r\n\r\n if (distSq < minDist) {\r\n minDist = distSq;\r\n minVertex = v;\r\n }\r\n }\r\n\r\n return minVertex;\r\n }\r\n\r\n /**\r\n * Add a ring to this representation of a molecule.\r\n *\r\n * @param {Ring} ring A new ring.\r\n * @returns {Number} The ring id of the new ring.\r\n */\r\n addRing(ring) {\r\n ring.id = this.ringIdCounter++;\r\n this.rings.push(ring);\r\n\r\n return ring.id;\r\n }\r\n\r\n /**\r\n * Removes a ring from the array of rings associated with the current molecule.\r\n *\r\n * @param {Number} ringId A ring id.\r\n */\r\n removeRing(ringId) {\r\n this.rings = this.rings.filter(function (item) {\r\n return item.id !== ringId;\r\n });\r\n\r\n // Also remove ring connections involving this ring\r\n this.ringConnections = this.ringConnections.filter(function (item) {\r\n return item.firstRingId !== ringId && item.secondRingId !== ringId;\r\n });\r\n\r\n // Remove the ring as neighbour of other rings\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let r = this.rings[i];\r\n r.neighbours = r.neighbours.filter(function (item) {\r\n return item !== ringId;\r\n });\r\n }\r\n }\r\n\r\n /**\r\n * Gets a ring object from the array of rings associated with the current molecule by its id. The ring id is not equal to the index, since rings can be added and removed when processing bridged rings.\r\n *\r\n * @param {Number} ringId A ring id.\r\n * @returns {Ring} A ring associated with the current molecule.\r\n */\r\n getRing(ringId) {\r\n for (var i = 0; i < this.rings.length; i++) {\r\n if (this.rings[i].id == ringId) {\r\n return this.rings[i];\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Add a ring connection to this representation of a molecule.\r\n *\r\n * @param {RingConnection} ringConnection A new ringConnection.\r\n * @returns {Number} The ring connection id of the new ring connection.\r\n */\r\n addRingConnection(ringConnection) {\r\n ringConnection.id = this.ringConnectionIdCounter++;\r\n this.ringConnections.push(ringConnection);\r\n\r\n return ringConnection.id;\r\n }\r\n\r\n /**\r\n * Removes a ring connection from the array of rings connections associated with the current molecule.\r\n *\r\n * @param {Number} ringConnectionId A ring connection id.\r\n */\r\n removeRingConnection(ringConnectionId) {\r\n this.ringConnections = this.ringConnections.filter(function (item) {\r\n return item.id !== ringConnectionId;\r\n });\r\n }\r\n\r\n /**\r\n * Removes all ring connections between two vertices.\r\n *\r\n * @param {Number} vertexIdA A vertex id.\r\n * @param {Number} vertexIdB A vertex id.\r\n */\r\n removeRingConnectionsBetween(vertexIdA, vertexIdB) {\r\n let toRemove = Array();\r\n for (var i = 0; i < this.ringConnections.length; i++) {\r\n let ringConnection = this.ringConnections[i];\r\n\r\n if (ringConnection.firstRingId === vertexIdA && ringConnection.secondRingId === vertexIdB ||\r\n ringConnection.firstRingId === vertexIdB && ringConnection.secondRingId === vertexIdA) {\r\n toRemove.push(ringConnection.id);\r\n }\r\n }\r\n\r\n for (var i = 0; i < toRemove.length; i++) {\r\n this.removeRingConnection(toRemove[i]);\r\n }\r\n }\r\n\r\n /**\r\n * Get a ring connection with a given id.\r\n * \r\n * @param {Number} id \r\n * @returns {RingConnection} The ring connection with the specified id.\r\n */\r\n getRingConnection(id) {\r\n for (var i = 0; i < this.ringConnections.length; i++) {\r\n if (this.ringConnections[i].id == id) {\r\n return this.ringConnections[i];\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Get the ring connections between a ring and a set of rings.\r\n *\r\n * @param {Number} ringId A ring id.\r\n * @param {Number[]} ringIds An array of ring ids.\r\n * @returns {Number[]} An array of ring connection ids.\r\n */\r\n getRingConnections(ringId, ringIds) {\r\n let ringConnections = Array();\r\n\r\n for (var i = 0; i < this.ringConnections.length; i++) {\r\n let rc = this.ringConnections[i];\r\n\r\n for (var j = 0; j < ringIds.length; j++) {\r\n let id = ringIds[j];\r\n\r\n if (rc.firstRingId === ringId && rc.secondRingId === id ||\r\n rc.firstRingId === id && rc.secondRingId === ringId) {\r\n ringConnections.push(rc.id);\r\n }\r\n }\r\n }\r\n\r\n return ringConnections;\r\n }\r\n\r\n /**\r\n * Returns the overlap score of the current molecule based on its positioned vertices. The higher the score, the more overlaps occur in the structure drawing.\r\n *\r\n * @returns {Object} Returns the total overlap score and the overlap score of each vertex sorted by score (higher to lower). Example: { total: 99, scores: [ { id: 0, score: 22 }, ... ] }\r\n */\r\n getOverlapScore() {\r\n let total = 0.0;\r\n let overlapScores = new Float32Array(this.graph.vertices.length);\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n overlapScores[i] = 0;\r\n }\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n var j = this.graph.vertices.length;\r\n while (--j > i) {\r\n let a = this.graph.vertices[i];\r\n let b = this.graph.vertices[j];\r\n\r\n if (!a.value.isDrawn || !b.value.isDrawn) {\r\n continue;\r\n }\r\n\r\n let dist = Vector2.subtract(a.position, b.position).lengthSq();\r\n\r\n if (dist < this.opts.bondLengthSq) {\r\n let weighted = (this.opts.bondLength - Math.sqrt(dist)) / this.opts.bondLength;\r\n total += weighted;\r\n overlapScores[i] += weighted;\r\n overlapScores[j] += weighted;\r\n }\r\n }\r\n }\r\n\r\n let sortable = Array();\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n sortable.push({\r\n id: i,\r\n score: overlapScores[i]\r\n });\r\n }\r\n\r\n sortable.sort(function (a, b) {\r\n return b.score - a.score;\r\n });\r\n\r\n return {\r\n total: total,\r\n scores: sortable,\r\n vertexScores: overlapScores\r\n };\r\n }\r\n\r\n /**\r\n * When drawing a double bond, choose the side to place the double bond. E.g. a double bond should always been drawn inside a ring.\r\n *\r\n * @param {Vertex} vertexA A vertex.\r\n * @param {Vertex} vertexB A vertex.\r\n * @param {Vector2[]} sides An array containing the two normals of the line spanned by the two provided vertices.\r\n * @returns {Object} Returns an object containing the following information: {\r\n totalSideCount: Counts the sides of each vertex in the molecule, is an array [ a, b ],\r\n totalPosition: Same as position, but based on entire molecule,\r\n sideCount: Counts the sides of each neighbour, is an array [ a, b ],\r\n position: which side to position the second bond, is 0 or 1, represents the index in the normal array. This is based on only the neighbours\r\n anCount: the number of neighbours of vertexA,\r\n bnCount: the number of neighbours of vertexB\r\n }\r\n */\r\n chooseSide(vertexA, vertexB, sides) {\r\n // Check which side has more vertices\r\n // Get all the vertices connected to the both ends\r\n let an = vertexA.getNeighbours(vertexB.id);\r\n let bn = vertexB.getNeighbours(vertexA.id);\r\n let anCount = an.length;\r\n let bnCount = bn.length;\r\n\r\n // All vertices connected to the edge vertexA to vertexB\r\n let tn = ArrayHelper.merge(an, bn);\r\n\r\n // Only considering the connected vertices\r\n let sideCount = [0, 0];\r\n\r\n for (var i = 0; i < tn.length; i++) {\r\n let v = this.graph.vertices[tn[i]].position;\r\n\r\n if (v.sameSideAs(vertexA.position, vertexB.position, sides[0])) {\r\n sideCount[0]++;\r\n } else {\r\n sideCount[1]++;\r\n }\r\n }\r\n\r\n // Considering all vertices in the graph, this is to resolve ties\r\n // from the above side counts\r\n let totalSideCount = [0, 0];\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let v = this.graph.vertices[i].position;\r\n\r\n if (v.sameSideAs(vertexA.position, vertexB.position, sides[0])) {\r\n totalSideCount[0]++;\r\n } else {\r\n totalSideCount[1]++;\r\n }\r\n }\r\n\r\n return {\r\n totalSideCount: totalSideCount,\r\n totalPosition: totalSideCount[0] > totalSideCount[1] ? 0 : 1,\r\n sideCount: sideCount,\r\n position: sideCount[0] > sideCount[1] ? 0 : 1,\r\n anCount: anCount,\r\n bnCount: bnCount\r\n };\r\n }\r\n\r\n /**\r\n * Sets the center for a ring.\r\n *\r\n * @param {Ring} ring A ring.\r\n */\r\n setRingCenter(ring) {\r\n let ringSize = ring.getSize();\r\n let total = new Vector2(0, 0);\r\n\r\n for (var i = 0; i < ringSize; i++) {\r\n total.add(this.graph.vertices[ring.members[i]].position);\r\n }\r\n\r\n ring.center = total.divide(ringSize);\r\n }\r\n\r\n /**\r\n * Gets the center of a ring contained within a bridged ring and containing a given vertex.\r\n *\r\n * @param {Ring} ring A bridged ring.\r\n * @param {Vertex} vertex A vertex.\r\n * @returns {Vector2} The center of the subring that containing the vertex.\r\n */\r\n getSubringCenter(ring, vertex) {\r\n let rings = vertex.value.originalRings;\r\n let center = ring.center;\r\n let smallest = Number.MAX_VALUE;\r\n\r\n // Always get the smallest ring.\r\n for (var i = 0; i < rings.length; i++) {\r\n for (var j = 0; j < ring.rings.length; j++) {\r\n if (rings[i] === ring.rings[j].id) {\r\n if (ring.rings[j].getSize() < smallest) {\r\n center = ring.rings[j].center;\r\n smallest = ring.rings[j].getSize();\r\n }\r\n }\r\n }\r\n }\r\n\r\n return center;\r\n }\r\n\r\n /**\r\n * Draw the actual edges as bonds to the canvas.\r\n *\r\n * @param {Boolean} debug A boolean indicating whether or not to draw debug helpers.\r\n */\r\n drawEdges(debug) {\r\n let that = this;\r\n let drawn = Array(this.graph.edges.length);\r\n drawn.fill(false);\r\n\r\n this.graph.traverseBF(0, function (vertex) {\r\n let edges = that.graph.getEdges(vertex.id);\r\n for (var i = 0; i < edges.length; i++) {\r\n let edgeId = edges[i];\r\n if (!drawn[edgeId]) {\r\n drawn[edgeId] = true;\r\n that.drawEdge(edgeId, debug);\r\n }\r\n }\r\n });\r\n\r\n // Draw ring for implicitly defined aromatic rings\r\n if (!this.bridgedRing) {\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let ring = this.rings[i];\r\n\r\n if (this.isRingAromatic(ring)) {\r\n this.canvasWrapper.drawAromaticityRing(ring);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Draw the an edge as a bonds to the canvas.\r\n *\r\n * @param {Number} edgeId An edge id.\r\n * @param {Boolean} debug A boolean indicating whether or not to draw debug helpers.\r\n */\r\n drawEdge(edgeId, debug) {\r\n let that = this;\r\n let edge = this.graph.edges[edgeId];\r\n let vertexA = this.graph.vertices[edge.sourceId];\r\n let vertexB = this.graph.vertices[edge.targetId];\r\n let elementA = vertexA.value.element;\r\n let elementB = vertexB.value.element;\r\n\r\n if ((!vertexA.value.isDrawn || !vertexB.value.isDrawn) && this.opts.atomVisualization === 'default') {\r\n return;\r\n }\r\n\r\n let a = vertexA.position;\r\n let b = vertexB.position;\r\n let normals = this.getEdgeNormals(edge);\r\n\r\n // Create a point on each side of the line\r\n let sides = ArrayHelper.clone(normals);\r\n\r\n sides[0].multiplyScalar(10).add(a);\r\n sides[1].multiplyScalar(10).add(a);\r\n\r\n if (edge.bondType === '=' || this.getRingbondType(vertexA, vertexB) === '=' ||\r\n (edge.isPartOfAromaticRing && this.bridgedRing)) {\r\n // Always draw double bonds inside the ring\r\n let inRing = this.areVerticesInSameRing(vertexA, vertexB);\r\n let s = this.chooseSide(vertexA, vertexB, sides);\r\n\r\n if (inRing) {\r\n // Always draw double bonds inside a ring\r\n // if the bond is shared by two rings, it is drawn in the larger\r\n // problem: smaller ring is aromatic, bond is still drawn in larger -> fix this\r\n let lcr = this.getLargestOrAromaticCommonRing(vertexA, vertexB);\r\n let center = lcr.center;\r\n\r\n normals[0].multiplyScalar(that.opts.bondSpacing);\r\n normals[1].multiplyScalar(that.opts.bondSpacing);\r\n\r\n // Choose the normal that is on the same side as the center\r\n let line = null;\r\n\r\n if (center.sameSideAs(vertexA.position, vertexB.position, Vector2.add(a, normals[0]))) {\r\n line = new Line(Vector2.add(a, normals[0]), Vector2.add(b, normals[0]), elementA, elementB);\r\n } else {\r\n line = new Line(Vector2.add(a, normals[1]), Vector2.add(b, normals[1]), elementA, elementB);\r\n }\r\n\r\n line.shorten(this.opts.bondLength - this.opts.shortBondLength * this.opts.bondLength);\r\n\r\n // The shortened edge\r\n if (edge.isPartOfAromaticRing) {\r\n this.canvasWrapper.drawLine(line, true);\r\n } else {\r\n this.canvasWrapper.drawLine(line);\r\n }\r\n\r\n // The normal edge\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB));\r\n } else if (edge.center || vertexA.isTerminal() && vertexB.isTerminal()) {\r\n normals[0].multiplyScalar(that.opts.halfBondSpacing);\r\n normals[1].multiplyScalar(that.opts.halfBondSpacing);\r\n\r\n let lineA = new Line(Vector2.add(a, normals[0]), Vector2.add(b, normals[0]), elementA, elementB);\r\n let lineB = new Line(Vector2.add(a, normals[1]), Vector2.add(b, normals[1]), elementA, elementB);\r\n\r\n this.canvasWrapper.drawLine(lineA);\r\n this.canvasWrapper.drawLine(lineB);\r\n } else if (s.anCount == 0 && s.bnCount > 1 || s.bnCount == 0 && s.anCount > 1) {\r\n // Both lines are the same length here\r\n // Add the spacing to the edges (which are of unit length)\r\n normals[0].multiplyScalar(that.opts.halfBondSpacing);\r\n normals[1].multiplyScalar(that.opts.halfBondSpacing);\r\n\r\n let lineA = new Line(Vector2.add(a, normals[0]), Vector2.add(b, normals[0]), elementA, elementB);\r\n let lineB = new Line(Vector2.add(a, normals[1]), Vector2.add(b, normals[1]), elementA, elementB);\r\n\r\n this.canvasWrapper.drawLine(lineA);\r\n this.canvasWrapper.drawLine(lineB);\r\n } else if (s.sideCount[0] > s.sideCount[1]) {\r\n normals[0].multiplyScalar(that.opts.bondSpacing);\r\n normals[1].multiplyScalar(that.opts.bondSpacing);\r\n\r\n let line = new Line(Vector2.add(a, normals[0]), Vector2.add(b, normals[0]), elementA, elementB);\r\n\r\n line.shorten(this.opts.bondLength - this.opts.shortBondLength * this.opts.bondLength);\r\n this.canvasWrapper.drawLine(line);\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB));\r\n } else if (s.sideCount[0] < s.sideCount[1]) {\r\n normals[0].multiplyScalar(that.opts.bondSpacing);\r\n normals[1].multiplyScalar(that.opts.bondSpacing);\r\n\r\n let line = new Line(Vector2.add(a, normals[1]), Vector2.add(b, normals[1]), elementA, elementB);\r\n\r\n line.shorten(this.opts.bondLength - this.opts.shortBondLength * this.opts.bondLength);\r\n this.canvasWrapper.drawLine(line);\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB));\r\n } else if (s.totalSideCount[0] > s.totalSideCount[1]) {\r\n normals[0].multiplyScalar(that.opts.bondSpacing);\r\n normals[1].multiplyScalar(that.opts.bondSpacing);\r\n\r\n let line = new Line(Vector2.add(a, normals[0]), Vector2.add(b, normals[0]), elementA, elementB);\r\n\r\n line.shorten(this.opts.bondLength - this.opts.shortBondLength * this.opts.bondLength);\r\n this.canvasWrapper.drawLine(line);\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB));\r\n } else if (s.totalSideCount[0] <= s.totalSideCount[1]) {\r\n normals[0].multiplyScalar(that.opts.bondSpacing);\r\n normals[1].multiplyScalar(that.opts.bondSpacing);\r\n\r\n let line = new Line(Vector2.add(a, normals[1]), Vector2.add(b, normals[1]), elementA, elementB);\r\n\r\n line.shorten(this.opts.bondLength - this.opts.shortBondLength * this.opts.bondLength);\r\n this.canvasWrapper.drawLine(line);\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB));\r\n } else {\r\n\r\n }\r\n } else if (edge.bondType === '#') {\r\n normals[0].multiplyScalar(that.opts.bondSpacing / 1.5);\r\n normals[1].multiplyScalar(that.opts.bondSpacing / 1.5);\r\n\r\n let lineA = new Line(Vector2.add(a, normals[0]), Vector2.add(b, normals[0]), elementA, elementB);\r\n let lineB = new Line(Vector2.add(a, normals[1]), Vector2.add(b, normals[1]), elementA, elementB);\r\n\r\n this.canvasWrapper.drawLine(lineA);\r\n this.canvasWrapper.drawLine(lineB);\r\n\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB));\r\n } else if (edge.bondType === '.') {\r\n // TODO: Something... maybe... version 2?\r\n } else {\r\n let isChiralCenterA = vertexA.value.isStereoCenter;\r\n let isChiralCenterB = vertexB.value.isStereoCenter;\r\n\r\n if (edge.wedge === 'up') {\r\n this.canvasWrapper.drawWedge(new Line(a, b, elementA, elementB, isChiralCenterA, isChiralCenterB));\r\n } else if (edge.wedge === 'down') {\r\n this.canvasWrapper.drawDashedWedge(new Line(a, b, elementA, elementB, isChiralCenterA, isChiralCenterB));\r\n } else {\r\n this.canvasWrapper.drawLine(new Line(a, b, elementA, elementB, isChiralCenterA, isChiralCenterB));\r\n }\r\n }\r\n\r\n if (debug) {\r\n let midpoint = Vector2.midpoint(a, b);\r\n this.canvasWrapper.drawDebugText(midpoint.x, midpoint.y, 'e: ' + edgeId);\r\n }\r\n }\r\n\r\n /**\r\n * Draws the vertices representing atoms to the canvas.\r\n *\r\n * @param {Boolean} debug A boolean indicating whether or not to draw debug messages to the canvas.\r\n */\r\n drawVertices(debug) {\r\n var i = this.graph.vertices.length;\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertex = this.graph.vertices[i];\r\n let atom = vertex.value;\r\n let charge = 0;\r\n let isotope = 0;\r\n let bondCount = vertex.value.bondCount;\r\n let element = atom.element;\r\n let hydrogens = Atom.maxBonds[element] - bondCount;\r\n let dir = vertex.getTextDirection(this.graph.vertices);\r\n let isTerminal = this.opts.terminalCarbons || element !== 'C' || atom.hasAttachedPseudoElements ? vertex.isTerminal() : false;\r\n let isCarbon = atom.element === 'C';\r\n\r\n // This is a HACK to remove all hydrogens from nitrogens in aromatic rings, as this\r\n // should be the most common state. This has to be fixed by kekulization\r\n if (atom.element === 'N' && atom.isPartOfAromaticRing) {\r\n hydrogens = 0;\r\n }\r\n\r\n if (atom.bracket) {\r\n hydrogens = atom.bracket.hcount;\r\n charge = atom.bracket.charge;\r\n isotope = atom.bracket.isotope;\r\n }\r\n\r\n if (this.opts.atomVisualization === 'allballs') {\r\n this.canvasWrapper.drawBall(vertex.position.x, vertex.position.y, element);\r\n } else if ((atom.isDrawn && (!isCarbon || atom.drawExplicit || isTerminal || atom.hasAttachedPseudoElements)) || this.graph.vertices.length === 1) {\r\n if (this.opts.atomVisualization === 'default') {\r\n this.canvasWrapper.drawText(vertex.position.x, vertex.position.y,\r\n element, hydrogens, dir, isTerminal, charge, isotope, atom.getAttachedPseudoElements());\r\n } else if (this.opts.atomVisualization === 'balls') {\r\n this.canvasWrapper.drawBall(vertex.position.x, vertex.position.y, element);\r\n }\r\n } else if (vertex.getNeighbourCount() === 2 && vertex.forcePositioned == true) {\r\n // If there is a carbon which bonds are in a straight line, draw a dot\r\n let a = this.graph.vertices[vertex.neighbours[0]].position;\r\n let b = this.graph.vertices[vertex.neighbours[1]].position;\r\n let angle = Vector2.threePointangle(vertex.position, a, b);\r\n\r\n if (Math.abs(Math.PI - angle) < 0.1) {\r\n this.canvasWrapper.drawPoint(vertex.position.x, vertex.position.y, element);\r\n }\r\n }\r\n\r\n if (debug) {\r\n let value = 'v: ' + vertex.id + ' ' + ArrayHelper.print(atom.ringbonds);\r\n this.canvasWrapper.drawDebugText(vertex.position.x, vertex.position.y, value);\r\n } else {\r\n // this.canvasWrapper.drawDebugText(vertex.position.x, vertex.position.y, vertex.value.chirality);\r\n }\r\n }\r\n\r\n // Draw the ring centers for debug purposes\r\n if (this.opts.debug) {\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let center = this.rings[i].center;\r\n this.canvasWrapper.drawDebugPoint(center.x, center.y, 'r: ' + this.rings[i].id);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Position the vertices according to their bonds and properties.\r\n */\r\n position() {\r\n let startVertex = null;\r\n\r\n // Always start drawing at a bridged ring if there is one\r\n // If not, start with a ring\r\n // else, start with 0\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n if (this.graph.vertices[i].value.bridgedRing !== null) {\r\n startVertex = this.graph.vertices[i];\r\n break;\r\n }\r\n }\r\n\r\n for (var i = 0; i < this.rings.length; i++) {\r\n if (this.rings[i].isBridged) {\r\n startVertex = this.graph.vertices[this.rings[i].members[0]];\r\n }\r\n }\r\n\r\n if (this.rings.length > 0 && startVertex === null) {\r\n startVertex = this.graph.vertices[this.rings[0].members[0]];\r\n }\r\n\r\n if (startVertex === null) {\r\n startVertex = this.graph.vertices[0];\r\n }\r\n\r\n this.createNextBond(startVertex, null, 0.0);\r\n }\r\n\r\n /**\r\n * Stores the current information associated with rings.\r\n */\r\n backupRingInformation() {\r\n this.originalRings = Array();\r\n this.originalRingConnections = Array();\r\n\r\n for (var i = 0; i < this.rings.length; i++) {\r\n this.originalRings.push(this.rings[i]);\r\n }\r\n\r\n for (var i = 0; i < this.ringConnections.length; i++) {\r\n this.originalRingConnections.push(this.ringConnections[i]);\r\n }\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n this.graph.vertices[i].value.backupRings();\r\n }\r\n }\r\n\r\n /**\r\n * Restores the most recently backed up information associated with rings.\r\n */\r\n restoreRingInformation() {\r\n // Get the subring centers from the bridged rings\r\n let bridgedRings = this.getBridgedRings();\r\n\r\n this.rings = Array();\r\n this.ringConnections = Array();\r\n\r\n for (var i = 0; i < bridgedRings.length; i++) {\r\n let bridgedRing = bridgedRings[i];\r\n\r\n for (var j = 0; j < bridgedRing.rings.length; j++) {\r\n let ring = bridgedRing.rings[j];\r\n this.originalRings[ring.id].center = ring.center;\r\n }\r\n }\r\n\r\n for (var i = 0; i < this.originalRings.length; i++) {\r\n this.rings.push(this.originalRings[i]);\r\n }\r\n\r\n for (var i = 0; i < this.originalRingConnections.length; i++) {\r\n this.ringConnections.push(this.originalRingConnections[i]);\r\n }\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n this.graph.vertices[i].value.restoreRings();\r\n }\r\n }\r\n\r\n // TODO: This needs some cleaning up\r\n\r\n /**\r\n * Creates a new ring, that is, positiones all the vertices inside a ring.\r\n *\r\n * @param {Ring} ring The ring to position.\r\n * @param {(Vector2|null)} [center=null] The center of the ring to be created.\r\n * @param {(Vertex|null)} [startVertex=null] The first vertex to be positioned inside the ring.\r\n * @param {(Vertex|null)} [previousVertex=null] The last vertex that was positioned.\r\n * @param {Boolean} [previousVertex=false] A boolean indicating whether or not this ring was force positioned already - this is needed after force layouting a ring, in order to draw rings connected to it.\r\n */\r\n createRing(ring, center = null, startVertex = null, previousVertex = null) {\r\n if (ring.positioned) {\r\n return;\r\n }\r\n\r\n center = center ? center : new Vector2(0, 0);\r\n\r\n let orderedNeighbours = ring.getOrderedNeighbours(this.ringConnections);\r\n let startingAngle = startVertex ? Vector2.subtract(startVertex.position, center).angle() : 0;\r\n\r\n let radius = MathHelper.polyCircumradius(this.opts.bondLength, ring.getSize());\r\n let angle = MathHelper.centralAngle(ring.getSize());\r\n\r\n ring.centralAngle = angle;\r\n\r\n let a = startingAngle;\r\n let that = this;\r\n let startVertexId = (startVertex) ? startVertex.id : null;\r\n\r\n if (ring.members.indexOf(startVertexId) === -1) {\r\n if (startVertex) {\r\n startVertex.positioned = false;\r\n }\r\n\r\n startVertexId = ring.members[0];\r\n }\r\n\r\n // If the ring is bridged, then draw the vertices inside the ring\r\n // using a force based approach\r\n if (ring.isBridged) {\r\n this.graph.kkLayout(ring.members.slice(), center, startVertex.id, ring, this.opts.bondLength,\r\n this.opts.kkThreshold, this.opts.kkInnerThreshold, this.opts.kkMaxIteration, \r\n this.opts.kkMaxInnerIteration, this.opts.kkMaxEnergy);\r\n ring.positioned = true;\r\n\r\n // Update the center of the bridged ring\r\n this.setRingCenter(ring);\r\n center = ring.center;\r\n\r\n // Setting the centers for the subrings\r\n for (var i = 0; i < ring.rings.length; i++) {\r\n this.setRingCenter(ring.rings[i]);\r\n }\r\n } else {\r\n ring.eachMember(this.graph.vertices, function (v) {\r\n let vertex = that.graph.vertices[v];\r\n\r\n if (!vertex.positioned) {\r\n vertex.setPosition(center.x + Math.cos(a) * radius, center.y + Math.sin(a) * radius);\r\n }\r\n\r\n a += angle;\r\n\r\n if (!ring.isBridged || ring.rings.length < 3) {\r\n vertex.angle = a;\r\n vertex.positioned = true;\r\n }\r\n }, startVertexId, (previousVertex) ? previousVertex.id : null);\r\n }\r\n\r\n ring.positioned = true;\r\n ring.center = center;\r\n\r\n // Draw neighbours in decreasing order of connectivity\r\n for (var i = 0; i < orderedNeighbours.length; i++) {\r\n let neighbour = this.getRing(orderedNeighbours[i].neighbour);\r\n\r\n if (neighbour.positioned) {\r\n continue;\r\n }\r\n\r\n let vertices = RingConnection.getVertices(this.ringConnections, ring.id, neighbour.id);\r\n\r\n if (vertices.length === 2) {\r\n // This ring is a fused ring\r\n ring.isFused = true;\r\n neighbour.isFused = true;\r\n\r\n let vertexA = this.graph.vertices[vertices[0]];\r\n let vertexB = this.graph.vertices[vertices[1]];\r\n\r\n // Get middle between vertex A and B\r\n let midpoint = Vector2.midpoint(vertexA.position, vertexB.position);\r\n\r\n // Get the normals to the line between A and B\r\n let normals = Vector2.normals(vertexA.position, vertexB.position);\r\n\r\n // Normalize the normals\r\n normals[0].normalize();\r\n normals[1].normalize();\r\n\r\n // Set length from middle of side to center (the apothem)\r\n let r = MathHelper.polyCircumradius(this.opts.bondLength, neighbour.getSize());\r\n let apothem = MathHelper.apothem(r, neighbour.getSize());\r\n\r\n normals[0].multiplyScalar(apothem).add(midpoint);\r\n normals[1].multiplyScalar(apothem).add(midpoint);\r\n\r\n // Pick the normal which results in a larger distance to the previous center\r\n // Also check whether it's inside another ring\r\n let nextCenter = normals[0];\r\n if (Vector2.subtract(center, normals[1]).lengthSq() > Vector2.subtract(center, normals[0]).lengthSq()) {\r\n nextCenter = normals[1];\r\n }\r\n\r\n // Get the vertex (A or B) which is in clock-wise direction of the other\r\n let posA = Vector2.subtract(vertexA.position, nextCenter);\r\n let posB = Vector2.subtract(vertexB.position, nextCenter);\r\n\r\n if (posA.clockwise(posB) === -1) {\r\n if (!neighbour.positioned) {\r\n this.createRing(neighbour, nextCenter, vertexA, vertexB);\r\n }\r\n } else {\r\n if (!neighbour.positioned) {\r\n this.createRing(neighbour, nextCenter, vertexB, vertexA);\r\n }\r\n }\r\n } else if (vertices.length === 1) {\r\n // This ring is a spiro\r\n ring.isSpiro = true;\r\n neighbour.isSpiro = true;\r\n\r\n let vertexA = this.graph.vertices[vertices[0]];\r\n\r\n // Get the vector pointing from the shared vertex to the new centpositioner\r\n let nextCenter = Vector2.subtract(center, vertexA.position);\r\n\r\n nextCenter.invert();\r\n nextCenter.normalize();\r\n\r\n // Get the distance from the vertex to the center\r\n let r = MathHelper.polyCircumradius(this.opts.bondLength, neighbour.getSize());\r\n\r\n nextCenter.multiplyScalar(r);\r\n nextCenter.add(vertexA.position);\r\n\r\n if (!neighbour.positioned) {\r\n this.createRing(neighbour, nextCenter, vertexA);\r\n }\r\n }\r\n }\r\n\r\n // Next, draw atoms that are not part of a ring that are directly attached to this ring\r\n for (var i = 0; i < ring.members.length; i++) {\r\n let ringMember = this.graph.vertices[ring.members[i]];\r\n let ringMemberNeighbours = ringMember.neighbours;\r\n\r\n // If there are multiple, the ovlerap will be resolved in the appropriate step\r\n for (var j = 0; j < ringMemberNeighbours.length; j++) {\r\n let v = this.graph.vertices[ringMemberNeighbours[j]];\r\n\r\n if (v.positioned) {\r\n continue;\r\n }\r\n\r\n v.value.isConnectedToRing = true;\r\n this.createNextBond(v, ringMember, 0.0);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Rotate an entire subtree by an angle around a center.\r\n *\r\n * @param {Number} vertexId A vertex id (the root of the sub-tree).\r\n * @param {Number} parentVertexId A vertex id in the previous direction of the subtree that is to rotate.\r\n * @param {Number} angle An angle in randians.\r\n * @param {Vector2} center The rotational center.\r\n */\r\n rotateSubtree(vertexId, parentVertexId, angle, center) {\r\n let that = this;\r\n\r\n this.graph.traverseTree(vertexId, parentVertexId, function (vertex) {\r\n vertex.position.rotateAround(angle, center);\r\n\r\n for (var i = 0; i < vertex.value.anchoredRings.length; i++) {\r\n let ring = that.rings[vertex.value.anchoredRings[i]];\r\n\r\n if (ring) {\r\n ring.center.rotateAround(angle, center);\r\n }\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * Gets the overlap score of a subtree.\r\n *\r\n * @param {Number} vertexId A vertex id (the root of the sub-tree).\r\n * @param {Number} parentVertexId A vertex id in the previous direction of the subtree.\r\n * @param {Number[]} vertexOverlapScores An array containing the vertex overlap scores indexed by vertex id.\r\n * @returns {Object} An object containing the total overlap score and the center of mass of the subtree weighted by overlap score { value: 0.2, center: new Vector2() }.\r\n */\r\n getSubtreeOverlapScore(vertexId, parentVertexId, vertexOverlapScores) {\r\n let that = this;\r\n let score = 0;\r\n let center = new Vector2(0, 0);\r\n let count = 0;\r\n\r\n this.graph.traverseTree(vertexId, parentVertexId, function (vertex) {\r\n if (!vertex.value.isDrawn) {\r\n return;\r\n }\r\n\r\n let s = vertexOverlapScores[vertex.id];\r\n if (s > that.opts.overlapSensitivity) {\r\n score += s;\r\n count++;\r\n }\r\n\r\n let position = that.graph.vertices[vertex.id].position.clone();\r\n position.multiplyScalar(s)\r\n center.add(position);\r\n });\r\n\r\n center.divide(score);\r\n\r\n return {\r\n value: score / count,\r\n center: center\r\n };\r\n }\r\n\r\n /**\r\n * Returns the current (positioned vertices so far) center of mass.\r\n * \r\n * @returns {Vector2} The current center of mass.\r\n */\r\n getCurrentCenterOfMass() {\r\n let total = new Vector2(0, 0);\r\n let count = 0;\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertex = this.graph.vertices[i];\r\n\r\n if (vertex.positioned) {\r\n total.add(vertex.position);\r\n count++;\r\n }\r\n }\r\n\r\n return total.divide(count);\r\n }\r\n\r\n /**\r\n * Returns the current (positioned vertices so far) center of mass in the neighbourhood of a given position.\r\n *\r\n * @param {Vector2} vec The point at which to look for neighbours.\r\n * @param {Number} [r=currentBondLength*2.0] The radius of vertices to include.\r\n * @returns {Vector2} The current center of mass.\r\n */\r\n getCurrentCenterOfMassInNeigbourhood(vec, r = this.opts.bondLength * 2.0) {\r\n let total = new Vector2(0, 0);\r\n let count = 0;\r\n let rSq = r * r;\r\n\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertex = this.graph.vertices[i];\r\n\r\n if (vertex.positioned && vec.distanceSq(vertex.position) < rSq) {\r\n total.add(vertex.position);\r\n count++;\r\n }\r\n }\r\n\r\n return total.divide(count);\r\n }\r\n\r\n /**\r\n * Resolve primary (exact) overlaps, such as two vertices that are connected to the same ring vertex.\r\n */\r\n resolvePrimaryOverlaps() {\r\n let overlaps = Array();\r\n let done = Array(this.graph.vertices.length);\r\n\r\n // Looking for overlaps created by two bonds coming out of a ring atom, which both point straight\r\n // away from the ring and are thus perfectly overlapping.\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let ring = this.rings[i];\r\n\r\n for (var j = 0; j < ring.members.length; j++) {\r\n let vertex = this.graph.vertices[ring.members[j]];\r\n\r\n if (done[vertex.id]) {\r\n continue;\r\n }\r\n\r\n done[vertex.id] = true;\r\n\r\n let nonRingNeighbours = this.getNonRingNeighbours(vertex.id);\r\n\r\n if (nonRingNeighbours.length > 1) {\r\n // Look for rings where there are atoms with two bonds outside the ring (overlaps)\r\n let rings = Array();\r\n\r\n for (var k = 0; k < vertex.value.rings.length; k++) {\r\n rings.push(vertex.value.rings[k]);\r\n }\r\n\r\n overlaps.push({\r\n common: vertex,\r\n rings: rings,\r\n vertices: nonRingNeighbours\r\n });\r\n } else if (nonRingNeighbours.length === 1 && vertex.value.rings.length === 2) {\r\n // Look for bonds coming out of joined rings to adjust the angle, an example is: C1=CC(=CC=C1)[C@]12SCCN1CC1=CC=CC=C21\r\n // where the angle has to be adjusted to account for fused ring\r\n let rings = Array();\r\n \r\n for (var k = 0; k < vertex.value.rings.length; k++) {\r\n rings.push(vertex.value.rings[k]);\r\n }\r\n\r\n overlaps.push({\r\n common: vertex,\r\n rings: rings,\r\n vertices: nonRingNeighbours\r\n });\r\n }\r\n }\r\n }\r\n\r\n for (var i = 0; i < overlaps.length; i++) {\r\n let overlap = overlaps[i];\r\n\r\n if (overlap.vertices.length === 2) {\r\n let a = overlap.vertices[0];\r\n let b = overlap.vertices[1];\r\n\r\n if (!a.value.isDrawn || !b.value.isDrawn) {\r\n continue;\r\n }\r\n\r\n let angle = (2 * Math.PI - this.getRing(overlap.rings[0]).getAngle()) / 6.0;\r\n\r\n this.rotateSubtree(a.id, overlap.common.id, angle, overlap.common.position);\r\n this.rotateSubtree(b.id, overlap.common.id, -angle, overlap.common.position);\r\n\r\n // Decide which way to rotate the vertices depending on the effect it has on the overlap score\r\n let overlapScore = this.getOverlapScore();\r\n let subTreeOverlapA = this.getSubtreeOverlapScore(a.id, overlap.common.id, overlapScore.vertexScores);\r\n let subTreeOverlapB = this.getSubtreeOverlapScore(b.id, overlap.common.id, overlapScore.vertexScores);\r\n let total = subTreeOverlapA.value + subTreeOverlapB.value;\r\n\r\n this.rotateSubtree(a.id, overlap.common.id, -2.0 * angle, overlap.common.position);\r\n this.rotateSubtree(b.id, overlap.common.id, 2.0 * angle, overlap.common.position);\r\n\r\n overlapScore = this.getOverlapScore();\r\n subTreeOverlapA = this.getSubtreeOverlapScore(a.id, overlap.common.id, overlapScore.vertexScores);\r\n subTreeOverlapB = this.getSubtreeOverlapScore(b.id, overlap.common.id, overlapScore.vertexScores);\r\n\r\n if (subTreeOverlapA.value + subTreeOverlapB.value > total) {\r\n this.rotateSubtree(a.id, overlap.common.id, 2.0 * angle, overlap.common.position);\r\n this.rotateSubtree(b.id, overlap.common.id, -2.0 * angle, overlap.common.position);\r\n }\r\n } else if (overlap.vertices.length === 1) {\r\n if (overlap.rings.length === 2) {\r\n // TODO: Implement for more overlap resolution\r\n // console.log(overlap);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Resolve secondary overlaps. Those overlaps are due to the structure turning back on itself.\r\n *\r\n * @param {Object[]} scores An array of objects sorted descending by score.\r\n * @param {Number} scores[].id A vertex id.\r\n * @param {Number} scores[].score The overlap score associated with the vertex id.\r\n */\r\n resolveSecondaryOverlaps(scores) {\r\n for (var i = 0; i < scores.length; i++) {\r\n if (scores[i].score > this.opts.overlapSensitivity) {\r\n let vertex = this.graph.vertices[scores[i].id];\r\n\r\n if (vertex.isTerminal()) {\r\n let closest = this.getClosestVertex(vertex);\r\n\r\n if (closest) {\r\n // If one of the vertices is the first one, the previous vertex is not the central vertex but the dummy\r\n // so take the next rather than the previous, which is vertex 1\r\n let closestPosition = null;\r\n\r\n if (closest.isTerminal()) {\r\n closestPosition = closest.id === 0 ? this.graph.vertices[1].position : closest.previousPosition\r\n } else {\r\n closestPosition = closest.id === 0 ? this.graph.vertices[1].position : closest.position\r\n }\r\n\r\n let vertexPreviousPosition = vertex.id === 0 ? this.graph.vertices[1].position : vertex.previousPosition;\r\n\r\n vertex.position.rotateAwayFrom(closestPosition, vertexPreviousPosition, MathHelper.toRad(20));\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Get the last non-null or 0 angle vertex.\r\n * @param {Number} vertexId A vertex id.\r\n * @returns {Vertex} The last vertex with an angle that was not 0 or null.\r\n */\r\n getLastVertexWithAngle(vertexId) {\r\n let angle = 0;\r\n let vertex = null;\r\n\r\n while (!angle && vertexId) {\r\n vertex = this.graph.vertices[vertexId];\r\n angle = vertex.angle;\r\n vertexId = vertex.parentVertexId;\r\n }\r\n\r\n return vertex;\r\n }\r\n\r\n /**\r\n * Positiones the next vertex thus creating a bond.\r\n *\r\n * @param {Vertex} vertex A vertex.\r\n * @param {Vertex} [previousVertex=null] The previous vertex which has been positioned.\r\n * @param {Number} [angle=0.0] The (global) angle of the vertex.\r\n * @param {Boolean} [originShortest=false] Whether the origin is the shortest subtree in the branch.\r\n * @param {Boolean} [skipPositioning=false] Whether or not to skip positioning and just check the neighbours.\r\n */\r\n createNextBond(vertex, previousVertex = null, angle = 0.0, originShortest = false, skipPositioning = false) {\r\n if (vertex.positioned && !skipPositioning) {\r\n return;\r\n }\r\n\r\n // If the double bond config was set on this vertex, do not check later\r\n let doubleBondConfigSet = false;\r\n\r\n // Keeping track of configurations around double bonds\r\n if (previousVertex) {\r\n let edge = this.graph.getEdge(vertex.id, previousVertex.id);\r\n\r\n if ((edge.bondType === '/' || edge.bondType === '\\\\') && ++this.doubleBondConfigCount % 2 === 1) {\r\n if (this.doubleBondConfig === null) {\r\n this.doubleBondConfig = edge.bondType;\r\n doubleBondConfigSet = true;\r\n\r\n // Switch if the bond is a branch bond and previous vertex is the first\r\n // TODO: Why is it different with the first vertex?\r\n if (previousVertex.parentVertexId === null && vertex.value.branchBond) {\r\n if (this.doubleBondConfig === '/') {\r\n this.doubleBondConfig = '\\\\';\r\n } else if (this.doubleBondConfig === '\\\\') {\r\n this.doubleBondConfig = '/';\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n // If the current node is the member of one ring, then point straight away\r\n // from the center of the ring. However, if the current node is a member of\r\n // two rings, point away from the middle of the centers of the two rings\r\n if (!skipPositioning) {\r\n if (!previousVertex) {\r\n // Add a (dummy) previous position if there is no previous vertex defined\r\n // Since the first vertex is at (0, 0), create a vector at (bondLength, 0)\r\n // and rotate it by 90°\r\n\r\n let dummy = new Vector2(this.opts.bondLength, 0);\r\n dummy.rotate(MathHelper.toRad(-60));\r\n\r\n vertex.previousPosition = dummy;\r\n vertex.setPosition(this.opts.bondLength, 0);\r\n vertex.angle = MathHelper.toRad(-60);\r\n\r\n // Do not position the vertex if it belongs to a bridged ring that is positioned using a layout algorithm.\r\n if (vertex.value.bridgedRing === null) {\r\n vertex.positioned = true;\r\n }\r\n } else if (previousVertex.value.rings.length > 0) {\r\n let neighbours = previousVertex.neighbours;\r\n let joinedVertex = null;\r\n let pos = new Vector2(0.0, 0.0);\r\n\r\n if (previousVertex.value.bridgedRing === null && previousVertex.value.rings.length > 1) {\r\n for (var i = 0; i < neighbours.length; i++) {\r\n let neighbour = this.graph.vertices[neighbours[i]];\r\n if (ArrayHelper.containsAll(neighbour.value.rings, previousVertex.value.rings)) {\r\n joinedVertex = neighbour;\r\n break;\r\n }\r\n }\r\n }\r\n\r\n if (joinedVertex === null) {\r\n for (var i = 0; i < neighbours.length; i++) {\r\n let v = this.graph.vertices[neighbours[i]];\r\n\r\n if (v.positioned && this.areVerticesInSameRing(v, previousVertex)) {\r\n pos.add(Vector2.subtract(v.position, previousVertex.position));\r\n }\r\n }\r\n\r\n pos.invert().normalize().multiplyScalar(this.opts.bondLength).add(previousVertex.position);\r\n } else {\r\n pos = joinedVertex.position.clone().rotateAround(Math.PI, previousVertex.position);\r\n }\r\n\r\n vertex.previousPosition = previousVertex.position;\r\n vertex.setPositionFromVector(pos);\r\n vertex.positioned = true;\r\n } else {\r\n // If the previous vertex was not part of a ring, draw a bond based\r\n // on the global angle of the previous bond\r\n let v = new Vector2(this.opts.bondLength, 0);\r\n\r\n v.rotate(angle);\r\n v.add(previousVertex.position);\r\n\r\n vertex.setPositionFromVector(v);\r\n vertex.previousPosition = previousVertex.position;\r\n vertex.positioned = true;\r\n }\r\n }\r\n\r\n // Go to next vertex\r\n // If two rings are connected by a bond ...\r\n if (vertex.value.bridgedRing !== null) {\r\n let nextRing = this.getRing(vertex.value.bridgedRing);\r\n\r\n if (!nextRing.positioned) {\r\n let nextCenter = Vector2.subtract(vertex.previousPosition, vertex.position);\r\n\r\n nextCenter.invert();\r\n nextCenter.normalize();\r\n\r\n let r = MathHelper.polyCircumradius(this.opts.bondLength, nextRing.members.length);\r\n nextCenter.multiplyScalar(r);\r\n nextCenter.add(vertex.position);\r\n \r\n this.createRing(nextRing, nextCenter, vertex);\r\n }\r\n } else if (vertex.value.rings.length > 0) {\r\n let nextRing = this.getRing(vertex.value.rings[0]);\r\n \r\n if (!nextRing.positioned) {\r\n let nextCenter = Vector2.subtract(vertex.previousPosition, vertex.position);\r\n\r\n nextCenter.invert();\r\n nextCenter.normalize();\r\n\r\n let r = MathHelper.polyCircumradius(this.opts.bondLength, nextRing.getSize());\r\n\r\n nextCenter.multiplyScalar(r);\r\n nextCenter.add(vertex.position);\r\n \r\n this.createRing(nextRing, nextCenter, vertex);\r\n }\r\n } else {\r\n // Draw the non-ring vertices connected to this one \r\n let isStereoCenter = vertex.value.isStereoCenter;\r\n let tmpNeighbours = vertex.getNeighbours();\r\n let neighbours = Array();\r\n\r\n // Remove neighbours that are not drawn\r\n for (var i = 0; i < tmpNeighbours.length; i++) {\r\n if (this.graph.vertices[tmpNeighbours[i]].value.isDrawn) {\r\n neighbours.push(tmpNeighbours[i]);\r\n }\r\n }\r\n\r\n // Remove the previous vertex (which has already been drawn)\r\n if (previousVertex) {\r\n neighbours = ArrayHelper.remove(neighbours, previousVertex.id);\r\n }\r\n\r\n let previousAngle = vertex.getAngle();\r\n\r\n if (neighbours.length === 1) {\r\n let nextVertex = this.graph.vertices[neighbours[0]];\r\n\r\n // Make a single chain always cis except when there's a tribble (yes, this is a Star Trek reference) bond\r\n // or if there are successive double bonds. Added a ring check because if there is an aromatic ring the ring bond inside the ring counts as a double bond and leads to =-= being straight.\r\n if ((vertex.value.bondType === '#' || (previousVertex && previousVertex.value.bondType === '#')) ||\r\n vertex.value.bondType === '=' && previousVertex && previousVertex.value.rings.length === 0 &&\r\n previousVertex.value.bondType === '=' && vertex.value.branchBond !== '-') {\r\n vertex.value.drawExplicit = false;\r\n\r\n if (previousVertex) {\r\n let straightEdge1 = this.graph.getEdge(vertex.id, previousVertex.id);\r\n straightEdge1.center = true;\r\n }\r\n\r\n let straightEdge2 = this.graph.getEdge(vertex.id, nextVertex.id);\r\n straightEdge2.center = true;\r\n\r\n if (vertex.value.bondType === '#' || previousVertex && previousVertex.value.bondType === '#') {\r\n nextVertex.angle = 0.0;\r\n }\r\n\r\n nextVertex.drawExplicit = true;\r\n\r\n this.createNextBond(nextVertex, vertex, previousAngle + nextVertex.angle);\r\n } else if (previousVertex && previousVertex.value.rings.length > 0) {\r\n // If coming out of a ring, always draw away from the center of mass\r\n let proposedAngleA = MathHelper.toRad(60);\r\n let proposedAngleB = -proposedAngleA;\r\n\r\n let proposedVectorA = new Vector2(this.opts.bondLength, 0);\r\n let proposedVectorB = new Vector2(this.opts.bondLength, 0);\r\n\r\n proposedVectorA.rotate(proposedAngleA).add(vertex.position);\r\n proposedVectorB.rotate(proposedAngleB).add(vertex.position);\r\n\r\n // let centerOfMass = this.getCurrentCenterOfMassInNeigbourhood(vertex.position, 100);\r\n let centerOfMass = this.getCurrentCenterOfMass();\r\n let distanceA = proposedVectorA.distanceSq(centerOfMass);\r\n let distanceB = proposedVectorB.distanceSq(centerOfMass);\r\n\r\n nextVertex.angle = distanceA < distanceB ? proposedAngleB : proposedAngleA;\r\n\r\n this.createNextBond(nextVertex, vertex, previousAngle + nextVertex.angle);\r\n } else {\r\n let a = vertex.angle;\r\n // Take the min and max if the previous angle was in a 4-neighbourhood (90° angles)\r\n // TODO: If a is null or zero, it should be checked whether or not this one should go cis or trans, that is,\r\n // it should go into the oposite direction of the last non-null or 0 previous vertex / angle.\r\n if (previousVertex && previousVertex.neighbours.length > 3) {\r\n if (a > 0) {\r\n a = Math.min(1.0472, a);\r\n } else if (a < 0) {\r\n a = Math.max(-1.0472, a);\r\n } else {\r\n a = 1.0472;\r\n }\r\n } else if (!a) {\r\n let v = this.getLastVertexWithAngle(vertex.id);\r\n a = v.angle;\r\n\r\n if (!a) {\r\n a = 1.0472;\r\n }\r\n }\r\n\r\n // Handle configuration around double bonds\r\n if (previousVertex && !doubleBondConfigSet) {\r\n let bondType = this.graph.getEdge(vertex.id, nextVertex.id).bondType;\r\n\r\n if (bondType === '/') {\r\n if (this.doubleBondConfig === '/') {\r\n // Nothing to do since it will be trans per default\r\n } else if (this.doubleBondConfig === '\\\\') {\r\n a = -a;\r\n }\r\n this.doubleBondConfig = null;\r\n } else if (bondType === '\\\\') {\r\n if (this.doubleBondConfig === '/') {\r\n a = -a;\r\n } else if (this.doubleBondConfig === '\\\\') {\r\n // Nothing to do since it will be trans per default\r\n }\r\n this.doubleBondConfig = null;\r\n }\r\n }\r\n\r\n if (originShortest) {\r\n nextVertex.angle = a;\r\n } else {\r\n nextVertex.angle = -a;\r\n }\r\n\r\n this.createNextBond(nextVertex, vertex, previousAngle + nextVertex.angle);\r\n }\r\n } else if (neighbours.length === 2) {\r\n // If the previous vertex comes out of a ring, it doesn't have an angle set\r\n let a = vertex.angle;\r\n\r\n if (!a) {\r\n a = 1.0472;\r\n }\r\n\r\n // Check for the longer subtree - always go with cis for the longer subtree\r\n let subTreeDepthA = this.graph.getTreeDepth(neighbours[0], vertex.id);\r\n let subTreeDepthB = this.graph.getTreeDepth(neighbours[1], vertex.id);\r\n\r\n let l = this.graph.vertices[neighbours[0]];\r\n let r = this.graph.vertices[neighbours[1]];\r\n\r\n l.value.subtreeDepth = subTreeDepthA;\r\n r.value.subtreeDepth = subTreeDepthB;\r\n\r\n // Also get the subtree for the previous direction (this is important when\r\n // the previous vertex is the shortest path)\r\n let subTreeDepthC = this.graph.getTreeDepth(previousVertex ? previousVertex.id : null, vertex.id);\r\n if (previousVertex) {\r\n previousVertex.value.subtreeDepth = subTreeDepthC;\r\n }\r\n\r\n let cis = 0;\r\n let trans = 1;\r\n\r\n // Carbons go always cis\r\n if (r.value.element === 'C' && l.value.element !== 'C' && subTreeDepthB > 1 && subTreeDepthA < 5) {\r\n cis = 1;\r\n trans = 0;\r\n } else if (r.value.element !== 'C' && l.value.element === 'C' && subTreeDepthA > 1 && subTreeDepthB < 5) {\r\n cis = 0;\r\n trans = 1;\r\n } else if (subTreeDepthB > subTreeDepthA) {\r\n cis = 1;\r\n trans = 0;\r\n }\r\n\r\n let cisVertex = this.graph.vertices[neighbours[cis]];\r\n let transVertex = this.graph.vertices[neighbours[trans]];\r\n\r\n let edgeCis = this.graph.getEdge(vertex.id, cisVertex.id);\r\n let edgeTrans = this.graph.getEdge(vertex.id, transVertex.id);\r\n\r\n // If the origin tree is the shortest, make them the main chain\r\n let originShortest = false;\r\n if (subTreeDepthC < subTreeDepthA && subTreeDepthC < subTreeDepthB) {\r\n originShortest = true;\r\n }\r\n\r\n transVertex.angle = a;\r\n cisVertex.angle = -a;\r\n\r\n if (this.doubleBondConfig === '\\\\') {\r\n if (transVertex.value.branchBond === '\\\\') {\r\n transVertex.angle = -a;\r\n cisVertex.angle = a;\r\n }\r\n } else if (this.doubleBondConfig === '/') {\r\n if (transVertex.value.branchBond === '/') {\r\n transVertex.angle = -a;\r\n cisVertex.angle = a;\r\n }\r\n }\r\n\r\n this.createNextBond(transVertex, vertex, previousAngle + transVertex.angle, originShortest);\r\n this.createNextBond(cisVertex, vertex, previousAngle + cisVertex.angle, originShortest);\r\n } else if (neighbours.length === 3) {\r\n // The vertex with the longest sub-tree should always go straight\r\n let d1 = this.graph.getTreeDepth(neighbours[0], vertex.id);\r\n let d2 = this.graph.getTreeDepth(neighbours[1], vertex.id);\r\n let d3 = this.graph.getTreeDepth(neighbours[2], vertex.id);\r\n\r\n let s = this.graph.vertices[neighbours[0]];\r\n let l = this.graph.vertices[neighbours[1]];\r\n let r = this.graph.vertices[neighbours[2]];\r\n\r\n s.value.subtreeDepth = d1;\r\n l.value.subtreeDepth = d2;\r\n r.value.subtreeDepth = d3;\r\n\r\n if (d2 > d1 && d2 > d3) {\r\n s = this.graph.vertices[neighbours[1]];\r\n l = this.graph.vertices[neighbours[0]];\r\n r = this.graph.vertices[neighbours[2]];\r\n } else if (d3 > d1 && d3 > d2) {\r\n s = this.graph.vertices[neighbours[2]];\r\n l = this.graph.vertices[neighbours[0]];\r\n r = this.graph.vertices[neighbours[1]];\r\n }\r\n\r\n // Create a cross if more than one subtree is of length > 1\r\n // or the vertex is connected to a ring\r\n if (previousVertex &&\r\n previousVertex.value.rings.length < 1 &&\r\n s.value.rings.length < 1 &&\r\n l.value.rings.length < 1 &&\r\n r.value.rings.length < 1 &&\r\n this.graph.getTreeDepth(l.id, vertex.id) === 1 &&\r\n this.graph.getTreeDepth(r.id, vertex.id) === 1 &&\r\n this.graph.getTreeDepth(s.id, vertex.id) > 1) {\r\n\r\n s.angle = -vertex.angle;\r\n if (vertex.angle >= 0) {\r\n l.angle = MathHelper.toRad(30);\r\n r.angle = MathHelper.toRad(90);\r\n } else {\r\n l.angle = -MathHelper.toRad(30);\r\n r.angle = -MathHelper.toRad(90);\r\n }\r\n\r\n this.createNextBond(s, vertex, previousAngle + s.angle);\r\n this.createNextBond(l, vertex, previousAngle + l.angle);\r\n this.createNextBond(r, vertex, previousAngle + r.angle);\r\n } else {\r\n s.angle = 0.0;\r\n l.angle = MathHelper.toRad(90);\r\n r.angle = -MathHelper.toRad(90);\r\n\r\n this.createNextBond(s, vertex, previousAngle + s.angle);\r\n this.createNextBond(l, vertex, previousAngle + l.angle);\r\n this.createNextBond(r, vertex, previousAngle + r.angle);\r\n }\r\n } else if (neighbours.length === 4) {\r\n // The vertex with the longest sub-tree should always go to the reflected opposide direction\r\n let d1 = this.graph.getTreeDepth(neighbours[0], vertex.id);\r\n let d2 = this.graph.getTreeDepth(neighbours[1], vertex.id);\r\n let d3 = this.graph.getTreeDepth(neighbours[2], vertex.id);\r\n let d4 = this.graph.getTreeDepth(neighbours[3], vertex.id);\r\n\r\n let w = this.graph.vertices[neighbours[0]];\r\n let x = this.graph.vertices[neighbours[1]];\r\n let y = this.graph.vertices[neighbours[2]];\r\n let z = this.graph.vertices[neighbours[3]];\r\n\r\n w.value.subtreeDepth = d1;\r\n x.value.subtreeDepth = d2;\r\n y.value.subtreeDepth = d3;\r\n z.value.subtreeDepth = d4;\r\n\r\n if (d2 > d1 && d2 > d3 && d2 > d4) {\r\n w = this.graph.vertices[neighbours[1]];\r\n x = this.graph.vertices[neighbours[0]];\r\n y = this.graph.vertices[neighbours[2]];\r\n z = this.graph.vertices[neighbours[3]];\r\n } else if (d3 > d1 && d3 > d2 && d3 > d4) {\r\n w = this.graph.vertices[neighbours[2]];\r\n x = this.graph.vertices[neighbours[0]];\r\n y = this.graph.vertices[neighbours[1]];\r\n z = this.graph.vertices[neighbours[3]];\r\n } else if (d4 > d1 && d4 > d2 && d4 > d3) {\r\n w = this.graph.vertices[neighbours[3]];\r\n x = this.graph.vertices[neighbours[0]];\r\n y = this.graph.vertices[neighbours[1]];\r\n z = this.graph.vertices[neighbours[2]];\r\n }\r\n\r\n w.angle = -MathHelper.toRad(36);\r\n x.angle = MathHelper.toRad(36);\r\n y.angle = -MathHelper.toRad(108);\r\n z.angle = MathHelper.toRad(108);\r\n\r\n this.createNextBond(w, vertex, previousAngle + w.angle);\r\n this.createNextBond(x, vertex, previousAngle + x.angle);\r\n this.createNextBond(y, vertex, previousAngle + y.angle);\r\n this.createNextBond(z, vertex, previousAngle + z.angle);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Gets the vetex sharing the edge that is the common bond of two rings.\r\n *\r\n * @param {Vertex} vertex A vertex.\r\n * @returns {(Number|null)} The id of a vertex sharing the edge that is the common bond of two rings with the vertex provided or null, if none.\r\n */\r\n getCommonRingbondNeighbour(vertex) {\r\n let neighbours = vertex.neighbours;\r\n\r\n for (var i = 0; i < neighbours.length; i++) {\r\n let neighbour = this.graph.vertices[neighbours[i]];\r\n\r\n if (ArrayHelper.containsAll(neighbour.value.rings, vertex.value.rings)) {\r\n return neighbour;\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n\r\n /**\r\n * Check if a vector is inside any ring.\r\n *\r\n * @param {Vector2} vec A vector.\r\n * @returns {Boolean} A boolean indicating whether or not the point (vector) is inside any of the rings associated with the current molecule.\r\n */\r\n isPointInRing(vec) {\r\n for (var i = 0; i < this.rings.length; i++) {\r\n let ring = this.rings[i];\r\n\r\n if (!ring.positioned) {\r\n continue;\r\n }\r\n\r\n let radius = MathHelper.polyCircumradius(this.opts.bondLength, ring.getSize());\r\n let radiusSq = radius * radius;\r\n\r\n if (vec.distanceSq(ring.center) < radiusSq) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Check whether or not an edge is part of a ring.\r\n *\r\n * @param {Edge} edge An edge.\r\n * @returns {Boolean} A boolean indicating whether or not the edge is part of a ring.\r\n */\r\n isEdgeInRing(edge) {\r\n let source = this.graph.vertices[edge.sourceId];\r\n let target = this.graph.vertices[edge.targetId];\r\n\r\n return this.areVerticesInSameRing(source, target);\r\n }\r\n\r\n /**\r\n * Check whether or not an edge is rotatable.\r\n *\r\n * @param {Edge} edge An edge.\r\n * @returns {Boolean} A boolean indicating whether or not the edge is rotatable.\r\n */\r\n isEdgeRotatable(edge) {\r\n let vertexA = this.graph.vertices[edge.sourceId];\r\n let vertexB = this.graph.vertices[edge.targetId];\r\n\r\n // Only single bonds are rotatable\r\n if (edge.bondType !== '-') {\r\n return false;\r\n }\r\n\r\n // Do not rotate edges that have a further single bond to each side - do that!\r\n // If the bond is terminal, it doesn't make sense to rotate it\r\n // if (vertexA.getNeighbourCount() + vertexB.getNeighbourCount() < 5) {\r\n // return false;\r\n // }\r\n\r\n if (vertexA.isTerminal() || vertexB.isTerminal()) {\r\n return false;\r\n }\r\n\r\n // Ringbonds are not rotatable\r\n if (vertexA.value.rings.length > 0 && vertexB.value.rings.length > 0 &&\r\n this.areVerticesInSameRing(vertexA, vertexB)) {\r\n return false;\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Check whether or not a ring is an implicitly defined aromatic ring (lower case smiles).\r\n *\r\n * @param {Ring} ring A ring.\r\n * @returns {Boolean} A boolean indicating whether or not a ring is implicitly defined as aromatic.\r\n */\r\n isRingAromatic(ring) {\r\n for (var i = 0; i < ring.members.length; i++) {\r\n let vertex = this.graph.vertices[ring.members[i]];\r\n\r\n if (!vertex.value.isPartOfAromaticRing) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Get the normals of an edge.\r\n *\r\n * @param {Edge} edge An edge.\r\n * @returns {Vector2[]} An array containing two vectors, representing the normals.\r\n */\r\n getEdgeNormals(edge) {\r\n let v1 = this.graph.vertices[edge.sourceId].position;\r\n let v2 = this.graph.vertices[edge.targetId].position;\r\n\r\n // Get the normalized normals for the edge\r\n let normals = Vector2.units(v1, v2);\r\n\r\n return normals;\r\n }\r\n\r\n /**\r\n * Returns an array of vertices that are neighbouring a vertix but are not members of a ring (including bridges).\r\n *\r\n * @param {Number} vertexId A vertex id.\r\n * @returns {Vertex[]} An array of vertices.\r\n */\r\n getNonRingNeighbours(vertexId) {\r\n let nrneighbours = Array();\r\n let vertex = this.graph.vertices[vertexId];\r\n let neighbours = vertex.neighbours;\r\n\r\n for (var i = 0; i < neighbours.length; i++) {\r\n let neighbour = this.graph.vertices[neighbours[i]];\r\n let nIntersections = ArrayHelper.intersection(vertex.value.rings, neighbour.value.rings).length;\r\n\r\n if (nIntersections === 0 && neighbour.value.isBridge == false) {\r\n nrneighbours.push(neighbour);\r\n }\r\n }\r\n\r\n return nrneighbours;\r\n }\r\n\r\n /**\r\n * Annotaed stereochemistry information for visualization.\r\n */\r\n annotateStereochemistry() {\r\n let maxDepth = 10;\r\n\r\n // For each stereo-center\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n let vertex = this.graph.vertices[i];\r\n\r\n if (!vertex.value.isStereoCenter) {\r\n continue;\r\n }\r\n\r\n let neighbours = vertex.getNeighbours();\r\n let nNeighbours = neighbours.length;\r\n let priorities = Array(nNeighbours);\r\n\r\n for (var j = 0; j < nNeighbours; j++) {\r\n let visited = new Uint8Array(this.graph.vertices.length);\r\n let priority = Array(Array());\r\n visited[vertex.id] = 1;\r\n\r\n this.visitStereochemistry(neighbours[j], vertex.id, visited, priority, maxDepth, 0);\r\n\r\n // Sort each level according to atomic number\r\n for (var k = 0; k < priority.length; k++) {\r\n priority[k].sort(function (a, b) {\r\n return b - a\r\n });\r\n }\r\n\r\n priorities[j] = [j, priority];\r\n }\r\n\r\n let maxLevels = 0;\r\n let maxEntries = 0;\r\n for (var j = 0; j < priorities.length; j++) {\r\n if (priorities[j][1].length > maxLevels) {\r\n maxLevels = priorities[j][1].length;\r\n }\r\n\r\n for (var k = 0; k < priorities[j][1].length; k++) {\r\n if (priorities[j][1][k].length > maxEntries) {\r\n maxEntries = priorities[j][1][k].length;\r\n }\r\n }\r\n }\r\n\r\n for (var j = 0; j < priorities.length; j++) {\r\n let diff = maxLevels - priorities[j][1].length;\r\n for (var k = 0; k < diff; k++) {\r\n priorities[j][1].push([]);\r\n }\r\n\r\n // Break ties by the position in the SMILES string as per specification\r\n priorities[j][1].push([neighbours[j]]);\r\n\r\n // Make all same length. Fill with zeroes.\r\n for (var k = 0; k < priorities[j][1].length; k++) {\r\n let diff = maxEntries - priorities[j][1][k].length;\r\n\r\n for (var l = 0; l < diff; l++) {\r\n priorities[j][1][k].push(0);\r\n }\r\n }\r\n }\r\n\r\n priorities.sort(function (a, b) {\r\n for (var j = 0; j < a[1].length; j++) {\r\n for (var k = 0; k < a[1][j].length; k++) {\r\n if (a[1][j][k] > b[1][j][k]) {\r\n return -1;\r\n } else if (a[1][j][k] < b[1][j][k]) {\r\n return 1;\r\n }\r\n }\r\n }\r\n\r\n return 0;\r\n });\r\n\r\n let order = new Uint8Array(nNeighbours);\r\n for (var j = 0; j < nNeighbours; j++) {\r\n order[j] = priorities[j][0];\r\n vertex.value.priority = j;\r\n }\r\n\r\n // Check the angles between elements 0 and 1, and 0 and 2 to determine whether they are\r\n // drawn cw or ccw\r\n // TODO: OC(Cl)=[C@]=C(C)F currently fails here, however this is, IMHO, not a valid SMILES.\r\n let posA = this.graph.vertices[neighbours[order[0]]].position;\r\n let posB = this.graph.vertices[neighbours[order[1]]].position;\r\n let posC = this.graph.vertices[neighbours[order[2]]].position;\r\n\r\n let cwA = posA.relativeClockwise(posB, vertex.position);\r\n let cwB = posA.relativeClockwise(posC, vertex.position);\r\n\r\n // If the second priority is clockwise from the first, the ligands are drawn clockwise, since\r\n // The hydrogen can be drawn on either side\r\n let isCw = cwA === -1;\r\n\r\n let rotation = vertex.value.bracket.chirality === '@' ? -1 : 1;\r\n let rs = MathHelper.parityOfPermutation(order) * rotation === 1 ? 'R' : 'S';\r\n\r\n // Flip the hydrogen direction when the drawing doesn't match the chirality.\r\n let wedgeA = 'down';\r\n let wedgeB = 'up';\r\n if (isCw && rs !== 'R' || !isCw && rs !== 'S') {\r\n vertex.value.hydrogenDirection = 'up';\r\n wedgeA = 'up';\r\n wedgeB = 'down';\r\n }\r\n\r\n if (vertex.value.hasHydrogen) {\r\n this.graph.getEdge(vertex.id, neighbours[order[order.length - 1]]).wedge = wedgeA;\r\n }\r\n\r\n // Get the shortest subtree to flip up / down. Ignore lowest priority\r\n // The rules are following:\r\n // 1. Do not draw wedge between two stereocenters\r\n // 2. Heteroatoms\r\n // 3. Draw outside ring\r\n // 4. Shortest subtree\r\n\r\n let wedgeOrder = new Array(neighbours.length - 1);\r\n let showHydrogen = vertex.value.rings.length > 1 && vertex.value.hasHydrogen;\r\n let offset = vertex.value.hasHydrogen ? 1 : 0;\r\n\r\n for (var j = 0; j < order.length - offset; j++) {\r\n wedgeOrder[j] = new Uint32Array(2);\r\n let neighbour = this.graph.vertices[neighbours[order[j]]];\r\n wedgeOrder[j][0] += neighbour.value.isStereoCenter ? 0 : 100000;\r\n // wedgeOrder[j][0] += neighbour.value.rings.length > 0 ? 0 : 10000;\r\n // Only add if in same ring, unlike above\r\n wedgeOrder[j][0] += this.areVerticesInSameRing(neighbour, vertex) ? 0 : 10000;\r\n wedgeOrder[j][0] += neighbour.value.isHeteroAtom() ? 1000 : 0;\r\n wedgeOrder[j][0] -= neighbour.value.subtreeDepth === 0 ? 1000 : 0;\r\n wedgeOrder[j][0] += 1000 - neighbour.value.subtreeDepth;\r\n wedgeOrder[j][1] = neighbours[order[j]];\r\n }\r\n \r\n\r\n wedgeOrder.sort(function (a, b) {\r\n if (a[0] > b[0]) {\r\n return -1;\r\n } else if (a[0] < b[0]) {\r\n return 1;\r\n }\r\n return 0;\r\n });\r\n\r\n // If all neighbours are in a ring, do not draw wedge, the hydrogen will be drawn.\r\n if (!showHydrogen) {\r\n let wedgeId = wedgeOrder[0][1];\r\n\r\n if (vertex.value.hasHydrogen) {\r\n this.graph.getEdge(vertex.id, wedgeId).wedge = wedgeB; \r\n } else {\r\n let wedge = wedgeB; \r\n\r\n for (var j = order.length - 1; j >= 0; j--) {\r\n if (wedge === wedgeA) {\r\n wedge = wedgeB;\r\n } else {\r\n wedge = wedgeA;\r\n }\r\n if (neighbours[order[j]] === wedgeId) {\r\n break;\r\n }\r\n }\r\n\r\n this.graph.getEdge(vertex.id, wedgeId).wedge = wedge;\r\n }\r\n }\r\n\r\n vertex.value.chirality = rs;\r\n }\r\n }\r\n\r\n /**\r\n * \r\n * \r\n * @param {Number} vertexId The id of a vertex.\r\n * @param {(Number|null)} previousVertexId The id of the parent vertex of the vertex.\r\n * @param {Uint8Array} visited An array containing the visited flag for all vertices in the graph.\r\n * @param {Array} priority An array of arrays storing the atomic numbers for each level.\r\n * @param {Number} maxDepth The maximum depth.\r\n * @param {Number} depth The current depth.\r\n */\r\n visitStereochemistry(vertexId, previousVertexId, visited, priority, maxDepth, depth, parentAtomicNumber = 0) {\r\n visited[vertexId] = 1;\r\n let vertex = this.graph.vertices[vertexId];\r\n let atomicNumber = vertex.value.getAtomicNumber();\r\n\r\n if (priority.length <= depth) {\r\n priority.push(Array());\r\n }\r\n\r\n for (var i = 0; i < this.graph.getEdge(vertexId, previousVertexId).weight; i++) {\r\n priority[depth].push(parentAtomicNumber * 1000 + atomicNumber);\r\n }\r\n\r\n let neighbours = this.graph.vertices[vertexId].neighbours;\r\n\r\n for (var i = 0; i < neighbours.length; i++) {\r\n if (visited[neighbours[i]] !== 1 && depth < maxDepth - 1) {\r\n this.visitStereochemistry(neighbours[i], vertexId, visited.slice(), priority, maxDepth, depth + 1, atomicNumber);\r\n }\r\n }\r\n\r\n // Valences are filled with hydrogens and passed to the next level.\r\n if (depth < maxDepth - 1) {\r\n let bonds = 0;\r\n\r\n for (var i = 0; i < neighbours.length; i++) {\r\n bonds += this.graph.getEdge(vertexId, neighbours[i]).weight;\r\n }\r\n\r\n for (var i = 0; i < vertex.value.getMaxBonds() - bonds; i++) {\r\n if (priority.length <= depth + 1) {\r\n priority.push(Array());\r\n }\r\n\r\n priority[depth + 1].push(atomicNumber * 1000 + 1);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Creates pseudo-elements (such as Et, Me, Ac, Bz, ...) at the position of the carbon sets\r\n * the involved atoms not to be displayed.\r\n */\r\n initPseudoElements() {\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n const vertex = this.graph.vertices[i];\r\n const neighbourIds = vertex.neighbours;\r\n let neighbours = Array(neighbourIds.length);\r\n\r\n for (var j = 0; j < neighbourIds.length; j++) {\r\n neighbours[j] = this.graph.vertices[neighbourIds[j]];\r\n }\r\n\r\n // Ignore atoms that have less than 3 neighbours, except if\r\n // the vertex is connected to a ring and has two neighbours\r\n if (vertex.getNeighbourCount() < 3 || vertex.value.rings.length > 0) {\r\n continue;\r\n }\r\n\r\n // TODO: This exceptions should be handled more elegantly (via config file?)\r\n\r\n // Ignore phosphates (especially for triphosphates)\r\n if (vertex.value.element === 'P') {\r\n continue;\r\n }\r\n\r\n // Ignore also guanidine\r\n if (vertex.value.element === 'C' && neighbours.length === 3 &&\r\n neighbours[0].value.element === 'N' && neighbours[1].value.element === 'N' && neighbours[2].value.element === 'N') {\r\n continue;\r\n }\r\n\r\n // Continue if there are less than two heteroatoms\r\n // or if a neighbour has more than 1 neighbour\r\n let heteroAtomCount = 0;\r\n let ctn = 0;\r\n\r\n for (var j = 0; j < neighbours.length; j++) {\r\n let neighbour = neighbours[j];\r\n let neighbouringElement = neighbour.value.element;\r\n let neighbourCount = neighbour.getNeighbourCount();\r\n\r\n if (neighbouringElement !== 'C' && neighbouringElement !== 'H' &&\r\n neighbourCount === 1) {\r\n heteroAtomCount++;\r\n }\r\n\r\n if (neighbourCount > 1) {\r\n ctn++;\r\n }\r\n }\r\n\r\n if (ctn > 1 || heteroAtomCount < 2) {\r\n continue;\r\n }\r\n\r\n // Get the previous atom (the one which is not terminal)\r\n let previous = null;\r\n\r\n for (var j = 0; j < neighbours.length; j++) {\r\n let neighbour = neighbours[j];\r\n\r\n if (neighbour.getNeighbourCount() > 1) {\r\n previous = neighbour;\r\n }\r\n }\r\n\r\n for (var j = 0; j < neighbours.length; j++) {\r\n let neighbour = neighbours[j];\r\n\r\n if (neighbour.getNeighbourCount() > 1) {\r\n continue;\r\n }\r\n\r\n neighbour.value.isDrawn = false;\r\n\r\n let hydrogens = Atom.maxBonds[neighbour.value.element] - neighbour.value.bondCount;\r\n let charge = '';\r\n\r\n if (neighbour.value.bracket) {\r\n hydrogens = neighbour.value.bracket.hcount;\r\n charge = neighbour.value.bracket.charge || 0;\r\n }\r\n\r\n vertex.value.attachPseudoElement(neighbour.value.element, previous ? previous.value.element : null, hydrogens, charge);\r\n }\r\n }\r\n\r\n // The second pass\r\n for (var i = 0; i < this.graph.vertices.length; i++) {\r\n const vertex = this.graph.vertices[i];\r\n const atom = vertex.value;\r\n const element = atom.element;\r\n\r\n if (element === 'C' || element === 'H' || !atom.isDrawn) {\r\n continue;\r\n }\r\n\r\n const neighbourIds = vertex.neighbours;\r\n let neighbours = Array(neighbourIds.length);\r\n\r\n for (var j = 0; j < neighbourIds.length; j++) {\r\n neighbours[j] = this.graph.vertices[neighbourIds[j]];\r\n }\r\n\r\n for (var j = 0; j < neighbours.length; j++) {\r\n let neighbour = neighbours[j].value;\r\n\r\n if (!neighbour.hasAttachedPseudoElements || neighbour.getAttachedPseudoElementsCount() !== 2) {\r\n continue;\r\n }\r\n\r\n const pseudoElements = neighbour.getAttachedPseudoElements();\r\n\r\n if (pseudoElements.hasOwnProperty('0O') && pseudoElements.hasOwnProperty('3C')) {\r\n neighbour.isDrawn = false;\r\n vertex.value.attachPseudoElement('Ac', '', 0);\r\n }\r\n }\r\n }\r\n }\r\n}\r\n\r\nmodule.exports = Drawer;","//@ts-check\r\n\r\n/** \r\n * A class representing an edge. \r\n * \r\n * @property {Number} id The id of this edge.\r\n * @property {Number} sourceId The id of the source vertex.\r\n * @property {Number} targetId The id of the target vertex.\r\n * @property {Number} weight The weight of this edge. That is, the degree of the bond (single bond = 1, double bond = 2, etc).\r\n * @property {String} [bondType='-'] The bond type of this edge.\r\n * @property {Boolean} [isPartOfAromaticRing=false] Whether or not this edge is part of an aromatic ring.\r\n * @property {Boolean} [center=false] Wheter or not the bond is centered. For example, this affects straight double bonds.\r\n * @property {String} [wedge=''] Wedge direction. Either '', 'up' or 'down'\r\n */\r\nclass Edge {\r\n /**\r\n * The constructor for the class Edge.\r\n *\r\n * @param {Number} sourceId A vertex id.\r\n * @param {Number} targetId A vertex id.\r\n * @param {Number} [weight=1] The weight of the edge.\r\n */\r\n constructor(sourceId, targetId, weight = 1) {\r\n this.id = null;\r\n this.sourceId = sourceId;\r\n this.targetId = targetId;\r\n this.weight = weight;\r\n this.bondType = '-';\r\n this.isPartOfAromaticRing = false;\r\n this.center = false;\r\n this.wedge = '';\r\n }\r\n\r\n /**\r\n * Set the bond type of this edge. This also sets the edge weight.\r\n * @param {String} bondType \r\n */\r\n setBondType(bondType) {\r\n this.bondType = bondType;\r\n this.weight = Edge.bonds[bondType];\r\n }\r\n\r\n /**\r\n * An object mapping the bond type to the number of bonds.\r\n *\r\n * @returns {Object} The object containing the map.\r\n */\r\n static get bonds() {\r\n return {\r\n '-': 1,\r\n '/': 1,\r\n '\\\\': 1,\r\n '=': 2,\r\n '#': 3,\r\n '$': 4\r\n }\r\n }\r\n}\r\n\r\nmodule.exports = Edge","//@ts-check\r\nconst MathHelper = require('./MathHelper')\r\nconst Vector2 = require('./Vector2')\r\nconst Vertex = require('./Vertex')\r\nconst Edge = require('./Edge')\r\nconst Ring = require('./Ring')\r\nconst Atom = require('./Atom')\r\n\r\n/** \r\n * A class representing the molecular graph. \r\n * \r\n * @property {Vertex[]} vertices The vertices of the graph.\r\n * @property {Edge[]} edges The edges of this graph.\r\n * @property {Object} vertexIdsToEdgeId A map mapping vertex ids to the edge between the two vertices. The key is defined as vertexAId + '_' + vertexBId.\r\n * @property {Boolean} isometric A boolean indicating whether or not the SMILES associated with this graph is isometric.\r\n */\r\nclass Graph {\r\n /**\r\n * The constructor of the class Graph.\r\n * \r\n * @param {Object} parseTree A SMILES parse tree.\r\n * @param {Boolean} [isomeric=false] A boolean specifying whether or not the SMILES is isomeric.\r\n */\r\n constructor(parseTree, isomeric = false) {\r\n this.vertices = Array();\r\n this.edges = Array();\r\n this.vertexIdsToEdgeId = {};\r\n this.isomeric = isomeric;\r\n\r\n // Used for the bridge detection algorithm\r\n this._time = 0;\r\n this._init(parseTree);\r\n }\r\n\r\n /**\r\n * PRIVATE FUNCTION. Initializing the graph from the parse tree.\r\n *\r\n * @param {Object} node The current node in the parse tree.\r\n * @param {Number} parentVertexId=null The id of the previous vertex.\r\n * @param {Boolean} isBranch=false Whether or not the bond leading to this vertex is a branch bond. Branches are represented by parentheses in smiles (e.g. CC(O)C).\r\n */\r\n _init(node, order = 0, parentVertexId = null, isBranch = false) {\r\n // Create a new vertex object\r\n let atom = new Atom(node.atom.element ? node.atom.element : node.atom, node.bond);\r\n\r\n atom.branchBond = node.branchBond;\r\n atom.ringbonds = node.ringbonds;\r\n atom.bracket = node.atom.element ? node.atom : null;\r\n\r\n let vertex = new Vertex(atom);\r\n let parentVertex = this.vertices[parentVertexId];\r\n\r\n this.addVertex(vertex);\r\n\r\n // Add the id of this node to the parent as child\r\n if (parentVertexId !== null) {\r\n vertex.setParentVertexId(parentVertexId);\r\n vertex.value.addNeighbouringElement(parentVertex.value.element);\r\n parentVertex.addChild(vertex.id);\r\n parentVertex.value.addNeighbouringElement(atom.element);\r\n\r\n // In addition create a spanningTreeChildren property, which later will\r\n // not contain the children added through ringbonds\r\n parentVertex.spanningTreeChildren.push(vertex.id);\r\n\r\n // Add edge between this node and its parent\r\n let edge = new Edge(parentVertexId, vertex.id, 1);\r\n let vertexId = null;\r\n\r\n if (isBranch) {\r\n edge.setBondType(vertex.value.branchBond || '-');\r\n vertexId = vertex.id;\r\n edge.setBondType(vertex.value.branchBond || '-');\r\n vertexId = vertex.id;\r\n } else {\r\n edge.setBondType(parentVertex.value.bondType || '-');\r\n vertexId = parentVertex.id;\r\n }\r\n\r\n let edgeId = this.addEdge(edge);\r\n }\r\n\r\n let offset = node.ringbondCount + 1;\r\n\r\n if (atom.bracket) {\r\n offset += atom.bracket.hcount;\r\n }\r\n\r\n let stereoHydrogens = 0;\r\n if (atom.bracket && atom.bracket.chirality) {\r\n atom.isStereoCenter = true;\r\n stereoHydrogens = atom.bracket.hcount;\r\n for (var i = 0; i < stereoHydrogens; i++) {\r\n this._init({\r\n atom: 'H',\r\n isBracket: 'false',\r\n branches: Array(),\r\n branchCount: 0,\r\n ringbonds: Array(),\r\n ringbondCount: false,\r\n next: null,\r\n hasNext: false,\r\n bond: '-'\r\n }, i, vertex.id, true);\r\n }\r\n }\r\n\r\n for (var i = 0; i < node.branchCount; i++) {\r\n this._init(node.branches[i], i + offset, vertex.id, true);\r\n }\r\n\r\n if (node.hasNext) {\r\n this._init(node.next, node.branchCount + offset, vertex.id);\r\n }\r\n }\r\n\r\n /**\r\n * Clears all the elements in this graph (edges and vertices).\r\n */\r\n clear() {\r\n this.vertices = Array();\r\n this.edges = Array();\r\n this.vertexIdsToEdgeId = {};\r\n }\r\n\r\n /**\r\n * Add a vertex to the graph.\r\n *\r\n * @param {Vertex} vertex A new vertex.\r\n * @returns {Number} The vertex id of the new vertex.\r\n */\r\n addVertex(vertex) {\r\n vertex.id = this.vertices.length;\r\n this.vertices.push(vertex);\r\n\r\n return vertex.id;\r\n }\r\n\r\n /**\r\n * Add an edge to the graph.\r\n *\r\n * @param {Edge} edge A new edge.\r\n * @returns {Number} The edge id of the new edge.\r\n */\r\n addEdge(edge) {\r\n let source = this.vertices[edge.sourceId];\r\n let target = this.vertices[edge.targetId];\r\n\r\n edge.id = this.edges.length;\r\n this.edges.push(edge);\r\n\r\n this.vertexIdsToEdgeId[edge.sourceId + '_' + edge.targetId] = edge.id;\r\n this.vertexIdsToEdgeId[edge.targetId + '_' + edge.sourceId] = edge.id;\r\n edge.isPartOfAromaticRing = source.value.isPartOfAromaticRing && target.value.isPartOfAromaticRing;\r\n\r\n source.value.bondCount += edge.weight;\r\n target.value.bondCount += edge.weight;\r\n\r\n source.edges.push(edge.id);\r\n target.edges.push(edge.id);\r\n\r\n return edge.id;\r\n }\r\n\r\n /**\r\n * Returns the edge between two given vertices.\r\n *\r\n * @param {Number} vertexIdA A vertex id.\r\n * @param {Number} vertexIdB A vertex id.\r\n * @returns {(Edge|null)} The edge or, if no edge can be found, null.\r\n */\r\n getEdge(vertexIdA, vertexIdB) {\r\n let edgeId = this.vertexIdsToEdgeId[vertexIdA + '_' + vertexIdB];\r\n\r\n return edgeId === undefined ? null : this.edges[edgeId];\r\n }\r\n\r\n /**\r\n * Returns the ids of edges connected to a vertex.\r\n *\r\n * @param {Number} vertexId A vertex id.\r\n * @returns {Number[]} An array containing the ids of edges connected to the vertex.\r\n */\r\n getEdges(vertexId) {\r\n let edgeIds = Array();\r\n let vertex = this.vertices[vertexId];\r\n\r\n for (var i = 0; i < vertex.neighbours.length; i++) {\r\n edgeIds.push(this.vertexIdsToEdgeId[vertexId + '_' + vertex.neighbours[i]]);\r\n }\r\n\r\n return edgeIds;\r\n }\r\n\r\n\r\n /**\r\n * Check whether or not two vertices are connected by an edge.\r\n *\r\n * @param {Number} vertexIdA A vertex id.\r\n * @param {Number} vertexIdB A vertex id.\r\n * @returns {Boolean} A boolean indicating whether or not two vertices are connected by an edge.\r\n */\r\n hasEdge(vertexIdA, vertexIdB) {\r\n return this.vertexIdsToEdgeId[vertexIdA + '_' + vertexIdB] !== undefined\r\n }\r\n\r\n /**\r\n * Returns an array containing the vertex ids of this graph.\r\n * \r\n * @returns {Number[]} An array containing all vertex ids of this graph.\r\n */\r\n getVertexList() {\r\n let arr = [this.vertices.length];\r\n\r\n for (var i = 0; i < this.vertices.length; i++) {\r\n arr[i] = this.vertices[i].id;\r\n }\r\n\r\n return arr;\r\n }\r\n\r\n /**\r\n * Returns an array containing source, target arrays of this graphs edges.\r\n * \r\n * @returns {Array[]} An array containing source, target arrays of this graphs edges. Example: [ [ 2, 5 ], [ 6, 9 ] ].\r\n */\r\n getEdgeList() {\r\n let arr = Array(this.edges.length);\r\n\r\n for (var i = 0; i < this.edges.length; i++) {\r\n arr[i] = [this.edges[i].sourceId, this.edges[i].targetId];\r\n }\r\n\r\n return arr;\r\n }\r\n\r\n /**\r\n * Get the adjacency matrix of the graph.\r\n * \r\n * @returns {Array[]} The adjancency matrix of the molecular graph.\r\n */\r\n getAdjacencyMatrix() {\r\n let length = this.vertices.length;\r\n let adjacencyMatrix = Array(length);\r\n\r\n for (var i = 0; i < length; i++) {\r\n adjacencyMatrix[i] = new Array(length);\r\n adjacencyMatrix[i].fill(0);\r\n }\r\n\r\n for (var i = 0; i < this.edges.length; i++) {\r\n let edge = this.edges[i];\r\n\r\n adjacencyMatrix[edge.sourceId][edge.targetId] = 1;\r\n adjacencyMatrix[edge.targetId][edge.sourceId] = 1;\r\n }\r\n\r\n return adjacencyMatrix;\r\n }\r\n\r\n /**\r\n * Get the adjacency matrix of the graph with all bridges removed (thus the components). Thus the remaining vertices are all part of ring systems.\r\n * \r\n * @returns {Array[]} The adjancency matrix of the molecular graph with all bridges removed.\r\n */\r\n getComponentsAdjacencyMatrix() {\r\n let length = this.vertices.length;\r\n let adjacencyMatrix = Array(length);\r\n let bridges = this.getBridges();\r\n\r\n for (var i = 0; i < length; i++) {\r\n adjacencyMatrix[i] = new Array(length);\r\n adjacencyMatrix[i].fill(0);\r\n }\r\n\r\n for (var i = 0; i < this.edges.length; i++) {\r\n let edge = this.edges[i];\r\n\r\n adjacencyMatrix[edge.sourceId][edge.targetId] = 1;\r\n adjacencyMatrix[edge.targetId][edge.sourceId] = 1;\r\n }\r\n\r\n for (var i = 0; i < bridges.length; i++) {\r\n adjacencyMatrix[bridges[i][0]][bridges[i][1]] = 0;\r\n adjacencyMatrix[bridges[i][1]][bridges[i][0]] = 0;\r\n }\r\n\r\n return adjacencyMatrix;\r\n }\r\n\r\n /**\r\n * Get the adjacency matrix of a subgraph.\r\n * \r\n * @param {Number[]} vertexIds An array containing the vertex ids contained within the subgraph.\r\n * @returns {Array[]} The adjancency matrix of the subgraph.\r\n */\r\n getSubgraphAdjacencyMatrix(vertexIds) {\r\n let length = vertexIds.length;\r\n let adjacencyMatrix = Array(length);\r\n\r\n for (var i = 0; i < length; i++) {\r\n adjacencyMatrix[i] = new Array(length);\r\n adjacencyMatrix[i].fill(0);\r\n\r\n for (var j = 0; j < length; j++) {\r\n if (i === j) {\r\n continue;\r\n }\r\n\r\n if (this.hasEdge(vertexIds[i], vertexIds[j])) {\r\n adjacencyMatrix[i][j] = 1;\r\n }\r\n }\r\n }\r\n\r\n return adjacencyMatrix;\r\n }\r\n\r\n /**\r\n * Get the distance matrix of the graph.\r\n * \r\n * @returns {Array[]} The distance matrix of the graph.\r\n */\r\n getDistanceMatrix() {\r\n let length = this.vertices.length;\r\n let adja = this.getAdjacencyMatrix();\r\n let dist = Array(length);\r\n\r\n for (var i = 0; i < length; i++) {\r\n dist[i] = Array(length);\r\n dist[i].fill(Infinity);\r\n }\r\n\r\n for (var i = 0; i < length; i++) {\r\n for (var j = 0; j < length; j++) {\r\n if (adja[i][j] === 1) {\r\n dist[i][j] = 1;\r\n }\r\n }\r\n }\r\n\r\n for (var k = 0; k < length; k++) {\r\n for (var i = 0; i < length; i++) {\r\n for (var j = 0; j < length; j++) {\r\n if (dist[i][j] > dist[i][k] + dist[k][j]) {\r\n dist[i][j] = dist[i][k] + dist[k][j]\r\n }\r\n }\r\n }\r\n }\r\n\r\n return dist;\r\n }\r\n\r\n /**\r\n * Get the distance matrix of a subgraph.\r\n * \r\n * @param {Number[]} vertexIds An array containing the vertex ids contained within the subgraph.\r\n * @returns {Array[]} The distance matrix of the subgraph.\r\n */\r\n getSubgraphDistanceMatrix(vertexIds) {\r\n let length = vertexIds.length;\r\n let adja = this.getSubgraphAdjacencyMatrix(vertexIds);\r\n let dist = Array(length);\r\n\r\n for (var i = 0; i < length; i++) {\r\n dist[i] = Array(length);\r\n dist[i].fill(Infinity);\r\n }\r\n\r\n for (var i = 0; i < length; i++) {\r\n for (var j = 0; j < length; j++) {\r\n if (adja[i][j] === 1) {\r\n dist[i][j] = 1;\r\n }\r\n }\r\n }\r\n\r\n for (var k = 0; k < length; k++) {\r\n for (var i = 0; i < length; i++) {\r\n for (var j = 0; j < length; j++) {\r\n if (dist[i][j] > dist[i][k] + dist[k][j]) {\r\n dist[i][j] = dist[i][k] + dist[k][j]\r\n }\r\n }\r\n }\r\n }\r\n\r\n return dist;\r\n }\r\n\r\n /**\r\n * Get the adjacency list of the graph.\r\n * \r\n * @returns {Array[]} The adjancency list of the graph.\r\n */\r\n getAdjacencyList() {\r\n let length = this.vertices.length;\r\n let adjacencyList = Array(length);\r\n\r\n for (var i = 0; i < length; i++) {\r\n adjacencyList[i] = [];\r\n\r\n for (var j = 0; j < length; j++) {\r\n if (i === j) {\r\n continue;\r\n }\r\n\r\n if (this.hasEdge(this.vertices[i].id, this.vertices[j].id)) {\r\n adjacencyList[i].push(j);\r\n }\r\n }\r\n }\r\n\r\n return adjacencyList;\r\n }\r\n\r\n /**\r\n * Get the adjacency list of a subgraph.\r\n * \r\n * @param {Number[]} vertexIds An array containing the vertex ids contained within the subgraph.\r\n * @returns {Array[]} The adjancency list of the subgraph.\r\n */\r\n getSubgraphAdjacencyList(vertexIds) {\r\n let length = vertexIds.length;\r\n let adjacencyList = Array(length);\r\n\r\n for (var i = 0; i < length; i++) {\r\n adjacencyList[i] = Array();\r\n\r\n for (var j = 0; j < length; j++) {\r\n if (i === j) {\r\n continue;\r\n }\r\n\r\n if (this.hasEdge(vertexIds[i], vertexIds[j])) {\r\n adjacencyList[i].push(j);\r\n }\r\n }\r\n }\r\n\r\n return adjacencyList;\r\n }\r\n\r\n /**\r\n * Returns an array containing the edge ids of bridges. A bridge splits the graph into multiple components when removed.\r\n * \r\n * @returns {Number[]} An array containing the edge ids of the bridges.\r\n */\r\n getBridges() {\r\n let length = this.vertices.length;\r\n let visited = new Array(length);\r\n let disc = new Array(length);\r\n let low = new Array(length);\r\n let parent = new Array(length);\r\n let adj = this.getAdjacencyList();\r\n let outBridges = Array();\r\n\r\n visited.fill(false);\r\n parent.fill(null);\r\n this._time = 0;\r\n\r\n for (var i = 0; i < length; i++) {\r\n if (!visited[i]) {\r\n this._bridgeDfs(i, visited, disc, low, parent, adj, outBridges);\r\n }\r\n }\r\n\r\n return outBridges;\r\n }\r\n\r\n /**\r\n * Traverses the graph in breadth-first order.\r\n * \r\n * @param {Number} startVertexId The id of the starting vertex.\r\n * @param {Function} callback The callback function to be called on every vertex.\r\n */\r\n traverseBF(startVertexId, callback) {\r\n let length = this.vertices.length;\r\n let visited = new Array(length);\r\n\r\n visited.fill(false);\r\n\r\n var queue = [startVertexId];\r\n\r\n while (queue.length > 0) {\r\n // JavaScripts shift() is O(n) ... bad JavaScript, bad!\r\n let u = queue.shift();\r\n let vertex = this.vertices[u];\r\n\r\n callback(vertex);\r\n\r\n for (var i = 0; i < vertex.neighbours.length; i++) {\r\n let v = vertex.neighbours[i]\r\n if (!visited[v]) {\r\n visited[v] = true;\r\n queue.push(v);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Get the depth of a subtree in the direction opposite to the vertex specified as the parent vertex.\r\n *\r\n * @param {Number} vertexId A vertex id.\r\n * @param {Number} parentVertexId The id of a neighbouring vertex.\r\n * @returns {Number} The depth of the sub-tree.\r\n */\r\n getTreeDepth(vertexId, parentVertexId) {\r\n if (vertexId === null || parentVertexId === null) {\r\n return 0;\r\n }\r\n\r\n let neighbours = this.vertices[vertexId].getSpanningTreeNeighbours(parentVertexId);\r\n let max = 0;\r\n\r\n for (var i = 0; i < neighbours.length; i++) {\r\n let childId = neighbours[i];\r\n let d = this.getTreeDepth(childId, vertexId);\r\n\r\n if (d > max) {\r\n max = d;\r\n }\r\n }\r\n\r\n return max + 1;\r\n }\r\n\r\n /**\r\n * Traverse a sub-tree in the graph.\r\n *\r\n * @param {Number} vertexId A vertex id.\r\n * @param {Number} parentVertexId A neighbouring vertex.\r\n * @param {Function} callback The callback function that is called with each visited as an argument.\r\n * @param {Number} [maxDepth=Number.MAX_SAFE_INTEGER] The maximum depth of the recursion.\r\n * @param {Boolean} [ignoreFirst=false] Whether or not to ignore the starting vertex supplied as vertexId in the callback.\r\n * @param {Number} [depth=1] The current depth in the tree.\r\n * @param {Uint8Array} [visited=null] An array holding a flag on whether or not a node has been visited.\r\n */\r\n traverseTree(vertexId, parentVertexId, callback, maxDepth = Number.MAX_SAFE_INTEGER, ignoreFirst = false, depth = 1, visited = null) {\r\n if (visited === null) {\r\n visited = new Uint8Array(this.vertices.length);\r\n }\r\n\r\n if (depth > maxDepth + 1 || visited[vertexId] === 1) {\r\n return;\r\n }\r\n\r\n visited[vertexId] = 1;\r\n\r\n let vertex = this.vertices[vertexId];\r\n let neighbours = vertex.getNeighbours(parentVertexId);\r\n\r\n if (!ignoreFirst || depth > 1) {\r\n callback(vertex);\r\n }\r\n\r\n for (var i = 0; i < neighbours.length; i++) {\r\n this.traverseTree(neighbours[i], vertexId, callback, maxDepth, ignoreFirst, depth + 1, visited);\r\n }\r\n }\r\n\r\n /**\r\n * Positiones the (sub)graph using Kamada and Kawais algorithm for drawing general undirected graphs. https://pdfs.semanticscholar.org/b8d3/bca50ccc573c5cb99f7d201e8acce6618f04.pdf\r\n * There are undocumented layout parameters. They are undocumented for a reason, so be very careful.\r\n * \r\n * @param {Number[]} vertexIds An array containing vertexIds to be placed using the force based layout.\r\n * @param {Vector2} center The center of the layout.\r\n * @param {Number} startVertexId A vertex id. Should be the starting vertex - e.g. the first to be positioned and connected to a previously place vertex.\r\n * @param {Ring} ring The bridged ring associated with this force-based layout.\r\n */\r\n kkLayout(vertexIds, center, startVertexId, ring, bondLength,\r\n threshold = 0.1, innerThreshold = 0.1, maxIteration = 2000,\r\n maxInnerIteration = 50, maxEnergy = 1e9) {\r\n\r\n let edgeStrength = bondLength;\r\n\r\n // Add vertices that are directly connected to the ring\r\n var i = vertexIds.length;\r\n while (i--) {\r\n let vertex = this.vertices[vertexIds[i]];\r\n var j = vertex.neighbours.length;\r\n }\r\n\r\n let matDist = this.getSubgraphDistanceMatrix(vertexIds);\r\n let length = vertexIds.length;\r\n\r\n // Initialize the positions. Place all vertices on a ring around the center\r\n let radius = MathHelper.polyCircumradius(500, length);\r\n let angle = MathHelper.centralAngle(length);\r\n let a = 0.0;\r\n let arrPositionX = new Float32Array(length);\r\n let arrPositionY = new Float32Array(length);\r\n let arrPositioned = Array(length);\r\n\r\n i = length;\r\n while (i--) {\r\n let vertex = this.vertices[vertexIds[i]];\r\n if (!vertex.positioned) {\r\n arrPositionX[i] = center.x + Math.cos(a) * radius;\r\n arrPositionY[i] = center.y + Math.sin(a) * radius;\r\n } else {\r\n arrPositionX[i] = vertex.position.x;\r\n arrPositionY[i] = vertex.position.y;\r\n }\r\n arrPositioned[i] = vertex.positioned;\r\n a += angle;\r\n }\r\n\r\n // Create the matrix containing the lengths\r\n let matLength = Array(length);\r\n i = length;\r\n while (i--) {\r\n matLength[i] = new Array(length);\r\n var j = length;\r\n while (j--) {\r\n matLength[i][j] = bondLength * matDist[i][j];\r\n }\r\n }\r\n\r\n // Create the matrix containing the spring strenghts\r\n let matStrength = Array(length);\r\n i = length;\r\n while (i--) {\r\n matStrength[i] = Array(length);\r\n var j = length;\r\n while (j--) {\r\n matStrength[i][j] = edgeStrength * Math.pow(matDist[i][j], -2.0);\r\n }\r\n }\r\n\r\n // Create the matrix containing the energies\r\n let matEnergy = Array(length);\r\n let arrEnergySumX = new Float32Array(length);\r\n let arrEnergySumY = new Float32Array(length);\r\n i = length;\r\n while (i--) {\r\n matEnergy[i] = Array(length);\r\n }\r\n\r\n i = length;\r\n let ux, uy, dEx, dEy, vx, vy, denom;\r\n\r\n while (i--) {\r\n ux = arrPositionX[i];\r\n uy = arrPositionY[i];\r\n dEx = 0.0;\r\n dEy = 0.0;\r\n let j = length;\r\n while (j--) {\r\n if (i === j) {\r\n continue;\r\n }\r\n vx = arrPositionX[j];\r\n vy = arrPositionY[j];\r\n denom = 1.0 / Math.sqrt((ux - vx) * (ux - vx) + (uy - vy) * (uy - vy));\r\n matEnergy[i][j] = [\r\n matStrength[i][j] * ((ux - vx) - matLength[i][j] * (ux - vx) * denom),\r\n matStrength[i][j] * ((uy - vy) - matLength[i][j] * (uy - vy) * denom)\r\n ]\r\n matEnergy[j][i] = matEnergy[i][j];\r\n dEx += matEnergy[i][j][0];\r\n dEy += matEnergy[i][j][1];\r\n }\r\n arrEnergySumX[i] = dEx;\r\n arrEnergySumY[i] = dEy;\r\n }\r\n\r\n // Utility functions, maybe inline them later\r\n let energy = function (index) {\r\n return [arrEnergySumX[index] * arrEnergySumX[index] + arrEnergySumY[index] * arrEnergySumY[index], arrEnergySumX[index], arrEnergySumY[index]];\r\n }\r\n\r\n let highestEnergy = function () {\r\n let maxEnergy = 0.0;\r\n let maxEnergyId = 0;\r\n let maxDEX = 0.0;\r\n let maxDEY = 0.0\r\n\r\n i = length;\r\n while (i--) {\r\n let [delta, dEX, dEY] = energy(i);\r\n\r\n if (delta > maxEnergy && arrPositioned[i] === false) {\r\n maxEnergy = delta;\r\n maxEnergyId = i;\r\n maxDEX = dEX;\r\n maxDEY = dEY;\r\n }\r\n }\r\n\r\n return [maxEnergyId, maxEnergy, maxDEX, maxDEY];\r\n }\r\n\r\n let update = function (index, dEX, dEY) {\r\n let dxx = 0.0;\r\n let dyy = 0.0;\r\n let dxy = 0.0;\r\n let ux = arrPositionX[index];\r\n let uy = arrPositionY[index];\r\n let arrL = matLength[index];\r\n let arrK = matStrength[index];\r\n\r\n i = length;\r\n while (i--) {\r\n if (i === index) {\r\n continue;\r\n }\r\n\r\n let vx = arrPositionX[i];\r\n let vy = arrPositionY[i];\r\n let l = arrL[i];\r\n let k = arrK[i];\r\n let m = (ux - vx) * (ux - vx);\r\n let denom = 1.0 / Math.pow(m + (uy - vy) * (uy - vy), 1.5);\r\n \r\n dxx += k * (1 - l * (uy - vy) * (uy - vy) * denom);\r\n dyy += k * (1 - l * m * denom);\r\n dxy += k * (l * (ux - vx) * (uy - vy) * denom);\r\n }\r\n\r\n // Prevent division by zero\r\n if (dxx === 0) {\r\n dxx = 0.1;\r\n }\r\n\r\n if (dyy === 0) {\r\n dyy = 0.1;\r\n }\r\n\r\n if (dxy === 0) {\r\n dxy = 0.1;\r\n }\r\n\r\n let dy = (dEX / dxx + dEY / dxy);\r\n dy /= (dxy / dxx - dyy / dxy); // had to split this onto two lines because the syntax highlighter went crazy.\r\n let dx = -(dxy * dy + dEX) / dxx;\r\n\r\n arrPositionX[index] += dx;\r\n arrPositionY[index] += dy;\r\n\r\n // Update the energies\r\n let arrE = matEnergy[index];\r\n dEX = 0.0;\r\n dEY = 0.0;\r\n\r\n ux = arrPositionX[index];\r\n uy = arrPositionY[index];\r\n\r\n let vx, vy, prevEx, prevEy, denom;\r\n\r\n i = length;\r\n while (i--) {\r\n if (index === i) {\r\n continue;\r\n }\r\n vx = arrPositionX[i];\r\n vy = arrPositionY[i];\r\n // Store old energies\r\n prevEx = arrE[i][0];\r\n prevEy = arrE[i][1];\r\n denom = 1.0 / Math.sqrt((ux - vx) * (ux - vx) + (uy - vy) * (uy - vy));\r\n dx = arrK[i] * ((ux - vx) - arrL[i] * (ux - vx) * denom);\r\n dy = arrK[i] * ((uy - vy) - arrL[i] * (uy - vy) * denom);\r\n\r\n arrE[i] = [dx, dy];\r\n dEX += dx;\r\n dEY += dy;\r\n arrEnergySumX[i] += dx - prevEx;\r\n arrEnergySumY[i] += dy - prevEy;\r\n }\r\n arrEnergySumX[index] = dEX;\r\n arrEnergySumY[index] = dEY;\r\n }\r\n\r\n // Setting up variables for the while loops\r\n let maxEnergyId = 0;\r\n let dEX = 0.0;\r\n let dEY = 0.0;\r\n let delta = 0.0;\r\n let iteration = 0;\r\n let innerIteration = 0;\r\n\r\n while (maxEnergy > threshold && maxIteration > iteration) {\r\n iteration++;\r\n [maxEnergyId, maxEnergy, dEX, dEY] = highestEnergy();\r\n delta = maxEnergy;\r\n innerIteration = 0;\r\n while (delta > innerThreshold && maxInnerIteration > innerIteration) {\r\n innerIteration++;\r\n update(maxEnergyId, dEX, dEY);\r\n [delta, dEX, dEY] = energy(maxEnergyId);\r\n }\r\n }\r\n\r\n i = length;\r\n while (i--) {\r\n let index = vertexIds[i];\r\n let vertex = this.vertices[index];\r\n vertex.position.x = arrPositionX[i];\r\n vertex.position.y = arrPositionY[i];\r\n vertex.positioned = true;\r\n vertex.forcePositioned = true;\r\n }\r\n }\r\n\r\n /**\r\n * PRIVATE FUNCTION used by getBridges().\r\n */\r\n _bridgeDfs(u, visited, disc, low, parent, adj, outBridges) {\r\n visited[u] = true;\r\n disc[u] = low[u] = ++this._time;\r\n\r\n for (var i = 0; i < adj[u].length; i++) {\r\n let v = adj[u][i];\r\n\r\n if (!visited[v]) {\r\n parent[v] = u;\r\n\r\n this._bridgeDfs(v, visited, disc, low, parent, adj, outBridges);\r\n\r\n low[u] = Math.min(low[u], low[v]);\r\n\r\n // If low > disc, we have a bridge\r\n if (low[v] > disc[u]) {\r\n outBridges.push([u, v]);\r\n }\r\n } else if (v !== parent[u]) {\r\n low[u] = Math.min(low[u], disc[v]);\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Returns the connected components of the graph.\r\n * \r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @returns {Set[]} Connected components as sets.\r\n */\r\n static getConnectedComponents(adjacencyMatrix) {\r\n let length = adjacencyMatrix.length;\r\n let visited = new Array(length);\r\n let components = new Array();\r\n let count = 0;\r\n\r\n visited.fill(false);\r\n\r\n for (var u = 0; u < length; u++) {\r\n if (!visited[u]) {\r\n let component = Array();\r\n visited[u] = true;\r\n component.push(u);\r\n count++;\r\n Graph._ccGetDfs(u, visited, adjacencyMatrix, component);\r\n if (component.length > 1) {\r\n components.push(component);\r\n }\r\n }\r\n }\r\n\r\n return components;\r\n }\r\n\r\n /**\r\n * Returns the number of connected components for the graph. \r\n * \r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @returns {Number} The number of connected components of the supplied graph.\r\n */\r\n static getConnectedComponentCount(adjacencyMatrix) {\r\n let length = adjacencyMatrix.length;\r\n let visited = new Array(length);\r\n let count = 0;\r\n\r\n visited.fill(false);\r\n\r\n for (var u = 0; u < length; u++) {\r\n if (!visited[u]) {\r\n visited[u] = true;\r\n count++;\r\n Graph._ccCountDfs(u, visited, adjacencyMatrix);\r\n }\r\n }\r\n\r\n return count;\r\n }\r\n\r\n /**\r\n * PRIVATE FUNCTION used by getConnectedComponentCount().\r\n */\r\n static _ccCountDfs(u, visited, adjacencyMatrix) {\r\n for (var v = 0; v < adjacencyMatrix[u].length; v++) {\r\n let c = adjacencyMatrix[u][v];\r\n\r\n if (!c || visited[v] || u === v) {\r\n continue;\r\n }\r\n\r\n visited[v] = true;\r\n Graph._ccCountDfs(v, visited, adjacencyMatrix);\r\n }\r\n }\r\n\r\n /**\r\n * PRIVATE FUNCTION used by getConnectedComponents().\r\n */\r\n static _ccGetDfs(u, visited, adjacencyMatrix, component) {\r\n for (var v = 0; v < adjacencyMatrix[u].length; v++) {\r\n let c = adjacencyMatrix[u][v];\r\n\r\n if (!c || visited[v] || u === v) {\r\n continue;\r\n }\r\n\r\n visited[v] = true;\r\n component.push(v);\r\n Graph._ccGetDfs(v, visited, adjacencyMatrix, component);\r\n }\r\n }\r\n}\r\n\r\nmodule.exports = Graph","//@ts-check\r\nconst Vector2 = require('./Vector2')\r\n\r\n/** \r\n * A class representing a line.\r\n * \r\n * @property {Vector2} from The Vector2 defining the start of the line.\r\n * @property {Vector2} to The Vector2 defining the end of the line.\r\n * @property {String} elementFrom The element symbol associated with the start of the line.\r\n * @property {String} elementTo The element symbol associated with the end of the line.\r\n * @property {Boolean} chiralFrom A boolean indicating whether or not the source atom is a chiral center.\r\n * @property {Boolean} chiralTo A boolean indicating whether or tno the target atom is a chiral center.\r\n */\r\nclass Line {\r\n /**\r\n * The constructor for the class Line.\r\n *\r\n * @param {Vector2} [from=new Vector2(0, 0)] A vector marking the beginning of the line.\r\n * @param {Vector2} [to=new Vector2(0, 0)] A vector marking the end of the line.\r\n * @param {string} [elementFrom=null] A one-letter representation of the element associated with the vector marking the beginning of the line.\r\n * @param {string} [elementTo=null] A one-letter representation of the element associated with the vector marking the end of the line.\r\n * @param {Boolean} [chiralFrom=false] Whether or not the from atom is a chiral center.\r\n * @param {Boolean} [chiralTo=false] Whether or not the to atom is a chiral center.\r\n */\r\n constructor(from = new Vector2(0,0), to = new Vector2(0, 0), elementFrom = null, elementTo = null, chiralFrom = false, chiralTo = false) {\r\n this.from = from;\r\n this.to = to;\r\n this.elementFrom = elementFrom;\r\n this.elementTo = elementTo;\r\n this.chiralFrom = chiralFrom;\r\n this.chiralTo = chiralTo;\r\n }\r\n\r\n /**\r\n * Clones this line and returns the clone.\r\n *\r\n * @returns {Line} A clone of this line.\r\n */\r\n clone() {\r\n return new Line(this.from.clone(), this.to.clone(), this.elementFrom, this.elementTo);\r\n }\r\n\r\n /**\r\n * Returns the length of this line.\r\n *\r\n * @returns {Number} The length of this line.\r\n */\r\n getLength() {\r\n return Math.sqrt(Math.pow(this.to.x - this.from.x, 2) + \r\n Math.pow(this.to.y - this.from.y, 2));\r\n }\r\n\r\n\r\n /**\r\n * Returns the angle of the line in relation to the coordinate system (the x-axis).\r\n *\r\n * @returns {Number} The angle in radians.\r\n */\r\n getAngle() {\r\n // Get the angle between the line and the x-axis\r\n let diff = Vector2.subtract(this.getRightVector(), this.getLeftVector());\r\n return diff.angle();\r\n }\r\n\r\n /**\r\n * Returns the right vector (the vector with the larger x value).\r\n *\r\n * @returns {Vector2} The right vector.\r\n */\r\n getRightVector() {\r\n // Return the vector with the larger x value (the right one)\r\n if (this.from.x < this.to.x) {\r\n return this.to;\r\n } else {\r\n return this.from;\r\n }\r\n }\r\n \r\n /**\r\n * Returns the left vector (the vector with the smaller x value).\r\n *\r\n * @returns {Vector2} The left vector.\r\n */\r\n getLeftVector() {\r\n // Return the vector with the smaller x value (the left one)\r\n if (this.from.x < this.to.x) {\r\n return this.from;\r\n } else {\r\n return this.to;\r\n }\r\n }\r\n\r\n /**\r\n * Returns the element associated with the right vector (the vector with the larger x value).\r\n *\r\n * @returns {String} The element associated with the right vector.\r\n */\r\n getRightElement() {\r\n if (this.from.x < this.to.x) {\r\n return this.elementTo;\r\n } else {\r\n return this.elementFrom;\r\n }\r\n }\r\n\r\n /**\r\n * Returns the element associated with the left vector (the vector with the smaller x value).\r\n *\r\n * @returns {String} The element associated with the left vector.\r\n */\r\n getLeftElement() {\r\n if (this.from.x < this.to.x) {\r\n return this.elementFrom;\r\n } else {\r\n return this.elementTo;\r\n }\r\n }\r\n\r\n /**\r\n * Returns whether or not the atom associated with the right vector (the vector with the larger x value) is a chiral center.\r\n *\r\n * @returns {Boolean} Whether or not the atom associated with the right vector is a chiral center.\r\n */\r\n getRightChiral() {\r\n if (this.from.x < this.to.x) {\r\n return this.chiralTo;\r\n } else {\r\n return this.chiralFrom;\r\n }\r\n }\r\n\r\n /**\r\n * Returns whether or not the atom associated with the left vector (the vector with the smaller x value) is a chiral center.\r\n *\r\n * @returns {Boolean} Whether or not the atom associated with the left vector is a chiral center.\r\n */\r\n getLeftChiral() {\r\n if (this.from.x < this.to.x) {\r\n return this.chiralFrom;\r\n } else {\r\n return this.chiralTo;\r\n }\r\n }\r\n\r\n /**\r\n * Set the value of the right vector.\r\n *\r\n * @param {Number} x The x value.\r\n * @param {Number} y The y value.\r\n * @returns {Line} This line.\r\n */\r\n setRightVector(x, y) {\r\n if (this.from.x < this.to.x) {\r\n this.to.x = x;\r\n this.to.y = y;\r\n } else {\r\n this.from.x = x;\r\n this.from.y = y;\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Set the value of the left vector.\r\n *\r\n * @param {Number} x The x value.\r\n * @param {Number} y The y value.\r\n * @returns {Line} This line.\r\n */\r\n setLeftVector(x, y) {\r\n if (this.from.x < this.to.x) {\r\n this.from.x = x;\r\n this.from.y = y;\r\n } else {\r\n this.to.x = x;\r\n this.to.y = y;\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Rotates this line to be aligned with the x-axis. The center of rotation is the left vector.\r\n *\r\n * @returns {Line} This line.\r\n */\r\n rotateToXAxis() {\r\n let left = this.getLeftVector();\r\n \r\n this.setRightVector(left.x + this.getLength(), left.y);\r\n \r\n return this;\r\n }\r\n\r\n /**\r\n * Rotate the line by a given value (in radians). The center of rotation is the left vector.\r\n *\r\n * @param {Number} theta The angle (in radians) to rotate the line.\r\n * @returns {Line} This line.\r\n */\r\n rotate(theta) {\r\n let l = this.getLeftVector();\r\n let r = this.getRightVector();\r\n let sinTheta = Math.sin(theta);\r\n let cosTheta = Math.cos(theta);\r\n\r\n let x = cosTheta * (r.x - l.x) - sinTheta * (r.y - l.y) + l.x;\r\n let y = sinTheta * (r.x - l.x) - cosTheta * (r.y - l.y) + l.y;\r\n \r\n this.setRightVector(x, y);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Shortens this line from the \"from\" direction by a given value (in pixels).\r\n *\r\n * @param {Number} by The length in pixels to shorten the vector by.\r\n * @returns {Line} This line.\r\n */\r\n shortenFrom(by) {\r\n let f = Vector2.subtract(this.to, this.from);\r\n \r\n f.normalize();\r\n f.multiplyScalar(by);\r\n \r\n this.from.add(f);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Shortens this line from the \"to\" direction by a given value (in pixels).\r\n *\r\n * @param {Number} by The length in pixels to shorten the vector by.\r\n * @returns {Line} This line.\r\n */\r\n shortenTo(by) {\r\n let f = Vector2.subtract(this.from, this.to);\r\n \r\n f.normalize();\r\n f.multiplyScalar(by);\r\n \r\n this.to.add(f);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Shorten the right side.\r\n *\r\n * @param {Number} by The length in pixels to shorten the vector by.\r\n * @returns {Line} Returns itself.\r\n */\r\n shortenRight(by) {\r\n if (this.from.x < this.to.x) {\r\n this.shortenTo(by);\r\n } else {\r\n this.shortenFrom(by);\r\n }\r\n\r\n return this;\r\n }\r\n \r\n /**\r\n * Shorten the left side.\r\n * \r\n * @param {Number} by The length in pixels to shorten the vector by.\r\n * @returns {Line} Returns itself.\r\n */\r\n shortenLeft(by) {\r\n if (this.from.x < this.to.x) {\r\n this.shortenFrom(by);\r\n } else {\r\n this.shortenTo(by);\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Shortens this line from both directions by a given value (in pixels).\r\n *\r\n * @param {Number} by The length in pixels to shorten the vector by.\r\n * @returns {Line} This line.\r\n */\r\n shorten(by) {\r\n let f = Vector2.subtract(this.from, this.to);\r\n \r\n f.normalize();\r\n f.multiplyScalar(by / 2.0);\r\n \r\n this.to.add(f);\r\n this.from.subtract(f);\r\n\r\n return this;\r\n }\r\n}\r\n\r\nmodule.exports = Line;","/** \r\n * A static class containing helper functions for math-related tasks. \r\n */\r\nclass MathHelper {\r\n /**\r\n * Rounds a value to a given number of decimals.\r\n *\r\n * @static\r\n * @param {Number} value A number.\r\n * @param {Number} decimals The number of decimals.\r\n * @returns {Number} A number rounded to a given number of decimals.\r\n */\r\n static round(value, decimals) {\r\n decimals = decimals ? decimals : 1;\r\n return Number(Math.round(value + 'e' + decimals) + 'e-' + decimals);\r\n }\r\n\r\n /**\r\n * Returns the means of the angles contained in an array. In radians.\r\n *\r\n * @static\r\n * @param {Number[]} arr An array containing angles (in radians).\r\n * @returns {Number} The mean angle in radians.\r\n */\r\n static meanAngle(arr) {\r\n let sin = 0.0;\r\n let cos = 0.0;\r\n \r\n for (var i = 0; i < arr.length; i++) {\r\n sin += Math.sin(arr[i]);\r\n cos += Math.cos(arr[i]);\r\n }\r\n\r\n return Math.atan2(sin / arr.length, cos / arr.length);\r\n }\r\n\r\n /**\r\n * Returns the inner angle of a n-sided regular polygon.\r\n *\r\n * @static\r\n * @param {Number} n Number of sides of a regular polygon.\r\n * @returns {Number} The inner angle of a given regular polygon.\r\n */\r\n static innerAngle(n) {\r\n return MathHelper.toRad((n - 2) * 180 / n);\r\n }\r\n\r\n /**\r\n * Returns the circumradius of a n-sided regular polygon with a given side-length.\r\n *\r\n * @static\r\n * @param {Number} s The side length of the regular polygon.\r\n * @param {Number} n The number of sides.\r\n * @returns {Number} The circumradius of the regular polygon.\r\n */\r\n static polyCircumradius(s, n) {\r\n return s / (2 * Math.sin(Math.PI / n));\r\n }\r\n\r\n /**\r\n * Returns the apothem of a regular n-sided polygon based on its radius.\r\n *\r\n * @static\r\n * @param {Number} r The radius.\r\n * @param {Number} n The number of edges of the regular polygon.\r\n * @returns {Number} The apothem of a n-sided polygon based on its radius.\r\n */\r\n static apothem(r, n) {\r\n return r * Math.cos(Math.PI / n);\r\n }\r\n\r\n static apothemFromSideLength(s, n) {\r\n let r = MathHelper.polyCircumradius(s, n);\r\n \r\n return MathHelper.apothem(r, n);\r\n }\r\n\r\n /**\r\n * The central angle of a n-sided regular polygon. In radians.\r\n *\r\n * @static\r\n * @param {Number} n The number of sides of the regular polygon.\r\n * @returns {Number} The central angle of the n-sided polygon in radians.\r\n */\r\n static centralAngle(n) {\r\n return MathHelper.toRad(360 / n);\r\n }\r\n\r\n /**\r\n * Convertes radians to degrees.\r\n *\r\n * @static\r\n * @param {Number} rad An angle in radians.\r\n * @returns {Number} The angle in degrees.\r\n */\r\n static toDeg(rad) {\r\n return rad * MathHelper.degFactor;\r\n }\r\n\r\n /**\r\n * Converts degrees to radians.\r\n *\r\n * @static\r\n * @param {Number} deg An angle in degrees.\r\n * @returns {Number} The angle in radians.\r\n */\r\n static toRad(deg) {\r\n return deg * MathHelper.radFactor;\r\n }\r\n\r\n /**\r\n * Returns the parity of the permutation (1 or -1)\r\n * @param {(Array|Uint8Array)} arr An array containing the permutation.\r\n * @returns {Number} The parity of the permutation (1 or -1), where 1 means even and -1 means odd.\r\n */\r\n static parityOfPermutation(arr) {\r\n let visited = new Uint8Array(arr.length);\r\n let evenLengthCycleCount = 0;\r\n\r\n let traverseCycle = function(i, cycleLength = 0) {\r\n if (visited[i] === 1) {\r\n return cycleLength;\r\n }\r\n\r\n cycleLength++;\r\n\r\n visited[i] = 1;\r\n return traverseCycle(arr[i], cycleLength);\r\n }\r\n\r\n for (var i = 0; i < arr.length; i++) {\r\n if (visited[i] === 1) {\r\n continue;\r\n }\r\n\r\n let cycleLength = traverseCycle(i);\r\n evenLengthCycleCount += (1 - cycleLength % 2);\r\n }\r\n\r\n return evenLengthCycleCount % 2 ? -1 : 1;\r\n }\r\n\r\n /** The factor to convert degrees to radians. */\r\n static get radFactor() {\r\n return Math.PI / 180.0;\r\n }\r\n\r\n /** The factor to convert radians to degrees. */\r\n static get degFactor() {\r\n return 180.0 / Math.PI;\r\n }\r\n\r\n /** Two times PI. */\r\n static get twoPI() {\r\n return 2.0 * Math.PI;\r\n }\r\n}\r\n\r\nmodule.exports = MathHelper;","// WHEN REPLACING, CHECK FOR:\n// KEEP THIS WHEN REGENERATING THE PARSER !!\n\nmodule.exports = (function () {\n \"use strict\";\n\n /*\n * Generated by PEG.js 0.10.0.\n *\n * http://pegjs.org/\n */\n\n function peg$subclass(child, parent) {\n function ctor() {\n this.constructor = child;\n }\n ctor.prototype = parent.prototype;\n child.prototype = new ctor();\n }\n\n function peg$SyntaxError(message, expected, found, location) {\n this.message = message;\n this.expected = expected;\n this.found = found;\n this.location = location;\n this.name = \"SyntaxError\";\n\n if (typeof Error.captureStackTrace === \"function\") {\n Error.captureStackTrace(this, peg$SyntaxError);\n }\n }\n\n peg$subclass(peg$SyntaxError, Error);\n\n peg$SyntaxError.buildMessage = function (expected, found) {\n var DESCRIBE_EXPECTATION_FNS = {\n literal: function (expectation) {\n return \"\\\"\" + literalEscape(expectation.text) + \"\\\"\";\n },\n\n \"class\": function (expectation) {\n var escapedParts = \"\",\n i;\n\n for (i = 0; i < expectation.parts.length; i++) {\n escapedParts += expectation.parts[i] instanceof Array ?\n classEscape(expectation.parts[i][0]) + \"-\" + classEscape(expectation.parts[i][1]) :\n classEscape(expectation.parts[i]);\n }\n\n return \"[\" + (expectation.inverted ? \"^\" : \"\") + escapedParts + \"]\";\n },\n\n any: function (expectation) {\n return \"any character\";\n },\n\n end: function (expectation) {\n return \"end of input\";\n },\n\n other: function (expectation) {\n return expectation.description;\n }\n };\n\n function hex(ch) {\n return ch.charCodeAt(0).toString(16).toUpperCase();\n }\n\n function literalEscape(s) {\n return s\n .replace(/\\\\/g, '\\\\\\\\')\n .replace(/\"/g, '\\\\\"')\n .replace(/\\0/g, '\\\\0')\n .replace(/\\t/g, '\\\\t')\n .replace(/\\n/g, '\\\\n')\n .replace(/\\r/g, '\\\\r')\n .replace(/[\\x00-\\x0F]/g, function (ch) {\n return '\\\\x0' + hex(ch);\n })\n .replace(/[\\x10-\\x1F\\x7F-\\x9F]/g, function (ch) {\n return '\\\\x' + hex(ch);\n });\n }\n\n function classEscape(s) {\n return s\n .replace(/\\\\/g, '\\\\\\\\')\n .replace(/\\]/g, '\\\\]')\n .replace(/\\^/g, '\\\\^')\n .replace(/-/g, '\\\\-')\n .replace(/\\0/g, '\\\\0')\n .replace(/\\t/g, '\\\\t')\n .replace(/\\n/g, '\\\\n')\n .replace(/\\r/g, '\\\\r')\n .replace(/[\\x00-\\x0F]/g, function (ch) {\n return '\\\\x0' + hex(ch);\n })\n .replace(/[\\x10-\\x1F\\x7F-\\x9F]/g, function (ch) {\n return '\\\\x' + hex(ch);\n });\n }\n\n function describeExpectation(expectation) {\n return DESCRIBE_EXPECTATION_FNS[expectation.type](expectation);\n }\n\n function describeExpected(expected) {\n var descriptions = new Array(expected.length),\n i, j;\n\n for (i = 0; i < expected.length; i++) {\n descriptions[i] = describeExpectation(expected[i]);\n }\n\n descriptions.sort();\n\n if (descriptions.length > 0) {\n for (i = 1, j = 1; i < descriptions.length; i++) {\n if (descriptions[i - 1] !== descriptions[i]) {\n descriptions[j] = descriptions[i];\n j++;\n }\n }\n descriptions.length = j;\n }\n\n switch (descriptions.length) {\n case 1:\n return descriptions[0];\n\n case 2:\n return descriptions[0] + \" or \" + descriptions[1];\n\n default:\n return descriptions.slice(0, -1).join(\", \") +\n \", or \" +\n descriptions[descriptions.length - 1];\n }\n }\n\n function describeFound(found) {\n return found ? \"\\\"\" + literalEscape(found) + \"\\\"\" : \"end of input\";\n }\n\n return \"Expected \" + describeExpected(expected) + \" but \" + describeFound(found) + \" found.\";\n };\n\n function peg$parse(input, options) {\n options = options !== void 0 ? options : {};\n\n // KEEP THIS WHEN REGENERATING THE PARSER !!\n var nOpenParentheses = input.split('(').length - 1;\n var nCloseParentheses = input.split(')').length - 1;\n\n if (nOpenParentheses !== nCloseParentheses) {\n throw peg$buildSimpleError('The number of opening parentheses does not match the number of closing parentheses.', 0);\n }\n // KEEP THIS WHEN REGENERATING THE PARSER !!\n\n var peg$FAILED = {},\n\n peg$startRuleFunctions = {\n chain: peg$parsechain\n },\n peg$startRuleFunction = peg$parsechain,\n\n peg$c0 = function (s) {\n var branches = [];\n var rings = [];\n\n for (var i = 0; i < s[1].length; i++) {\n branches.push(s[1][i]);\n }\n\n\n for (var i = 0; i < s[2].length; i++) {\n var bond = (s[2][i][0]) ? s[2][i][0] : '-';\n rings.push({\n 'bond': bond,\n 'id': s[2][i][1]\n });\n }\n\n for (var i = 0; i < s[3].length; i++) {\n branches.push(s[3][i]);\n }\n\n for (var i = 0; i < s[6].length; i++) {\n branches.push(s[6][i]);\n }\n\n return {\n 'atom': s[0],\n 'isBracket': s[0].element ? true : false,\n 'branches': branches,\n 'branchCount': branches.length,\n 'ringbonds': rings,\n 'ringbondCount': rings.length,\n 'bond': s[4] ? s[4] : '-',\n 'next': s[5],\n 'hasNext': s[5] ? true : false\n }\n\n return s;\n },\n peg$c1 = \"(\",\n peg$c2 = peg$literalExpectation(\"(\", false),\n peg$c3 = \")\",\n peg$c4 = peg$literalExpectation(\")\", false),\n peg$c5 = function (b) {\n var bond = (b[1]) ? b[1] : '-';\n b[2].branchBond = bond;\n return b[2]\n },\n peg$c6 = function (a) {\n return a;\n },\n peg$c7 = /^[\\-=#$:\\/\\\\.]/,\n peg$c8 = peg$classExpectation([\"-\", \"=\", \"#\", \"$\", \":\", \"/\", \"\\\\\", \".\"], false, false),\n peg$c9 = function (b) {\n return b;\n },\n peg$c10 = \"[\",\n peg$c11 = peg$literalExpectation(\"[\", false),\n peg$c12 = \"se\",\n peg$c13 = peg$literalExpectation(\"se\", false),\n peg$c14 = \"as\",\n peg$c15 = peg$literalExpectation(\"as\", false),\n peg$c16 = \"]\",\n peg$c17 = peg$literalExpectation(\"]\", false),\n peg$c18 = function (b) {\n return {\n 'isotope': b[1],\n 'element': b[2],\n 'chirality': b[3],\n 'hcount': b[4],\n 'charge': b[5],\n 'class': b[6]\n }\n },\n peg$c19 = \"B\",\n peg$c20 = peg$literalExpectation(\"B\", false),\n peg$c21 = \"r\",\n peg$c22 = peg$literalExpectation(\"r\", false),\n peg$c23 = \"C\",\n peg$c24 = peg$literalExpectation(\"C\", false),\n peg$c25 = \"l\",\n peg$c26 = peg$literalExpectation(\"l\", false),\n peg$c27 = /^[NOPSFI]/,\n peg$c28 = peg$classExpectation([\"N\", \"O\", \"P\", \"S\", \"F\", \"I\"], false, false),\n peg$c29 = function (o) {\n if (o.length > 1) return o.join('');\n return o;\n },\n peg$c30 = /^[bcnops]/,\n peg$c31 = peg$classExpectation([\"b\", \"c\", \"n\", \"o\", \"p\", \"s\"], false, false),\n peg$c32 = \"*\",\n peg$c33 = peg$literalExpectation(\"*\", false),\n peg$c34 = function (w) {\n return w;\n },\n peg$c35 = /^[A-Z]/,\n peg$c36 = peg$classExpectation([\n [\"A\", \"Z\"]\n ], false, false),\n peg$c37 = /^[a-z]/,\n peg$c38 = peg$classExpectation([\n [\"a\", \"z\"]\n ], false, false),\n peg$c39 = function (e) {\n return e.join('');\n },\n peg$c40 = \"%\",\n peg$c41 = peg$literalExpectation(\"%\", false),\n peg$c42 = /^[1-9]/,\n peg$c43 = peg$classExpectation([\n [\"1\", \"9\"]\n ], false, false),\n peg$c44 = /^[0-9]/,\n peg$c45 = peg$classExpectation([\n [\"0\", \"9\"]\n ], false, false),\n peg$c46 = function (r) {\n if (r.length == 1) return Number(r);\n return Number(r.join('').replace('%', ''));\n },\n peg$c47 = \"@\",\n peg$c48 = peg$literalExpectation(\"@\", false),\n peg$c49 = \"TH\",\n peg$c50 = peg$literalExpectation(\"TH\", false),\n peg$c51 = /^[12]/,\n peg$c52 = peg$classExpectation([\"1\", \"2\"], false, false),\n peg$c53 = \"AL\",\n peg$c54 = peg$literalExpectation(\"AL\", false),\n peg$c55 = \"SP\",\n peg$c56 = peg$literalExpectation(\"SP\", false),\n peg$c57 = /^[1-3]/,\n peg$c58 = peg$classExpectation([\n [\"1\", \"3\"]\n ], false, false),\n peg$c59 = \"TB\",\n peg$c60 = peg$literalExpectation(\"TB\", false),\n peg$c61 = \"OH\",\n peg$c62 = peg$literalExpectation(\"OH\", false),\n peg$c63 = function (c) {\n if (!c[1]) return '@';\n if (c[1] == '@') return '@@';\n\n return c[1].join('').replace(',', '');\n },\n peg$c64 = function (c) {\n return c;\n },\n peg$c65 = \"+\",\n peg$c66 = peg$literalExpectation(\"+\", false),\n peg$c67 = function (c) {\n if (!c[1]) return 1;\n if (c[1] != '+') return Number(c[1].join(''));\n return 2;\n },\n peg$c68 = \"-\",\n peg$c69 = peg$literalExpectation(\"-\", false),\n peg$c70 = function (c) {\n if (!c[1]) return -1;\n if (c[1] != '-') return -Number(c[1].join(''));\n return -2;\n },\n peg$c71 = \"H\",\n peg$c72 = peg$literalExpectation(\"H\", false),\n peg$c73 = function (h) {\n if (h[1]) return Number(h[1]);\n return 1;\n },\n peg$c74 = \":\",\n peg$c75 = peg$literalExpectation(\":\", false),\n peg$c76 = /^[0]/,\n peg$c77 = peg$classExpectation([\"0\"], false, false),\n peg$c78 = function (c) {\n return Number(c[1][0] + c[1][1].join(''));\n },\n peg$c79 = function (i) {\n return Number(i.join(''));\n },\n\n peg$currPos = 0,\n peg$savedPos = 0,\n peg$posDetailsCache = [{\n line: 1,\n column: 1\n }],\n peg$maxFailPos = 0,\n peg$maxFailExpected = [],\n peg$silentFails = 0,\n\n peg$result;\n\n if (\"startRule\" in options) {\n if (!(options.startRule in peg$startRuleFunctions)) {\n throw new Error(\"Can't start parsing from rule \\\"\" + options.startRule + \"\\\".\");\n }\n\n peg$startRuleFunction = peg$startRuleFunctions[options.startRule];\n }\n\n function text() {\n return input.substring(peg$savedPos, peg$currPos);\n }\n\n function location() {\n return peg$computeLocation(peg$savedPos, peg$currPos);\n }\n\n function expected(description, location) {\n location = location !== void 0 ? location : peg$computeLocation(peg$savedPos, peg$currPos)\n\n throw peg$buildStructuredError(\n [peg$otherExpectation(description)],\n input.substring(peg$savedPos, peg$currPos),\n location\n );\n }\n\n function error(message, location) {\n location = location !== void 0 ? location : peg$computeLocation(peg$savedPos, peg$currPos)\n\n throw peg$buildSimpleError(message, location);\n }\n\n function peg$literalExpectation(text, ignoreCase) {\n return {\n type: \"literal\",\n text: text,\n ignoreCase: ignoreCase\n };\n }\n\n function peg$classExpectation(parts, inverted, ignoreCase) {\n return {\n type: \"class\",\n parts: parts,\n inverted: inverted,\n ignoreCase: ignoreCase\n };\n }\n\n function peg$anyExpectation() {\n return {\n type: \"any\"\n };\n }\n\n function peg$endExpectation() {\n return {\n type: \"end\"\n };\n }\n\n function peg$otherExpectation(description) {\n return {\n type: \"other\",\n description: description\n };\n }\n\n function peg$computePosDetails(pos) {\n var details = peg$posDetailsCache[pos],\n p;\n\n if (details) {\n return details;\n } else {\n p = pos - 1;\n while (!peg$posDetailsCache[p]) {\n p--;\n }\n\n details = peg$posDetailsCache[p];\n details = {\n line: details.line,\n column: details.column\n };\n\n while (p < pos) {\n if (input.charCodeAt(p) === 10) {\n details.line++;\n details.column = 1;\n } else {\n details.column++;\n }\n\n p++;\n }\n\n peg$posDetailsCache[pos] = details;\n return details;\n }\n }\n\n function peg$computeLocation(startPos, endPos) {\n var startPosDetails = peg$computePosDetails(startPos),\n endPosDetails = peg$computePosDetails(endPos);\n\n return {\n start: {\n offset: startPos,\n line: startPosDetails.line,\n column: startPosDetails.column\n },\n end: {\n offset: endPos,\n line: endPosDetails.line,\n column: endPosDetails.column\n }\n };\n }\n\n function peg$fail(expected) {\n if (peg$currPos < peg$maxFailPos) {\n return;\n }\n\n if (peg$currPos > peg$maxFailPos) {\n peg$maxFailPos = peg$currPos;\n peg$maxFailExpected = [];\n }\n\n peg$maxFailExpected.push(expected);\n }\n\n function peg$buildSimpleError(message, location) {\n return new peg$SyntaxError(message, null, null, location);\n }\n\n function peg$buildStructuredError(expected, found, location) {\n return new peg$SyntaxError(\n peg$SyntaxError.buildMessage(expected, found),\n expected,\n found,\n location\n );\n }\n\n function peg$parsechain() {\n var s0, s1, s2, s3, s4, s5, s6, s7, s8, s9;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n s2 = peg$parseatom();\n if (s2 !== peg$FAILED) {\n s3 = [];\n s4 = peg$parsebranch();\n while (s4 !== peg$FAILED) {\n s3.push(s4);\n s4 = peg$parsebranch();\n }\n if (s3 !== peg$FAILED) {\n s4 = [];\n s5 = peg$currPos;\n s6 = peg$parsebond();\n if (s6 === peg$FAILED) {\n s6 = null;\n }\n if (s6 !== peg$FAILED) {\n s7 = peg$parsering();\n if (s7 !== peg$FAILED) {\n s6 = [s6, s7];\n s5 = s6;\n } else {\n peg$currPos = s5;\n s5 = peg$FAILED;\n }\n } else {\n peg$currPos = s5;\n s5 = peg$FAILED;\n }\n while (s5 !== peg$FAILED) {\n s4.push(s5);\n s5 = peg$currPos;\n s6 = peg$parsebond();\n if (s6 === peg$FAILED) {\n s6 = null;\n }\n if (s6 !== peg$FAILED) {\n s7 = peg$parsering();\n if (s7 !== peg$FAILED) {\n s6 = [s6, s7];\n s5 = s6;\n } else {\n peg$currPos = s5;\n s5 = peg$FAILED;\n }\n } else {\n peg$currPos = s5;\n s5 = peg$FAILED;\n }\n }\n if (s4 !== peg$FAILED) {\n s5 = [];\n s6 = peg$parsebranch();\n while (s6 !== peg$FAILED) {\n s5.push(s6);\n s6 = peg$parsebranch();\n }\n if (s5 !== peg$FAILED) {\n s6 = peg$parsebond();\n if (s6 === peg$FAILED) {\n s6 = null;\n }\n if (s6 !== peg$FAILED) {\n s7 = peg$parsechain();\n if (s7 === peg$FAILED) {\n s7 = null;\n }\n if (s7 !== peg$FAILED) {\n s8 = [];\n s9 = peg$parsebranch();\n while (s9 !== peg$FAILED) {\n s8.push(s9);\n s9 = peg$parsebranch();\n }\n if (s8 !== peg$FAILED) {\n s2 = [s2, s3, s4, s5, s6, s7, s8];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c0(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsebranch() {\n var s0, s1, s2, s3, s4, s5;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 40) {\n s2 = peg$c1;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c2);\n }\n }\n if (s2 !== peg$FAILED) {\n s3 = peg$parsebond();\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s4 = peg$parsechain();\n if (s4 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 41) {\n s5 = peg$c3;\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c4);\n }\n }\n if (s5 !== peg$FAILED) {\n s2 = [s2, s3, s4, s5];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c5(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parseatom() {\n var s0, s1;\n\n s0 = peg$currPos;\n s1 = peg$parseorganicsymbol();\n if (s1 === peg$FAILED) {\n s1 = peg$parsearomaticsymbol();\n if (s1 === peg$FAILED) {\n s1 = peg$parsebracketatom();\n if (s1 === peg$FAILED) {\n s1 = peg$parsewildcard();\n }\n }\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c6(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsebond() {\n var s0, s1;\n\n s0 = peg$currPos;\n if (peg$c7.test(input.charAt(peg$currPos))) {\n s1 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s1 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c8);\n }\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c9(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsebracketatom() {\n var s0, s1, s2, s3, s4, s5, s6, s7, s8, s9;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 91) {\n s2 = peg$c10;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c11);\n }\n }\n if (s2 !== peg$FAILED) {\n s3 = peg$parseisotope();\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n if (input.substr(peg$currPos, 2) === peg$c12) {\n s4 = peg$c12;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c13);\n }\n }\n if (s4 === peg$FAILED) {\n if (input.substr(peg$currPos, 2) === peg$c14) {\n s4 = peg$c14;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c15);\n }\n }\n if (s4 === peg$FAILED) {\n s4 = peg$parsearomaticsymbol();\n if (s4 === peg$FAILED) {\n s4 = peg$parseelementsymbol();\n if (s4 === peg$FAILED) {\n s4 = peg$parsewildcard();\n }\n }\n }\n }\n if (s4 !== peg$FAILED) {\n s5 = peg$parsechiral();\n if (s5 === peg$FAILED) {\n s5 = null;\n }\n if (s5 !== peg$FAILED) {\n s6 = peg$parsehcount();\n if (s6 === peg$FAILED) {\n s6 = null;\n }\n if (s6 !== peg$FAILED) {\n s7 = peg$parsecharge();\n if (s7 === peg$FAILED) {\n s7 = null;\n }\n if (s7 !== peg$FAILED) {\n s8 = peg$parseclass();\n if (s8 === peg$FAILED) {\n s8 = null;\n }\n if (s8 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 93) {\n s9 = peg$c16;\n peg$currPos++;\n } else {\n s9 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c17);\n }\n }\n if (s9 !== peg$FAILED) {\n s2 = [s2, s3, s4, s5, s6, s7, s8, s9];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c18(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parseorganicsymbol() {\n var s0, s1, s2, s3;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 66) {\n s2 = peg$c19;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c20);\n }\n }\n if (s2 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 114) {\n s3 = peg$c21;\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c22);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 === peg$FAILED) {\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 67) {\n s2 = peg$c23;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c24);\n }\n }\n if (s2 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 108) {\n s3 = peg$c25;\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c26);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 === peg$FAILED) {\n if (peg$c27.test(input.charAt(peg$currPos))) {\n s1 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s1 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c28);\n }\n }\n }\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c29(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsearomaticsymbol() {\n var s0, s1;\n\n s0 = peg$currPos;\n if (peg$c30.test(input.charAt(peg$currPos))) {\n s1 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s1 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c31);\n }\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c6(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsewildcard() {\n var s0, s1;\n\n s0 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 42) {\n s1 = peg$c32;\n peg$currPos++;\n } else {\n s1 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c33);\n }\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c34(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parseelementsymbol() {\n var s0, s1, s2, s3;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (peg$c35.test(input.charAt(peg$currPos))) {\n s2 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c36);\n }\n }\n if (s2 !== peg$FAILED) {\n if (peg$c37.test(input.charAt(peg$currPos))) {\n s3 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c38);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c39(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsering() {\n var s0, s1, s2, s3, s4;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 37) {\n s2 = peg$c40;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c41);\n }\n }\n if (s2 !== peg$FAILED) {\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s3 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s3 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s4 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s4 !== peg$FAILED) {\n s2 = [s2, s3, s4];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 === peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s1 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s1 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c46(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsechiral() {\n var s0, s1, s2, s3, s4, s5, s6;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 64) {\n s2 = peg$c47;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c48);\n }\n }\n if (s2 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 64) {\n s3 = peg$c47;\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c48);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (input.substr(peg$currPos, 2) === peg$c49) {\n s4 = peg$c49;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c50);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c51.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c52);\n }\n }\n if (s5 !== peg$FAILED) {\n s4 = [s4, s5];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (input.substr(peg$currPos, 2) === peg$c53) {\n s4 = peg$c53;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c54);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c51.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c52);\n }\n }\n if (s5 !== peg$FAILED) {\n s4 = [s4, s5];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (input.substr(peg$currPos, 2) === peg$c55) {\n s4 = peg$c55;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c56);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c57.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c58);\n }\n }\n if (s5 !== peg$FAILED) {\n s4 = [s4, s5];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (input.substr(peg$currPos, 2) === peg$c59) {\n s4 = peg$c59;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c60);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s5 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s6 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s6 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s6 === peg$FAILED) {\n s6 = null;\n }\n if (s6 !== peg$FAILED) {\n s4 = [s4, s5, s6];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (input.substr(peg$currPos, 2) === peg$c61) {\n s4 = peg$c61;\n peg$currPos += 2;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c62);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s5 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s6 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s6 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s6 === peg$FAILED) {\n s6 = null;\n }\n if (s6 !== peg$FAILED) {\n s4 = [s4, s5, s6];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n }\n }\n }\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c63(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsecharge() {\n var s0, s1;\n\n s0 = peg$currPos;\n s1 = peg$parseposcharge();\n if (s1 === peg$FAILED) {\n s1 = peg$parsenegcharge();\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c64(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parseposcharge() {\n var s0, s1, s2, s3, s4, s5;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 43) {\n s2 = peg$c65;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c66);\n }\n }\n if (s2 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 43) {\n s3 = peg$c65;\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c66);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s4 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s5 === peg$FAILED) {\n s5 = null;\n }\n if (s5 !== peg$FAILED) {\n s4 = [s4, s5];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c67(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsenegcharge() {\n var s0, s1, s2, s3, s4, s5;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 45) {\n s2 = peg$c68;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c69);\n }\n }\n if (s2 !== peg$FAILED) {\n if (input.charCodeAt(peg$currPos) === 45) {\n s3 = peg$c68;\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c69);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = peg$currPos;\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s4 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s4 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s5 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s5 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s5 === peg$FAILED) {\n s5 = null;\n }\n if (s5 !== peg$FAILED) {\n s4 = [s4, s5];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c70(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parsehcount() {\n var s0, s1, s2, s3;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 72) {\n s2 = peg$c71;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c72);\n }\n }\n if (s2 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s3 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c73(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parseclass() {\n var s0, s1, s2, s3, s4, s5, s6;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (input.charCodeAt(peg$currPos) === 58) {\n s2 = peg$c74;\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c75);\n }\n }\n if (s2 !== peg$FAILED) {\n s3 = peg$currPos;\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s4 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s4 !== peg$FAILED) {\n s5 = [];\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s6 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s6 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n while (s6 !== peg$FAILED) {\n s5.push(s6);\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s6 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s6 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n }\n if (s5 !== peg$FAILED) {\n s4 = [s4, s5];\n s3 = s4;\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n } else {\n peg$currPos = s3;\n s3 = peg$FAILED;\n }\n if (s3 === peg$FAILED) {\n if (peg$c76.test(input.charAt(peg$currPos))) {\n s3 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c77);\n }\n }\n }\n if (s3 !== peg$FAILED) {\n s2 = [s2, s3];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c78(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n function peg$parseisotope() {\n var s0, s1, s2, s3, s4;\n\n s0 = peg$currPos;\n s1 = peg$currPos;\n if (peg$c42.test(input.charAt(peg$currPos))) {\n s2 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s2 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c43);\n }\n }\n if (s2 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s3 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s3 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s3 === peg$FAILED) {\n s3 = null;\n }\n if (s3 !== peg$FAILED) {\n if (peg$c44.test(input.charAt(peg$currPos))) {\n s4 = input.charAt(peg$currPos);\n peg$currPos++;\n } else {\n s4 = peg$FAILED;\n if (peg$silentFails === 0) {\n peg$fail(peg$c45);\n }\n }\n if (s4 === peg$FAILED) {\n s4 = null;\n }\n if (s4 !== peg$FAILED) {\n s2 = [s2, s3, s4];\n s1 = s2;\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n } else {\n peg$currPos = s1;\n s1 = peg$FAILED;\n }\n if (s1 !== peg$FAILED) {\n peg$savedPos = s0;\n s1 = peg$c79(s1);\n }\n s0 = s1;\n\n return s0;\n }\n\n peg$result = peg$startRuleFunction();\n\n if (peg$result !== peg$FAILED && peg$currPos === input.length) {\n return peg$result;\n } else {\n if (peg$result !== peg$FAILED && peg$currPos < input.length) {\n peg$fail(peg$endExpectation());\n }\n\n throw peg$buildStructuredError(\n peg$maxFailExpected,\n peg$maxFailPos < input.length ? input.charAt(peg$maxFailPos) : null,\n peg$maxFailPos < input.length ?\n peg$computeLocation(peg$maxFailPos, peg$maxFailPos + 1) :\n peg$computeLocation(peg$maxFailPos, peg$maxFailPos)\n );\n }\n }\n\n return {\n SyntaxError: peg$SyntaxError,\n parse: peg$parse\n };\n})();","//@ts-check\r\nconst ArrayHelper = require('./ArrayHelper')\r\nconst Vector2 = require('./Vector2')\r\nconst Vertex = require('./Vertex')\r\nconst RingConnection = require('./RingConnection')\r\n\r\n/** \r\n * A class representing a ring.\r\n * \r\n * @property {Number} id The id of this ring.\r\n * @property {Number[]} members An array containing the vertex ids of the ring members.\r\n * @property {Number[]} edges An array containing the edge ids of the edges between the ring members.\r\n * @property {Number[]} insiders An array containing the vertex ids of the vertices contained within the ring if it is a bridged ring.\r\n * @property {Number[]} neighbours An array containing the ids of neighbouring rings.\r\n * @property {Boolean} positioned A boolean indicating whether or not this ring has been positioned.\r\n * @property {Vector2} center The center of this ring.\r\n * @property {Ring[]} rings The rings contained within this ring if this ring is bridged.\r\n * @property {Boolean} isBridged A boolean whether or not this ring is bridged.\r\n * @property {Boolean} isPartOfBridged A boolean whether or not this ring is part of a bridge ring.\r\n * @property {Boolean} isSpiro A boolean whether or not this ring is part of a spiro.\r\n * @property {Boolean} isFused A boolean whether or not this ring is part of a fused ring.\r\n * @property {Number} centralAngle The central angle of this ring.\r\n * @property {Boolean} canFlip A boolean indicating whether or not this ring allows flipping of attached vertices to the inside of the ring.\r\n */\r\nclass Ring {\r\n /**\r\n * The constructor for the class Ring.\r\n *\r\n * @param {Number[]} members An array containing the vertex ids of the members of the ring to be created.\r\n */\r\n constructor(members) {\r\n this.id = null;\r\n this.members = members;\r\n this.edges = [];\r\n this.insiders = [];\r\n this.neighbours = [];\r\n this.positioned = false;\r\n this.center = new Vector2(0, 0);\r\n this.rings = [];\r\n this.isBridged = false;\r\n this.isPartOfBridged = false;\r\n this.isSpiro = false;\r\n this.isFused = false;\r\n this.centralAngle = 0.0;\r\n this.canFlip = true;\r\n }\r\n \r\n /**\r\n * Clones this ring and returns the clone.\r\n *\r\n * @returns {Ring} A clone of this ring.\r\n */\r\n clone() {\r\n let clone = new Ring(this.members);\r\n\r\n clone.id = this.id;\r\n clone.insiders = ArrayHelper.clone(this.insiders);\r\n clone.neighbours = ArrayHelper.clone(this.neighbours);\r\n clone.positioned = this.positioned;\r\n clone.center = this.center.clone();\r\n clone.rings = ArrayHelper.clone(this.rings);\r\n clone.isBridged = this.isBridged;\r\n clone.isPartOfBridged = this.isPartOfBridged;\r\n clone.isSpiro = this.isSpiro;\r\n clone.isFused = this.isFused;\r\n clone.centralAngle = this.centralAngle;\r\n clone.canFlip = this.canFlip;\r\n\r\n return clone;\r\n }\r\n\r\n /**\r\n * Returns the size (number of members) of this ring.\r\n *\r\n * @returns {Number} The size (number of members) of this ring.\r\n */\r\n getSize() {\r\n return this.members.length;\r\n }\r\n\r\n /**\r\n * Gets the polygon representation (an array of the ring-members positional vectors) of this ring.\r\n *\r\n * @param {Vertex[]} vertices An array of vertices representing the current molecule.\r\n * @returns {Vector2[]} An array of the positional vectors of the ring members.\r\n */\r\n getPolygon(vertices) {\r\n let polygon = [];\r\n\r\n for (let i = 0; i < this.members.length; i++) {\r\n polygon.push(vertices[this.members[i]].position);\r\n }\r\n\r\n return polygon;\r\n }\r\n\r\n /**\r\n * Returns the angle of this ring in relation to the coordinate system.\r\n *\r\n * @returns {Number} The angle in radians.\r\n */\r\n getAngle() {\r\n return Math.PI - this.centralAngle;\r\n }\r\n\r\n /**\r\n * Loops over the members of this ring from a given start position in a direction opposite to the vertex id passed as the previousId.\r\n *\r\n * @param {Vertex[]} vertices The vertices associated with the current molecule.\r\n * @param {Function} callback A callback with the current vertex id as a parameter.\r\n * @param {Number} startVertexId The vertex id of the start vertex.\r\n * @param {Number} previousVertexId The vertex id of the previous vertex (the loop calling the callback function will run in the opposite direction of this vertex).\r\n */\r\n eachMember(vertices, callback, startVertexId, previousVertexId) {\r\n startVertexId = startVertexId || startVertexId === 0 ? startVertexId : this.members[0];\r\n let current = startVertexId;\r\n let max = 0;\r\n\r\n while (current != null && max < 100) {\r\n let prev = current;\r\n \r\n callback(prev);\r\n current = vertices[current].getNextInRing(vertices, this.id, previousVertexId);\r\n previousVertexId = prev;\r\n \r\n // Stop while loop when arriving back at the start vertex\r\n if (current == startVertexId) {\r\n current = null;\r\n }\r\n\r\n max++;\r\n }\r\n }\r\n\r\n /**\r\n * Returns an array containing the neighbouring rings of this ring ordered by ring size.\r\n *\r\n * @param {RingConnection[]} ringConnections An array of ring connections associated with the current molecule.\r\n * @returns {Object[]} An array of neighbouring rings sorted by ring size. Example: { n: 5, neighbour: 1 }.\r\n */\r\n getOrderedNeighbours(ringConnections) {\r\n let orderedNeighbours = Array(this.neighbours.length);\r\n \r\n for (let i = 0; i < this.neighbours.length; i++) {\r\n let vertices = RingConnection.getVertices(ringConnections, this.id, this.neighbours[i]);\r\n \r\n orderedNeighbours[i] = {\r\n n: vertices.length,\r\n neighbour: this.neighbours[i]\r\n };\r\n }\r\n\r\n orderedNeighbours.sort(function (a, b) {\r\n // Sort highest to lowest\r\n return b.n - a.n;\r\n });\r\n\r\n return orderedNeighbours;\r\n }\r\n\r\n /**\r\n * Check whether this ring is an implicitly defined benzene-like (e.g. C1=CC=CC=C1) with 6 members and 3 double bonds.\r\n *\r\n * @param {Vertex[]} vertices An array of vertices associated with the current molecule.\r\n * @returns {Boolean} A boolean indicating whether or not this ring is an implicitly defined benzene-like.\r\n */\r\n isBenzeneLike(vertices) {\r\n let db = this.getDoubleBondCount(vertices);\r\n let length = this.members.length;\r\n\r\n return db === 3 && length === 6 ||\r\n db === 2 && length === 5 ;\r\n }\r\n\r\n /**\r\n * Get the number of double bonds inside this ring.\r\n *\r\n * @param {Vertex[]} vertices An array of vertices associated with the current molecule.\r\n * @returns {Number} The number of double bonds inside this ring.\r\n */\r\n getDoubleBondCount(vertices) {\r\n let doubleBondCount = 0;\r\n\r\n for (let i = 0; i < this.members.length; i++) {\r\n let atom = vertices[this.members[i]].value;\r\n\r\n if (atom.bondType === '=' || atom.branchBond === '=') {\r\n doubleBondCount++;\r\n }\r\n }\r\n\r\n return doubleBondCount;\r\n }\r\n\r\n /**\r\n * Checks whether or not this ring contains a member with a given vertex id.\r\n *\r\n * @param {Number} vertexId A vertex id.\r\n * @returns {Boolean} A boolean indicating whether or not this ring contains a member with the given vertex id.\r\n */\r\n contains(vertexId) {\r\n for (let i = 0; i < this.members.length; i++) {\r\n if (this.members[i] == vertexId) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n}\r\n\r\nmodule.exports = Ring;","//@ts-check\r\nconst Vertex = require('./Vertex')\r\nconst Ring = require('./Ring')\r\n\r\n/** \r\n * A class representing a ring connection.\r\n * \r\n * @property {Number} id The id of this ring connection.\r\n * @property {Number} firstRingId A ring id.\r\n * @property {Number} secondRingId A ring id.\r\n * @property {Set} vertices A set containing the vertex ids participating in the ring connection.\r\n */\r\nclass RingConnection {\r\n /**\r\n * The constructor for the class RingConnection.\r\n *\r\n * @param {Ring} firstRing A ring.\r\n * @param {Ring} secondRing A ring.\r\n */\r\n constructor(firstRing, secondRing) {\r\n this.id = null;\r\n this.firstRingId = firstRing.id;\r\n this.secondRingId = secondRing.id;\r\n this.vertices = new Set();\r\n\r\n for (var m = 0; m < firstRing.members.length; m++) {\r\n let c = firstRing.members[m];\r\n\r\n for (let n = 0; n < secondRing.members.length; n++) {\r\n let d = secondRing.members[n];\r\n\r\n if (c === d) {\r\n this.addVertex(c);\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * Adding a vertex to the ring connection.\r\n *\r\n * @param {Number} vertexId A vertex id.\r\n */\r\n addVertex(vertexId) {\r\n this.vertices.add(vertexId);\r\n }\r\n\r\n /**\r\n * Update the ring id of this ring connection that is not the ring id supplied as the second argument.\r\n *\r\n * @param {Number} ringId A ring id. The new ring id to be set.\r\n * @param {Number} otherRingId A ring id. The id that is NOT to be updated.\r\n */\r\n updateOther(ringId, otherRingId) {\r\n if (this.firstRingId === otherRingId) {\r\n this.secondRingId = ringId;\r\n } else {\r\n this.firstRingId = ringId;\r\n }\r\n }\r\n\r\n /**\r\n * Returns a boolean indicating whether or not a ring with a given id is participating in this ring connection.\r\n * \r\n * @param {Number} ringId A ring id.\r\n * @returns {Boolean} A boolean indicating whether or not a ring with a given id participates in this ring connection.\r\n */\r\n containsRing(ringId) {\r\n return this.firstRingId === ringId || this.secondRingId === ringId;\r\n }\r\n\r\n /**\r\n * Checks whether or not this ring connection is a bridge in a bridged ring.\r\n *\r\n * @param {Vertex[]} vertices The array of vertices associated with the current molecule.\r\n * @returns {Boolean} A boolean indicating whether or not this ring connection is a bridge.\r\n */\r\n isBridge(vertices) {\r\n if (this.vertices.size > 2) {\r\n return true;\r\n }\r\n\r\n for (let vertexId of this.vertices) {\r\n if(vertices[vertexId].value.rings.length > 2) {\r\n return true;\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Checks whether or not two rings are connected by a bridged bond.\r\n *\r\n * @static\r\n * @param {RingConnection[]} ringConnections An array of ring connections containing the ring connections associated with the current molecule.\r\n * @param {Vertex[]} vertices An array of vertices containing the vertices associated with the current molecule.\r\n * @param {Number} firstRingId A ring id.\r\n * @param {Number} secondRingId A ring id.\r\n * @returns {Boolean} A boolean indicating whether or not two rings ar connected by a bridged bond.\r\n */\r\n static isBridge(ringConnections, vertices, firstRingId, secondRingId) {\r\n let ringConnection = null;\r\n \r\n for (let i = 0; i < ringConnections.length; i++) {\r\n ringConnection = ringConnections[i];\r\n\r\n if (ringConnection.firstRingId === firstRingId && ringConnection.secondRingId === secondRingId ||\r\n ringConnection.firstRingId === secondRingId && ringConnection.secondRingId === firstRingId) {\r\n return ringConnection.isBridge(vertices);\r\n }\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Retruns the neighbouring rings of a given ring.\r\n *\r\n * @static\r\n * @param {RingConnection[]} ringConnections An array of ring connections containing ring connections associated with the current molecule.\r\n * @param {Number} ringId A ring id.\r\n * @returns {Number[]} An array of ring ids of neighbouring rings.\r\n */\r\n static getNeighbours(ringConnections, ringId) {\r\n let neighbours = [];\r\n\r\n for (let i = 0; i < ringConnections.length; i++) {\r\n let ringConnection = ringConnections[i];\r\n \r\n if (ringConnection.firstRingId === ringId) {\r\n neighbours.push(ringConnection.secondRingId);\r\n } else if (ringConnection.secondRingId === ringId) {\r\n neighbours.push(ringConnection.firstRingId);\r\n }\r\n }\r\n\r\n return neighbours;\r\n }\r\n\r\n /**\r\n * Returns an array of vertex ids associated with a given ring connection.\r\n *\r\n * @static\r\n * @param {RingConnection[]} ringConnections An array of ring connections containing ring connections associated with the current molecule.\r\n * @param {Number} firstRingId A ring id.\r\n * @param {Number} secondRingId A ring id.\r\n * @returns {Number[]} An array of vertex ids associated with the ring connection.\r\n */\r\n static getVertices(ringConnections, firstRingId, secondRingId) {\r\n for (let i = 0; i < ringConnections.length; i++) {\r\n let ringConnection = ringConnections[i];\r\n if (ringConnection.firstRingId === firstRingId && ringConnection.secondRingId === secondRingId ||\r\n ringConnection.firstRingId === secondRingId && ringConnection.secondRingId === firstRingId) {\r\n return [...ringConnection.vertices];\r\n }\r\n }\r\n }\r\n}\r\n\r\nmodule.exports = RingConnection","//@ts-check\r\nconst Graph = require('./Graph')\r\n\r\n/** A class encapsulating the functionality to find the smallest set of smallest rings in a graph. */\r\nclass SSSR {\r\n /**\r\n * Returns an array containing arrays, each representing a ring from the smallest set of smallest rings in the graph.\r\n * \r\n * @param {Graph} graph A Graph object.\r\n * @param {Boolean} [experimental=false] Whether or not to use experimental SSSR.\r\n * @returns {Array[]} An array containing arrays, each representing a ring from the smallest set of smallest rings in the group.\r\n */\r\n static getRings(graph, experimental=false) {\r\n let adjacencyMatrix = graph.getComponentsAdjacencyMatrix();\r\n if (adjacencyMatrix.length === 0) {\r\n return null;\r\n }\r\n\r\n let connectedComponents = Graph.getConnectedComponents(adjacencyMatrix);\r\n let rings = Array();\r\n\r\n for (var i = 0; i < connectedComponents.length; i++) {\r\n let connectedComponent = connectedComponents[i];\r\n let ccAdjacencyMatrix = graph.getSubgraphAdjacencyMatrix([...connectedComponent]);\r\n\r\n let arrBondCount = new Uint16Array(ccAdjacencyMatrix.length);\r\n let arrRingCount = new Uint16Array(ccAdjacencyMatrix.length);\r\n\r\n for (var j = 0; j < ccAdjacencyMatrix.length; j++) {\r\n arrRingCount[j] = 0;\r\n arrBondCount[j] = 0;\r\n\r\n for (var k = 0; k < ccAdjacencyMatrix[j].length; k++) {\r\n arrBondCount[j] += ccAdjacencyMatrix[j][k];\r\n }\r\n }\r\n\r\n // Get the edge number and the theoretical number of rings in SSSR\r\n let nEdges = 0;\r\n\r\n for (var j = 0; j < ccAdjacencyMatrix.length; j++) {\r\n for (var k = j + 1; k < ccAdjacencyMatrix.length; k++) {\r\n nEdges += ccAdjacencyMatrix[j][k];\r\n }\r\n }\r\n\r\n let nSssr = nEdges - ccAdjacencyMatrix.length + 1;\r\n\r\n // console.log(nEdges, ccAdjacencyMatrix.length, nSssr);\r\n // console.log(SSSR.getEdgeList(ccAdjacencyMatrix));\r\n // console.log(ccAdjacencyMatrix);\r\n\r\n // If all vertices have 3 incident edges, calculate with different formula (see Euler)\r\n let allThree = true;\r\n for (var j = 0; j < arrBondCount.length; j++) {\r\n if (arrBondCount[j] !== 3) {\r\n allThree = false;\r\n }\r\n }\r\n\r\n if (allThree) {\r\n nSssr = 2.0 + nEdges - ccAdjacencyMatrix.length;\r\n }\r\n\r\n // All vertices are part of one ring if theres only one ring.\r\n if (nSssr === 1) {\r\n rings.push([...connectedComponent]);\r\n continue;\r\n }\r\n \r\n if (experimental) {\r\n nSssr = 999;\r\n }\r\n\r\n let { d, pe, pe_prime } = SSSR.getPathIncludedDistanceMatrices(ccAdjacencyMatrix);\r\n let c = SSSR.getRingCandidates(d, pe, pe_prime);\r\n let sssr = SSSR.getSSSR(c, d, ccAdjacencyMatrix, pe, pe_prime, arrBondCount, arrRingCount, nSssr);\r\n\r\n for (var j = 0; j < sssr.length; j++) {\r\n let ring = Array(sssr[j].size);\r\n let index = 0;\r\n\r\n for (let val of sssr[j]) {\r\n // Get the original id of the vertex back\r\n ring[index++] = connectedComponent[val];\r\n }\r\n\r\n rings.push(ring);\r\n }\r\n }\r\n \r\n\r\n // So, for some reason, this would return three rings for C1CCCC2CC1CCCC2, which is wrong\r\n // As I don't have time to fix this properly, it will stay in. I'm sorry next person who works\r\n // on it. At that point it might be best to reimplement the whole SSSR thing...\r\n rings.pop();\r\n return rings;\r\n }\r\n\r\n /**\r\n * Creates a printable string from a matrix (2D array).\r\n * \r\n * @param {Array[]} matrix A 2D array.\r\n * @returns {String} A string representing the matrix.\r\n */\r\n static matrixToString(matrix) {\r\n let str = '';\r\n\r\n for (var i = 0; i < matrix.length; i++) {\r\n for (var j = 0; j < matrix[i].length; j++) {\r\n str += matrix[i][j] + ' ';\r\n }\r\n\r\n str += '\\n';\r\n }\r\n\r\n return str;\r\n }\r\n\r\n /**\r\n * Returnes the two path-included distance matrices used to find the sssr.\r\n * \r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @returns {Object} The path-included distance matrices. { p1, p2 }\r\n */\r\n static getPathIncludedDistanceMatrices(adjacencyMatrix) {\r\n let length = adjacencyMatrix.length;\r\n let d = Array(length);\r\n let pe = Array(length);\r\n let pe_prime = Array(length);\r\n var l = 0;\r\n var m = 0;\r\n var n = 0;\r\n\r\n var i = length;\r\n while (i--) {\r\n d[i] = Array(length);\r\n pe[i] = Array(length);\r\n pe_prime[i] = Array(length);\r\n\r\n var j = length;\r\n while (j--) {\r\n d[i][j] = (i === j || adjacencyMatrix[i][j] === 1) ? adjacencyMatrix[i][j] : Number.POSITIVE_INFINITY;\r\n\r\n if (d[i][j] === 1) {\r\n pe[i][j] = [[[i, j]]];\r\n } else {\r\n pe[i][j] = Array();\r\n }\r\n\r\n pe_prime[i][j] = Array();\r\n }\r\n }\r\n\r\n var k = length;\r\n var j;\r\n while (k--) {\r\n i = length;\r\n while (i--) {\r\n j = length;\r\n while (j--) {\r\n const previousPathLength = d[i][j];\r\n const newPathLength = d[i][k] + d[k][j];\r\n\r\n if (previousPathLength > newPathLength) {\r\n var l, m, n;\r\n if (previousPathLength === newPathLength + 1) {\r\n pe_prime[i][j] = [pe[i][j].length];\r\n l = pe[i][j].length\r\n while (l--) {\r\n pe_prime[i][j][l] = [pe[i][j][l].length];\r\n m = pe[i][j][l].length\r\n while (m--) {\r\n pe_prime[i][j][l][m] = [pe[i][j][l][m].length];\r\n n = pe[i][j][l][m].length;\r\n while (n--) {\r\n pe_prime[i][j][l][m][n] = [pe[i][j][l][m][0], pe[i][j][l][m][1]];\r\n }\r\n }\r\n }\r\n } else {\r\n pe_prime[i][j] = Array();\r\n }\r\n\r\n d[i][j] = newPathLength;\r\n\r\n pe[i][j] = [[]];\r\n\r\n l = pe[i][k][0].length;\r\n while (l--) {\r\n pe[i][j][0].push(pe[i][k][0][l]);\r\n }\r\n\r\n l = pe[k][j][0].length;\r\n while (l--) {\r\n pe[i][j][0].push(pe[k][j][0][l]);\r\n }\r\n } else if (previousPathLength === newPathLength) {\r\n if (pe[i][k].length && pe[k][j].length) {\r\n var l;\r\n if (pe[i][j].length) {\r\n let tmp = Array();\r\n\r\n l = pe[i][k][0].length;\r\n while (l--) {\r\n tmp.push(pe[i][k][0][l]);\r\n }\r\n\r\n l = pe[k][j][0].length;\r\n while (l--) {\r\n tmp.push(pe[k][j][0][l]);\r\n }\r\n\r\n pe[i][j].push(tmp);\r\n } else {\r\n let tmp = Array();\r\n l = pe[i][k][0].length;\r\n while (l--) {\r\n tmp.push(pe[i][k][0][l]);\r\n }\r\n\r\n l = pe[k][j][0].length;\r\n while (l--) {\r\n tmp.push(pe[k][j][0][l]);\r\n }\r\n\r\n pe[i][j][0] = tmp\r\n }\r\n }\r\n } else if (previousPathLength === newPathLength - 1) {\r\n var l;\r\n if (pe_prime[i][j].length) {\r\n let tmp = Array();\r\n\r\n l = pe[i][k][0].length;\r\n while (l--) {\r\n tmp.push(pe[i][k][0][l]);\r\n }\r\n\r\n l = pe[k][j][0].length;\r\n while (l--) {\r\n tmp.push(pe[k][j][0][l]);\r\n }\r\n\r\n pe_prime[i][j].push(tmp);\r\n } else {\r\n let tmp = Array();\r\n\r\n l = pe[i][k][0].length;\r\n while (l--) {\r\n tmp.push(pe[i][k][0][l]);\r\n }\r\n\r\n l = pe[k][j][0].length;\r\n while (l--) {\r\n tmp.push(pe[k][j][0][l]);\r\n }\r\n\r\n pe_prime[i][j][0] = tmp;\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n return {\r\n d: d,\r\n pe: pe,\r\n pe_prime: pe_prime\r\n };\r\n }\r\n\r\n /**\r\n * Get the ring candidates from the path-included distance matrices.\r\n * \r\n * @param {Array[]} d The distance matrix.\r\n * @param {Array[]} pe A matrix containing the shortest paths.\r\n * @param {Array[]} pe_prime A matrix containing the shortest paths + one vertex.\r\n * @returns {Array[]} The ring candidates.\r\n */\r\n static getRingCandidates(d, pe, pe_prime) {\r\n let length = d.length;\r\n let candidates = Array();\r\n let c = 0;\r\n\r\n for (let i = 0; i < length; i++) {\r\n for (let j = 0; j < length; j++) {\r\n if (d[i][j] === 0 || (pe[i][j].length === 1 && pe_prime[i][j] === 0)) {\r\n continue;\r\n } else {\r\n // c is the number of vertices in the cycle.\r\n if (pe_prime[i][j].length !== 0) {\r\n c = 2 * (d[i][j] + 0.5);\r\n } else {\r\n c = 2 * d[i][j];\r\n }\r\n\r\n if (c !== Infinity) {\r\n candidates.push([c, pe[i][j], pe_prime[i][j]]);\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Candidates have to be sorted by c\r\n candidates.sort(function (a, b) {\r\n return a[0] - b[0];\r\n });\r\n\r\n return candidates;\r\n }\r\n\r\n /**\r\n * Searches the candidates for the smallest set of smallest rings.\r\n * \r\n * @param {Array[]} c The candidates.\r\n * @param {Array[]} d The distance matrix.\r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @param {Array[]} pe A matrix containing the shortest paths.\r\n * @param {Array[]} pe_prime A matrix containing the shortest paths + one vertex.\r\n * @param {Uint16Array} arrBondCount A matrix containing the bond count of each vertex.\r\n * @param {Uint16Array} arrRingCount A matrix containing the number of rings associated with each vertex.\r\n * @param {Number} nsssr The theoretical number of rings in the graph.\r\n * @returns {Set[]} The smallest set of smallest rings.\r\n */\r\n static getSSSR(c, d, adjacencyMatrix, pe, pe_prime, arrBondCount, arrRingCount, nsssr) {\r\n let cSssr = Array();\r\n let allBonds = Array();\r\n\r\n for (let i = 0; i < c.length; i++) {\r\n if (c[i][0] % 2 !== 0) {\r\n for (let j = 0; j < c[i][2].length; j++) {\r\n let bonds = c[i][1][0].concat(c[i][2][j]);\r\n // Some bonds are added twice, resulting in [[u, v], [u, v]] instead of [u, v].\r\n // TODO: This is a workaround, fix later. Probably should be a set rather than an array, however the computational overhead\r\n // is probably bigger compared to leaving it like this.\r\n for (var k = 0; k < bonds.length; k++) {\r\n if (bonds[k][0].constructor === Array) bonds[k] = bonds[k][0];\r\n }\r\n\r\n let atoms = SSSR.bondsToAtoms(bonds);\r\n\r\n if (SSSR.getBondCount(atoms, adjacencyMatrix) === atoms.size && !SSSR.pathSetsContain(cSssr, atoms, bonds, allBonds, arrBondCount, arrRingCount)) {\r\n cSssr.push(atoms);\r\n allBonds = allBonds.concat(bonds);\r\n }\r\n\r\n if (cSssr.length > nsssr) {\r\n return cSssr;\r\n }\r\n }\r\n } else {\r\n for (let j = 0; j < c[i][1].length - 1; j++) {\r\n let bonds = c[i][1][j].concat(c[i][1][j + 1]);\r\n // Some bonds are added twice, resulting in [[u, v], [u, v]] instead of [u, v].\r\n // TODO: This is a workaround, fix later. Probably should be a set rather than an array, however the computational overhead\r\n // is probably bigger compared to leaving it like this.\r\n for (var k = 0; k < bonds.length; k++) {\r\n if (bonds[k][0].constructor === Array) bonds[k] = bonds[k][0];\r\n }\r\n\r\n let atoms = SSSR.bondsToAtoms(bonds);\r\n\r\n if (SSSR.getBondCount(atoms, adjacencyMatrix) === atoms.size && !SSSR.pathSetsContain(cSssr, atoms, bonds, allBonds, arrBondCount, arrRingCount)) {\r\n cSssr.push(atoms);\r\n allBonds = allBonds.concat(bonds);\r\n }\r\n\r\n if (cSssr.length > nsssr) {\r\n return cSssr;\r\n }\r\n }\r\n }\r\n }\r\n\r\n return cSssr;\r\n }\r\n\r\n /**\r\n * Returns the number of edges in a graph defined by an adjacency matrix.\r\n * \r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @returns {Number} The number of edges in the graph defined by the adjacency matrix.\r\n */\r\n static getEdgeCount(adjacencyMatrix) {\r\n let edgeCount = 0;\r\n let length = adjacencyMatrix.length;\r\n\r\n var i = length - 1;\r\n while (i--) {\r\n var j = length;\r\n while (j--) {\r\n if (adjacencyMatrix[i][j] === 1) {\r\n edgeCount++;\r\n }\r\n }\r\n }\r\n\r\n return edgeCount;\r\n }\r\n\r\n /**\r\n * Returns an edge list constructed form an adjacency matrix.\r\n * \r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @returns {Array[]} An edge list. E.g. [ [ 0, 1 ], ..., [ 16, 2 ] ]\r\n */\r\n static getEdgeList(adjacencyMatrix) {\r\n let length = adjacencyMatrix.length;\r\n let edgeList = Array();\r\n\r\n var i = length - 1;\r\n while (i--) {\r\n var j = length;\r\n while (j--) {\r\n if (adjacencyMatrix[i][j] === 1) {\r\n edgeList.push([i, j]);\r\n }\r\n }\r\n }\r\n\r\n return edgeList;\r\n }\r\n\r\n /**\r\n * Return a set of vertex indices contained in an array of bonds.\r\n * \r\n * @param {Array} bonds An array of bonds. A bond is defined as [ sourceVertexId, targetVertexId ].\r\n * @returns {Set} An array of vertices.\r\n */\r\n static bondsToAtoms(bonds) {\r\n let atoms = new Set();\r\n\r\n var i = bonds.length;\r\n while (i--) {\r\n atoms.add(bonds[i][0]);\r\n atoms.add(bonds[i][1]);\r\n }\r\n return atoms;\r\n }\r\n\r\n /**\r\n * Returns the number of bonds within a set of atoms.\r\n * \r\n * @param {Set} atoms An array of atom ids.\r\n * @param {Array[]} adjacencyMatrix An adjacency matrix.\r\n * @returns {Number} The number of bonds in a set of atoms.\r\n */\r\n static getBondCount(atoms, adjacencyMatrix) {\r\n let count = 0;\r\n for (let u of atoms) {\r\n for (let v of atoms) {\r\n if (u === v) {\r\n continue;\r\n }\r\n count += adjacencyMatrix[u][v]\r\n }\r\n }\r\n\r\n return count / 2;\r\n }\r\n\r\n /**\r\n * Checks whether or not a given path already exists in an array of paths.\r\n * \r\n * @param {Set[]} pathSets An array of sets each representing a path.\r\n * @param {Set} pathSet A set representing a path.\r\n * @param {Array[]} bonds The bonds associated with the current path.\r\n * @param {Array[]} allBonds All bonds currently associated with rings in the SSSR set.\r\n * @param {Uint16Array} arrBondCount A matrix containing the bond count of each vertex.\r\n * @param {Uint16Array} arrRingCount A matrix containing the number of rings associated with each vertex.\r\n * @returns {Boolean} A boolean indicating whether or not a give path is contained within a set.\r\n */\r\n static pathSetsContain(pathSets, pathSet, bonds, allBonds, arrBondCount, arrRingCount) {\r\n var i = pathSets.length;\r\n while (i--) {\r\n if (SSSR.isSupersetOf(pathSet, pathSets[i])) {\r\n return true;\r\n }\r\n\r\n if (pathSets[i].size !== pathSet.size) {\r\n continue;\r\n }\r\n\r\n if (SSSR.areSetsEqual(pathSets[i], pathSet)) {\r\n return true;\r\n }\r\n }\r\n\r\n // Check if the edges from the candidate are already all contained within the paths of the set of paths.\r\n // TODO: For some reason, this does not replace the isSupersetOf method above -> why?\r\n let count = 0;\r\n let allContained = false;\r\n i = bonds.length;\r\n while (i--) {\r\n var j = allBonds.length;\r\n while (j--) {\r\n if (bonds[i][0] === allBonds[j][0] && bonds[i][1] === allBonds[j][1] ||\r\n bonds[i][1] === allBonds[j][0] && bonds[i][0] === allBonds[j][1]) {\r\n count++;\r\n }\r\n\r\n if (count === bonds.length) {\r\n allContained = true;\r\n }\r\n }\r\n }\r\n\r\n // If all the bonds and thus vertices are already contained within other rings\r\n // check if there's one vertex with ringCount < bondCount\r\n let specialCase = false;\r\n if (allContained) {\r\n for (let element of pathSet) {\r\n if (arrRingCount[element] < arrBondCount[element]) {\r\n specialCase = true;\r\n break;\r\n }\r\n }\r\n }\r\n\r\n if (allContained && !specialCase) {\r\n return true;\r\n }\r\n\r\n // Update the ring counts for the vertices\r\n for (let element of pathSet) {\r\n arrRingCount[element]++;\r\n }\r\n\r\n return false;\r\n }\r\n\r\n /**\r\n * Checks whether or not two sets are equal (contain the same elements).\r\n * \r\n * @param {Set} setA A set.\r\n * @param {Set} setB A set.\r\n * @returns {Boolean} A boolean indicating whether or not the two sets are equal.\r\n */\r\n static areSetsEqual(setA, setB) {\r\n if (setA.size !== setB.size) {\r\n return false;\r\n }\r\n\r\n for (let element of setA) {\r\n if (!setB.has(element)) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n\r\n /**\r\n * Checks whether or not a set (setA) is a superset of another set (setB).\r\n * \r\n * @param {Set} setA A set.\r\n * @param {Set} setB A set.\r\n * @returns {Boolean} A boolean indicating whether or not setB is a superset of setA.\r\n */\r\n static isSupersetOf(setA, setB) {\r\n for (var element of setB) {\r\n if (!setA.has(element)) {\r\n return false;\r\n }\r\n }\r\n\r\n return true;\r\n }\r\n}\r\n\r\nmodule.exports = SSSR;","//@ts-check\r\n\r\n/** \r\n * A class representing a 2D vector.\r\n * \r\n * @property {Number} x The x component of the vector.\r\n * @property {Number} y The y component of the vector.\r\n */\r\nclass Vector2 {\r\n /**\r\n * The constructor of the class Vector2.\r\n *\r\n * @param {(Number|Vector2)} x The initial x coordinate value or, if the single argument, a Vector2 object.\r\n * @param {Number} y The initial y coordinate value.\r\n */\r\n constructor(x, y) {\r\n if (arguments.length == 0) {\r\n this.x = 0;\r\n this.y = 0;\r\n } else if (arguments.length == 1) {\r\n this.x = x.x;\r\n this.y = x.y;\r\n } else {\r\n this.x = x;\r\n this.y = y;\r\n }\r\n }\r\n\r\n /**\r\n * Clones this vector and returns the clone.\r\n *\r\n * @returns {Vector2} The clone of this vector.\r\n */\r\n clone() {\r\n return new Vector2(this.x, this.y);\r\n }\r\n\r\n /**\r\n * Returns a string representation of this vector.\r\n *\r\n * @returns {String} A string representation of this vector.\r\n */\r\n toString() {\r\n return '(' + this.x + ',' + this.y + ')';\r\n }\r\n\r\n /**\r\n * Add the x and y coordinate values of a vector to the x and y coordinate values of this vector.\r\n *\r\n * @param {Vector2} vec Another vector.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n add(vec) {\r\n this.x += vec.x;\r\n this.y += vec.y;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Subtract the x and y coordinate values of a vector from the x and y coordinate values of this vector.\r\n *\r\n * @param {Vector2} vec Another vector.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n subtract(vec) {\r\n this.x -= vec.x;\r\n this.y -= vec.y;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Divide the x and y coordinate values of this vector by a scalar.\r\n *\r\n * @param {Number} scalar The scalar.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n divide(scalar) {\r\n this.x /= scalar;\r\n this.y /= scalar;\r\n\r\n return this;\r\n }\r\n \r\n /**\r\n * Multiply the x and y coordinate values of this vector by the values of another vector.\r\n *\r\n * @param {Vector2} v A vector.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n multiply(v) {\r\n this.x *= v.x;\r\n this.y *= v.y;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Multiply the x and y coordinate values of this vector by a scalar.\r\n *\r\n * @param {Number} scalar The scalar.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n multiplyScalar(scalar) {\r\n this.x *= scalar;\r\n this.y *= scalar;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Inverts this vector. Same as multiply(-1.0).\r\n *\r\n * @returns {Vector2} Returns itself.\r\n */\r\n invert() {\r\n this.x = -this.x;\r\n this.y = -this.y;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns the angle of this vector in relation to the coordinate system.\r\n *\r\n * @returns {Number} The angle in radians.\r\n */\r\n angle() {\r\n return Math.atan2(this.y, this.x);\r\n }\r\n\r\n /**\r\n * Returns the euclidean distance between this vector and another vector.\r\n *\r\n * @param {Vector2} vec A vector.\r\n * @returns {Number} The euclidean distance between the two vectors.\r\n */\r\n distance(vec) {\r\n return Math.sqrt((vec.x - this.x) * (vec.x - this.x) + (vec.y - this.y) * (vec.y - this.y));\r\n }\r\n\r\n /**\r\n * Returns the squared euclidean distance between this vector and another vector. When only the relative distances of a set of vectors are needed, this is is less expensive than using distance(vec).\r\n *\r\n * @param {Vector2} vec Another vector.\r\n * @returns {Number} The squared euclidean distance of the two vectors.\r\n */\r\n distanceSq(vec) {\r\n return (vec.x - this.x) * (vec.x - this.x) + (vec.y - this.y) * (vec.y - this.y);\r\n }\r\n\r\n /**\r\n * Checks whether or not this vector is in a clockwise or counter-clockwise rotational direction compared to another vector in relation to the coordinate system.\r\n *\r\n * @param {Vector2} vec Another vector.\r\n * @returns {Number} Returns -1, 0 or 1 if the vector supplied as an argument is clockwise, neutral or counter-clockwise respectively to this vector in relation to the coordinate system.\r\n */\r\n clockwise(vec) {\r\n let a = this.y * vec.x;\r\n let b = this.x * vec.y;\r\n \r\n if (a > b) {\r\n return -1;\r\n }\r\n else if (a === b) {\r\n return 0;\r\n }\r\n\r\n return 1;\r\n }\r\n\r\n /**\r\n * Checks whether or not this vector is in a clockwise or counter-clockwise rotational direction compared to another vector in relation to an arbitrary third vector.\r\n *\r\n * @param {Vector2} center The central vector.\r\n * @param {Vector2} vec Another vector.\r\n * @returns {Number} Returns -1, 0 or 1 if the vector supplied as an argument is clockwise, neutral or counter-clockwise respectively to this vector in relation to an arbitrary third vector.\r\n */\r\n relativeClockwise(center, vec) {\r\n let a = (this.y - center.y) * (vec.x - center.x);\r\n let b = (this.x - center.x) * (vec.y - center.y);\r\n \r\n if (a > b) {\r\n return -1;\r\n }\r\n else if (a === b) {\r\n return 0;\r\n }\r\n\r\n return 1;\r\n }\r\n\r\n /**\r\n * Rotates this vector by a given number of radians around the origin of the coordinate system.\r\n *\r\n * @param {Number} angle The angle in radians to rotate the vector.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n rotate(angle) {\r\n let tmp = new Vector2(0, 0);\r\n let cosAngle = Math.cos(angle);\r\n let sinAngle = Math.sin(angle);\r\n\r\n tmp.x = this.x * cosAngle - this.y * sinAngle;\r\n tmp.y = this.x * sinAngle + this.y * cosAngle;\r\n \r\n this.x = tmp.x;\r\n this.y = tmp.y;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Rotates this vector around another vector.\r\n *\r\n * @param {Number} angle The angle in radians to rotate the vector.\r\n * @param {Vector2} vec The vector which is used as the rotational center.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n rotateAround(angle, vec) {\r\n let s = Math.sin(angle);\r\n let c = Math.cos(angle);\r\n\r\n this.x -= vec.x;\r\n this.y -= vec.y;\r\n\r\n let x = this.x * c - this.y * s;\r\n let y = this.x * s + this.y * c;\r\n\r\n this.x = x + vec.x;\r\n this.y = y + vec.y;\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Rotate a vector around a given center to the same angle as another vector (so that the two vectors and the center are in a line, with both vectors on one side of the center), keeps the distance from this vector to the center.\r\n *\r\n * @param {Vector2} vec The vector to rotate this vector to.\r\n * @param {Vector2} center The rotational center.\r\n * @param {Number} [offsetAngle=0.0] An additional amount of radians to rotate the vector.\r\n * @returns {Vector2} Returns itself.\r\n */\r\n rotateTo(vec, center, offsetAngle = 0.0) {\r\n // Problem if this is first position\r\n this.x += 0.001;\r\n this.y -= 0.001;\r\n\r\n let a = Vector2.subtract(this, center);\r\n let b = Vector2.subtract(vec, center);\r\n let angle = Vector2.angle(b, a);\r\n\r\n this.rotateAround(angle + offsetAngle, center);\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Rotates the vector away from a specified vector around a center.\r\n * \r\n * @param {Vector2} vec The vector this one is rotated away from.\r\n * @param {Vector2} center The rotational center.\r\n * @param {Number} angle The angle by which to rotate.\r\n */\r\n rotateAwayFrom(vec, center, angle) {\r\n this.rotateAround(angle, center);\r\n \r\n let distSqA = this.distanceSq(vec);\r\n \r\n this.rotateAround(-2.0 * angle, center);\r\n\r\n let distSqB = this.distanceSq(vec);\r\n\r\n // If it was rotated towards the other vertex, rotate in other direction by same amount.\r\n if (distSqB < distSqA) {\r\n this.rotateAround(2.0 * angle, center);\r\n }\r\n }\r\n\r\n /**\r\n * Returns the angle in radians used to rotate this vector away from a given vector.\r\n * \r\n * @param {Vector2} vec The vector this one is rotated away from.\r\n * @param {Vector2} center The rotational center.\r\n * @param {Number} angle The angle by which to rotate.\r\n * @returns {Number} The angle in radians.\r\n */\r\n getRotateAwayFromAngle(vec, center, angle) {\r\n let tmp = this.clone();\r\n\r\n tmp.rotateAround(angle, center);\r\n \r\n let distSqA = tmp.distanceSq(vec);\r\n \r\n tmp.rotateAround(-2.0 * angle, center);\r\n\r\n let distSqB = tmp.distanceSq(vec);\r\n\r\n if (distSqB < distSqA) {\r\n return angle;\r\n } else {\r\n return -angle;\r\n }\r\n }\r\n\r\n /**\r\n * Returns the angle in radians used to rotate this vector towards a given vector.\r\n * \r\n * @param {Vector2} vec The vector this one is rotated towards to.\r\n * @param {Vector2} center The rotational center.\r\n * @param {Number} angle The angle by which to rotate.\r\n * @returns {Number} The angle in radians.\r\n */\r\n getRotateTowardsAngle(vec, center, angle) {\r\n let tmp = this.clone();\r\n\r\n tmp.rotateAround(angle, center);\r\n \r\n let distSqA = tmp.distanceSq(vec);\r\n \r\n tmp.rotateAround(-2.0 * angle, center);\r\n\r\n let distSqB = tmp.distanceSq(vec);\r\n\r\n if (distSqB > distSqA) {\r\n return angle;\r\n } else {\r\n return -angle;\r\n }\r\n }\r\n\r\n /**\r\n * Gets the angles between this vector and another vector around a common center of rotation.\r\n *\r\n * @param {Vector2} vec Another vector.\r\n * @param {Vector2} center The center of rotation.\r\n * @returns {Number} The angle between this vector and another vector around a center of rotation in radians.\r\n */\r\n getRotateToAngle(vec, center) {\r\n let a = Vector2.subtract(this, center);\r\n let b = Vector2.subtract(vec, center);\r\n let angle = Vector2.angle(b, a);\r\n \r\n return Number.isNaN(angle) ? 0.0 : angle;\r\n }\r\n\r\n /**\r\n * Checks whether a vector lies within a polygon spanned by a set of vectors.\r\n *\r\n * @param {Vector2[]} polygon An array of vectors spanning the polygon.\r\n * @returns {Boolean} A boolean indicating whether or not this vector is within a polygon.\r\n */\r\n isInPolygon(polygon) {\r\n let inside = false;\r\n\r\n // Its not always a given, that the polygon is convex (-> sugars)\r\n for (let i = 0, j = polygon.length - 1; i < polygon.length; j = i++) {\r\n if (((polygon[i].y > this.y) != (polygon[j].y > this.y)) &&\r\n (this.x < (polygon[j].x - polygon[i].x) * (this.y - polygon[i].y) /\r\n (polygon[j].y - polygon[i].y) + polygon[i].x)) {\r\n inside = !inside;\r\n }\r\n }\r\n\r\n\r\n return inside;\r\n }\r\n\r\n /**\r\n * Returns the length of this vector.\r\n *\r\n * @returns {Number} The length of this vector.\r\n */\r\n length() {\r\n return Math.sqrt((this.x * this.x) + (this.y * this.y));\r\n }\r\n\r\n /**\r\n * Returns the square of the length of this vector.\r\n *\r\n * @returns {Number} The square of the length of this vector.\r\n */\r\n lengthSq() {\r\n return (this.x * this.x) + (this.y * this.y);\r\n }\r\n\r\n /**\r\n * Normalizes this vector.\r\n *\r\n * @returns {Vector2} Returns itself.\r\n */\r\n normalize() {\r\n this.divide(this.length());\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Returns a normalized copy of this vector.\r\n *\r\n * @returns {Vector2} A normalized copy of this vector.\r\n */\r\n normalized() {\r\n return Vector2.divideScalar(this, this.length());\r\n }\r\n\r\n /**\r\n * Calculates which side of a line spanned by two vectors this vector is.\r\n *\r\n * @param {Vector2} vecA A vector.\r\n * @param {Vector2} vecB A vector.\r\n * @returns {Number} A number indicating the side of this vector, given a line spanned by two other vectors.\r\n */\r\n whichSide(vecA, vecB) {\r\n return (this.x - vecA.x) * (vecB.y - vecA.y) - (this.y - vecA.y) * (vecB.x - vecA.x);\r\n }\r\n\r\n /**\r\n * Checks whether or not this vector is on the same side of a line spanned by two vectors as another vector.\r\n *\r\n * @param {Vector2} vecA A vector spanning the line.\r\n * @param {Vector2} vecB A vector spanning the line.\r\n * @param {Vector2} vecC A vector to check whether or not it is on the same side as this vector.\r\n * @returns {Boolean} Returns a boolean indicating whether or not this vector is on the same side as another vector.\r\n */\r\n sameSideAs(vecA, vecB, vecC) {\r\n let d = this.whichSide(vecA, vecB);\r\n let dRef = vecC.whichSide(vecA, vecB);\r\n\r\n return d < 0 && dRef < 0 || d == 0 && dRef == 0 || d > 0 && dRef > 0;\r\n }\r\n\r\n /**\r\n * Adds two vectors and returns the result as a new vector.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A summand.\r\n * @param {Vector2} vecB A summand.\r\n * @returns {Vector2} Returns the sum of two vectors.\r\n */\r\n static add(vecA, vecB) {\r\n return new Vector2(vecA.x + vecB.x, vecA.y + vecB.y);\r\n }\r\n\r\n /**\r\n * Subtracts one vector from another and returns the result as a new vector.\r\n *\r\n * @static\r\n * @param {Vector2} vecA The minuend.\r\n * @param {Vector2} vecB The subtrahend.\r\n * @returns {Vector2} Returns the difference of two vectors.\r\n */\r\n static subtract(vecA, vecB) {\r\n return new Vector2(vecA.x - vecB.x, vecA.y - vecB.y);\r\n }\r\n\r\n /**\r\n * Multiplies two vectors (value by value) and returns the result.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector.\r\n * @param {Vector2} vecB A vector.\r\n * @returns {Vector2} Returns the product of two vectors.\r\n */\r\n static multiply(vecA, vecB) {\r\n return new Vector2(vecA.x * vecB.x, vecA.y * vecB.y);\r\n }\r\n\r\n /**\r\n * Multiplies two vectors (value by value) and returns the result.\r\n *\r\n * @static\r\n * @param {Vector2} vec A vector.\r\n * @param {Number} scalar A scalar.\r\n * @returns {Vector2} Returns the product of two vectors.\r\n */\r\n static multiplyScalar(vec, scalar) {\r\n return new Vector2(vec.x, vec.y).multiplyScalar(scalar);\r\n }\r\n\r\n /**\r\n * Returns the midpoint of a line spanned by two vectors.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector spanning the line.\r\n * @param {Vector2} vecB A vector spanning the line.\r\n * @returns {Vector2} The midpoint of the line spanned by two vectors.\r\n */\r\n static midpoint(vecA, vecB) {\r\n return new Vector2((vecA.x + vecB.x) / 2, (vecA.y + vecB.y) / 2);\r\n }\r\n\r\n /**\r\n * Returns the normals of a line spanned by two vectors.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector spanning the line.\r\n * @param {Vector2} vecB A vector spanning the line.\r\n * @returns {Vector2[]} An array containing the two normals, each represented by a vector.\r\n */\r\n static normals(vecA, vecB) {\r\n let delta = Vector2.subtract(vecB, vecA);\r\n\r\n return [\r\n new Vector2(-delta.y, delta.x),\r\n new Vector2(delta.y, -delta.x)\r\n ];\r\n }\r\n\r\n /**\r\n * Returns the unit (normalized normal) vectors of a line spanned by two vectors.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector spanning the line.\r\n * @param {Vector2} vecB A vector spanning the line.\r\n * @returns {Vector2[]} An array containing the two unit vectors.\r\n */\r\n static units(vecA, vecB) {\r\n let delta = Vector2.subtract(vecB, vecA);\r\n\r\n return [\r\n (new Vector2(-delta.y, delta.x)).normalize(),\r\n (new Vector2(delta.y, -delta.x)).normalize()\r\n ];\r\n }\r\n\r\n /**\r\n * Divides a vector by another vector and returns the result as new vector.\r\n *\r\n * @static\r\n * @param {Vector2} vecA The dividend.\r\n * @param {Vector2} vecB The divisor.\r\n * @returns {Vector2} The fraction of the two vectors.\r\n */\r\n static divide(vecA, vecB) {\r\n return new Vector2(vecA.x / vecB.x, vecA.y / vecB.y);\r\n }\r\n\r\n /**\r\n * Divides a vector by a scalar and returns the result as new vector.\r\n *\r\n * @static\r\n * @param {Vector2} vecA The dividend.\r\n * @param {Number} s The scalar.\r\n * @returns {Vector2} The fraction of the two vectors.\r\n */\r\n static divideScalar(vecA, s) {\r\n return new Vector2(vecA.x / s, vecA.y / s);\r\n }\r\n\r\n /**\r\n * Returns the dot product of two vectors.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector.\r\n * @param {Vector2} vecB A vector.\r\n * @returns {Number} The dot product of two vectors.\r\n */\r\n static dot(vecA, vecB) {\r\n return vecA.x * vecB.x + vecA.y * vecB.y;\r\n }\r\n\r\n /**\r\n * Returns the angle between two vectors.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector.\r\n * @param {Vector2} vecB A vector.\r\n * @returns {Number} The angle between two vectors in radians.\r\n */\r\n static angle(vecA, vecB) {\r\n let dot = Vector2.dot(vecA, vecB);\r\n\r\n return Math.acos(dot / (vecA.length() * vecB.length()));\r\n }\r\n\r\n /**\r\n * Returns the angle between two vectors based on a third vector in between.\r\n *\r\n * @static\r\n * @param {Vector2} vecA A vector.\r\n * @param {Vector2} vecB A (central) vector.\r\n * @param {Vector2} vecC A vector.\r\n * @returns {Number} The angle in radians.\r\n */\r\n static threePointangle(vecA, vecB, vecC) {\r\n let ab = Vector2.subtract(vecB, vecA);\r\n let bc = Vector2.subtract(vecC, vecB);\r\n let abLength = vecA.distance(vecB);\r\n let bcLength = vecB.distance(vecC);\r\n\r\n return Math.acos(Vector2.dot(ab, bc) / (abLength * bcLength));\r\n }\r\n \r\n /**\r\n * Returns the scalar projection of a vector on another vector.\r\n *\r\n * @static\r\n * @param {Vector2} vecA The vector to be projected.\r\n * @param {Vector2} vecB The vector to be projection upon.\r\n * @returns {Number} The scalar component.\r\n */\r\n static scalarProjection(vecA, vecB) {\r\n let unit = vecB.normalized();\r\n \r\n return Vector2.dot(vecA, unit);\r\n }\r\n\r\n /**\r\n * Returns the average vector (normalized) of the input vectors.\r\n *\r\n * @static\r\n * @param {Array} vecs An array containing vectors.\r\n * @returns {Vector2} The resulting vector (normalized).\r\n */\r\n static averageDirection(vecs) {\r\n let avg = new Vector2(0.0, 0.0);\r\n\r\n for (var i = 0; i < vecs.length; i++) {\r\n let vec = vecs[i];\r\n avg.add(vec);\r\n }\r\n\r\n return avg.normalize();\r\n }\r\n}\r\n\r\nmodule.exports = Vector2;","//@ts-check\r\nconst MathHelper = require('./MathHelper')\r\nconst ArrayHelper = require('./ArrayHelper')\r\nconst Vector2 = require('./Vector2')\r\nconst Atom = require('./Atom')\r\n\r\n/** \r\n * A class representing a vertex.\r\n * \r\n * @property {Number} id The id of this vertex.\r\n * @property {Atom} value The atom associated with this vertex.\r\n * @property {Vector2} position The position of this vertex.\r\n * @property {Vector2} previousPosition The position of the previous vertex.\r\n * @property {Number|null} parentVertexId The id of the previous vertex.\r\n * @property {Number[]} children The ids of the children of this vertex.\r\n * @property {Number[]} spanningTreeChildren The ids of the children of this vertex as defined in the spanning tree defined by the SMILES.\r\n * @property {Number[]} edges The ids of edges associated with this vertex.\r\n * @property {Boolean} positioned A boolean indicating whether or not this vertex has been positioned.\r\n * @property {Number} angle The angle of this vertex.\r\n * @property {Number} dir The direction of this vertex.\r\n * @property {Number} neighbourCount The number of neighbouring vertices.\r\n * @property {Number[]} neighbours The vertex ids of neighbouring vertices.\r\n * @property {String[]} neighbouringElements The element symbols associated with neighbouring vertices.\r\n * @property {Boolean} forcePositioned A boolean indicating whether or not this vertex was positioned using a force-based approach.\r\n */\r\n\r\nclass Vertex {\r\n /**\r\n * The constructor for the class Vertex.\r\n *\r\n * @param {Atom} value The value associated with this vertex.\r\n * @param {Number} [x=0] The initial x coordinate of the positional vector of this vertex.\r\n * @param {Number} [y=0] The initial y coordinate of the positional vector of this vertex.\r\n */\r\n constructor(value, x = 0, y = 0) {\r\n this.id = null;\r\n this.value = value;\r\n this.position = new Vector2(x ? x : 0, y ? y : 0);\r\n this.previousPosition = new Vector2(0, 0);\r\n this.parentVertexId = null;\r\n this.children = Array();\r\n this.spanningTreeChildren = Array();\r\n this.edges = Array();\r\n this.positioned = false;\r\n this.angle = null;\r\n this.dir = 1.0;\r\n this.neighbourCount = 0;\r\n this.neighbours = Array();\r\n this.neighbouringElements = Array();\r\n this.forcePositioned = false;\r\n }\r\n\r\n /**\r\n * Set the 2D coordinates of the vertex.\r\n * \r\n * @param {Number} x The x component of the coordinates.\r\n * @param {Number} y The y component of the coordinates.\r\n * \r\n */\r\n setPosition(x, y) {\r\n this.position.x = x;\r\n this.position.y = y;\r\n }\r\n\r\n /**\r\n * Set the 2D coordinates of the vertex from a Vector2.\r\n * \r\n * @param {Vector2} v A 2D vector.\r\n * \r\n */\r\n setPositionFromVector(v) {\r\n this.position.x = v.x;\r\n this.position.y = v.y;\r\n }\r\n\r\n /**\r\n * Add a child vertex id to this vertex.\r\n * @param {Number} vertexId The id of a vertex to be added as a child to this vertex.\r\n */\r\n addChild(vertexId) {\r\n this.children.push(vertexId);\r\n this.neighbours.push(vertexId);\r\n\r\n this.neighbourCount++;\r\n }\r\n\r\n /**\r\n * Add a child vertex id to this vertex as the second child of the neighbours array,\r\n * except this vertex is the first vertex of the SMILE string, then it is added as the first.\r\n * This is used to get the correct ordering of neighbours for parity calculations.\r\n * If a hydrogen is implicitly attached to the chiral center, insert as the third child.\r\n * @param {Number} vertexId The id of a vertex to be added as a child to this vertex.\r\n * @param {Number} ringbondIndex The index of the ringbond.\r\n */\r\n addRingbondChild(vertexId, ringbondIndex) {\r\n this.children.push(vertexId);\r\n\r\n if (this.value.bracket) {\r\n let index = 1;\r\n\r\n if (this.id === 0 && this.value.bracket.hcount === 0) {\r\n index = 0;\r\n }\r\n \r\n if (this.value.bracket.hcount === 1 && ringbondIndex === 0) {\r\n index = 2;\r\n }\r\n\r\n if (this.value.bracket.hcount === 1 && ringbondIndex === 1) {\r\n if (this.neighbours.length < 3) {\r\n index = 2;\r\n } else {\r\n index = 3;\r\n }\r\n }\r\n\r\n if (this.value.bracket.hcount === null && ringbondIndex === 0) {\r\n index = 1;\r\n }\r\n\r\n if (this.value.bracket.hcount === null && ringbondIndex === 1) {\r\n if (this.neighbours.length < 3) {\r\n index = 1;\r\n } else {\r\n index = 2;\r\n }\r\n }\r\n \r\n this.neighbours.splice(index, 0, vertexId);\r\n } else {\r\n this.neighbours.push(vertexId);\r\n }\r\n\r\n this.neighbourCount++;\r\n }\r\n\r\n /**\r\n * Set the vertex id of the parent.\r\n * \r\n * @param {Number} parentVertexId The parents vertex id.\r\n */\r\n setParentVertexId(parentVertexId) {\r\n this.neighbourCount++;\r\n this.parentVertexId = parentVertexId;\r\n this.neighbours.push(parentVertexId);\r\n }\r\n\r\n /**\r\n * Returns true if this vertex is terminal (has no parent or child vertices), otherwise returns false. Always returns true if associated value has property hasAttachedPseudoElements set to true.\r\n *\r\n * @returns {Boolean} A boolean indicating whether or not this vertex is terminal.\r\n */\r\n isTerminal() {\r\n if (this.value.hasAttachedPseudoElements) {\r\n return true;\r\n }\r\n\r\n return (this.parentVertexId === null && this.children.length < 2) || this.children.length === 0;\r\n }\r\n\r\n /**\r\n * Clones this vertex and returns the clone.\r\n *\r\n * @returns {Vertex} A clone of this vertex.\r\n */\r\n clone() {\r\n let clone = new Vertex(this.value, this.position.x, this.position.y);\r\n clone.id = this.id;\r\n clone.previousPosition = new Vector2(this.previousPosition.x, this.previousPosition.y);\r\n clone.parentVertexId = this.parentVertexId;\r\n clone.children = ArrayHelper.clone(this.children);\r\n clone.spanningTreeChildren = ArrayHelper.clone(this.spanningTreeChildren);\r\n clone.edges = ArrayHelper.clone(this.edges);\r\n clone.positioned = this.positioned;\r\n clone.angle = this.angle;\r\n clone.forcePositioned = this.forcePositioned;\r\n return clone;\r\n }\r\n\r\n /**\r\n * Returns true if this vertex and the supplied vertex both have the same id, else returns false.\r\n *\r\n * @param {Vertex} vertex The vertex to check.\r\n * @returns {Boolean} A boolean indicating whether or not the two vertices have the same id.\r\n */\r\n equals(vertex) {\r\n return this.id === vertex.id;\r\n }\r\n\r\n /**\r\n * Returns the angle of this vertexes positional vector. If a reference vector is supplied in relations to this vector, else in relations to the coordinate system.\r\n *\r\n * @param {Vector2} [referenceVector=null] - The reference vector.\r\n * @param {Boolean} [returnAsDegrees=false] - If true, returns angle in degrees, else in radians.\r\n * @returns {Number} The angle of this vertex.\r\n */\r\n getAngle(referenceVector = null, returnAsDegrees = false) {\r\n let u = null;\r\n\r\n if (!referenceVector) {\r\n u = Vector2.subtract(this.position, this.previousPosition);\r\n } else {\r\n u = Vector2.subtract(this.position, referenceVector);\r\n }\r\n\r\n if (returnAsDegrees) {\r\n return MathHelper.toDeg(u.angle());\r\n }\r\n\r\n return u.angle();\r\n }\r\n\r\n /**\r\n * Returns the suggested text direction when text is added at the position of this vertex.\r\n *\r\n * @param {Vertex[]} vertices The array of vertices for the current molecule.\r\n * @returns {String} The suggested direction of the text.\r\n */\r\n getTextDirection(vertices) {\r\n let neighbours = this.getDrawnNeighbours(vertices);\r\n let angles = Array();\r\n\r\n for (let i = 0; i < neighbours.length; i++) {\r\n angles.push(this.getAngle(vertices[neighbours[i]].position));\r\n }\r\n\r\n let textAngle = MathHelper.meanAngle(angles);\r\n\r\n // Round to 0, 90, 180 or 270 degree\r\n let halfPi = Math.PI / 2.0;\r\n textAngle = Math.round(Math.round(textAngle / halfPi) * halfPi);\r\n\r\n if (textAngle === 2) {\r\n return 'down';\r\n } else if (textAngle === -2) {\r\n return 'up';\r\n } else if (textAngle === 0 || textAngle === -0) {\r\n return 'right'; // is checking for -0 necessary?\r\n } else if (textAngle === 3 || textAngle === -3) {\r\n return 'left';\r\n } else {\r\n return 'down'; // default to down\r\n }\r\n }\r\n\r\n /**\r\n * Returns an array of ids of neighbouring vertices.\r\n *\r\n * @param {Number} [vertexId=null] If a value is supplied, the vertex with this id is excluded from the returned indices.\r\n * @returns {Number[]} An array containing the ids of neighbouring vertices.\r\n */\r\n getNeighbours(vertexId = null) {\r\n if (vertexId === null) {\r\n return this.neighbours.slice();\r\n }\r\n\r\n let arr = Array();\r\n\r\n for (let i = 0; i < this.neighbours.length; i++) {\r\n if (this.neighbours[i] !== vertexId) {\r\n arr.push(this.neighbours[i]);\r\n }\r\n }\r\n\r\n return arr;\r\n }\r\n\r\n /**\r\n * Returns an array of ids of neighbouring vertices that will be drawn (vertex.value.isDrawn === true).\r\n * \r\n * @param {Vertex[]} vertices An array containing the vertices associated with the current molecule.\r\n * @returns {Number[]} An array containing the ids of neighbouring vertices that will be drawn.\r\n */\r\n getDrawnNeighbours(vertices) {\r\n let arr = Array();\r\n\r\n for (let i = 0; i < this.neighbours.length; i++) {\r\n if (vertices[this.neighbours[i]].value.isDrawn) {\r\n arr.push(this.neighbours[i]);\r\n }\r\n }\r\n\r\n return arr;\r\n }\r\n\r\n /**\r\n * Returns the number of neighbours of this vertex.\r\n *\r\n * @returns {Number} The number of neighbours.\r\n */\r\n getNeighbourCount() {\r\n return this.neighbourCount;\r\n }\r\n\r\n /**\r\n * Returns a list of ids of vertices neighbouring this one in the original spanning tree, excluding the ringbond connections.\r\n *\r\n * @param {Number} [vertexId=null] If supplied, the vertex with this id is excluded from the array returned.\r\n * @returns {Number[]} An array containing the ids of the neighbouring vertices.\r\n */\r\n getSpanningTreeNeighbours(vertexId = null) {\r\n let neighbours = Array();\r\n\r\n for (let i = 0; i < this.spanningTreeChildren.length; i++) {\r\n if (vertexId === undefined || vertexId != this.spanningTreeChildren[i]) {\r\n neighbours.push(this.spanningTreeChildren[i]);\r\n }\r\n }\r\n\r\n if (this.parentVertexId != null) {\r\n if (vertexId === undefined || vertexId != this.parentVertexId) {\r\n neighbours.push(this.parentVertexId);\r\n }\r\n }\r\n\r\n return neighbours;\r\n }\r\n\r\n /**\r\n * Gets the next vertex in the ring in opposide direction to the supplied vertex id.\r\n *\r\n * @param {Vertex[]} vertices The array of vertices for the current molecule.\r\n * @param {Number} ringId The id of the ring containing this vertex.\r\n * @param {Number} previousVertexId The id of the previous vertex. The next vertex will be opposite from the vertex with this id as seen from this vertex.\r\n * @returns {Number} The id of the next vertex in the ring.\r\n */\r\n getNextInRing(vertices, ringId, previousVertexId) {\r\n let neighbours = this.getNeighbours();\r\n\r\n for (let i = 0; i < neighbours.length; i++) {\r\n if (ArrayHelper.contains(vertices[neighbours[i]].value.rings, {\r\n value: ringId\r\n }) &&\r\n neighbours[i] != previousVertexId) {\r\n return neighbours[i];\r\n }\r\n }\r\n\r\n return null;\r\n }\r\n}\r\n\r\nmodule.exports = Vertex;"]} \ No newline at end of file diff --git a/doc/ArrayHelper.html b/doc/ArrayHelper.html index 4018247b..2104e375 100644 --- a/doc/ArrayHelper.html +++ b/doc/ArrayHelper.html @@ -3398,7 +3398,7 @@
Returns:

- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/ArrayHelper.js.html b/doc/ArrayHelper.js.html index c628999b..effe1e57 100644 --- a/doc/ArrayHelper.js.html +++ b/doc/ArrayHelper.js.html @@ -438,7 +438,7 @@

ArrayHelper.js


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Atom.html b/doc/Atom.html index e0505858..161b5235 100644 --- a/doc/Atom.html +++ b/doc/Atom.html @@ -2906,7 +2906,7 @@

restoreRi
- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Atom.js.html b/doc/Atom.js.html index 09cd0bc5..67d92872 100644 --- a/doc/Atom.js.html +++ b/doc/Atom.js.html @@ -596,7 +596,7 @@

Atom.js


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/CanvasWrapper.html b/doc/CanvasWrapper.html index 42dee0e6..76294db3 100644 --- a/doc/CanvasWrapper.html +++ b/doc/CanvasWrapper.html @@ -3737,7 +3737,7 @@
Parameters:

- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/CanvasWrapper.js.html b/doc/CanvasWrapper.js.html index 6a332b1c..16e40430 100644 --- a/doc/CanvasWrapper.js.html +++ b/doc/CanvasWrapper.js.html @@ -946,7 +946,7 @@

CanvasWrapper.js


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Drawer.html b/doc/Drawer.html index 6d6e5c98..66cb0129 100644 --- a/doc/Drawer.html +++ b/doc/Drawer.html @@ -445,7 +445,7 @@

addRingSource:
@@ -603,7 +603,7 @@

addR
Source:
@@ -761,7 +761,7 @@

Source:
@@ -848,7 +848,7 @@

Source:
@@ -1029,7 +1029,7 @@

Source:
@@ -1116,7 +1116,7 @@

chooseSide<
Source:
@@ -1327,7 +1327,7 @@

crea
Source:
@@ -1508,7 +1508,7 @@

createN
Source:
@@ -1816,7 +1816,7 @@

createRing<
Source:
@@ -2133,7 +2133,7 @@

drawSource:
@@ -2363,7 +2363,7 @@

drawEdgeSource:
@@ -2522,7 +2522,7 @@

drawEdgesSource:
@@ -2658,7 +2658,7 @@

drawVerti
Source:
@@ -2794,7 +2794,7 @@

edgeRing
Source:
@@ -2952,7 +2952,7 @@

extendSource:
@@ -3039,7 +3039,7 @@

ge
Source:
@@ -3197,7 +3197,7 @@

getBri
Source:
@@ -3306,7 +3306,7 @@

getCl
Source:
@@ -3464,7 +3464,7 @@

Source:
@@ -3625,7 +3625,7 @@

getComm
Source:
@@ -3806,7 +3806,7 @@

Source:
@@ -3915,7 +3915,7 @@

Source:
@@ -4128,7 +4128,7 @@

getEdge
Source:
@@ -4286,7 +4286,7 @@

getFused
Source:
@@ -4395,7 +4395,7 @@

getH
Source:
@@ -4504,7 +4504,7 @@

Source:
@@ -4688,7 +4688,7 @@

Source:
@@ -4846,7 +4846,7 @@

ge
Source:
@@ -4955,7 +4955,7 @@

g
Source:
@@ -5113,7 +5113,7 @@

getOve
Source:
@@ -5222,7 +5222,7 @@

getRingSource:
@@ -5380,7 +5380,7 @@

getRin
Source:
@@ -5564,7 +5564,7 @@

getR
Source:
@@ -5722,7 +5722,7 @@

get
Source:
@@ -5903,7 +5903,7 @@

getRingCo
Source:
@@ -6012,7 +6012,7 @@

getSpirosSource:
@@ -6121,7 +6121,7 @@

getSu
Source:
@@ -6302,7 +6302,7 @@

Source:
@@ -6506,7 +6506,7 @@

g
Source:
@@ -6615,7 +6615,7 @@

getVerti
Source:
@@ -6819,7 +6819,7 @@

hasBrid
Source:
@@ -6928,7 +6928,7 @@

ini
Source:
@@ -7016,7 +7016,7 @@

initRingsSource:
@@ -7103,7 +7103,7 @@

isEdgeInR
Source:
@@ -7261,7 +7261,7 @@

isEdge
Source:
@@ -7419,7 +7419,7 @@

is
Source:
@@ -7577,7 +7577,7 @@

isPointI
Source:
@@ -7735,7 +7735,7 @@

isRingA
Source:
@@ -7893,7 +7893,7 @@

positionSource:
@@ -7980,7 +7980,7 @@

printRin
Source:
@@ -8089,7 +8089,7 @@

removeRing<
Source:
@@ -8225,7 +8225,7 @@

r
Source:
@@ -8361,7 +8361,7 @@

Source:
@@ -8520,7 +8520,7 @@

Source:
@@ -8607,7 +8607,7 @@

Source:
@@ -8815,7 +8815,7 @@

Source:
@@ -8902,7 +8902,7 @@

rotateDr
Source:
@@ -8989,7 +8989,7 @@

rotateSu
Source:
@@ -9194,7 +9194,7 @@

setRingC
Source:
@@ -9330,7 +9330,7 @@

v
Source:
@@ -9583,7 +9583,7 @@

Parameters:

- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Drawer.js.html b/doc/Drawer.js.html index 1c6dd123..1eb5511f 100644 --- a/doc/Drawer.js.html +++ b/doc/Drawer.js.html @@ -101,6 +101,12 @@

Drawer.js

fontSizeLarge: 5, fontSizeSmall: 3, padding: 20.0, + experimentalSSSR: false, + kkThreshold: 0.1, + kkInnerThreshold: 0.1, + kkMaxIteration: 20000, + kkMaxInnerIteration: 50, + kkMaxEnergy: 1e9, themes: { dark: { C: '#fff', @@ -327,6 +333,12 @@

Drawer.js

this.drawEdges(this.opts.debug); this.drawVertices(this.opts.debug); this.canvasWrapper.reset(); + + if (this.opts.debug) { + console.log(this.graph); + console.log(this.rings); + console.log(this.ringConnections); + } } } @@ -672,7 +684,7 @@

Drawer.js

} // Get the rings in the graph (the SSSR) - let rings = SSSR.getRings(this.graph); + let rings = SSSR.getRings(this.graph, this.opts.experimentalSSSR); if (rings === null) { return; @@ -896,6 +908,7 @@

Drawer.js

// Create the ring let ring = new Ring([...ringMembers]); + this.addRing(ring); ring.isBridged = true; ring.neighbours = [...neighbours]; @@ -904,8 +917,6 @@

Drawer.js

ring.rings.push(this.getRing(ringIds[i]).clone()); } - this.addRing(ring); - for (var i = 0; i < ring.members.length; i++) { this.graph.vertices[ring.members[i]].value.bridgedRing = ring.id; } @@ -1749,7 +1760,9 @@

Drawer.js

// If the ring is bridged, then draw the vertices inside the ring // using a force based approach if (ring.isBridged) { - this.graph.kkLayout(ring.members.slice(), center, startVertex.id, ring, this.opts.bondLength); + this.graph.kkLayout(ring.members.slice(), center, startVertex.id, ring, this.opts.bondLength, + this.opts.kkThreshold, this.opts.kkInnerThreshold, this.opts.kkMaxIteration, + this.opts.kkMaxInnerIteration, this.opts.kkMaxEnergy); ring.positioned = true; // Update the center of the bridged ring @@ -3076,7 +3089,7 @@

Drawer.js


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Edge.html b/doc/Edge.html index addfa22c..b01e97d4 100644 --- a/doc/Edge.html +++ b/doc/Edge.html @@ -841,7 +841,7 @@
Parameters:

- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Edge.js.html b/doc/Edge.js.html index d962d13b..12ac0527 100644 --- a/doc/Edge.js.html +++ b/doc/Edge.js.html @@ -115,7 +115,7 @@

Edge.js


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Graph.html b/doc/Graph.html index f81a034c..b17c0cdb 100644 --- a/doc/Graph.html +++ b/doc/Graph.html @@ -408,7 +408,7 @@

(static)
Source:
@@ -495,7 +495,7 @@

(static) _c
Source:
@@ -582,7 +582,7 @@

(
Source:
@@ -740,7 +740,7 @@

(stat
Source:
@@ -898,7 +898,7 @@

_bridgeDfs<
Source:
@@ -3529,7 +3529,7 @@

kkLayoutSource:
@@ -3569,7 +3569,8 @@

kkLayout -

Positiones the (sub)graph using Kamada and Kawais algorithm for drawing general undirected graphs. https://pdfs.semanticscholar.org/b8d3/bca50ccc573c5cb99f7d201e8acce6618f04.pdf

+

Positiones the (sub)graph using Kamada and Kawais algorithm for drawing general undirected graphs. https://pdfs.semanticscholar.org/b8d3/bca50ccc573c5cb99f7d201e8acce6618f04.pdf +There are undocumented layout parameters. They are undocumented for a reason, so be very careful.

@@ -4274,7 +4275,7 @@
Parameters:

- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Graph.js.html b/doc/Graph.js.html index 0f93d433..cce8d3b8 100644 --- a/doc/Graph.js.html +++ b/doc/Graph.js.html @@ -607,13 +607,17 @@

Graph.js

/** * Positiones the (sub)graph using Kamada and Kawais algorithm for drawing general undirected graphs. https://pdfs.semanticscholar.org/b8d3/bca50ccc573c5cb99f7d201e8acce6618f04.pdf + * There are undocumented layout parameters. They are undocumented for a reason, so be very careful. * * @param {Number[]} vertexIds An array containing vertexIds to be placed using the force based layout. * @param {Vector2} center The center of the layout. * @param {Number} startVertexId A vertex id. Should be the starting vertex - e.g. the first to be positioned and connected to a previously place vertex. * @param {Ring} ring The bridged ring associated with this force-based layout. */ - kkLayout(vertexIds, center, startVertexId, ring, bondLength) { + kkLayout(vertexIds, center, startVertexId, ring, bondLength, + threshold = 0.1, innerThreshold = 0.1, maxIteration = 2000, + maxInnerIteration = 50, maxEnergy = 1e9) { + let edgeStrength = bondLength; // Add vertices that are directly connected to the ring @@ -814,13 +818,6 @@

Graph.js

arrEnergySumY[index] = dEY; } - // Setting parameters - let threshold = 0.1; - let innerThreshold = 0.1; - let maxIteration = 2000; - let maxInnerIteration = 50; - let maxEnergy = 1e9; - // Setting up variables for the while loops let maxEnergyId = 0; let dEX = 0.0; @@ -981,7 +978,7 @@

Graph.js


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Line.html b/doc/Line.html index 10d7c5e9..9d247561 100644 --- a/doc/Line.html +++ b/doc/Line.html @@ -3017,7 +3017,7 @@
Returns:

- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Line.js.html b/doc/Line.js.html index c9763a73..f7404852 100644 --- a/doc/Line.js.html +++ b/doc/Line.js.html @@ -356,7 +356,7 @@

Line.js


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/MathHelper.html b/doc/MathHelper.html index 3dcd80fa..d5d542c3 100644 --- a/doc/MathHelper.html +++ b/doc/MathHelper.html @@ -1869,7 +1869,7 @@
Returns:

- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/MathHelper.js.html b/doc/MathHelper.js.html index 32914438..ba09c06f 100644 --- a/doc/MathHelper.js.html +++ b/doc/MathHelper.js.html @@ -214,7 +214,7 @@

MathHelper.js


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Ring.html b/doc/Ring.html index 7d161805..116b1152 100644 --- a/doc/Ring.html +++ b/doc/Ring.html @@ -1908,7 +1908,7 @@
Returns:

- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Ring.js.html b/doc/Ring.js.html index 7a3d75d8..6431390c 100644 --- a/doc/Ring.js.html +++ b/doc/Ring.js.html @@ -267,7 +267,7 @@

Ring.js


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/RingConnection.html b/doc/RingConnection.html index c4f31cae..ddb54f98 100644 --- a/doc/RingConnection.html +++ b/doc/RingConnection.html @@ -1602,7 +1602,7 @@
Parameters:

- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/RingConnection.js.html b/doc/RingConnection.js.html index 91ac344f..fe17340e 100644 --- a/doc/RingConnection.js.html +++ b/doc/RingConnection.js.html @@ -216,7 +216,7 @@

RingConnection.js


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/SSSR.html b/doc/SSSR.html index c2af41e0..8514ea04 100644 --- a/doc/SSSR.html +++ b/doc/SSSR.html @@ -179,7 +179,7 @@

(static) Source:
@@ -360,7 +360,7 @@

(static) Source:
@@ -518,7 +518,7 @@

(static) Source:
@@ -699,7 +699,7 @@

(static) Source:
@@ -857,7 +857,7 @@

(static)
Source:
@@ -1015,7 +1015,7 @@

Source:
@@ -1173,7 +1173,7 @@

(static) <
Source:
@@ -1365,7 +1365,7 @@
Returns:
-

(static) getRings(graph) → {Array.<Array>}

+

(static) getRings(graph, experimentalopt) → {Array.<Array>}

@@ -1377,7 +1377,7 @@

(static) get
Source:
@@ -1443,8 +1443,12 @@

Parameters:
Type + Attributes + + Default + Description @@ -1468,13 +1472,64 @@
Parameters:
+ + + + + + + + + + + +

A Graph object.

+ + + + experimental + + + + + +Boolean + + + + + + + + + <optional>
+ + + + + + + + + + + + false + + + + +

Whether or not to use experimental SSSR.

+ + + @@ -1535,7 +1590,7 @@

(static) getS
Source:
@@ -1854,7 +1909,7 @@

(static) Source:
@@ -2035,7 +2090,7 @@

(static) Source:
@@ -2193,7 +2248,7 @@

(static) Source:
@@ -2469,7 +2524,7 @@
Returns:

- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/SSSR.js.html b/doc/SSSR.js.html index bfa9b301..8ec05261 100644 --- a/doc/SSSR.js.html +++ b/doc/SSSR.js.html @@ -51,9 +51,10 @@

SSSR.js

* Returns an array containing arrays, each representing a ring from the smallest set of smallest rings in the graph. * * @param {Graph} graph A Graph object. + * @param {Boolean} [experimental=false] Whether or not to use experimental SSSR. * @returns {Array[]} An array containing arrays, each representing a ring from the smallest set of smallest rings in the group. */ - static getRings(graph) { + static getRings(graph, experimental=false) { let adjacencyMatrix = graph.getComponentsAdjacencyMatrix(); if (adjacencyMatrix.length === 0) { return null; @@ -110,6 +111,10 @@

SSSR.js

rings.push([...connectedComponent]); continue; } + + if (experimental) { + nSssr = 999; + } let { d, pe, pe_prime } = SSSR.getPathIncludedDistanceMatrices(ccAdjacencyMatrix); let c = SSSR.getRingCandidates(d, pe, pe_prime); @@ -127,7 +132,12 @@

SSSR.js

rings.push(ring); } } + + // So, for some reason, this would return three rings for C1CCCC2CC1CCCC2, which is wrong + // As I don't have time to fix this properly, it will stay in. I'm sorry next person who works + // on it. At that point it might be best to reimplement the whole SSSR thing... + rings.pop(); return rings; } @@ -617,7 +627,7 @@

SSSR.js


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Vector2.html b/doc/Vector2.html index 246ed462..02fa62c2 100644 --- a/doc/Vector2.html +++ b/doc/Vector2.html @@ -7088,7 +7088,7 @@
Returns:

- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Vector2.js.html b/doc/Vector2.js.html index 80c939c3..1e2bbe6e 100644 --- a/doc/Vector2.js.html +++ b/doc/Vector2.js.html @@ -684,7 +684,7 @@

Vector2.js


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Vertex.html b/doc/Vertex.html index a0638dc6..2124c341 100644 --- a/doc/Vertex.html +++ b/doc/Vertex.html @@ -3013,7 +3013,7 @@
Parameters:

- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/Vertex.js.html b/doc/Vertex.js.html index 8d0dce77..2bec1b8d 100644 --- a/doc/Vertex.js.html +++ b/doc/Vertex.js.html @@ -398,7 +398,7 @@

Vertex.js


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/doc/all.md b/doc/all.md index 4cf1a477..bc264ddf 100644 --- a/doc/all.md +++ b/doc/all.md @@ -67,7 +67,7 @@ A static class containing helper functions for array-related tasks. * [.removeAll(arrA, arrB)](#ArrayHelper.removeAll) ⇒ Array * [.merge(arrA, arrB)](#ArrayHelper.merge) ⇒ Array * [.containsAll(arrA, arrB)](#ArrayHelper.containsAll) ⇒ Boolean - * [.sortByAtomicNumberDesc(arr)](#ArrayHelper.sortByAtomicNumberDesc) ⇒ Array.<Object> + * [.sortByAtomicNumberDesc(arr)](#ArrayHelper.sortByAtomicNumberDesc) ⇒ [ 'Array' ].<Object> * [.deepCopy(arr)](#ArrayHelper.deepCopy) ⇒ Array @@ -105,7 +105,7 @@ Returns a string representation of an array. If the array contains objects with | Param | Type | Description | | --- | --- | --- | -| arr | Array.<Object> | An array. | +| arr | [ 'Array' ].<Object> | An array. | | arr[].id | \* | If the array contains an object with the property 'id', the properties value is printed. Else, the array elements value is printend. | @@ -268,15 +268,15 @@ Checks whether or not an array contains all the elements of another array, witho -### ArrayHelper.sortByAtomicNumberDesc(arr) ⇒ Array.<Object> +### ArrayHelper.sortByAtomicNumberDesc(arr) ⇒ [ 'Array' ].<Object> Sort an array of atomic number information. Where the number is indicated as x, x.y, x.y.z, ... **Kind**: static method of [ArrayHelper](#ArrayHelper) -**Returns**: Array.<Object> - The array sorted by atomic number. Example of an array entry: { atomicNumber: 2, vertexId: 5 }. +**Returns**: [ 'Array' ].<Object> - The array sorted by atomic number. Example of an array entry: { atomicNumber: 2, vertexId: 5 }. | Param | Type | Description | | --- | --- | --- | -| arr | Array.<Object> | An array of vertex ids with their associated atomic numbers. | +| arr | [ 'Array' ].<Object> | An array of vertex ids with their associated atomic numbers. | | arr[].vertexId | Number | A vertex id. | | arr[].atomicNumber | String | The atomic number associated with the vertex id. | @@ -304,27 +304,27 @@ A class representing an atom. | --- | --- | --- | | element | String | The element symbol of this atom. Single-letter symbols are always uppercase. Examples: H, C, F, Br, Si, ... | | drawExplicit | Boolean | A boolean indicating whether or not this atom is drawn explicitly (for example, a carbon atom). This overrides the default behaviour. | -| ringbonds | Array.<Object> | An array containing the ringbond ids and bond types as specified in the original SMILE. | +| ringbonds | [ 'Array' ].<Object> | An array containing the ringbond ids and bond types as specified in the original SMILE. | | branchBond | String | The branch bond as defined in the SMILES. | | ringbonds[].id | Number | The ringbond id as defined in the SMILES. | | ringbonds[].bondType | String | The bond type of the ringbond as defined in the SMILES. | -| rings | Array.<Number> | The ids of rings which contain this atom. | +| rings | [ 'Array' ].<Number> | The ids of rings which contain this atom. | | bondType | String | The bond type associated with this array. Examples: -, =, #, ... | | isBridge | Boolean | A boolean indicating whether or not this atom is part of a bridge in a bridged ring (contained by the largest ring). | | isBridgeNode | Boolean | A boolean indicating whether or not this atom is a bridge node (a member of the largest ring in a bridged ring which is connected to a bridge-atom). | -| originalRings | Array.<Number> | Used to back up rings when they are replaced by a bridged ring. | +| originalRings | [ 'Array' ].<Number> | Used to back up rings when they are replaced by a bridged ring. | | bridgedRing | Number | The id of the bridged ring if the atom is part of a bridged ring. | -| anchoredRings | Array.<Number> | The ids of the rings that are anchored to this atom. The centers of anchored rings are translated when this atom is translated. | +| anchoredRings | [ 'Array' ].<Number> | The ids of the rings that are anchored to this atom. The centers of anchored rings are translated when this atom is translated. | | bracket | Object | If this atom is defined as a bracket atom in the original SMILES, this object contains all the bracket information. Example: { hcount: {Number}, charge: ['--', '-', '+', '++'], isotope: {Number} }. | | plane | Number | Specifies on which "plane" the atoms is in stereochemical deptictions (-1 back, 0 middle, 1 front). | -| attachedPseudoElements | Array.<Object> | A map with containing information for pseudo elements or concatinated elements. The key is comprised of the element symbol and the hydrogen count. | +| attachedPseudoElements | [ 'Array' ].<Object> | A map with containing information for pseudo elements or concatinated elements. The key is comprised of the element symbol and the hydrogen count. | | attachedPseudoElement[].element | String | The element symbol. | | attachedPseudoElement[].count | Number | The number of occurences that match the key. | | attachedPseudoElement[].hyrogenCount | Number | The number of hydrogens attached to each atom matching the key. | | hasAttachedPseudoElements | Boolean | A boolean indicating whether or not this attom will be drawn with an attached pseudo element or concatinated elements. | | isDrawn | Boolean | A boolean indicating whether or not this atom is drawn. In contrast to drawExplicit, the bond is drawn neither. | | isConnectedToRing | Boolean | A boolean indicating whether or not this atom is directly connected (but not a member of) a ring. | -| neighbouringElements | Array.<String> | An array containing the element symbols of neighbouring atoms. | +| neighbouringElements | [ 'Array' ].<String> | An array containing the element symbols of neighbouring atoms. | | isPartOfAromaticRing | Boolean | A boolean indicating whether or not this atom is part of an explicitly defined aromatic ring. Example: c1ccccc1. | | bondCount | Number | The number of bonds in which this atom is participating. | | chirality | String | The chirality of this atom if it is a stereocenter (R or S). | @@ -465,7 +465,7 @@ Check whether or not the neighbouring elements of this atom equal the supplied a | Param | Type | Description | | --- | --- | --- | -| arr | Array.<String> | An array containing all the elements that are neighbouring this atom. E.g. ['C', 'O', 'O', 'N'] | +| arr | [ 'Array' ].<String> | An array containing all the elements that are neighbouring this atom. E.g. ['C', 'O', 'O', 'N'] | @@ -585,7 +585,7 @@ Scale the canvas based on vertex positions. | Param | Type | Description | | --- | --- | --- | -| vertices | [Array.<Vertex>](#Vertex) | An array of vertices containing the vertices associated with the current molecule. | +| vertices | [[ 'Array' ].<Vertex>](#Vertex) | An array of vertices containing the vertices associated with the current molecule. | @@ -787,9 +787,9 @@ The main class of the application representing the smiles drawer * [.extend()](#Drawer+extend) * [.draw(data, target, themeName, infoOnly)](#Drawer+draw) * [.edgeRingCount(edgeId)](#Drawer+edgeRingCount) ⇒ Number - * [.getBridgedRings()](#Drawer+getBridgedRings) ⇒ [Array.<Ring>](#Ring) - * [.getFusedRings()](#Drawer+getFusedRings) ⇒ [Array.<Ring>](#Ring) - * [.getSpiros()](#Drawer+getSpiros) ⇒ [Array.<Ring>](#Ring) + * [.getBridgedRings()](#Drawer+getBridgedRings) ⇒ [[ 'Array' ].<Ring>](#Ring) + * [.getFusedRings()](#Drawer+getFusedRings) ⇒ [[ 'Array' ].<Ring>](#Ring) + * [.getSpiros()](#Drawer+getSpiros) ⇒ [[ 'Array' ].<Ring>](#Ring) * [.printRingInfo()](#Drawer+printRingInfo) ⇒ String * [.rotateDrawing()](#Drawer+rotateDrawing) * [.getTotalOverlapScore()](#Drawer+getTotalOverlapScore) ⇒ Number @@ -799,13 +799,13 @@ The main class of the application representing the smiles drawer * [.getMolecularFormula()](#Drawer+getMolecularFormula) ⇒ String * [.getRingbondType(vertexA, vertexB)](#Drawer+getRingbondType) ⇒ String | null * [.initRings()](#Drawer+initRings) - * [.getBridgedRingRings(ringId)](#Drawer+getBridgedRingRings) ⇒ Array.<Number> + * [.getBridgedRingRings(ringId)](#Drawer+getBridgedRingRings) ⇒ [ 'Array' ].<Number> * [.isPartOfBridgedRing(ringId)](#Drawer+isPartOfBridgedRing) ⇒ Boolean * [.createBridgedRing(ringIds, sourceVertexId)](#Drawer+createBridgedRing) ⇒ [Ring](#Ring) * [.areVerticesInSameRing(vertexA, vertexB)](#Drawer+areVerticesInSameRing) ⇒ Boolean - * [.getCommonRings(vertexA, vertexB)](#Drawer+getCommonRings) ⇒ Array.<Number> + * [.getCommonRings(vertexA, vertexB)](#Drawer+getCommonRings) ⇒ [ 'Array' ].<Number> * [.getLargestOrAromaticCommonRing(vertexA, vertexB)](#Drawer+getLargestOrAromaticCommonRing) ⇒ [Ring](#Ring) | null - * [.getVerticesAt(position, radius, excludeVertexId)](#Drawer+getVerticesAt) ⇒ Array.<Number> + * [.getVerticesAt(position, radius, excludeVertexId)](#Drawer+getVerticesAt) ⇒ [ 'Array' ].<Number> * [.getClosestVertex(vertex)](#Drawer+getClosestVertex) ⇒ [Vertex](#Vertex) * [.addRing(ring)](#Drawer+addRing) ⇒ Number * [.removeRing(ringId)](#Drawer+removeRing) @@ -814,7 +814,7 @@ The main class of the application representing the smiles drawer * [.removeRingConnection(ringConnectionId)](#Drawer+removeRingConnection) * [.removeRingConnectionsBetween(vertexIdA, vertexIdB)](#Drawer+removeRingConnectionsBetween) * [.getRingConnection(id)](#Drawer+getRingConnection) ⇒ [RingConnection](#RingConnection) - * [.getRingConnections(ringId, ringIds)](#Drawer+getRingConnections) ⇒ Array.<Number> + * [.getRingConnections(ringId, ringIds)](#Drawer+getRingConnections) ⇒ [ 'Array' ].<Number> * [.getOverlapScore()](#Drawer+getOverlapScore) ⇒ Object * [.chooseSide(vertexA, vertexB, sides)](#Drawer+chooseSide) ⇒ Object * [.setRingCenter(ring)](#Drawer+setRingCenter) @@ -839,8 +839,8 @@ The main class of the application representing the smiles drawer * [.isEdgeInRing(edge)](#Drawer+isEdgeInRing) ⇒ Boolean * [.isEdgeRotatable(edge)](#Drawer+isEdgeRotatable) ⇒ Boolean * [.isRingAromatic(ring)](#Drawer+isRingAromatic) ⇒ Boolean - * [.getEdgeNormals(edge)](#Drawer+getEdgeNormals) ⇒ [Array.<Vector2>](#Vector2) - * [.getNonRingNeighbours(vertexId)](#Drawer+getNonRingNeighbours) ⇒ [Array.<Vertex>](#Vertex) + * [.getEdgeNormals(edge)](#Drawer+getEdgeNormals) ⇒ [[ 'Array' ].<Vector2>](#Vector2) + * [.getNonRingNeighbours(vertexId)](#Drawer+getNonRingNeighbours) ⇒ [[ 'Array' ].<Vertex>](#Vertex) * [.annotateStereochemistry()](#Drawer+annotateStereochemistry) * [.visitStereochemistry(vertexId, previousVertexId, visited, priority, maxDepth, depth)](#Drawer+visitStereochemistry) * [.initPseudoElements()](#Drawer+initPseudoElements) @@ -889,25 +889,25 @@ Returns the number of rings this edge is a part of. -### drawer.getBridgedRings() ⇒ [Array.<Ring>](#Ring) +### drawer.getBridgedRings() ⇒ [[ 'Array' ].<Ring>](#Ring) Returns an array containing the bridged rings associated with this molecule. **Kind**: instance method of [Drawer](#Drawer) -**Returns**: [Array.<Ring>](#Ring) - An array containing all bridged rings associated with this molecule. +**Returns**: [[ 'Array' ].<Ring>](#Ring) - An array containing all bridged rings associated with this molecule. -### drawer.getFusedRings() ⇒ [Array.<Ring>](#Ring) +### drawer.getFusedRings() ⇒ [[ 'Array' ].<Ring>](#Ring) Returns an array containing all fused rings associated with this molecule. **Kind**: instance method of [Drawer](#Drawer) -**Returns**: [Array.<Ring>](#Ring) - An array containing all fused rings associated with this molecule. +**Returns**: [[ 'Array' ].<Ring>](#Ring) - An array containing all fused rings associated with this molecule. -### drawer.getSpiros() ⇒ [Array.<Ring>](#Ring) +### drawer.getSpiros() ⇒ [[ 'Array' ].<Ring>](#Ring) Returns an array containing all spiros associated with this molecule. **Kind**: instance method of [Drawer](#Drawer) -**Returns**: [Array.<Ring>](#Ring) - An array containing all spiros associated with this molecule. +**Returns**: [[ 'Array' ].<Ring>](#Ring) - An array containing all spiros associated with this molecule. ### drawer.printRingInfo() ⇒ String @@ -977,11 +977,11 @@ Initializes rings and ringbonds for the current molecule. **Kind**: instance method of [Drawer](#Drawer) -### drawer.getBridgedRingRings(ringId) ⇒ Array.<Number> +### drawer.getBridgedRingRings(ringId) ⇒ [ 'Array' ].<Number> Returns all rings connected by bridged bonds starting from the ring with the supplied ring id. **Kind**: instance method of [Drawer](#Drawer) -**Returns**: Array.<Number> - An array containing all ring ids of rings part of a bridged ring system. +**Returns**: [ 'Array' ].<Number> - An array containing all ring ids of rings part of a bridged ring system. | Param | Type | Description | | --- | --- | --- | @@ -1009,7 +1009,7 @@ Creates a bridged ring. | Param | Type | Description | | --- | --- | --- | -| ringIds | Array.<Number> | An array of ids of rings involved in the bridged ring. | +| ringIds | [ 'Array' ].<Number> | An array of ids of rings involved in the bridged ring. | | sourceVertexId | Number | The vertex id to start the bridged ring discovery from. | @@ -1027,11 +1027,11 @@ Checks whether or not two vertices are in the same ring. -### drawer.getCommonRings(vertexA, vertexB) ⇒ Array.<Number> +### drawer.getCommonRings(vertexA, vertexB) ⇒ [ 'Array' ].<Number> Returns an array of ring ids shared by both vertices. **Kind**: instance method of [Drawer](#Drawer) -**Returns**: Array.<Number> - An array of ids of rings shared by the two vertices. +**Returns**: [ 'Array' ].<Number> - An array of ids of rings shared by the two vertices. | Param | Type | Description | | --- | --- | --- | @@ -1053,11 +1053,11 @@ Returns the aromatic or largest ring shared by the two vertices. -### drawer.getVerticesAt(position, radius, excludeVertexId) ⇒ Array.<Number> +### drawer.getVerticesAt(position, radius, excludeVertexId) ⇒ [ 'Array' ].<Number> Returns an array of vertices positioned at a specified location. **Kind**: instance method of [Drawer](#Drawer) -**Returns**: Array.<Number> - An array containing vertex ids in a given location. +**Returns**: [ 'Array' ].<Number> - An array containing vertex ids in a given location. | Param | Type | Description | | --- | --- | --- | @@ -1161,16 +1161,16 @@ Get a ring connection with a given id. -### drawer.getRingConnections(ringId, ringIds) ⇒ Array.<Number> +### drawer.getRingConnections(ringId, ringIds) ⇒ [ 'Array' ].<Number> Get the ring connections between a ring and a set of rings. **Kind**: instance method of [Drawer](#Drawer) -**Returns**: Array.<Number> - An array of ring connection ids. +**Returns**: [ 'Array' ].<Number> - An array of ring connection ids. | Param | Type | Description | | --- | --- | --- | | ringId | Number | A ring id. | -| ringIds | Array.<Number> | An array of ring ids. | +| ringIds | [ 'Array' ].<Number> | An array of ring ids. | @@ -1198,7 +1198,7 @@ When drawing a double bond, choose the side to place the double bond. E.g. a dou | --- | --- | --- | | vertexA | [Vertex](#Vertex) | A vertex. | | vertexB | [Vertex](#Vertex) | A vertex. | -| sides | [Array.<Vector2>](#Vector2) | An array containing the two normals of the line spanned by the two provided vertices. | +| sides | [[ 'Array' ].<Vector2>](#Vector2) | An array containing the two normals of the line spanned by the two provided vertices. | @@ -1317,7 +1317,7 @@ Gets the overlap score of a subtree. | --- | --- | --- | | vertexId | Number | A vertex id (the root of the sub-tree). | | parentVertexId | Number | A vertex id in the previous direction of the subtree. | -| vertexOverlapScores | Array.<Number> | An array containing the vertex overlap scores indexed by vertex id. | +| vertexOverlapScores | [ 'Array' ].<Number> | An array containing the vertex overlap scores indexed by vertex id. | @@ -1354,7 +1354,7 @@ Resolve secondary overlaps. Those overlaps are due to the structure turning back | Param | Type | Description | | --- | --- | --- | -| scores | Array.<Object> | An array of objects sorted descending by score. | +| scores | [ 'Array' ].<Object> | An array of objects sorted descending by score. | | scores[].id | Number | A vertex id. | | scores[].score | Number | The overlap score associated with the vertex id. | @@ -1447,11 +1447,11 @@ Check whether or not a ring is an implicitly defined aromatic ring (lower case s -### drawer.getEdgeNormals(edge) ⇒ [Array.<Vector2>](#Vector2) +### drawer.getEdgeNormals(edge) ⇒ [[ 'Array' ].<Vector2>](#Vector2) Get the normals of an edge. **Kind**: instance method of [Drawer](#Drawer) -**Returns**: [Array.<Vector2>](#Vector2) - An array containing two vectors, representing the normals. +**Returns**: [[ 'Array' ].<Vector2>](#Vector2) - An array containing two vectors, representing the normals. | Param | Type | Description | | --- | --- | --- | @@ -1459,11 +1459,11 @@ Get the normals of an edge. -### drawer.getNonRingNeighbours(vertexId) ⇒ [Array.<Vertex>](#Vertex) +### drawer.getNonRingNeighbours(vertexId) ⇒ [[ 'Array' ].<Vertex>](#Vertex) Returns an array of vertices that are neighbouring a vertix but are not members of a ring (including bridges). **Kind**: instance method of [Drawer](#Drawer) -**Returns**: [Array.<Vertex>](#Vertex) - An array of vertices. +**Returns**: [[ 'Array' ].<Vertex>](#Vertex) - An array of vertices. | Param | Type | Description | | --- | --- | --- | @@ -1562,8 +1562,8 @@ A class representing the molecular graph. | Name | Type | Description | | --- | --- | --- | -| vertices | [Array.<Vertex>](#Vertex) | The vertices of the graph. | -| edges | [Array.<Edge>](#Edge) | The edges of this graph. | +| vertices | [[ 'Array' ].<Vertex>](#Vertex) | The vertices of the graph. | +| edges | [[ 'Array' ].<Edge>](#Edge) | The edges of this graph. | | vertexIdsToEdgeId | Object | A map mapping vertex ids to the edge between the two vertices. The key is defined as vertexAId + '_' + vertexBId. | | isometric | Boolean | A boolean indicating whether or not the SMILES associated with this graph is isometric. | @@ -1576,25 +1576,25 @@ A class representing the molecular graph. * [.addVertex(vertex)](#Graph+addVertex) ⇒ Number * [.addEdge(edge)](#Graph+addEdge) ⇒ Number * [.getEdge(vertexIdA, vertexIdB)](#Graph+getEdge) ⇒ [Edge](#Edge) | null - * [.getEdges(vertexId)](#Graph+getEdges) ⇒ Array.<Number> + * [.getEdges(vertexId)](#Graph+getEdges) ⇒ [ 'Array' ].<Number> * [.hasEdge(vertexIdA, vertexIdB)](#Graph+hasEdge) ⇒ Boolean - * [.getVertexList()](#Graph+getVertexList) ⇒ Array.<Number> - * [.getEdgeList()](#Graph+getEdgeList) ⇒ Array.<Array> - * [.getAdjacencyMatrix()](#Graph+getAdjacencyMatrix) ⇒ Array.<Array> - * [.getComponentsAdjacencyMatrix()](#Graph+getComponentsAdjacencyMatrix) ⇒ Array.<Array> - * [.getSubgraphAdjacencyMatrix(vertexIds)](#Graph+getSubgraphAdjacencyMatrix) ⇒ Array.<Array> - * [.getDistanceMatrix()](#Graph+getDistanceMatrix) ⇒ Array.<Array> - * [.getSubgraphDistanceMatrix(vertexIds)](#Graph+getSubgraphDistanceMatrix) ⇒ Array.<Array> - * [.getAdjacencyList()](#Graph+getAdjacencyList) ⇒ Array.<Array> - * [.getSubgraphAdjacencyList(vertexIds)](#Graph+getSubgraphAdjacencyList) ⇒ Array.<Array> - * [.getBridges()](#Graph+getBridges) ⇒ Array.<Number> + * [.getVertexList()](#Graph+getVertexList) ⇒ [ 'Array' ].<Number> + * [.getEdgeList()](#Graph+getEdgeList) ⇒ [ 'Array' ].<Array> + * [.getAdjacencyMatrix()](#Graph+getAdjacencyMatrix) ⇒ [ 'Array' ].<Array> + * [.getComponentsAdjacencyMatrix()](#Graph+getComponentsAdjacencyMatrix) ⇒ [ 'Array' ].<Array> + * [.getSubgraphAdjacencyMatrix(vertexIds)](#Graph+getSubgraphAdjacencyMatrix) ⇒ [ 'Array' ].<Array> + * [.getDistanceMatrix()](#Graph+getDistanceMatrix) ⇒ [ 'Array' ].<Array> + * [.getSubgraphDistanceMatrix(vertexIds)](#Graph+getSubgraphDistanceMatrix) ⇒ [ 'Array' ].<Array> + * [.getAdjacencyList()](#Graph+getAdjacencyList) ⇒ [ 'Array' ].<Array> + * [.getSubgraphAdjacencyList(vertexIds)](#Graph+getSubgraphAdjacencyList) ⇒ [ 'Array' ].<Array> + * [.getBridges()](#Graph+getBridges) ⇒ [ 'Array' ].<Number> * [.traverseBF(startVertexId, callback)](#Graph+traverseBF) * [.getTreeDepth(vertexId, parentVertexId)](#Graph+getTreeDepth) ⇒ Number * [.traverseTree(vertexId, parentVertexId, callback, [maxDepth], [ignoreFirst], [depth], [visited])](#Graph+traverseTree) * [.kkLayout(vertexIds, center, startVertexId, ring)](#Graph+kkLayout) * [._bridgeDfs()](#Graph+_bridgeDfs) * _static_ - * [.getConnectedComponents(adjacencyMatrix)](#Graph.getConnectedComponents) ⇒ Array.<Set> + * [.getConnectedComponents(adjacencyMatrix)](#Graph.getConnectedComponents) ⇒ [ 'Array' ].<Set> * [.getConnectedComponentCount(adjacencyMatrix)](#Graph.getConnectedComponentCount) ⇒ Number * [._ccCountDfs()](#Graph._ccCountDfs) * [._ccGetDfs()](#Graph._ccGetDfs) @@ -1668,11 +1668,11 @@ Returns the edge between two given vertices. -### graph.getEdges(vertexId) ⇒ Array.<Number> +### graph.getEdges(vertexId) ⇒ [ 'Array' ].<Number> Returns the ids of edges connected to a vertex. **Kind**: instance method of [Graph](#Graph) -**Returns**: Array.<Number> - An array containing the ids of edges connected to the vertex. +**Returns**: [ 'Array' ].<Number> - An array containing the ids of edges connected to the vertex. | Param | Type | Description | | --- | --- | --- | @@ -1693,89 +1693,89 @@ Check whether or not two vertices are connected by an edge. -### graph.getVertexList() ⇒ Array.<Number> +### graph.getVertexList() ⇒ [ 'Array' ].<Number> Returns an array containing the vertex ids of this graph. **Kind**: instance method of [Graph](#Graph) -**Returns**: Array.<Number> - An array containing all vertex ids of this graph. +**Returns**: [ 'Array' ].<Number> - An array containing all vertex ids of this graph. -### graph.getEdgeList() ⇒ Array.<Array> +### graph.getEdgeList() ⇒ [ 'Array' ].<Array> Returns an array containing source, target arrays of this graphs edges. **Kind**: instance method of [Graph](#Graph) -**Returns**: Array.<Array> - An array containing source, target arrays of this graphs edges. Example: [ [ 2, 5 ], [ 6, 9 ] ]. +**Returns**: [ 'Array' ].<Array> - An array containing source, target arrays of this graphs edges. Example: [ [ 2, 5 ], [ 6, 9 ] ]. -### graph.getAdjacencyMatrix() ⇒ Array.<Array> +### graph.getAdjacencyMatrix() ⇒ [ 'Array' ].<Array> Get the adjacency matrix of the graph. **Kind**: instance method of [Graph](#Graph) -**Returns**: Array.<Array> - The adjancency matrix of the molecular graph. +**Returns**: [ 'Array' ].<Array> - The adjancency matrix of the molecular graph. -### graph.getComponentsAdjacencyMatrix() ⇒ Array.<Array> +### graph.getComponentsAdjacencyMatrix() ⇒ [ 'Array' ].<Array> Get the adjacency matrix of the graph with all bridges removed (thus the components). Thus the remaining vertices are all part of ring systems. **Kind**: instance method of [Graph](#Graph) -**Returns**: Array.<Array> - The adjancency matrix of the molecular graph with all bridges removed. +**Returns**: [ 'Array' ].<Array> - The adjancency matrix of the molecular graph with all bridges removed. -### graph.getSubgraphAdjacencyMatrix(vertexIds) ⇒ Array.<Array> +### graph.getSubgraphAdjacencyMatrix(vertexIds) ⇒ [ 'Array' ].<Array> Get the adjacency matrix of a subgraph. **Kind**: instance method of [Graph](#Graph) -**Returns**: Array.<Array> - The adjancency matrix of the subgraph. +**Returns**: [ 'Array' ].<Array> - The adjancency matrix of the subgraph. | Param | Type | Description | | --- | --- | --- | -| vertexIds | Array.<Number> | An array containing the vertex ids contained within the subgraph. | +| vertexIds | [ 'Array' ].<Number> | An array containing the vertex ids contained within the subgraph. | -### graph.getDistanceMatrix() ⇒ Array.<Array> +### graph.getDistanceMatrix() ⇒ [ 'Array' ].<Array> Get the distance matrix of the graph. **Kind**: instance method of [Graph](#Graph) -**Returns**: Array.<Array> - The distance matrix of the graph. +**Returns**: [ 'Array' ].<Array> - The distance matrix of the graph. -### graph.getSubgraphDistanceMatrix(vertexIds) ⇒ Array.<Array> +### graph.getSubgraphDistanceMatrix(vertexIds) ⇒ [ 'Array' ].<Array> Get the distance matrix of a subgraph. **Kind**: instance method of [Graph](#Graph) -**Returns**: Array.<Array> - The distance matrix of the subgraph. +**Returns**: [ 'Array' ].<Array> - The distance matrix of the subgraph. | Param | Type | Description | | --- | --- | --- | -| vertexIds | Array.<Number> | An array containing the vertex ids contained within the subgraph. | +| vertexIds | [ 'Array' ].<Number> | An array containing the vertex ids contained within the subgraph. | -### graph.getAdjacencyList() ⇒ Array.<Array> +### graph.getAdjacencyList() ⇒ [ 'Array' ].<Array> Get the adjacency list of the graph. **Kind**: instance method of [Graph](#Graph) -**Returns**: Array.<Array> - The adjancency list of the graph. +**Returns**: [ 'Array' ].<Array> - The adjancency list of the graph. -### graph.getSubgraphAdjacencyList(vertexIds) ⇒ Array.<Array> +### graph.getSubgraphAdjacencyList(vertexIds) ⇒ [ 'Array' ].<Array> Get the adjacency list of a subgraph. **Kind**: instance method of [Graph](#Graph) -**Returns**: Array.<Array> - The adjancency list of the subgraph. +**Returns**: [ 'Array' ].<Array> - The adjancency list of the subgraph. | Param | Type | Description | | --- | --- | --- | -| vertexIds | Array.<Number> | An array containing the vertex ids contained within the subgraph. | +| vertexIds | [ 'Array' ].<Number> | An array containing the vertex ids contained within the subgraph. | -### graph.getBridges() ⇒ Array.<Number> +### graph.getBridges() ⇒ [ 'Array' ].<Number> Returns an array containing the edge ids of bridges. A bridge splits the graph into multiple components when removed. **Kind**: instance method of [Graph](#Graph) -**Returns**: Array.<Number> - An array containing the edge ids of the bridges. +**Returns**: [ 'Array' ].<Number> - An array containing the edge ids of the bridges. ### graph.traverseBF(startVertexId, callback) @@ -1827,7 +1827,7 @@ Positiones the (sub)graph using Kamada and Kawais algorithm for drawing general | Param | Type | Description | | --- | --- | --- | -| vertexIds | Array.<Number> | An array containing vertexIds to be placed using the force based layout. | +| vertexIds | [ 'Array' ].<Number> | An array containing vertexIds to be placed using the force based layout. | | center | [Vector2](#Vector2) | The center of the layout. | | startVertexId | Number | A vertex id. Should be the starting vertex - e.g. the first to be positioned and connected to a previously place vertex. | | ring | [Ring](#Ring) | The bridged ring associated with this force-based layout. | @@ -1840,15 +1840,15 @@ PRIVATE FUNCTION used by getBridges(). **Kind**: instance method of [Graph](#Graph) -### Graph.getConnectedComponents(adjacencyMatrix) ⇒ Array.<Set> +### Graph.getConnectedComponents(adjacencyMatrix) ⇒ [ 'Array' ].<Set> Returns the connected components of the graph. **Kind**: static method of [Graph](#Graph) -**Returns**: Array.<Set> - Connected components as sets. +**Returns**: [ 'Array' ].<Set> - Connected components as sets. | Param | Type | Description | | --- | --- | --- | -| adjacencyMatrix | Array.<Array> | An adjacency matrix. | +| adjacencyMatrix | [ 'Array' ].<Array> | An adjacency matrix. | @@ -1860,7 +1860,7 @@ Returns the number of connected components for the graph. | Param | Type | Description | | --- | --- | --- | -| adjacencyMatrix | Array.<Array> | An adjacency matrix. | +| adjacencyMatrix | [ 'Array' ].<Array> | An adjacency matrix. | @@ -2158,7 +2158,7 @@ Returns the means of the angles contained in an array. In radians. | Param | Type | Description | | --- | --- | --- | -| arr | Array.<Number> | An array containing angles (in radians). | +| arr | [ 'Array' ].<Number> | An array containing angles (in radians). | @@ -2257,13 +2257,13 @@ A class representing a ring. | Name | Type | Description | | --- | --- | --- | | id | Number | The id of this ring. | -| members | Array.<Number> | An array containing the vertex ids of the ring members. | -| edges | Array.<Number> | An array containing the edge ids of the edges between the ring members. | -| insiders | Array.<Number> | An array containing the vertex ids of the vertices contained within the ring if it is a bridged ring. | -| neighbours | Array.<Number> | An array containing the ids of neighbouring rings. | +| members | [ 'Array' ].<Number> | An array containing the vertex ids of the ring members. | +| edges | [ 'Array' ].<Number> | An array containing the edge ids of the edges between the ring members. | +| insiders | [ 'Array' ].<Number> | An array containing the vertex ids of the vertices contained within the ring if it is a bridged ring. | +| neighbours | [ 'Array' ].<Number> | An array containing the ids of neighbouring rings. | | positioned | Boolean | A boolean indicating whether or not this ring has been positioned. | | center | [Vector2](#Vector2) | The center of this ring. | -| rings | [Array.<Ring>](#Ring) | The rings contained within this ring if this ring is bridged. | +| rings | [[ 'Array' ].<Ring>](#Ring) | The rings contained within this ring if this ring is bridged. | | isBridged | Boolean | A boolean whether or not this ring is bridged. | | isPartOfBridged | Boolean | A boolean whether or not this ring is part of a bridge ring. | | isSpiro | Boolean | A boolean whether or not this ring is part of a spiro. | @@ -2276,10 +2276,10 @@ A class representing a ring. * [new Ring(members)](#new_Ring_new) * [.clone()](#Ring+clone) ⇒ [Ring](#Ring) * [.getSize()](#Ring+getSize) ⇒ Number - * [.getPolygon(vertices)](#Ring+getPolygon) ⇒ [Array.<Vector2>](#Vector2) + * [.getPolygon(vertices)](#Ring+getPolygon) ⇒ [[ 'Array' ].<Vector2>](#Vector2) * [.getAngle()](#Ring+getAngle) ⇒ Number * [.eachMember(vertices, callback, startVertexId, previousVertexId)](#Ring+eachMember) - * [.getOrderedNeighbours(ringConnections)](#Ring+getOrderedNeighbours) ⇒ Array.<Object> + * [.getOrderedNeighbours(ringConnections)](#Ring+getOrderedNeighbours) ⇒ [ 'Array' ].<Object> * [.isBenzeneLike(vertices)](#Ring+isBenzeneLike) ⇒ Boolean * [.getDoubleBondCount(vertices)](#Ring+getDoubleBondCount) ⇒ Number * [.contains(vertexId)](#Ring+contains) ⇒ Boolean @@ -2292,7 +2292,7 @@ The constructor for the class Ring. | Param | Type | Description | | --- | --- | --- | -| members | Array.<Number> | An array containing the vertex ids of the members of the ring to be created. | +| members | [ 'Array' ].<Number> | An array containing the vertex ids of the members of the ring to be created. | @@ -2310,15 +2310,15 @@ Returns the size (number of members) of this ring. **Returns**: Number - The size (number of members) of this ring. -### ring.getPolygon(vertices) ⇒ [Array.<Vector2>](#Vector2) +### ring.getPolygon(vertices) ⇒ [[ 'Array' ].<Vector2>](#Vector2) Gets the polygon representation (an array of the ring-members positional vectors) of this ring. **Kind**: instance method of [Ring](#Ring) -**Returns**: [Array.<Vector2>](#Vector2) - An array of the positional vectors of the ring members. +**Returns**: [[ 'Array' ].<Vector2>](#Vector2) - An array of the positional vectors of the ring members. | Param | Type | Description | | --- | --- | --- | -| vertices | [Array.<Vertex>](#Vertex) | An array of vertices representing the current molecule. | +| vertices | [[ 'Array' ].<Vertex>](#Vertex) | An array of vertices representing the current molecule. | @@ -2336,22 +2336,22 @@ Loops over the members of this ring from a given start position in a direction o | Param | Type | Description | | --- | --- | --- | -| vertices | [Array.<Vertex>](#Vertex) | The vertices associated with the current molecule. | +| vertices | [[ 'Array' ].<Vertex>](#Vertex) | The vertices associated with the current molecule. | | callback | function | A callback with the current vertex id as a parameter. | | startVertexId | Number | The vertex id of the start vertex. | | previousVertexId | Number | The vertex id of the previous vertex (the loop calling the callback function will run in the opposite direction of this vertex). | -### ring.getOrderedNeighbours(ringConnections) ⇒ Array.<Object> +### ring.getOrderedNeighbours(ringConnections) ⇒ [ 'Array' ].<Object> Returns an array containing the neighbouring rings of this ring ordered by ring size. **Kind**: instance method of [Ring](#Ring) -**Returns**: Array.<Object> - An array of neighbouring rings sorted by ring size. Example: { n: 5, neighbour: 1 }. +**Returns**: [ 'Array' ].<Object> - An array of neighbouring rings sorted by ring size. Example: { n: 5, neighbour: 1 }. | Param | Type | Description | | --- | --- | --- | -| ringConnections | [Array.<RingConnection>](#RingConnection) | An array of ring connections associated with the current molecule. | +| ringConnections | [[ 'Array' ].<RingConnection>](#RingConnection) | An array of ring connections associated with the current molecule. | @@ -2363,7 +2363,7 @@ Check whether this ring is an implicitly defined benzene-like (e.g. C1=CC=CC=C1) | Param | Type | Description | | --- | --- | --- | -| vertices | [Array.<Vertex>](#Vertex) | An array of vertices associated with the current molecule. | +| vertices | [[ 'Array' ].<Vertex>](#Vertex) | An array of vertices associated with the current molecule. | @@ -2375,7 +2375,7 @@ Get the number of double bonds inside this ring. | Param | Type | Description | | --- | --- | --- | -| vertices | [Array.<Vertex>](#Vertex) | An array of vertices associated with the current molecule. | +| vertices | [[ 'Array' ].<Vertex>](#Vertex) | An array of vertices associated with the current molecule. | @@ -2402,7 +2402,7 @@ A class representing a ring connection. | id | Number | The id of this ring connection. | | firstRingId | Number | A ring id. | | secondRingId | Number | A ring id. | -| vertices | Set.<Number> | A set containing the vertex ids participating in the ring connection. | +| vertices | [ 'Set' ].<Number> | A set containing the vertex ids participating in the ring connection. | * [RingConnection](#RingConnection) @@ -2414,8 +2414,8 @@ A class representing a ring connection. * [.isBridge(vertices)](#RingConnection+isBridge) ⇒ Boolean * _static_ * [.isBridge(ringConnections, vertices, firstRingId, secondRingId)](#RingConnection.isBridge) ⇒ Boolean - * [.getNeighbours(ringConnections, ringId)](#RingConnection.getNeighbours) ⇒ Array.<Number> - * [.getVertices(ringConnections, firstRingId, secondRingId)](#RingConnection.getVertices) ⇒ Array.<Number> + * [.getNeighbours(ringConnections, ringId)](#RingConnection.getNeighbours) ⇒ [ 'Array' ].<Number> + * [.getVertices(ringConnections, firstRingId, secondRingId)](#RingConnection.getVertices) ⇒ [ 'Array' ].<Number> @@ -2473,7 +2473,7 @@ Checks whether or not this ring connection is a bridge in a bridged ring. | Param | Type | Description | | --- | --- | --- | -| vertices | [Array.<Vertex>](#Vertex) | The array of vertices associated with the current molecule. | +| vertices | [[ 'Array' ].<Vertex>](#Vertex) | The array of vertices associated with the current molecule. | @@ -2485,35 +2485,35 @@ Checks whether or not two rings are connected by a bridged bond. | Param | Type | Description | | --- | --- | --- | -| ringConnections | [Array.<RingConnection>](#RingConnection) | An array of ring connections containing the ring connections associated with the current molecule. | -| vertices | [Array.<Vertex>](#Vertex) | An array of vertices containing the vertices associated with the current molecule. | +| ringConnections | [[ 'Array' ].<RingConnection>](#RingConnection) | An array of ring connections containing the ring connections associated with the current molecule. | +| vertices | [[ 'Array' ].<Vertex>](#Vertex) | An array of vertices containing the vertices associated with the current molecule. | | firstRingId | Number | A ring id. | | secondRingId | Number | A ring id. | -### RingConnection.getNeighbours(ringConnections, ringId) ⇒ Array.<Number> +### RingConnection.getNeighbours(ringConnections, ringId) ⇒ [ 'Array' ].<Number> Retruns the neighbouring rings of a given ring. **Kind**: static method of [RingConnection](#RingConnection) -**Returns**: Array.<Number> - An array of ring ids of neighbouring rings. +**Returns**: [ 'Array' ].<Number> - An array of ring ids of neighbouring rings. | Param | Type | Description | | --- | --- | --- | -| ringConnections | [Array.<RingConnection>](#RingConnection) | An array of ring connections containing ring connections associated with the current molecule. | +| ringConnections | [[ 'Array' ].<RingConnection>](#RingConnection) | An array of ring connections containing ring connections associated with the current molecule. | | ringId | Number | A ring id. | -### RingConnection.getVertices(ringConnections, firstRingId, secondRingId) ⇒ Array.<Number> +### RingConnection.getVertices(ringConnections, firstRingId, secondRingId) ⇒ [ 'Array' ].<Number> Returns an array of vertex ids associated with a given ring connection. **Kind**: static method of [RingConnection](#RingConnection) -**Returns**: Array.<Number> - An array of vertex ids associated with the ring connection. +**Returns**: [ 'Array' ].<Number> - An array of vertex ids associated with the ring connection. | Param | Type | Description | | --- | --- | --- | -| ringConnections | [Array.<RingConnection>](#RingConnection) | An array of ring connections containing ring connections associated with the current molecule. | +| ringConnections | [[ 'Array' ].<RingConnection>](#RingConnection) | An array of ring connections containing ring connections associated with the current molecule. | | firstRingId | Number | A ring id. | | secondRingId | Number | A ring id. | @@ -2525,14 +2525,14 @@ A class encapsulating the functionality to find the smallest set of smallest rin **Kind**: global class * [SSSR](#SSSR) - * [.getRings(graph)](#SSSR.getRings) ⇒ Array.<Array> + * [.getRings(graph)](#SSSR.getRings) ⇒ [ 'Array' ].<Array> * [.matrixToString(matrix)](#SSSR.matrixToString) ⇒ String * [.getPathIncludedDistanceMatrices(adjacencyMatrix)](#SSSR.getPathIncludedDistanceMatrices) ⇒ Object - * [.getRingCandidates(d, pe, pe_prime)](#SSSR.getRingCandidates) ⇒ Array.<Array> - * [.getSSSR(c, d, adjacencyMatrix, pe, pe_prime, arrBondCount, arrRingCount, nsssr)](#SSSR.getSSSR) ⇒ Array.<Set> + * [.getRingCandidates(d, pe, pe_prime)](#SSSR.getRingCandidates) ⇒ [ 'Array' ].<Array> + * [.getSSSR(c, d, adjacencyMatrix, pe, pe_prime, arrBondCount, arrRingCount, nsssr)](#SSSR.getSSSR) ⇒ [ 'Array' ].<Set> * [.getEdgeCount(adjacencyMatrix)](#SSSR.getEdgeCount) ⇒ Number - * [.getEdgeList(adjacencyMatrix)](#SSSR.getEdgeList) ⇒ Array.<Array> - * [.bondsToAtoms(bonds)](#SSSR.bondsToAtoms) ⇒ Set.<Number> + * [.getEdgeList(adjacencyMatrix)](#SSSR.getEdgeList) ⇒ [ 'Array' ].<Array> + * [.bondsToAtoms(bonds)](#SSSR.bondsToAtoms) ⇒ [ 'Set' ].<Number> * [.getBondCount(atoms, adjacencyMatrix)](#SSSR.getBondCount) ⇒ Number * [.pathSetsContain(pathSets, pathSet, bonds, allBonds, arrBondCount, arrRingCount)](#SSSR.pathSetsContain) ⇒ Boolean * [.areSetsEqual(setA, setB)](#SSSR.areSetsEqual) ⇒ Boolean @@ -2540,11 +2540,11 @@ A class encapsulating the functionality to find the smallest set of smallest rin -### SSSR.getRings(graph) ⇒ Array.<Array> +### SSSR.getRings(graph) ⇒ [ 'Array' ].<Array> Returns an array containing arrays, each representing a ring from the smallest set of smallest rings in the graph. **Kind**: static method of [SSSR](#SSSR) -**Returns**: Array.<Array> - An array containing arrays, each representing a ring from the smallest set of smallest rings in the group. +**Returns**: [ 'Array' ].<Array> - An array containing arrays, each representing a ring from the smallest set of smallest rings in the group. | Param | Type | Description | | --- | --- | --- | @@ -2560,7 +2560,7 @@ Creates a printable string from a matrix (2D array). | Param | Type | Description | | --- | --- | --- | -| matrix | Array.<Array> | A 2D array. | +| matrix | [ 'Array' ].<Array> | A 2D array. | @@ -2572,37 +2572,37 @@ Returnes the two path-included distance matrices used to find the sssr. | Param | Type | Description | | --- | --- | --- | -| adjacencyMatrix | Array.<Array> | An adjacency matrix. | +| adjacencyMatrix | [ 'Array' ].<Array> | An adjacency matrix. | -### SSSR.getRingCandidates(d, pe, pe_prime) ⇒ Array.<Array> +### SSSR.getRingCandidates(d, pe, pe_prime) ⇒ [ 'Array' ].<Array> Get the ring candidates from the path-included distance matrices. **Kind**: static method of [SSSR](#SSSR) -**Returns**: Array.<Array> - The ring candidates. +**Returns**: [ 'Array' ].<Array> - The ring candidates. | Param | Type | Description | | --- | --- | --- | -| d | Array.<Array> | The distance matrix. | -| pe | Array.<Array> | A matrix containing the shortest paths. | -| pe_prime | Array.<Array> | A matrix containing the shortest paths + one vertex. | +| d | [ 'Array' ].<Array> | The distance matrix. | +| pe | [ 'Array' ].<Array> | A matrix containing the shortest paths. | +| pe_prime | [ 'Array' ].<Array> | A matrix containing the shortest paths + one vertex. | -### SSSR.getSSSR(c, d, adjacencyMatrix, pe, pe_prime, arrBondCount, arrRingCount, nsssr) ⇒ Array.<Set> +### SSSR.getSSSR(c, d, adjacencyMatrix, pe, pe_prime, arrBondCount, arrRingCount, nsssr) ⇒ [ 'Array' ].<Set> Searches the candidates for the smallest set of smallest rings. **Kind**: static method of [SSSR](#SSSR) -**Returns**: Array.<Set> - The smallest set of smallest rings. +**Returns**: [ 'Array' ].<Set> - The smallest set of smallest rings. | Param | Type | Description | | --- | --- | --- | -| c | Array.<Array> | The candidates. | -| d | Array.<Array> | The distance matrix. | -| adjacencyMatrix | Array.<Array> | An adjacency matrix. | -| pe | Array.<Array> | A matrix containing the shortest paths. | -| pe_prime | Array.<Array> | A matrix containing the shortest paths + one vertex. | +| c | [ 'Array' ].<Array> | The candidates. | +| d | [ 'Array' ].<Array> | The distance matrix. | +| adjacencyMatrix | [ 'Array' ].<Array> | An adjacency matrix. | +| pe | [ 'Array' ].<Array> | A matrix containing the shortest paths. | +| pe_prime | [ 'Array' ].<Array> | A matrix containing the shortest paths + one vertex. | | arrBondCount | Uint16Array | A matrix containing the bond count of each vertex. | | arrRingCount | Uint16Array | A matrix containing the number of rings associated with each vertex. | | nsssr | Number | The theoretical number of rings in the graph. | @@ -2617,27 +2617,27 @@ Returns the number of edges in a graph defined by an adjacency matrix. | Param | Type | Description | | --- | --- | --- | -| adjacencyMatrix | Array.<Array> | An adjacency matrix. | +| adjacencyMatrix | [ 'Array' ].<Array> | An adjacency matrix. | -### SSSR.getEdgeList(adjacencyMatrix) ⇒ Array.<Array> +### SSSR.getEdgeList(adjacencyMatrix) ⇒ [ 'Array' ].<Array> Returns an edge list constructed form an adjacency matrix. **Kind**: static method of [SSSR](#SSSR) -**Returns**: Array.<Array> - An edge list. E.g. [ [ 0, 1 ], ..., [ 16, 2 ] ] +**Returns**: [ 'Array' ].<Array> - An edge list. E.g. [ [ 0, 1 ], ..., [ 16, 2 ] ] | Param | Type | Description | | --- | --- | --- | -| adjacencyMatrix | Array.<Array> | An adjacency matrix. | +| adjacencyMatrix | [ 'Array' ].<Array> | An adjacency matrix. | -### SSSR.bondsToAtoms(bonds) ⇒ Set.<Number> +### SSSR.bondsToAtoms(bonds) ⇒ [ 'Set' ].<Number> Return a set of vertex indices contained in an array of bonds. **Kind**: static method of [SSSR](#SSSR) -**Returns**: Set.<Number> - An array of vertices. +**Returns**: [ 'Set' ].<Number> - An array of vertices. | Param | Type | Description | | --- | --- | --- | @@ -2653,8 +2653,8 @@ Returns the number of bonds within a set of atoms. | Param | Type | Description | | --- | --- | --- | -| atoms | Set.<Number> | An array of atom ids. | -| adjacencyMatrix | Array.<Array> | An adjacency matrix. | +| atoms | [ 'Set' ].<Number> | An array of atom ids. | +| adjacencyMatrix | [ 'Array' ].<Array> | An adjacency matrix. | @@ -2666,10 +2666,10 @@ Checks whether or not a given path already exists in an array of paths. | Param | Type | Description | | --- | --- | --- | -| pathSets | Array.<Set> | An array of sets each representing a path. | -| pathSet | Set.<Number> | A set representing a path. | -| bonds | Array.<Array> | The bonds associated with the current path. | -| allBonds | Array.<Array> | All bonds currently associated with rings in the SSSR set. | +| pathSets | [ 'Array' ].<Set> | An array of sets each representing a path. | +| pathSet | [ 'Set' ].<Number> | A set representing a path. | +| bonds | [ 'Array' ].<Array> | The bonds associated with the current path. | +| allBonds | [ 'Array' ].<Array> | All bonds currently associated with rings in the SSSR set. | | arrBondCount | Uint16Array | A matrix containing the bond count of each vertex. | | arrRingCount | Uint16Array | A matrix containing the number of rings associated with each vertex. | @@ -2683,8 +2683,8 @@ Checks whether or not two sets are equal (contain the same elements). | Param | Type | Description | | --- | --- | --- | -| setA | Set.<Number> | A set. | -| setB | Set.<Number> | A set. | +| setA | [ 'Set' ].<Number> | A set. | +| setB | [ 'Set' ].<Number> | A set. | @@ -2696,8 +2696,8 @@ Checks whether or not a set (setA) is a superset of another set (setB). | Param | Type | Description | | --- | --- | --- | -| setA | Set.<Number> | A set. | -| setB | Set.<Number> | A set. | +| setA | [ 'Set' ].<Number> | A set. | +| setB | [ 'Set' ].<Number> | A set. | @@ -2749,8 +2749,8 @@ A class representing a 2D vector. * [.multiply(vecA, vecB)](#Vector2.multiply) ⇒ [Vector2](#Vector2) * [.multiplyScalar(vec, scalar)](#Vector2.multiplyScalar) ⇒ [Vector2](#Vector2) * [.midpoint(vecA, vecB)](#Vector2.midpoint) ⇒ [Vector2](#Vector2) - * [.normals(vecA, vecB)](#Vector2.normals) ⇒ [Array.<Vector2>](#Vector2) - * [.units(vecA, vecB)](#Vector2.units) ⇒ [Array.<Vector2>](#Vector2) + * [.normals(vecA, vecB)](#Vector2.normals) ⇒ [[ 'Array' ].<Vector2>](#Vector2) + * [.units(vecA, vecB)](#Vector2.units) ⇒ [[ 'Array' ].<Vector2>](#Vector2) * [.divide(vecA, vecB)](#Vector2.divide) ⇒ [Vector2](#Vector2) * [.divideScalar(vecA, s)](#Vector2.divideScalar) ⇒ [Vector2](#Vector2) * [.dot(vecA, vecB)](#Vector2.dot) ⇒ Number @@ -3010,7 +3010,7 @@ Checks whether a vector lies within a polygon spanned by a set of vectors. | Param | Type | Description | | --- | --- | --- | -| polygon | [Array.<Vector2>](#Vector2) | An array of vectors spanning the polygon. | +| polygon | [[ 'Array' ].<Vector2>](#Vector2) | An array of vectors spanning the polygon. | @@ -3134,11 +3134,11 @@ Returns the midpoint of a line spanned by two vectors. -### Vector2.normals(vecA, vecB) ⇒ [Array.<Vector2>](#Vector2) +### Vector2.normals(vecA, vecB) ⇒ [[ 'Array' ].<Vector2>](#Vector2) Returns the normals of a line spanned by two vectors. **Kind**: static method of [Vector2](#Vector2) -**Returns**: [Array.<Vector2>](#Vector2) - An array containing the two normals, each represented by a vector. +**Returns**: [[ 'Array' ].<Vector2>](#Vector2) - An array containing the two normals, each represented by a vector. | Param | Type | Description | | --- | --- | --- | @@ -3147,11 +3147,11 @@ Returns the normals of a line spanned by two vectors. -### Vector2.units(vecA, vecB) ⇒ [Array.<Vector2>](#Vector2) +### Vector2.units(vecA, vecB) ⇒ [[ 'Array' ].<Vector2>](#Vector2) Returns the unit (normalized normal) vectors of a line spanned by two vectors. **Kind**: static method of [Vector2](#Vector2) -**Returns**: [Array.<Vector2>](#Vector2) - An array containing the two unit vectors. +**Returns**: [[ 'Array' ].<Vector2>](#Vector2) - An array containing the two unit vectors. | Param | Type | Description | | --- | --- | --- | @@ -3264,15 +3264,15 @@ A class representing a vertex. | position | [Vector2](#Vector2) | The position of this vertex. | | previousPosition | [Vector2](#Vector2) | The position of the previous vertex. | | parentVertexId | Number | null | The id of the previous vertex. | -| children | Array.<Number> | The ids of the children of this vertex. | -| spanningTreeChildren | Array.<Number> | The ids of the children of this vertex as defined in the spanning tree defined by the SMILES. | -| edges | Array.<Number> | The ids of edges associated with this vertex. | +| children | [ 'Array' ].<Number> | The ids of the children of this vertex. | +| spanningTreeChildren | [ 'Array' ].<Number> | The ids of the children of this vertex as defined in the spanning tree defined by the SMILES. | +| edges | [ 'Array' ].<Number> | The ids of edges associated with this vertex. | | positioned | Boolean | A boolean indicating whether or not this vertex has been positioned. | | angle | Number | The angle of this vertex. | | dir | Number | The direction of this vertex. | | neighbourCount | Number | The number of neighbouring vertices. | -| neighbours | Array.<Number> | The vertex ids of neighbouring vertices. | -| neighbouringElements | Array.<String> | The element symbols associated with neighbouring vertices. | +| neighbours | [ 'Array' ].<Number> | The vertex ids of neighbouring vertices. | +| neighbouringElements | [ 'Array' ].<String> | The element symbols associated with neighbouring vertices. | | forcePositioned | Boolean | A boolean indicating whether or not this vertex was positioned using a force-based approach. | @@ -3288,10 +3288,10 @@ A class representing a vertex. * [.equals(vertex)](#Vertex+equals) ⇒ Boolean * [.getAngle([referenceVector], [returnAsDegrees])](#Vertex+getAngle) ⇒ Number * [.getTextDirection(vertices)](#Vertex+getTextDirection) ⇒ String - * [.getNeighbours([vertexId])](#Vertex+getNeighbours) ⇒ Array.<Number> - * [.getDrawnNeighbours(vertices)](#Vertex+getDrawnNeighbours) ⇒ Array.<Number> + * [.getNeighbours([vertexId])](#Vertex+getNeighbours) ⇒ [ 'Array' ].<Number> + * [.getDrawnNeighbours(vertices)](#Vertex+getDrawnNeighbours) ⇒ [ 'Array' ].<Number> * [.getNeighbourCount()](#Vertex+getNeighbourCount) ⇒ Number - * [.getSpanningTreeNeighbours([vertexId])](#Vertex+getSpanningTreeNeighbours) ⇒ Array.<Number> + * [.getSpanningTreeNeighbours([vertexId])](#Vertex+getSpanningTreeNeighbours) ⇒ [ 'Array' ].<Number> * [.getNextInRing(vertices, ringId, previousVertexId)](#Vertex+getNextInRing) ⇒ Number @@ -3412,15 +3412,15 @@ Returns the suggested text direction when text is added at the position of this | Param | Type | Description | | --- | --- | --- | -| vertices | [Array.<Vertex>](#Vertex) | The array of vertices for the current molecule. | +| vertices | [[ 'Array' ].<Vertex>](#Vertex) | The array of vertices for the current molecule. | -### vertex.getNeighbours([vertexId]) ⇒ Array.<Number> +### vertex.getNeighbours([vertexId]) ⇒ [ 'Array' ].<Number> Returns an array of ids of neighbouring vertices. **Kind**: instance method of [Vertex](#Vertex) -**Returns**: Array.<Number> - An array containing the ids of neighbouring vertices. +**Returns**: [ 'Array' ].<Number> - An array containing the ids of neighbouring vertices. | Param | Type | Default | Description | | --- | --- | --- | --- | @@ -3428,15 +3428,15 @@ Returns an array of ids of neighbouring vertices. -### vertex.getDrawnNeighbours(vertices) ⇒ Array.<Number> +### vertex.getDrawnNeighbours(vertices) ⇒ [ 'Array' ].<Number> Returns an array of ids of neighbouring vertices that will be drawn (vertex.value.isDrawn === true). **Kind**: instance method of [Vertex](#Vertex) -**Returns**: Array.<Number> - An array containing the ids of neighbouring vertices that will be drawn. +**Returns**: [ 'Array' ].<Number> - An array containing the ids of neighbouring vertices that will be drawn. | Param | Type | Description | | --- | --- | --- | -| vertices | [Array.<Vertex>](#Vertex) | An array containing the vertices associated with the current molecule. | +| vertices | [[ 'Array' ].<Vertex>](#Vertex) | An array containing the vertices associated with the current molecule. | @@ -3447,11 +3447,11 @@ Returns the number of neighbours of this vertex. **Returns**: Number - The number of neighbours. -### vertex.getSpanningTreeNeighbours([vertexId]) ⇒ Array.<Number> +### vertex.getSpanningTreeNeighbours([vertexId]) ⇒ [ 'Array' ].<Number> Returns a list of ids of vertices neighbouring this one in the original spanning tree, excluding the ringbond connections. **Kind**: instance method of [Vertex](#Vertex) -**Returns**: Array.<Number> - An array containing the ids of the neighbouring vertices. +**Returns**: [ 'Array' ].<Number> - An array containing the ids of the neighbouring vertices. | Param | Type | Default | Description | | --- | --- | --- | --- | @@ -3467,7 +3467,7 @@ Gets the next vertex in the ring in opposide direction to the supplied vertex id | Param | Type | Description | | --- | --- | --- | -| vertices | [Array.<Vertex>](#Vertex) | The array of vertices for the current molecule. | +| vertices | [[ 'Array' ].<Vertex>](#Vertex) | The array of vertices for the current molecule. | | ringId | Number | The id of the ring containing this vertex. | | previousVertexId | Number | The id of the previous vertex. The next vertex will be opposite from the vertex with this id as seen from this vertex. | diff --git a/doc/index.html b/doc/index.html index 09e0625f..69b7e140 100644 --- a/doc/index.html +++ b/doc/index.html @@ -365,7 +365,7 @@

Contributors


- Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 10:25:20 GMT+0200 (Central European Summer Time) using the docdash theme. + Documentation generated by JSDoc 3.6.2 on Thu Jun 06 2019 11:56:56 GMT+0200 (Central European Summer Time) using the docdash theme.
diff --git a/example/index_experimental.html b/example/index_experimental.html new file mode 100644 index 00000000..41203a41 --- /dev/null +++ b/example/index_experimental.html @@ -0,0 +1,301 @@ + + + + + + SMILES Parser Demo + + + + + + + + + + +
+ +

SmilesDrawer

+ +
 
+
+ +
+ +
+ +
+ + + + + + + + + + + + +
+ + + + + + + + + + + + + + + + + + + + +
+ +
+ +
+
+ + + + + \ No newline at end of file diff --git a/src/Drawer.js b/src/Drawer.js index 410b54e5..2b59f207 100644 --- a/src/Drawer.js +++ b/src/Drawer.js @@ -57,6 +57,12 @@ class Drawer { fontSizeLarge: 5, fontSizeSmall: 3, padding: 20.0, + experimentalSSSR: false, + kkThreshold: 0.1, + kkInnerThreshold: 0.1, + kkMaxIteration: 20000, + kkMaxInnerIteration: 50, + kkMaxEnergy: 1e9, themes: { dark: { C: '#fff', @@ -283,6 +289,12 @@ class Drawer { this.drawEdges(this.opts.debug); this.drawVertices(this.opts.debug); this.canvasWrapper.reset(); + + if (this.opts.debug) { + console.log(this.graph); + console.log(this.rings); + console.log(this.ringConnections); + } } } @@ -628,7 +640,7 @@ class Drawer { } // Get the rings in the graph (the SSSR) - let rings = SSSR.getRings(this.graph); + let rings = SSSR.getRings(this.graph, this.opts.experimentalSSSR); if (rings === null) { return; @@ -852,6 +864,7 @@ class Drawer { // Create the ring let ring = new Ring([...ringMembers]); + this.addRing(ring); ring.isBridged = true; ring.neighbours = [...neighbours]; @@ -860,8 +873,6 @@ class Drawer { ring.rings.push(this.getRing(ringIds[i]).clone()); } - this.addRing(ring); - for (var i = 0; i < ring.members.length; i++) { this.graph.vertices[ring.members[i]].value.bridgedRing = ring.id; } @@ -1705,7 +1716,9 @@ class Drawer { // If the ring is bridged, then draw the vertices inside the ring // using a force based approach if (ring.isBridged) { - this.graph.kkLayout(ring.members.slice(), center, startVertex.id, ring, this.opts.bondLength); + this.graph.kkLayout(ring.members.slice(), center, startVertex.id, ring, this.opts.bondLength, + this.opts.kkThreshold, this.opts.kkInnerThreshold, this.opts.kkMaxIteration, + this.opts.kkMaxInnerIteration, this.opts.kkMaxEnergy); ring.positioned = true; // Update the center of the bridged ring diff --git a/src/Graph.js b/src/Graph.js index b57934cd..891b4c0c 100644 --- a/src/Graph.js +++ b/src/Graph.js @@ -563,13 +563,17 @@ class Graph { /** * Positiones the (sub)graph using Kamada and Kawais algorithm for drawing general undirected graphs. https://pdfs.semanticscholar.org/b8d3/bca50ccc573c5cb99f7d201e8acce6618f04.pdf + * There are undocumented layout parameters. They are undocumented for a reason, so be very careful. * * @param {Number[]} vertexIds An array containing vertexIds to be placed using the force based layout. * @param {Vector2} center The center of the layout. * @param {Number} startVertexId A vertex id. Should be the starting vertex - e.g. the first to be positioned and connected to a previously place vertex. * @param {Ring} ring The bridged ring associated with this force-based layout. */ - kkLayout(vertexIds, center, startVertexId, ring, bondLength) { + kkLayout(vertexIds, center, startVertexId, ring, bondLength, + threshold = 0.1, innerThreshold = 0.1, maxIteration = 2000, + maxInnerIteration = 50, maxEnergy = 1e9) { + let edgeStrength = bondLength; // Add vertices that are directly connected to the ring @@ -770,13 +774,6 @@ class Graph { arrEnergySumY[index] = dEY; } - // Setting parameters - let threshold = 0.1; - let innerThreshold = 0.1; - let maxIteration = 2000; - let maxInnerIteration = 50; - let maxEnergy = 1e9; - // Setting up variables for the while loops let maxEnergyId = 0; let dEX = 0.0; diff --git a/src/SSSR.js b/src/SSSR.js index 6501a5ba..5e1b4eb6 100644 --- a/src/SSSR.js +++ b/src/SSSR.js @@ -7,9 +7,10 @@ class SSSR { * Returns an array containing arrays, each representing a ring from the smallest set of smallest rings in the graph. * * @param {Graph} graph A Graph object. + * @param {Boolean} [experimental=false] Whether or not to use experimental SSSR. * @returns {Array[]} An array containing arrays, each representing a ring from the smallest set of smallest rings in the group. */ - static getRings(graph) { + static getRings(graph, experimental=false) { let adjacencyMatrix = graph.getComponentsAdjacencyMatrix(); if (adjacencyMatrix.length === 0) { return null; @@ -66,6 +67,10 @@ class SSSR { rings.push([...connectedComponent]); continue; } + + if (experimental) { + nSssr = 999; + } let { d, pe, pe_prime } = SSSR.getPathIncludedDistanceMatrices(ccAdjacencyMatrix); let c = SSSR.getRingCandidates(d, pe, pe_prime); @@ -83,7 +88,12 @@ class SSSR { rings.push(ring); } } + + // So, for some reason, this would return three rings for C1CCCC2CC1CCCC2, which is wrong + // As I don't have time to fix this properly, it will stay in. I'm sorry next person who works + // on it. At that point it might be best to reimplement the whole SSSR thing... + rings.pop(); return rings; }