Mercurial > repos > crs4 > taxonomy_krona_chart
comparison test-data/krona_test1.html @ 10:e9005d1f3cfd draft default tip
"planemo upload for repository https://github.com/galaxyproject/tools-iuc/tree/master/tools/taxonomy_krona_chart commit 46ae76d42d29ce02cf05b6ab735e0c305a86f9cd"
author | iuc |
---|---|
date | Fri, 18 Dec 2020 16:16:12 +0000 |
parents | 1334cb4c6b68 |
children |
comparison
equal
deleted
inserted
replaced
9:1334cb4c6b68 | 10:e9005d1f3cfd |
---|---|
1 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"> | |
2 <html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en"> | |
3 <head> | |
4 <meta charset="utf-8"/> | |
5 <link rel="shortcut icon" href=""/> | |
6 <script id="notfound" type="text/javascript">window.onload=function(){document.body.innerHTML=""}</script> | |
7 <script language="javascript" type="text/javascript"> | |
8 {//----------------------------------------------------------------------------- | |
9 // | |
10 // PURPOSE | |
11 // | |
12 // Krona is a flexible tool for exploring the relative proportions of | |
13 // hierarchical data, such as metagenomic classifications, using a | |
14 // radial, space-filling display. It is implemented using HTML5 and | |
15 // JavaScript, allowing charts to be explored locally or served over the | |
16 // Internet, requiring only a current version of any major web | |
17 // browser. Krona charts can be created using an Excel template or from | |
18 // common bioinformatic formats using the provided conversion scripts. | |
19 // | |
20 // | |
21 // COPYRIGHT LICENSE | |
22 // | |
23 // Copyright (c) 2011, Battelle National Biodefense Institute (BNBI); | |
24 // all rights reserved. Authored by: Brian Ondov, Nicholas Bergman, and | |
25 // Adam Phillippy | |
26 // | |
27 // This Software was prepared for the Department of Homeland Security | |
28 // (DHS) by the Battelle National Biodefense Institute, LLC (BNBI) as | |
29 // part of contract HSHQDC-07-C-00020 to manage and operate the National | |
30 // Biodefense Analysis and Countermeasures Center (NBACC), a Federally | |
31 // Funded Research and Development Center. | |
32 // | |
33 // Redistribution and use in source and binary forms, with or without | |
34 // modification, are permitted provided that the following conditions are | |
35 // met: | |
36 // | |
37 // * Redistributions of source code must retain the above copyright | |
38 // notice, this list of conditions and the following disclaimer. | |
39 // | |
40 // * Redistributions in binary form must reproduce the above copyright | |
41 // notice, this list of conditions and the following disclaimer in the | |
42 // documentation and/or other materials provided with the distribution. | |
43 // | |
44 // * Neither the name of the Battelle National Biodefense Institute nor | |
45 // the names of its contributors may be used to endorse or promote | |
46 // products derived from this software without specific prior written | |
47 // permission. | |
48 // | |
49 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | |
50 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | |
51 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | |
52 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | |
53 // HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |
54 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |
55 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | |
56 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | |
57 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |
58 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |
59 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
60 // | |
61 // | |
62 // TRADEMARK LICENSE | |
63 // | |
64 // KRONA(TM) is a trademark of the Department of Homeland Security, and use | |
65 // of the trademark is subject to the following conditions: | |
66 // | |
67 // * Distribution of the unchanged, official code/software using the | |
68 // KRONA(TM) mark is hereby permitted by the Department of Homeland | |
69 // Security, provided that the software is distributed without charge | |
70 // and modification. | |
71 // | |
72 // * Distribution of altered source code/software using the KRONA(TM) mark | |
73 // is not permitted unless written permission has been granted by the | |
74 // Department of Homeland Security. | |
75 // | |
76 // | |
77 // FOR MORE INFORMATION VISIT | |
78 // | |
79 // https://github.com/marbl/Krona/wiki/ | |
80 // | |
81 //----------------------------------------------------------------------------- | |
82 } | |
83 | |
84 | |
85 var canvas; | |
86 var context; | |
87 var svg; // for snapshot mode | |
88 var collapse = true; | |
89 var collapseCheckBox; | |
90 var collapseLast; | |
91 var compress; | |
92 var compressCheckBox; | |
93 var maxAbsoluteDepthText; | |
94 var maxAbsoluteDepthButtonDecrease; | |
95 var maxAbsoluteDepthButtonIncrease; | |
96 var fontSize = 11; | |
97 var fontSizeText; | |
98 var fontSizeButtonDecrease; | |
99 var fontSizeButtonIncrease; | |
100 var fontSizeLast; | |
101 var radiusButtonDecrease; | |
102 var radiusButtonIncrease; | |
103 var shorten; | |
104 var shortenCheckBox; | |
105 var maxAbsoluteDepth; | |
106 var backButton; | |
107 var upButton; | |
108 var forwardButton; | |
109 var snapshotButton; | |
110 var snapshotMode = false; | |
111 var details; | |
112 var detailsName; | |
113 var search; | |
114 var searchResults; | |
115 var nSearchResults; | |
116 var useHueCheckBox; | |
117 var useHueDiv; | |
118 var datasetDropDown; | |
119 var datasetButtonLast; | |
120 var datasetButtonPrev; | |
121 var datasetButtonNext; | |
122 var keyControl; | |
123 var showKeys = true; | |
124 var linkButton; | |
125 var linkText; | |
126 var frame; | |
127 | |
128 // Node references. Note that the meanings of 'selected' and 'focused' are | |
129 // swapped in the docs. | |
130 // | |
131 var head; // the root of the entire tree | |
132 var selectedNode = 0; // the root of the current view | |
133 var focusNode = 0; // a node chosen for more info (single-click) | |
134 var highlightedNode = 0; // mouse hover node | |
135 var highlightingHidden = false; | |
136 var nodes = new Array(); | |
137 var currentNodeID = 0; // to iterate while loading | |
138 | |
139 var nodeHistory = new Array(); | |
140 var nodeHistoryPosition = 0; | |
141 | |
142 var dataEnabled = false; // true when supplemental files are present | |
143 | |
144 // store non-Krona GET variables so they can be passed on to links | |
145 // | |
146 var getVariables = new Array(); | |
147 | |
148 // selectedNodeLast is separate from the history, since we need to check | |
149 // properties of the last node viewed when browsing through the history | |
150 // | |
151 var selectedNodeLast = 0; | |
152 var zoomOut = false; | |
153 | |
154 // temporary zoom-in while holding the mouse button on a wedge | |
155 // | |
156 var quickLook = false; // true when in quick look state | |
157 var mouseDown = false; | |
158 var mouseDownTime; // to detect mouse button hold | |
159 var quickLookHoldLength = 200; | |
160 | |
161 var imageWidth; | |
162 var imageHeight; | |
163 var centerX; | |
164 var centerY; | |
165 var gRadius; | |
166 var updateViewNeeded = false; | |
167 | |
168 // Determines the angle that the pie chart starts at. 90 degrees makes the | |
169 // center label consistent with the children. | |
170 // | |
171 var rotationOffset = Math.PI / 2; | |
172 | |
173 var buffer; | |
174 var bufferFactor = .1; | |
175 | |
176 // The maps are the small pie charts showing the current slice being viewed. | |
177 // | |
178 var mapBuffer = 10; | |
179 var mapRadius = 0; | |
180 var maxMapRadius = 25; | |
181 var mapWidth = 150; | |
182 var maxLabelOverhang = Math.PI * 4.18; | |
183 | |
184 // Keys are the labeled boxes for slices in the highest level that are too thin | |
185 // to label. | |
186 // | |
187 var maxKeySizeFactor = 2; // will be multiplied by font size | |
188 var keySize; | |
189 var keys; | |
190 var keyBuffer = 10; | |
191 var currentKey; | |
192 var keyMinTextLeft; | |
193 var keyMinAngle; | |
194 | |
195 var minRingWidthFactor = 5; // will be multiplied by font size | |
196 var maxPossibleDepth; // the theoretical max that can be displayed | |
197 var maxDisplayDepth; // the actual depth that will be displayed | |
198 var headerHeight = 0;//document.getElementById('options').clientHeight; | |
199 var historySpacingFactor = 1.6; // will be multiplied by font size | |
200 var historyAlphaDelta = .25; | |
201 | |
202 // appearance | |
203 // | |
204 var lineOpacity = 0.3; | |
205 var saturation = 0.5; | |
206 var lightnessBase = 0.6; | |
207 var lightnessMax = .8; | |
208 var thinLineWidth = .3; | |
209 var highlightLineWidth = 1.5; | |
210 var labelBoxBuffer = 6; | |
211 var labelBoxRounding = 15; | |
212 var labelWidthFudge = 1.05; // The width of unshortened labels are set slightly | |
213 // longer than the name width so the animation | |
214 // finishes faster. | |
215 var fontNormal; | |
216 var fontBold; | |
217 var fontFamily = 'sans-serif'; | |
218 //var fontFaceBold = 'bold Arial'; | |
219 var nodeRadius; | |
220 var angleFactor; | |
221 var tickLength; | |
222 var compressedRadii; | |
223 | |
224 // colors | |
225 // | |
226 var highlightFill = 'rgba(255, 255, 255, .3)'; | |
227 var colorUnclassified = 'rgb(220,220,220)'; | |
228 | |
229 // label staggering | |
230 // | |
231 var labelOffsets; // will store the current offset at each depth | |
232 // | |
233 // This will store pointers to the last node that had a label in each offset (or "track") of a | |
234 // each depth. These will be used to shorten neighboring labels that would overlap. | |
235 // The [nLabelNodes] index will store the last node with a radial label. | |
236 // labelFirstNodes is the same, but to check for going all the way around and | |
237 // overlapping the first labels. | |
238 // | |
239 var labelLastNodes; | |
240 var labelFirstNodes; | |
241 // | |
242 var nLabelOffsets = 3; // the number of offsets to use | |
243 | |
244 var mouseX = -1; | |
245 var mouseY = -1; | |
246 var mouseXRel = -1; | |
247 var mouseYRel = -1; | |
248 | |
249 // tweening | |
250 // | |
251 var progress = 0; // for tweening; goes from 0 to 1. | |
252 var progressLast = 0; | |
253 var tweenFactor = 0; // progress converted by a curve for a smoother effect. | |
254 var tweenLength = 850; // in ms | |
255 var tweenCurvature = 13; | |
256 // | |
257 // tweenMax is used to scale the sigmoid function so its range is [0,1] for the | |
258 // domain [0,1] | |
259 // | |
260 var tweenMax = 1 / (1 + Math.exp(-tweenCurvature / 2)); | |
261 // | |
262 var tweenStartTime; | |
263 | |
264 // for framerate debug | |
265 // | |
266 var tweenFrames = 0; | |
267 var fpsDisplay = document.getElementById('frameRate'); | |
268 | |
269 // Arrays to translate xml attribute names into displayable attribute names | |
270 // | |
271 var attributes = new Array(); | |
272 // | |
273 var magnitudeIndex; // the index of attribute arrays used for magnitude | |
274 var membersAssignedIndex; | |
275 var membersSummaryIndex; | |
276 | |
277 // For defining gradients | |
278 // | |
279 var hueDisplayName; | |
280 var hueStopPositions; | |
281 var hueStopHues; | |
282 var hueStopText; | |
283 | |
284 // multiple datasets | |
285 // | |
286 var currentDataset = 0; | |
287 var lastDataset = 0; | |
288 var datasets = 1; | |
289 var datasetNames; | |
290 var datasetSelectSize = 30; | |
291 var datasetAlpha = new Tween(0, 0); | |
292 var datasetWidths = new Array(); | |
293 var datasetChanged; | |
294 var datasetSelectWidth = 50; | |
295 | |
296 window.onload = load; | |
297 | |
298 var image; | |
299 var hiddenPattern; | |
300 var loadingImage; | |
301 var logoImage; | |
302 | |
303 function backingScale() | |
304 { | |
305 if ('devicePixelRatio' in window) | |
306 { | |
307 if (window.devicePixelRatio > 1) | |
308 { | |
309 return window.devicePixelRatio; | |
310 } | |
311 } | |
312 | |
313 return 1; | |
314 } | |
315 | |
316 function resize() | |
317 { | |
318 imageWidth = window.innerWidth; | |
319 imageHeight = window.innerHeight; | |
320 | |
321 if ( ! snapshotMode ) | |
322 { | |
323 context.canvas.width = imageWidth * backingScale(); | |
324 context.canvas.height = imageHeight * backingScale(); | |
325 context.canvas.style.width = imageWidth + "px" | |
326 context.canvas.style.height = imageHeight + "px" | |
327 context.scale(backingScale(), backingScale()); | |
328 } | |
329 | |
330 if ( datasetDropDown ) | |
331 { | |
332 var ratio = | |
333 (datasetDropDown.offsetTop + datasetDropDown.clientHeight) * 2 / | |
334 imageHeight; | |
335 | |
336 if ( ratio > 1 ) | |
337 { | |
338 ratio = 1; | |
339 } | |
340 | |
341 ratio = Math.sqrt(ratio); | |
342 | |
343 datasetSelectWidth = | |
344 (datasetDropDown.offsetLeft + datasetDropDown.clientWidth) * ratio; | |
345 } | |
346 var leftMargin = datasets > 1 ? datasetSelectWidth + 30 : 0; | |
347 var minDimension = imageWidth - mapWidth - leftMargin > imageHeight ? | |
348 imageHeight : | |
349 imageWidth - mapWidth - leftMargin; | |
350 | |
351 maxMapRadius = minDimension * .03; | |
352 buffer = minDimension * bufferFactor; | |
353 margin = minDimension * .015; | |
354 centerX = (imageWidth - mapWidth - leftMargin) / 2 + leftMargin; | |
355 centerY = imageHeight / 2; | |
356 gRadius = minDimension / 2 - buffer; | |
357 //context.font = '11px sans-serif'; | |
358 } | |
359 | |
360 function handleResize() | |
361 { | |
362 updateViewNeeded = true; | |
363 } | |
364 | |
365 function Attribute() | |
366 { | |
367 } | |
368 | |
369 function Tween(start, end) | |
370 { | |
371 this.start = start; | |
372 this.end = end; | |
373 this.current = this.start; | |
374 | |
375 this.current = function() | |
376 { | |
377 if ( progress == 1 || this.start == this.end ) | |
378 { | |
379 return this.end; | |
380 } | |
381 else | |
382 { | |
383 return this.start + tweenFactor * (this.end - this.start); | |
384 } | |
385 }; | |
386 | |
387 this.setTarget = function(target) | |
388 { | |
389 this.start = this.current(); | |
390 this.end = target; | |
391 } | |
392 } | |
393 | |
394 function Node() | |
395 { | |
396 this.id = currentNodeID; | |
397 currentNodeID++; | |
398 nodes[this.id] = this; | |
399 | |
400 this.angleStart = new Tween(Math.PI, 0); | |
401 this.angleEnd = new Tween(Math.PI, 0); | |
402 this.radiusInner = new Tween(1, 1); | |
403 this.labelRadius = new Tween(1, 1); | |
404 this.labelWidth = new Tween(0, 0); | |
405 this.scale = new Tween(1, 1); // TEMP | |
406 this.radiusOuter = new Tween(1, 1); | |
407 | |
408 this.r = new Tween(255, 255); | |
409 this.g = new Tween(255, 255); | |
410 this.b = new Tween(255, 255); | |
411 | |
412 this.alphaLabel = new Tween(0, 1); | |
413 this.alphaLine = new Tween(0, 1); | |
414 this.alphaArc = new Tween(0, 0); | |
415 this.alphaWedge = new Tween(0, 1); | |
416 this.alphaOther = new Tween(0, 1); | |
417 this.alphaPattern = new Tween(0, 0); | |
418 this.children = Array(); | |
419 this.parent = 0; | |
420 | |
421 this.attributes = new Array(attributes.length); | |
422 | |
423 this.addChild = function(child) | |
424 { | |
425 this.children.push(child); | |
426 }; | |
427 | |
428 this.addLabelNode = function(depth, labelOffset) | |
429 { | |
430 if ( labelHeadNodes[depth][labelOffset] == 0 ) | |
431 { | |
432 // this will become the head node for this list | |
433 | |
434 labelHeadNodes[depth][labelOffset] = this; | |
435 this.labelPrev = this; | |
436 } | |
437 | |
438 var head = labelHeadNodes[depth][labelOffset]; | |
439 | |
440 this.labelNext = head; | |
441 this.labelPrev = head.labelPrev; | |
442 head.labelPrev.labelNext = this; | |
443 head.labelPrev = this; | |
444 } | |
445 | |
446 this.canDisplayDepth = function() | |
447 { | |
448 // whether this node is at a depth that can be displayed, according | |
449 // to the max absolute depth | |
450 | |
451 return this.depth <= maxAbsoluteDepth; | |
452 } | |
453 | |
454 this.canDisplayHistory = function() | |
455 { | |
456 var radiusInner; | |
457 | |
458 if ( compress ) | |
459 { | |
460 radiusInner = compressedRadii[0]; | |
461 } | |
462 else | |
463 { | |
464 radiusInner = nodeRadius; | |
465 } | |
466 | |
467 return ( | |
468 -this.labelRadius.end * gRadius + | |
469 historySpacingFactor * fontSize / 2 < | |
470 radiusInner * gRadius | |
471 ); | |
472 } | |
473 | |
474 this.canDisplayLabelCurrent = function() | |
475 { | |
476 return ( | |
477 (this.angleEnd.current() - this.angleStart.current()) * | |
478 (this.radiusInner.current() * gRadius + gRadius) >= | |
479 minWidth()); | |
480 } | |
481 | |
482 this.checkHighlight = function() | |
483 { | |
484 if ( this.children.length == 0 && this == focusNode ) | |
485 { | |
486 //return false; | |
487 } | |
488 | |
489 if ( this.hide ) | |
490 { | |
491 return false; | |
492 } | |
493 | |
494 if ( this.radiusInner.end == 1 ) | |
495 { | |
496 // compressed to the outside; don't check | |
497 | |
498 return false; | |
499 } | |
500 | |
501 var highlighted = false; | |
502 | |
503 var angleStartCurrent = this.angleStart.current() + rotationOffset; | |
504 var angleEndCurrent = this.angleEnd.current() + rotationOffset; | |
505 var radiusInner = this.radiusInner.current() * gRadius; | |
506 | |
507 for ( var i = 0; i < this.children.length; i++ ) | |
508 { | |
509 highlighted = this.children[i].checkHighlight(); | |
510 | |
511 if ( highlighted ) | |
512 { | |
513 return true; | |
514 } | |
515 } | |
516 | |
517 if ( this.radial ) | |
518 { | |
519 var angleText = (angleStartCurrent + angleEndCurrent) / 2; | |
520 var radiusText = (gRadius + radiusInner) / 2; | |
521 | |
522 context.rotate(angleText); | |
523 context.beginPath(); | |
524 context.moveTo(radiusText, -fontSize); | |
525 context.lineTo(radiusText, fontSize); | |
526 context.lineTo(radiusText + centerX, fontSize); | |
527 context.lineTo(radiusText + centerX, -fontSize); | |
528 context.closePath(); | |
529 context.rotate(-angleText); | |
530 | |
531 if ( context.isPointInPath(mouseXRel, mouseYRel) ) | |
532 { | |
533 var label = String(this.getPercentage()) + '%' + ' ' + this.name; | |
534 | |
535 if ( this.searchResultChildren() ) | |
536 { | |
537 label += searchResultString(this.searchResultChildren()); | |
538 } | |
539 | |
540 if | |
541 ( | |
542 Math.sqrt((mouseXRel) * (mouseXRel) + (mouseYRel) * (mouseYRel)) / backingScale() < | |
543 radiusText + measureText(label) | |
544 ) | |
545 { | |
546 highlighted = true; | |
547 } | |
548 } | |
549 } | |
550 else | |
551 { | |
552 for ( var i = 0; i < this.hiddenLabels.length; i++ ) | |
553 { | |
554 var hiddenLabel = this.hiddenLabels[i]; | |
555 | |
556 context.rotate(hiddenLabel.angle); | |
557 context.beginPath(); | |
558 context.moveTo(gRadius, -fontSize); | |
559 context.lineTo(gRadius, fontSize); | |
560 context.lineTo(gRadius + centerX, fontSize); | |
561 context.lineTo(gRadius + centerX, -fontSize); | |
562 context.closePath(); | |
563 context.rotate(-hiddenLabel.angle); | |
564 | |
565 if ( context.isPointInPath(mouseXRel, mouseYRel) ) | |
566 { | |
567 var label = String(hiddenLabel.value) + ' more'; | |
568 | |
569 if ( hiddenLabel.search ) | |
570 { | |
571 label += searchResultString(hiddenLabel.search); | |
572 } | |
573 | |
574 if | |
575 ( | |
576 Math.sqrt((mouseXRel) * (mouseXRel) + (mouseYRel) * (mouseYRel)) / backingScale() < | |
577 gRadius + fontSize + measureText(label) | |
578 ) | |
579 { | |
580 highlighted = true; | |
581 break; | |
582 } | |
583 } | |
584 } | |
585 } | |
586 | |
587 if ( ! highlighted && this != selectedNode && ! this.getCollapse() ) | |
588 { | |
589 context.beginPath(); | |
590 context.arc(0, 0, radiusInner, angleStartCurrent, angleEndCurrent, false); | |
591 context.arc(0, 0, gRadius, angleEndCurrent, angleStartCurrent, true); | |
592 context.closePath(); | |
593 | |
594 if ( context.isPointInPath(mouseXRel, mouseYRel) ) | |
595 { | |
596 highlighted = true; | |
597 } | |
598 | |
599 if | |
600 ( | |
601 ! highlighted && | |
602 (angleEndCurrent - angleStartCurrent) * | |
603 (radiusInner + gRadius) < | |
604 minWidth() && | |
605 this.getDepth() == selectedNode.getDepth() + 1 | |
606 ) | |
607 { | |
608 if ( showKeys && this.checkHighlightKey() ) | |
609 { | |
610 highlighted = true; | |
611 } | |
612 } | |
613 } | |
614 | |
615 if ( highlighted ) | |
616 { | |
617 if ( this != highlightedNode ) | |
618 { | |
619 // document.body.style.cursor='pointer'; | |
620 } | |
621 | |
622 highlightedNode = this; | |
623 } | |
624 | |
625 return highlighted; | |
626 } | |
627 | |
628 this.checkHighlightCenter = function() | |
629 { | |
630 if ( ! this.canDisplayHistory() ) | |
631 { | |
632 return; | |
633 } | |
634 | |
635 var cx = centerX; | |
636 var cy = centerY - this.labelRadius.end * gRadius; | |
637 //var dim = context.measureText(this.name); | |
638 | |
639 var width = this.nameWidth; | |
640 | |
641 if ( this.searchResultChildren() ) | |
642 { | |
643 var results = searchResultString(this.searchResultChildren()); | |
644 var dim = context.measureText(results); | |
645 width += dim.width; | |
646 } | |
647 | |
648 if | |
649 ( | |
650 mouseX > cx - width / 2 && | |
651 mouseX < cx + width / 2 && | |
652 mouseY > cy - historySpacingFactor * fontSize / 2 && | |
653 mouseY < cy + historySpacingFactor * fontSize / 2 | |
654 ) | |
655 { | |
656 highlightedNode = this; | |
657 return; | |
658 } | |
659 | |
660 if ( this.getParent() ) | |
661 { | |
662 this.getParent().checkHighlightCenter(); | |
663 } | |
664 } | |
665 | |
666 this.checkHighlightKey = function() | |
667 { | |
668 var offset = keyOffset(); | |
669 | |
670 var xMin = imageWidth - keySize - margin - this.keyNameWidth - keyBuffer; | |
671 var xMax = imageWidth - margin; | |
672 var yMin = offset; | |
673 var yMax = offset + keySize; | |
674 | |
675 currentKey++; | |
676 | |
677 return ( | |
678 mouseX > xMin && | |
679 mouseX < xMax && | |
680 mouseY > yMin && | |
681 mouseY < yMax); | |
682 } | |
683 | |
684 this.checkHighlightMap = function() | |
685 { | |
686 if ( this.parent ) | |
687 { | |
688 this.parent.checkHighlightMap(); | |
689 } | |
690 | |
691 if ( this.getCollapse() || this == focusNode ) | |
692 { | |
693 return; | |
694 } | |
695 | |
696 var box = this.getMapPosition(); | |
697 | |
698 if | |
699 ( | |
700 mouseX > box.x - mapRadius && | |
701 mouseX < box.x + mapRadius && | |
702 mouseY > box.y - mapRadius && | |
703 mouseY < box.y + mapRadius | |
704 ) | |
705 { | |
706 highlightedNode = this; | |
707 } | |
708 } | |
709 | |
710 /* this.collapse = function() | |
711 { | |
712 for (var i = 0; i < this.children.length; i++ ) | |
713 { | |
714 this.children[i] = this.children[i].collapse(); | |
715 } | |
716 | |
717 if | |
718 ( | |
719 this.children.length == 1 && | |
720 this.children[0].magnitude == this.magnitude | |
721 ) | |
722 { | |
723 this.children[0].parent = this.parent; | |
724 this.children[0].getDepth() = this.parent.getDepth() + 1; | |
725 return this.children[0]; | |
726 } | |
727 else | |
728 { | |
729 return this; | |
730 } | |
731 } | |
732 */ | |
733 this.draw = function(labelMode, selected, searchHighlighted) | |
734 { | |
735 var depth = this.getDepth() - selectedNode.getDepth() + 1; | |
736 // var hidden = false; | |
737 | |
738 if ( selectedNode == this ) | |
739 { | |
740 selected = true; | |
741 } | |
742 | |
743 var angleStartCurrent = this.angleStart.current() + rotationOffset; | |
744 var angleEndCurrent = this.angleEnd.current() + rotationOffset; | |
745 var radiusInner = this.radiusInner.current() * gRadius; | |
746 var canDisplayLabelCurrent = this.canDisplayLabelCurrent(); | |
747 var hiddenSearchResults = false; | |
748 | |
749 /* if ( ! this.hide ) | |
750 { | |
751 for ( var i = 0; i < this.children.length; i++ ) | |
752 { | |
753 if ( this.children[i].hide && this.children[i].searchResults ) | |
754 { | |
755 hiddenSearchResults = true; | |
756 } | |
757 } | |
758 } | |
759 */ | |
760 var drawChildren = | |
761 ( ! this.hide || ! this.hidePrev && progress < 1 ) && | |
762 ( ! this.hideAlone || ! this.hideAlonePrev && progress < 1 ); | |
763 | |
764 // if ( this.alphaWedge.current() > 0 || this.alphaLabel.current() > 0 ) | |
765 { | |
766 var lastChildAngleEnd = angleStartCurrent; | |
767 | |
768 if ( this.hasChildren() )//canDisplayChildren ) | |
769 { | |
770 lastChildAngleEnd = | |
771 this.children[this.children.length - 1].angleEnd.current() | |
772 + rotationOffset; | |
773 } | |
774 | |
775 if ( labelMode ) | |
776 { | |
777 var drawRadial = | |
778 !( | |
779 this.parent && | |
780 this.parent != selectedNode && | |
781 angleEndCurrent == this.parent.angleEnd.current() + rotationOffset | |
782 ); | |
783 | |
784 //if ( angleStartCurrent != angleEndCurrent ) | |
785 { | |
786 this.drawLines(angleStartCurrent, angleEndCurrent, radiusInner, drawRadial, selected); | |
787 } | |
788 | |
789 var alphaOtherCurrent = this.alphaOther.current(); | |
790 var childRadiusInner; | |
791 | |
792 if ( this == selectedNode || alphaOtherCurrent ) | |
793 { | |
794 childRadiusInner = | |
795 this.children.length ? | |
796 this.children[this.children.length - 1].radiusInner.current() * gRadius | |
797 : radiusInner | |
798 } | |
799 | |
800 if ( this == selectedNode ) | |
801 { | |
802 this.drawReferenceRings(childRadiusInner); | |
803 } | |
804 | |
805 if | |
806 ( | |
807 selected && | |
808 ! searchHighlighted && | |
809 this != selectedNode && | |
810 ( | |
811 this.isSearchResult || | |
812 this.hideAlone && this.searchResultChildren() || | |
813 false | |
814 // this.hide && | |
815 // this.containsSearchResult | |
816 ) | |
817 ) | |
818 { | |
819 context.globalAlpha = this.alphaWedge.current(); | |
820 | |
821 drawWedge | |
822 ( | |
823 angleStartCurrent, | |
824 angleEndCurrent, | |
825 radiusInner, | |
826 gRadius, | |
827 highlightFill, | |
828 0, | |
829 true | |
830 ); | |
831 | |
832 if | |
833 ( | |
834 this.keyed && | |
835 ! showKeys && | |
836 this.searchResults && | |
837 ! searchHighlighted && | |
838 this != highlightedNode && | |
839 this != focusNode | |
840 ) | |
841 { | |
842 var angle = (angleEndCurrent + angleStartCurrent) / 2; | |
843 this.drawLabel(angle, true, false, true, true); | |
844 } | |
845 | |
846 //this.drawHighlight(false); | |
847 searchHighlighted = true; | |
848 } | |
849 | |
850 if | |
851 ( | |
852 this == selectedNode || | |
853 // true | |
854 //(canDisplayLabelCurrent) && | |
855 this != highlightedNode && | |
856 this != focusNode | |
857 ) | |
858 { | |
859 if ( this.radial != this.radialPrev && this.alphaLabel.end == 1 ) | |
860 { | |
861 context.globalAlpha = tweenFactor; | |
862 } | |
863 else | |
864 { | |
865 context.globalAlpha = this.alphaLabel.current(); | |
866 } | |
867 | |
868 this.drawLabel | |
869 ( | |
870 (angleStartCurrent + angleEndCurrent) / 2, | |
871 this.hideAlone && this.searchResultChildren() || | |
872 (this.isSearchResult || hiddenSearchResults) && selected, | |
873 this == selectedNode && ! this.radial, | |
874 selected, | |
875 this.radial | |
876 ); | |
877 | |
878 if ( this.radial != this.radialPrev && this.alphaLabel.start == 1 && progress < 1 ) | |
879 { | |
880 context.globalAlpha = 1 - tweenFactor; | |
881 | |
882 this.drawLabel | |
883 ( | |
884 (angleStartCurrent + angleEndCurrent) / 2, | |
885 (this.isSearchResult || hiddenSearchResults) && selected, | |
886 this == selectedNodeLast && ! this.radialPrev, | |
887 selected, | |
888 this.radialPrev | |
889 ); | |
890 } | |
891 } | |
892 | |
893 if | |
894 ( | |
895 alphaOtherCurrent && | |
896 lastChildAngleEnd != null | |
897 ) | |
898 { | |
899 if | |
900 ( | |
901 (angleEndCurrent - lastChildAngleEnd) * | |
902 (childRadiusInner + gRadius) >= | |
903 minWidth() | |
904 ) | |
905 { | |
906 //context.font = fontNormal; | |
907 context.globalAlpha = this.alphaOther.current(); | |
908 | |
909 drawTextPolar | |
910 ( | |
911 this.getUnclassifiedText(), | |
912 this.getUnclassifiedPercentage(), | |
913 (lastChildAngleEnd + angleEndCurrent) / 2, | |
914 (childRadiusInner + gRadius) / 2, | |
915 true, | |
916 false, | |
917 false, | |
918 0, | |
919 0 | |
920 ); | |
921 } | |
922 } | |
923 | |
924 if ( this == selectedNode && this.keyUnclassified && showKeys ) | |
925 { | |
926 this.drawKey | |
927 ( | |
928 (lastChildAngleEnd + angleEndCurrent) / 2, | |
929 false, | |
930 false | |
931 ); | |
932 } | |
933 } | |
934 else | |
935 { | |
936 var alphaWedgeCurrent = this.alphaWedge.current(); | |
937 | |
938 if ( alphaWedgeCurrent || this.alphaOther.current() ) | |
939 { | |
940 var currentR = this.r.current(); | |
941 var currentG = this.g.current(); | |
942 var currentB = this.b.current(); | |
943 | |
944 var fill = rgbText(currentR, currentG, currentB); | |
945 | |
946 var radiusOuter; | |
947 var lastChildAngle; | |
948 var truncateWedge = | |
949 ( | |
950 (this.hasChildren() || this == selectedNode ) && | |
951 ! this.keyed && | |
952 (compress || depth < maxDisplayDepth) && | |
953 drawChildren | |
954 ); | |
955 | |
956 if ( truncateWedge ) | |
957 { | |
958 radiusOuter = this.children.length ? this.children[0].radiusInner.current() * gRadius : radiusInner; | |
959 } | |
960 else | |
961 { | |
962 radiusOuter = gRadius; | |
963 } | |
964 /* | |
965 if ( this.hasChildren() ) | |
966 { | |
967 radiusOuter = this.children[0].getUncollapsed().radiusInner.current() * gRadius + 1; | |
968 } | |
969 else | |
970 { // TEMP | |
971 radiusOuter = radiusInner + nodeRadius * gRadius; | |
972 | |
973 if ( radiusOuter > gRadius ) | |
974 { | |
975 radiusOuter = gRadius; | |
976 } | |
977 } | |
978 */ | |
979 context.globalAlpha = alphaWedgeCurrent; | |
980 | |
981 if ( radiusInner != radiusOuter || truncateWedge ) | |
982 { | |
983 drawWedge | |
984 ( | |
985 angleStartCurrent, | |
986 angleEndCurrent, | |
987 radiusInner, | |
988 radiusOuter,//this.radiusOuter.current() * gRadius, | |
989 //'rgba(0, 200, 0, .1)', | |
990 fill, | |
991 this.alphaPattern.current() | |
992 ); | |
993 | |
994 if ( truncateWedge ) | |
995 { | |
996 // fill in the extra space if the sum of our childrens' | |
997 // magnitudes is less than ours | |
998 | |
999 if ( lastChildAngleEnd < angleEndCurrent )//&& false) // TEMP | |
1000 { | |
1001 if ( radiusOuter > 1 ) | |
1002 { | |
1003 // overlap slightly to hide the seam | |
1004 | |
1005 // radiusOuter -= 1; | |
1006 } | |
1007 | |
1008 if ( alphaWedgeCurrent < 1 ) | |
1009 { | |
1010 context.globalAlpha = this.alphaOther.current(); | |
1011 drawWedge | |
1012 ( | |
1013 lastChildAngleEnd, | |
1014 angleEndCurrent, | |
1015 radiusOuter, | |
1016 gRadius, | |
1017 colorUnclassified, | |
1018 0 | |
1019 ); | |
1020 context.globalAlpha = alphaWedgeCurrent; | |
1021 } | |
1022 | |
1023 drawWedge | |
1024 ( | |
1025 lastChildAngleEnd, | |
1026 angleEndCurrent, | |
1027 radiusOuter, | |
1028 gRadius,//this.radiusOuter.current() * gRadius, | |
1029 //'rgba(200, 0, 0, .1)', | |
1030 fill, | |
1031 this.alphaPattern.current() | |
1032 ); | |
1033 } | |
1034 } | |
1035 | |
1036 if ( radiusOuter < gRadius ) | |
1037 { | |
1038 // patch up the seam | |
1039 // | |
1040 context.beginPath(); | |
1041 context.arc(0, 0, radiusOuter, angleStartCurrent/*lastChildAngleEnd*/, angleEndCurrent, false); | |
1042 context.strokeStyle = fill; | |
1043 context.lineWidth = 1; | |
1044 context.stroke(); | |
1045 } | |
1046 } | |
1047 | |
1048 if ( this.keyed && selected && showKeys )//&& progress == 1 ) | |
1049 { | |
1050 this.drawKey | |
1051 ( | |
1052 (angleStartCurrent + angleEndCurrent) / 2, | |
1053 ( | |
1054 this == highlightedNode || | |
1055 this == focusNode || | |
1056 this.searchResults | |
1057 ), | |
1058 this == highlightedNode || this == focusNode | |
1059 ); | |
1060 } | |
1061 } | |
1062 } | |
1063 } | |
1064 | |
1065 this.hiddenLabels = Array(); | |
1066 | |
1067 if ( drawChildren ) | |
1068 { | |
1069 // draw children | |
1070 // | |
1071 for ( var i = 0; i < this.children.length; i++ ) | |
1072 { | |
1073 if ( this.drawHiddenChildren(i, selected, labelMode, searchHighlighted) ) | |
1074 { | |
1075 i = this.children[i].hiddenEnd; | |
1076 } | |
1077 else | |
1078 { | |
1079 this.children[i].draw(labelMode, selected, searchHighlighted); | |
1080 } | |
1081 } | |
1082 } | |
1083 }; | |
1084 | |
1085 this.drawHiddenChildren = function | |
1086 ( | |
1087 firstHiddenChild, | |
1088 selected, | |
1089 labelMode, | |
1090 searchHighlighted | |
1091 ) | |
1092 { | |
1093 var firstChild = this.children[firstHiddenChild]; | |
1094 | |
1095 if ( firstChild.hiddenEnd == null || firstChild.radiusInner.current() == 1 ) | |
1096 { | |
1097 return false; | |
1098 } | |
1099 | |
1100 for ( var i = firstHiddenChild; i < firstChild.hiddenEnd; i++ ) | |
1101 { | |
1102 if ( ! this.children[i].hide || ! this.children[i].hidePrev && progress < 1 ) | |
1103 { | |
1104 return false; | |
1105 } | |
1106 } | |
1107 | |
1108 var angleStart = firstChild.angleStart.current() + rotationOffset; | |
1109 var lastChild = this.children[firstChild.hiddenEnd]; | |
1110 var angleEnd = lastChild.angleEnd.current() + rotationOffset; | |
1111 var radiusInner = gRadius * firstChild.radiusInner.current(); | |
1112 var hiddenChildren = firstChild.hiddenEnd - firstHiddenChild + 1; | |
1113 | |
1114 if ( labelMode ) | |
1115 { | |
1116 var hiddenSearchResults = 0; | |
1117 | |
1118 for ( var i = firstHiddenChild; i <= firstChild.hiddenEnd; i++ ) | |
1119 { | |
1120 hiddenSearchResults += this.children[i].searchResults; | |
1121 | |
1122 if ( this.children[i].magnitude == 0 ) | |
1123 { | |
1124 hiddenChildren--; | |
1125 } | |
1126 } | |
1127 | |
1128 if | |
1129 ( | |
1130 selected && | |
1131 (angleEnd - angleStart) * | |
1132 (gRadius + gRadius) >= | |
1133 minWidth() || | |
1134 this == highlightedNode && | |
1135 hiddenChildren || | |
1136 hiddenSearchResults | |
1137 ) | |
1138 { | |
1139 context.globalAlpha = this.alphaWedge.current(); | |
1140 | |
1141 this.drawHiddenLabel | |
1142 ( | |
1143 angleStart, | |
1144 angleEnd, | |
1145 hiddenChildren, | |
1146 hiddenSearchResults | |
1147 ); | |
1148 } | |
1149 } | |
1150 | |
1151 var drawWedges = true; | |
1152 | |
1153 for ( var i = firstHiddenChild; i <= firstChild.hiddenEnd; i++ ) | |
1154 { | |
1155 // all hidden children must be completely hidden to draw together | |
1156 | |
1157 if ( this.children[i].alphaPattern.current() != this.children[i].alphaWedge.current() ) | |
1158 { | |
1159 drawWedges = false; | |
1160 break; | |
1161 } | |
1162 } | |
1163 | |
1164 if ( labelMode ) | |
1165 { | |
1166 if ( drawWedges ) | |
1167 { | |
1168 var drawRadial = (angleEnd < this.angleEnd.current() + rotationOffset); | |
1169 this.drawLines(angleStart, angleEnd, radiusInner, drawRadial); | |
1170 } | |
1171 | |
1172 if ( hiddenSearchResults && ! searchHighlighted ) | |
1173 { | |
1174 drawWedge | |
1175 ( | |
1176 angleStart, | |
1177 angleEnd, | |
1178 radiusInner, | |
1179 gRadius,//this.radiusOuter.current() * gRadius, | |
1180 highlightFill, | |
1181 0, | |
1182 true | |
1183 ); | |
1184 } | |
1185 } | |
1186 else if ( drawWedges ) | |
1187 { | |
1188 context.globalAlpha = this.alphaWedge.current(); | |
1189 | |
1190 var fill = rgbText | |
1191 ( | |
1192 firstChild.r.current(), | |
1193 firstChild.g.current(), | |
1194 firstChild.b.current() | |
1195 ); | |
1196 | |
1197 drawWedge | |
1198 ( | |
1199 angleStart, | |
1200 angleEnd, | |
1201 radiusInner, | |
1202 gRadius,//this.radiusOuter.current() * gRadius, | |
1203 fill, | |
1204 context.globalAlpha, | |
1205 false | |
1206 ); | |
1207 } | |
1208 | |
1209 return drawWedges; | |
1210 } | |
1211 | |
1212 this.drawHiddenLabel = function(angleStart, angleEnd, value, hiddenSearchResults) | |
1213 { | |
1214 var textAngle = (angleStart + angleEnd) / 2; | |
1215 var labelRadius = gRadius + fontSize;//(radiusInner + radius) / 2; | |
1216 | |
1217 var hiddenLabel = Array(); | |
1218 | |
1219 hiddenLabel.value = value; | |
1220 hiddenLabel.angle = textAngle; | |
1221 hiddenLabel.search = hiddenSearchResults; | |
1222 | |
1223 this.hiddenLabels.push(hiddenLabel); | |
1224 | |
1225 drawTick(gRadius - fontSize * .75, fontSize * 1.5, textAngle); | |
1226 drawTextPolar | |
1227 ( | |
1228 value.toString() + ' more', | |
1229 0, // inner text | |
1230 textAngle, | |
1231 labelRadius, | |
1232 true, // radial | |
1233 hiddenSearchResults, // bubble | |
1234 this == highlightedNode || this == focusNode, // bold | |
1235 false, | |
1236 hiddenSearchResults | |
1237 ); | |
1238 } | |
1239 | |
1240 this.drawHighlight = function(bold) | |
1241 { | |
1242 var angleStartCurrent = this.angleStart.current() + rotationOffset; | |
1243 var angleEndCurrent = this.angleEnd.current() + rotationOffset; | |
1244 var radiusInner = this.radiusInner.current() * gRadius; | |
1245 | |
1246 //this.setHighlightStyle(); | |
1247 | |
1248 if ( this == focusNode && this == highlightedNode && this.hasChildren() ) | |
1249 { | |
1250 // context.fillStyle = "rgba(255, 255, 255, .3)"; | |
1251 arrow | |
1252 ( | |
1253 angleStartCurrent, | |
1254 angleEndCurrent, | |
1255 radiusInner | |
1256 ); | |
1257 } | |
1258 else | |
1259 { | |
1260 drawWedge | |
1261 ( | |
1262 angleStartCurrent, | |
1263 angleEndCurrent, | |
1264 radiusInner, | |
1265 gRadius, | |
1266 highlightFill, | |
1267 0, | |
1268 true | |
1269 ); | |
1270 } | |
1271 | |
1272 // check if hidden children should be highlighted | |
1273 // | |
1274 for ( var i = 0; i < this.children.length; i++ ) | |
1275 { | |
1276 if | |
1277 ( | |
1278 this.children[i].getDepth() - selectedNode.getDepth() + 1 <= | |
1279 maxDisplayDepth && | |
1280 this.children[i].hiddenEnd != null | |
1281 ) | |
1282 { | |
1283 var firstChild = this.children[i]; | |
1284 var lastChild = this.children[firstChild.hiddenEnd]; | |
1285 var hiddenAngleStart = firstChild.angleStart.current() + rotationOffset; | |
1286 var hiddenAngleEnd = lastChild.angleEnd.current() + rotationOffset; | |
1287 var hiddenRadiusInner = gRadius * firstChild.radiusInner.current(); | |
1288 | |
1289 drawWedge | |
1290 ( | |
1291 hiddenAngleStart, | |
1292 hiddenAngleEnd, | |
1293 hiddenRadiusInner, | |
1294 gRadius, | |
1295 'rgba(255, 255, 255, .3)', | |
1296 0, | |
1297 true | |
1298 ); | |
1299 | |
1300 if ( false && ! this.searchResults ) | |
1301 { | |
1302 this.drawHiddenLabel | |
1303 ( | |
1304 hiddenAngleStart, | |
1305 hiddenAngleEnd, | |
1306 firstChild.hiddenEnd - i + 1 | |
1307 ); | |
1308 } | |
1309 | |
1310 i = firstChild.hiddenEnd; | |
1311 } | |
1312 } | |
1313 | |
1314 // context.strokeStyle = 'black'; | |
1315 context.fillStyle = 'black'; | |
1316 | |
1317 var highlight = ! ( progress < 1 && zoomOut && this == selectedNodeLast ); | |
1318 | |
1319 var angle = (angleEndCurrent + angleStartCurrent) / 2; | |
1320 | |
1321 if ( ! (this.keyed && showKeys) ) | |
1322 { | |
1323 this.drawLabel(angle, true, bold, true, this.radial); | |
1324 } | |
1325 } | |
1326 | |
1327 this.drawHighlightCenter = function() | |
1328 { | |
1329 if ( ! this.canDisplayHistory() ) | |
1330 { | |
1331 return; | |
1332 } | |
1333 | |
1334 context.lineWidth = highlightLineWidth; | |
1335 context.strokeStyle = 'black'; | |
1336 context.fillStyle = "rgba(255, 255, 255, .6)"; | |
1337 | |
1338 context.fillStyle = 'black'; | |
1339 this.drawLabel(3 * Math.PI / 2, true, true, false); | |
1340 context.font = fontNormal; | |
1341 } | |
1342 | |
1343 this.drawKey = function(angle, highlight, bold) | |
1344 { | |
1345 var offset = keyOffset(); | |
1346 var color; | |
1347 var colorText = this.magnitude == 0 ? 'gray' : 'black'; | |
1348 var patternAlpha = this.alphaPattern.end; | |
1349 var boxLeft = imageWidth - keySize - margin; | |
1350 var textY = offset + keySize / 2; | |
1351 | |
1352 var label; | |
1353 var keyNameWidth; | |
1354 | |
1355 if ( this == selectedNode ) | |
1356 { | |
1357 color = colorUnclassified; | |
1358 label = | |
1359 this.getUnclassifiedText() + | |
1360 ' ' + | |
1361 this.getUnclassifiedPercentage(); | |
1362 keyNameWidth = measureText(label, false); | |
1363 } | |
1364 else | |
1365 { | |
1366 label = this.keyLabel; | |
1367 color = rgbText(this.r.end, this.g.end, this.b.end); | |
1368 | |
1369 if ( highlight ) | |
1370 { | |
1371 if ( this.searchResultChildren() ) | |
1372 { | |
1373 label = label + searchResultString(this.searchResultChildren()); | |
1374 } | |
1375 | |
1376 keyNameWidth = measureText(label, bold); | |
1377 } | |
1378 else | |
1379 { | |
1380 keyNameWidth = this.keyNameWidth; | |
1381 } | |
1382 } | |
1383 | |
1384 var textLeft = boxLeft - keyBuffer - keyNameWidth - fontSize / 2; | |
1385 var labelLeft = textLeft; | |
1386 | |
1387 if ( labelLeft > keyMinTextLeft - fontSize / 2 ) | |
1388 { | |
1389 keyMinTextLeft -= fontSize / 2; | |
1390 | |
1391 if ( keyMinTextLeft < centerX - gRadius + fontSize / 2 ) | |
1392 { | |
1393 keyMinTextLeft = centerX - gRadius + fontSize / 2; | |
1394 } | |
1395 | |
1396 labelLeft = keyMinTextLeft; | |
1397 } | |
1398 | |
1399 var lineX = new Array(); | |
1400 var lineY = new Array(); | |
1401 | |
1402 var bendRadius; | |
1403 var keyAngle = Math.atan((textY - centerY) / (labelLeft - centerX)); | |
1404 var arcAngle; | |
1405 | |
1406 if ( keyAngle < 0 ) | |
1407 { | |
1408 keyAngle += Math.PI; | |
1409 } | |
1410 | |
1411 if ( keyMinAngle == 0 || angle < keyMinAngle ) | |
1412 { | |
1413 keyMinAngle = angle; | |
1414 } | |
1415 | |
1416 if ( angle > Math.PI && keyMinAngle > Math.PI ) | |
1417 { | |
1418 // allow lines to come underneath the chart | |
1419 | |
1420 angle -= Math.PI * 2; | |
1421 } | |
1422 | |
1423 lineX.push(Math.cos(angle) * gRadius); | |
1424 lineY.push(Math.sin(angle) * gRadius); | |
1425 | |
1426 if ( angle < keyAngle && textY > centerY + Math.sin(angle) * (gRadius + buffer * (currentKey - 1) / (keys + 1) / 2 + buffer / 2) ) | |
1427 { | |
1428 bendRadius = gRadius + buffer - buffer * currentKey / (keys + 1) / 2; | |
1429 } | |
1430 else | |
1431 { | |
1432 bendRadius = gRadius + buffer * currentKey / (keys + 1) / 2 + buffer / 2; | |
1433 } | |
1434 | |
1435 var outside = | |
1436 Math.sqrt | |
1437 ( | |
1438 Math.pow(labelLeft - centerX, 2) + | |
1439 Math.pow(textY - centerY, 2) | |
1440 ) > bendRadius; | |
1441 | |
1442 if ( ! outside ) | |
1443 { | |
1444 arcAngle = Math.asin((textY - centerY) / bendRadius); | |
1445 | |
1446 keyMinTextLeft = min(keyMinTextLeft, centerX + bendRadius * Math.cos(arcAngle) - fontSize / 2); | |
1447 | |
1448 if ( labelLeft < textLeft && textLeft > centerX + bendRadius * Math.cos(arcAngle) ) | |
1449 { | |
1450 lineX.push(textLeft - centerX); | |
1451 lineY.push(textY - centerY); | |
1452 } | |
1453 } | |
1454 else | |
1455 { | |
1456 keyMinTextLeft = min(keyMinTextLeft, labelLeft - fontSize / 2); | |
1457 | |
1458 if ( angle < keyAngle ) | |
1459 { | |
1460 // flip everything over y = x | |
1461 // | |
1462 arcAngle = Math.PI / 2 - keyLineAngle | |
1463 ( | |
1464 Math.PI / 2 - angle, | |
1465 Math.PI / 2 - keyAngle, | |
1466 bendRadius, | |
1467 textY - centerY, | |
1468 labelLeft - centerX, | |
1469 lineY, | |
1470 lineX | |
1471 ); | |
1472 | |
1473 } | |
1474 else | |
1475 { | |
1476 arcAngle = keyLineAngle | |
1477 ( | |
1478 angle, | |
1479 keyAngle, | |
1480 bendRadius, | |
1481 labelLeft - centerX, | |
1482 textY - centerY, | |
1483 lineX, | |
1484 lineY | |
1485 ); | |
1486 } | |
1487 } | |
1488 | |
1489 if ( labelLeft > centerX + bendRadius * Math.cos(arcAngle) || | |
1490 textY > centerY + bendRadius * Math.sin(arcAngle) + .01) | |
1491 // if ( outside || ) | |
1492 { | |
1493 lineX.push(labelLeft - centerX); | |
1494 lineY.push(textY - centerY); | |
1495 | |
1496 if ( textLeft != labelLeft ) | |
1497 { | |
1498 lineX.push(textLeft - centerX); | |
1499 lineY.push(textY - centerY); | |
1500 } | |
1501 } | |
1502 | |
1503 context.globalAlpha = this.alphaWedge.current(); | |
1504 | |
1505 if ( snapshotMode ) | |
1506 { | |
1507 var labelSVG; | |
1508 | |
1509 if ( this == selectedNode ) | |
1510 { | |
1511 labelSVG = | |
1512 this.getUnclassifiedText() + | |
1513 spacer() + | |
1514 this.getUnclassifiedPercentage(); | |
1515 } | |
1516 else | |
1517 { | |
1518 labelSVG = this.name + spacer() + this.getPercentage() + '%'; | |
1519 } | |
1520 | |
1521 svg += | |
1522 '<rect fill="' + color + '" ' + | |
1523 'x="' + boxLeft + '" y="' + offset + | |
1524 '" width="' + keySize + '" height="' + keySize + '"/>'; | |
1525 | |
1526 if ( patternAlpha ) | |
1527 { | |
1528 svg += | |
1529 '<rect fill="url(#hiddenPattern)" style="stroke:none" ' + | |
1530 'x="' + boxLeft + '" y="' + offset + | |
1531 '" width="' + keySize + '" height="' + keySize + '"/>'; | |
1532 } | |
1533 | |
1534 svg += | |
1535 '<path class="line' + | |
1536 (highlight ? ' highlight' : '') + | |
1537 '" d="M ' + (lineX[0] + centerX) + ',' + | |
1538 (lineY[0] + centerY); | |
1539 | |
1540 if ( angle != arcAngle ) | |
1541 { | |
1542 svg += | |
1543 ' L ' + (centerX + bendRadius * Math.cos(angle)) + ',' + | |
1544 (centerY + bendRadius * Math.sin(angle)) + | |
1545 ' A ' + bendRadius + ',' + bendRadius + ' 0 ' + | |
1546 '0,' + (angle > arcAngle ? '0' : '1') + ' ' + | |
1547 (centerX + bendRadius * Math.cos(arcAngle)) + ',' + | |
1548 (centerY + bendRadius * Math.sin(arcAngle)); | |
1549 } | |
1550 | |
1551 for ( var i = 1; i < lineX.length; i++ ) | |
1552 { | |
1553 svg += | |
1554 ' L ' + (centerX + lineX[i]) + ',' + | |
1555 (centerY + lineY[i]); | |
1556 } | |
1557 | |
1558 svg += '"/>'; | |
1559 | |
1560 if ( highlight ) | |
1561 { | |
1562 if ( this.searchResultChildren() ) | |
1563 { | |
1564 labelSVG = labelSVG + searchResultString(this.searchResultChildren()); | |
1565 } | |
1566 | |
1567 drawBubbleSVG | |
1568 ( | |
1569 boxLeft - keyBuffer - keyNameWidth - fontSize / 2, | |
1570 textY - fontSize, | |
1571 keyNameWidth + fontSize, | |
1572 fontSize * 2, | |
1573 fontSize, | |
1574 0 | |
1575 ); | |
1576 | |
1577 if ( this.isSearchResult ) | |
1578 { | |
1579 drawSearchHighlights | |
1580 ( | |
1581 label, | |
1582 boxLeft - keyBuffer - keyNameWidth, | |
1583 textY, | |
1584 0 | |
1585 ) | |
1586 } | |
1587 } | |
1588 | |
1589 svg += svgText(labelSVG, boxLeft - keyBuffer, textY, 'end', bold, colorText); | |
1590 } | |
1591 else | |
1592 { | |
1593 context.fillStyle = color; | |
1594 context.translate(-centerX, -centerY); | |
1595 context.strokeStyle = 'black'; | |
1596 context.globalAlpha = 1;//this.alphaWedge.current(); | |
1597 | |
1598 context.fillRect(boxLeft, offset, keySize, keySize); | |
1599 | |
1600 if ( patternAlpha ) | |
1601 { | |
1602 context.globalAlpha = patternAlpha; | |
1603 context.fillStyle = hiddenPattern; | |
1604 | |
1605 // make clipping box for Firefox performance | |
1606 context.beginPath(); | |
1607 context.moveTo(boxLeft, offset); | |
1608 context.lineTo(boxLeft + keySize, offset); | |
1609 context.lineTo(boxLeft + keySize, offset + keySize); | |
1610 context.lineTo(boxLeft, offset + keySize); | |
1611 context.closePath(); | |
1612 context.save(); | |
1613 context.clip(); | |
1614 | |
1615 context.fillRect(boxLeft, offset, keySize, keySize); | |
1616 context.fillRect(boxLeft, offset, keySize, keySize); | |
1617 | |
1618 context.restore(); // remove clipping region | |
1619 } | |
1620 | |
1621 if ( highlight ) | |
1622 { | |
1623 this.setHighlightStyle(); | |
1624 context.fillRect(boxLeft, offset, keySize, keySize); | |
1625 } | |
1626 else | |
1627 { | |
1628 context.lineWidth = thinLineWidth; | |
1629 } | |
1630 | |
1631 context.strokeRect(boxLeft, offset, keySize, keySize); | |
1632 | |
1633 if ( lineX.length ) | |
1634 { | |
1635 context.beginPath(); | |
1636 context.moveTo(lineX[0] + centerX, lineY[0] + centerY); | |
1637 | |
1638 context.arc(centerX, centerY, bendRadius, angle, arcAngle, angle > arcAngle); | |
1639 | |
1640 for ( var i = 1; i < lineX.length; i++ ) | |
1641 { | |
1642 context.lineTo(lineX[i] + centerX, lineY[i] + centerY); | |
1643 } | |
1644 | |
1645 context.globalAlpha = this == selectedNode ? | |
1646 this.children[0].alphaWedge.current() : | |
1647 this.alphaWedge.current(); | |
1648 context.lineWidth = highlight ? highlightLineWidth : thinLineWidth; | |
1649 context.stroke(); | |
1650 context.globalAlpha = 1; | |
1651 } | |
1652 | |
1653 if ( highlight ) | |
1654 { | |
1655 drawBubbleCanvas | |
1656 ( | |
1657 boxLeft - keyBuffer - keyNameWidth - fontSize / 2, | |
1658 textY - fontSize, | |
1659 keyNameWidth + fontSize, | |
1660 fontSize * 2, | |
1661 fontSize, | |
1662 0 | |
1663 ); | |
1664 | |
1665 if ( this.isSearchResult ) | |
1666 { | |
1667 drawSearchHighlights | |
1668 ( | |
1669 label, | |
1670 boxLeft - keyBuffer - keyNameWidth, | |
1671 textY, | |
1672 0 | |
1673 ) | |
1674 } | |
1675 } | |
1676 | |
1677 drawText(label, boxLeft - keyBuffer, offset + keySize / 2, 0, 'end', bold, colorText); | |
1678 | |
1679 context.translate(centerX, centerY); | |
1680 } | |
1681 | |
1682 currentKey++; | |
1683 } | |
1684 | |
1685 this.drawLabel = function(angle, bubble, bold, selected, radial) | |
1686 { | |
1687 if ( context.globalAlpha == 0 ) | |
1688 { | |
1689 return; | |
1690 } | |
1691 | |
1692 var innerText; | |
1693 var label; | |
1694 var radius; | |
1695 | |
1696 if ( radial ) | |
1697 { | |
1698 radius = (this.radiusInner.current() + 1) * gRadius / 2; | |
1699 } | |
1700 else | |
1701 { | |
1702 radius = this.labelRadius.current() * gRadius; | |
1703 } | |
1704 | |
1705 if ( radial && (selected || bubble ) ) | |
1706 { | |
1707 var percentage = this.getPercentage(); | |
1708 innerText = percentage + '%'; | |
1709 } | |
1710 | |
1711 if | |
1712 ( | |
1713 ! radial && | |
1714 this != selectedNode && | |
1715 ! bubble && | |
1716 ( !zoomOut || this != selectedNodeLast) | |
1717 ) | |
1718 { | |
1719 label = this.shortenLabel(); | |
1720 } | |
1721 else | |
1722 { | |
1723 label = this.name; | |
1724 } | |
1725 | |
1726 var flipped = drawTextPolar | |
1727 ( | |
1728 label, | |
1729 innerText, | |
1730 angle, | |
1731 radius, | |
1732 radial, | |
1733 bubble, | |
1734 bold, | |
1735 // this.isSearchResult && this.shouldAddSearchResultsString() && (!selected || this == selectedNode || highlight), | |
1736 this.isSearchResult && (!selected || this == selectedNode || bubble), | |
1737 (this.hideAlone || !selected || this == selectedNode ) ? this.searchResultChildren() : 0 | |
1738 ); | |
1739 | |
1740 var depth = this.getDepth() - selectedNode.getDepth() + 1; | |
1741 | |
1742 if | |
1743 ( | |
1744 ! radial && | |
1745 ! bubble && | |
1746 this != selectedNode && | |
1747 this.angleEnd.end != this.angleStart.end && | |
1748 nLabelOffsets[depth - 2] > 2 && | |
1749 this.labelWidth.current() > (this.angleEnd.end - this.angleStart.end) * Math.abs(radius) && | |
1750 ! ( zoomOut && this == selectedNodeLast ) && | |
1751 this.labelRadius.end > 0 | |
1752 ) | |
1753 { | |
1754 // name extends beyond wedge; draw tick mark towards the central | |
1755 // radius for easier identification | |
1756 | |
1757 var radiusCenter = compress ? | |
1758 (compressedRadii[depth - 1] + compressedRadii[depth - 2]) / 2 : | |
1759 (depth - .5) * nodeRadius; | |
1760 | |
1761 if ( this.labelRadius.end > radiusCenter ) | |
1762 { | |
1763 if ( flipped ) | |
1764 { | |
1765 drawTick(radius - tickLength * 1.4 , tickLength, angle); | |
1766 } | |
1767 else | |
1768 { | |
1769 drawTick(radius - tickLength * 1.7, tickLength, angle); | |
1770 } | |
1771 } | |
1772 else | |
1773 { | |
1774 if ( flipped ) | |
1775 { | |
1776 drawTick(radius + tickLength * .7, tickLength, angle); | |
1777 } | |
1778 else | |
1779 { | |
1780 drawTick(radius + tickLength * .4, tickLength, angle); | |
1781 } | |
1782 } | |
1783 } | |
1784 } | |
1785 | |
1786 this.drawLines = function(angleStart, angleEnd, radiusInner, drawRadial, selected) | |
1787 { | |
1788 if ( snapshotMode ) | |
1789 { | |
1790 if ( this != selectedNode) | |
1791 { | |
1792 if ( angleEnd == angleStart + Math.PI * 2 ) | |
1793 { | |
1794 // fudge to prevent overlap, which causes arc ambiguity | |
1795 // | |
1796 angleEnd -= .1 / gRadius; | |
1797 } | |
1798 | |
1799 var longArc = angleEnd - angleStart > Math.PI ? 1 : 0; | |
1800 | |
1801 var x1 = centerX + radiusInner * Math.cos(angleStart); | |
1802 var y1 = centerY + radiusInner * Math.sin(angleStart); | |
1803 | |
1804 var x2 = centerX + gRadius * Math.cos(angleStart); | |
1805 var y2 = centerY + gRadius * Math.sin(angleStart); | |
1806 | |
1807 var x3 = centerX + gRadius * Math.cos(angleEnd); | |
1808 var y3 = centerY + gRadius * Math.sin(angleEnd); | |
1809 | |
1810 var x4 = centerX + radiusInner * Math.cos(angleEnd); | |
1811 var y4 = centerY + radiusInner * Math.sin(angleEnd); | |
1812 | |
1813 if ( this.alphaArc.end ) | |
1814 { | |
1815 var dArray = | |
1816 [ | |
1817 " M ", x4, ",", y4, | |
1818 " A ", radiusInner, ",", radiusInner, " 0 ", longArc, | |
1819 " 0 ", x1, ",", y1 | |
1820 ]; | |
1821 | |
1822 svg += '<path class="line" d="' + dArray.join('') + '"/>'; | |
1823 } | |
1824 | |
1825 if ( drawRadial && this.alphaLine.end ) | |
1826 { | |
1827 svg += '<line x1="' + x3 + '" y1="' + y3 + '" x2="' + x4 + '" y2="' + y4 + '"/>'; | |
1828 } | |
1829 } | |
1830 } | |
1831 else | |
1832 { | |
1833 context.lineWidth = thinLineWidth; | |
1834 context.strokeStyle = 'black'; | |
1835 context.beginPath(); | |
1836 context.arc(0, 0, radiusInner, angleStart, angleEnd, false); | |
1837 context.globalAlpha = this.alphaArc.current(); | |
1838 context.stroke(); | |
1839 | |
1840 if ( drawRadial ) | |
1841 { | |
1842 var x1 = radiusInner * Math.cos(angleEnd); | |
1843 var y1 = radiusInner * Math.sin(angleEnd); | |
1844 var x2 = gRadius * Math.cos(angleEnd); | |
1845 var y2 = gRadius * Math.sin(angleEnd); | |
1846 | |
1847 context.beginPath(); | |
1848 context.moveTo(x1, y1); | |
1849 context.lineTo(x2, y2); | |
1850 | |
1851 // if ( this.getCollapse() )//( selected && this != selectedNode ) | |
1852 { | |
1853 context.globalAlpha = this.alphaLine.current(); | |
1854 } | |
1855 | |
1856 context.stroke(); | |
1857 } | |
1858 } | |
1859 } | |
1860 | |
1861 this.drawMap = function(child) | |
1862 { | |
1863 if ( this.parent ) | |
1864 { | |
1865 this.parent.drawMap(child); | |
1866 } | |
1867 | |
1868 if ( this.getCollapse() && this != child || this == focusNode ) | |
1869 { | |
1870 return; | |
1871 } | |
1872 | |
1873 var angleStart = | |
1874 (child.baseMagnitude - this.baseMagnitude) / this.magnitude * Math.PI * 2 + | |
1875 rotationOffset; | |
1876 var angleEnd = | |
1877 (child.baseMagnitude - this.baseMagnitude + child.magnitude) / | |
1878 this.magnitude * Math.PI * 2 + | |
1879 rotationOffset; | |
1880 | |
1881 var box = this.getMapPosition(); | |
1882 | |
1883 context.save(); | |
1884 context.fillStyle = 'black'; | |
1885 context.textAlign = 'end'; | |
1886 context.textBaseline = 'middle'; | |
1887 | |
1888 var textX = box.x - mapRadius - mapBuffer; | |
1889 var percentage = getPercentage(child.magnitude / this.magnitude); | |
1890 | |
1891 var highlight = this == selectedNode || this == highlightedNode; | |
1892 | |
1893 if ( highlight ) | |
1894 { | |
1895 context.font = fontBold; | |
1896 } | |
1897 else | |
1898 { | |
1899 context.font = fontNormal; | |
1900 } | |
1901 | |
1902 context.fillText(percentage + '% of', textX, box.y - mapRadius / 3); | |
1903 context.fillText(this.name, textX, box.y + mapRadius / 3); | |
1904 | |
1905 if ( highlight ) | |
1906 { | |
1907 context.font = fontNormal; | |
1908 } | |
1909 | |
1910 if ( this == highlightedNode && this != selectedNode ) | |
1911 { | |
1912 context.fillStyle = 'rgb(245, 245, 245)'; | |
1913 // context.fillStyle = 'rgb(200, 200, 200)'; | |
1914 } | |
1915 else | |
1916 { | |
1917 context.fillStyle = 'rgb(255, 255, 255)'; | |
1918 } | |
1919 | |
1920 context.beginPath(); | |
1921 context.arc(box.x, box.y, mapRadius, 0, Math.PI * 2, true); | |
1922 context.closePath(); | |
1923 context.fill(); | |
1924 | |
1925 if ( this == selectedNode ) | |
1926 { | |
1927 context.lineWidth = 1; | |
1928 context.fillStyle = 'rgb(100, 100, 100)'; | |
1929 } | |
1930 else | |
1931 { | |
1932 if ( this == highlightedNode ) | |
1933 { | |
1934 context.lineWidth = .2; | |
1935 context.fillStyle = 'rgb(190, 190, 190)'; | |
1936 } | |
1937 else | |
1938 { | |
1939 context.lineWidth = .2; | |
1940 context.fillStyle = 'rgb(200, 200, 200)'; | |
1941 } | |
1942 } | |
1943 | |
1944 var maxDepth = this.getMaxDepth(); | |
1945 | |
1946 if ( ! compress && maxDepth > maxPossibleDepth + this.getDepth() - 1 ) | |
1947 { | |
1948 maxDepth = maxPossibleDepth + this.getDepth() - 1; | |
1949 } | |
1950 | |
1951 if ( this.getDepth() < selectedNode.getDepth() ) | |
1952 { | |
1953 if ( child.getDepth() - 1 >= maxDepth ) | |
1954 { | |
1955 maxDepth = child.getDepth(); | |
1956 } | |
1957 } | |
1958 | |
1959 var radiusInner; | |
1960 | |
1961 if ( compress ) | |
1962 { | |
1963 radiusInner = 0; | |
1964 // Math.atan(child.getDepth() - this.getDepth()) / | |
1965 // Math.PI * 2 * .9; | |
1966 } | |
1967 else | |
1968 { | |
1969 radiusInner = | |
1970 (child.getDepth() - this.getDepth()) / | |
1971 (maxDepth - this.getDepth() + 1); | |
1972 } | |
1973 | |
1974 context.stroke(); | |
1975 context.beginPath(); | |
1976 | |
1977 if ( radiusInner == 0 ) | |
1978 { | |
1979 context.moveTo(box.x, box.y); | |
1980 } | |
1981 else | |
1982 { | |
1983 context.arc(box.x, box.y, mapRadius * radiusInner, angleEnd, angleStart, true); | |
1984 } | |
1985 | |
1986 context.arc(box.x, box.y, mapRadius, angleStart, angleEnd, false); | |
1987 context.closePath(); | |
1988 context.fill(); | |
1989 | |
1990 if ( this == highlightedNode && this != selectedNode ) | |
1991 { | |
1992 context.lineWidth = 1; | |
1993 context.stroke(); | |
1994 } | |
1995 | |
1996 context.restore(); | |
1997 } | |
1998 | |
1999 this.drawReferenceRings = function(childRadiusInner) | |
2000 { | |
2001 if ( snapshotMode ) | |
2002 { | |
2003 svg += | |
2004 '<circle cx="' + centerX + '" cy="' + centerY + | |
2005 '" r="' + childRadiusInner + '"/>'; | |
2006 svg += | |
2007 '<circle cx="' + centerX + '" cy="' + centerY + | |
2008 '" r="' + gRadius + '"/>'; | |
2009 } | |
2010 else | |
2011 { | |
2012 context.globalAlpha = 1 - this.alphaLine.current();//this.getUncollapsed().alphaLine.current(); | |
2013 context.beginPath(); | |
2014 context.arc(0, 0, childRadiusInner, 0, Math.PI * 2, false); | |
2015 context.stroke(); | |
2016 context.beginPath(); | |
2017 context.arc(0, 0, gRadius, 0, Math.PI * 2, false); | |
2018 context.stroke(); | |
2019 } | |
2020 } | |
2021 | |
2022 this.getCollapse = function() | |
2023 { | |
2024 return ( | |
2025 collapse && | |
2026 this.collapse && | |
2027 this.depth != maxAbsoluteDepth | |
2028 ); | |
2029 } | |
2030 | |
2031 this.getDepth = function() | |
2032 { | |
2033 if ( collapse ) | |
2034 { | |
2035 return this.depthCollapsed; | |
2036 } | |
2037 else | |
2038 { | |
2039 return this.depth; | |
2040 } | |
2041 } | |
2042 | |
2043 this.getMagnitude = function() | |
2044 { | |
2045 return this.attributes[magnitudeIndex][currentDataset]; | |
2046 } | |
2047 | |
2048 this.getMapPosition = function() | |
2049 { | |
2050 return { | |
2051 x : (details.offsetLeft + details.clientWidth - mapRadius), | |
2052 y : ((focusNode.getDepth() - this.getDepth()) * | |
2053 (mapBuffer + mapRadius * 2) - mapRadius) + | |
2054 details.clientHeight + details.offsetTop | |
2055 }; | |
2056 } | |
2057 | |
2058 this.getMaxDepth = function(limit) | |
2059 { | |
2060 var max; | |
2061 | |
2062 if ( collapse ) | |
2063 { | |
2064 return this.maxDepthCollapsed; | |
2065 } | |
2066 else | |
2067 { | |
2068 if ( this.maxDepth > maxAbsoluteDepth ) | |
2069 { | |
2070 return maxAbsoluteDepth; | |
2071 } | |
2072 else | |
2073 { | |
2074 return this.maxDepth; | |
2075 } | |
2076 } | |
2077 } | |
2078 | |
2079 this.getData = function(index, summary) | |
2080 { | |
2081 var files = new Array(); | |
2082 | |
2083 if | |
2084 ( | |
2085 this.attributes[index] != null && | |
2086 this.attributes[index][currentDataset] != null && | |
2087 this.attributes[index][currentDataset] != '' | |
2088 ) | |
2089 { | |
2090 files.push | |
2091 ( | |
2092 document.location + | |
2093 '.files/' + | |
2094 this.attributes[index][currentDataset] | |
2095 ); | |
2096 } | |
2097 | |
2098 if ( summary ) | |
2099 { | |
2100 for ( var i = 0; i < this.children.length; i++ ) | |
2101 { | |
2102 files = files.concat(this.children[i].getData(index, true)); | |
2103 } | |
2104 } | |
2105 | |
2106 return files; | |
2107 } | |
2108 | |
2109 this.getList = function(index, summary) | |
2110 { | |
2111 var list; | |
2112 | |
2113 if | |
2114 ( | |
2115 this.attributes[index] != null && | |
2116 this.attributes[index][currentDataset] != null | |
2117 ) | |
2118 { | |
2119 list = this.attributes[index][currentDataset]; | |
2120 } | |
2121 else | |
2122 { | |
2123 list = new Array(); | |
2124 } | |
2125 | |
2126 if ( summary ) | |
2127 { | |
2128 for ( var i = 0; i < this.children.length; i++ ) | |
2129 { | |
2130 list = list.concat(this.children[i].getList(index, true)); | |
2131 } | |
2132 } | |
2133 | |
2134 return list; | |
2135 } | |
2136 | |
2137 this.getParent = function() | |
2138 { | |
2139 // returns parent, accounting for collapsing or 0 if doesn't exist | |
2140 | |
2141 var parent = this.parent; | |
2142 | |
2143 while ( parent != 0 && parent.getCollapse() ) | |
2144 { | |
2145 parent = parent.parent; | |
2146 } | |
2147 | |
2148 return parent; | |
2149 } | |
2150 | |
2151 this.getPercentage = function() | |
2152 { | |
2153 return getPercentage(this.magnitude / selectedNode.magnitude); | |
2154 } | |
2155 | |
2156 this.getUnclassifiedPercentage = function() | |
2157 { | |
2158 if ( this.children.length ) | |
2159 { | |
2160 var lastChild = this.children[this.children.length - 1]; | |
2161 | |
2162 return getPercentage | |
2163 ( | |
2164 ( | |
2165 this.baseMagnitude + | |
2166 this.magnitude - | |
2167 lastChild.magnitude - | |
2168 lastChild.baseMagnitude | |
2169 ) / this.magnitude | |
2170 ) + '%'; | |
2171 } | |
2172 else | |
2173 { | |
2174 return '100%'; | |
2175 } | |
2176 } | |
2177 | |
2178 this.getUnclassifiedText = function() | |
2179 { | |
2180 return '[other '+ this.name + ']'; | |
2181 } | |
2182 | |
2183 this.getUncollapsed = function() | |
2184 { | |
2185 // recurse through collapsed children until uncollapsed node is found | |
2186 | |
2187 if ( this.getCollapse() ) | |
2188 { | |
2189 return this.children[0].getUncollapsed(); | |
2190 } | |
2191 else | |
2192 { | |
2193 return this; | |
2194 } | |
2195 } | |
2196 | |
2197 this.hasChildren = function() | |
2198 { | |
2199 return this.children.length && this.depth < maxAbsoluteDepth && this.magnitude; | |
2200 } | |
2201 | |
2202 this.hasParent = function(parent) | |
2203 { | |
2204 if ( this.parent ) | |
2205 { | |
2206 if ( this.parent == parent ) | |
2207 { | |
2208 return true; | |
2209 } | |
2210 else | |
2211 { | |
2212 return this.parent.hasParent(parent); | |
2213 } | |
2214 } | |
2215 else | |
2216 { | |
2217 return false; | |
2218 } | |
2219 } | |
2220 | |
2221 this.maxVisibleDepth = function(maxDepth) | |
2222 { | |
2223 var childInnerRadius; | |
2224 var depth = this.getDepth() - selectedNode.getDepth() + 1; | |
2225 var currentMaxDepth = depth; | |
2226 | |
2227 if ( this.hasChildren() && depth < maxDepth) | |
2228 { | |
2229 var lastChild = this.children[this.children.length - 1]; | |
2230 | |
2231 if ( this.name == 'Pseudomonadaceae' ) | |
2232 { | |
2233 var x = 3; | |
2234 } | |
2235 | |
2236 if | |
2237 ( | |
2238 lastChild.baseMagnitude + lastChild.magnitude < | |
2239 this.baseMagnitude + this.magnitude | |
2240 ) | |
2241 { | |
2242 currentMaxDepth++; | |
2243 } | |
2244 | |
2245 if ( compress ) | |
2246 { | |
2247 childInnerRadius = compressedRadii[depth - 1]; | |
2248 } | |
2249 else | |
2250 { | |
2251 childInnerRadius = (depth) / maxDepth; | |
2252 } | |
2253 | |
2254 for ( var i = 0; i < this.children.length; i++ ) | |
2255 { | |
2256 if | |
2257 (//true || | |
2258 this.children[i].magnitude * | |
2259 angleFactor * | |
2260 (childInnerRadius + 1) * | |
2261 gRadius >= | |
2262 minWidth() | |
2263 ) | |
2264 { | |
2265 var childMaxDepth = this.children[i].maxVisibleDepth(maxDepth); | |
2266 | |
2267 if ( childMaxDepth > currentMaxDepth ) | |
2268 { | |
2269 currentMaxDepth = childMaxDepth; | |
2270 } | |
2271 } | |
2272 } | |
2273 } | |
2274 | |
2275 return currentMaxDepth; | |
2276 } | |
2277 | |
2278 this.resetLabelWidth = function() | |
2279 { | |
2280 var nameWidthOld = this.nameWidth; | |
2281 | |
2282 if ( true || ! this.radial )//&& fontSize != fontSizeLast ) | |
2283 { | |
2284 var dim = context.measureText(this.name); | |
2285 this.nameWidth = dim.width; | |
2286 } | |
2287 | |
2288 if ( fontSize != fontSizeLast && this.labelWidth.end == nameWidthOld * labelWidthFudge ) | |
2289 { | |
2290 // font size changed; adjust start of tween to match | |
2291 | |
2292 this.labelWidth.start = this.nameWidth * labelWidthFudge; | |
2293 } | |
2294 else | |
2295 { | |
2296 this.labelWidth.start = this.labelWidth.current(); | |
2297 } | |
2298 | |
2299 this.labelWidth.end = this.nameWidth * labelWidthFudge; | |
2300 } | |
2301 | |
2302 this.restrictLabelWidth = function(width) | |
2303 { | |
2304 if ( width < this.labelWidth.end ) | |
2305 { | |
2306 this.labelWidth.end = width; | |
2307 } | |
2308 } | |
2309 | |
2310 this.search = function() | |
2311 { | |
2312 this.isSearchResult = false; | |
2313 this.searchResults = 0; | |
2314 | |
2315 if | |
2316 ( | |
2317 ! this.getCollapse() && | |
2318 search.value != '' && | |
2319 this.name.toLowerCase().indexOf(search.value.toLowerCase()) != -1 | |
2320 ) | |
2321 { | |
2322 this.isSearchResult = true; | |
2323 this.searchResults = 1; | |
2324 nSearchResults++; | |
2325 } | |
2326 | |
2327 for ( var i = 0; i < this.children.length; i++ ) | |
2328 { | |
2329 this.searchResults += this.children[i].search(); | |
2330 } | |
2331 | |
2332 return this.searchResults; | |
2333 } | |
2334 | |
2335 this.searchResultChildren = function() | |
2336 { | |
2337 if ( this.isSearchResult ) | |
2338 { | |
2339 return this.searchResults - 1; | |
2340 } | |
2341 else | |
2342 { | |
2343 return this.searchResults; | |
2344 } | |
2345 } | |
2346 | |
2347 this.setDepth = function(depth, depthCollapsed) | |
2348 { | |
2349 this.depth = depth; | |
2350 this.depthCollapsed = depthCollapsed; | |
2351 | |
2352 if | |
2353 ( | |
2354 this.children.length == 1 && | |
2355 // this.magnitude > 0 && | |
2356 this.children[0].magnitude == this.magnitude && | |
2357 ( head.children.length > 1 || this.children[0].children.length ) | |
2358 ) | |
2359 { | |
2360 this.collapse = true; | |
2361 } | |
2362 else | |
2363 { | |
2364 this.collapse = false; | |
2365 depthCollapsed++; | |
2366 } | |
2367 | |
2368 for ( var i = 0; i < this.children.length; i++ ) | |
2369 { | |
2370 this.children[i].setDepth(depth + 1, depthCollapsed); | |
2371 } | |
2372 } | |
2373 | |
2374 this.setHighlightStyle = function() | |
2375 { | |
2376 context.lineWidth = highlightLineWidth; | |
2377 | |
2378 if ( this.hasChildren() || this != focusNode || this != highlightedNode ) | |
2379 { | |
2380 context.strokeStyle = 'black'; | |
2381 context.fillStyle = "rgba(255, 255, 255, .3)"; | |
2382 } | |
2383 else | |
2384 { | |
2385 context.strokeStyle = 'rgb(90,90,90)'; | |
2386 context.fillStyle = "rgba(155, 155, 155, .3)"; | |
2387 } | |
2388 } | |
2389 | |
2390 this.setLabelWidth = function(node) | |
2391 { | |
2392 if ( ! shorten || this.radial ) | |
2393 { | |
2394 return; // don't need to set width | |
2395 } | |
2396 | |
2397 if ( node.hide ) | |
2398 { | |
2399 alert('wtf'); | |
2400 return; | |
2401 } | |
2402 | |
2403 var angle = (this.angleStart.end + this.angleEnd.end) / 2; | |
2404 var a; // angle difference | |
2405 | |
2406 if ( node == selectedNode ) | |
2407 { | |
2408 a = Math.abs(angle - node.angleOther); | |
2409 } | |
2410 else | |
2411 { | |
2412 a = Math.abs(angle - (node.angleStart.end + node.angleEnd.end) / 2); | |
2413 } | |
2414 | |
2415 if ( a == 0 ) | |
2416 { | |
2417 return; | |
2418 } | |
2419 | |
2420 if ( a > Math.PI ) | |
2421 { | |
2422 a = 2 * Math.PI - a; | |
2423 } | |
2424 | |
2425 if ( node.radial || node == selectedNode ) | |
2426 { | |
2427 var nodeLabelRadius; | |
2428 | |
2429 if ( node == selectedNode ) | |
2430 { | |
2431 // radial 'other' label | |
2432 | |
2433 nodeLabelRadius = (node.children[0].radiusInner.end + 1) / 2; | |
2434 } | |
2435 else | |
2436 { | |
2437 nodeLabelRadius = (node.radiusInner.end + 1) / 2; | |
2438 } | |
2439 | |
2440 if ( a < Math.PI / 2 ) | |
2441 { | |
2442 var r = this.labelRadius.end * gRadius + .5 * fontSize | |
2443 var hypotenuse = r / Math.cos(a); | |
2444 var opposite = r * Math.tan(a); | |
2445 var fontRadius = .8 * fontSize; | |
2446 | |
2447 if | |
2448 ( | |
2449 nodeLabelRadius * gRadius < hypotenuse && | |
2450 this.labelWidth.end / 2 + fontRadius > opposite | |
2451 ) | |
2452 { | |
2453 this.labelWidth.end = 2 * (opposite - fontRadius); | |
2454 } | |
2455 } | |
2456 } | |
2457 else if | |
2458 ( | |
2459 this.labelRadius.end == node.labelRadius.end && | |
2460 a < Math.PI / 4 | |
2461 ) | |
2462 { | |
2463 // same radius with small angle; use circumferential approximation | |
2464 | |
2465 var dist = a * this.labelRadius.end * gRadius - fontSize * (1 - a * 4 / Math.PI) * 1.3; | |
2466 | |
2467 if ( this.labelWidth.end < dist ) | |
2468 { | |
2469 node.restrictLabelWidth((dist - this.labelWidth.end / 2) * 2); | |
2470 } | |
2471 else if ( node.labelWidth.end < dist ) | |
2472 { | |
2473 this.restrictLabelWidth((dist - node.labelWidth.end / 2) * 2); | |
2474 } | |
2475 else | |
2476 { | |
2477 // both labels reach halfway point; restrict both | |
2478 | |
2479 this.labelWidth.end = dist; | |
2480 node.labelWidth.end = dist | |
2481 } | |
2482 } | |
2483 else | |
2484 { | |
2485 var r1 = this.labelRadius.end * gRadius; | |
2486 var r2 = node.labelRadius.end * gRadius; | |
2487 | |
2488 // first adjust the radii to account for the height of the font by shifting them | |
2489 // toward each other | |
2490 // | |
2491 var fontFudge = .35 * fontSize; | |
2492 // | |
2493 if ( this.labelRadius.end < node.labelRadius.end ) | |
2494 { | |
2495 r1 += fontFudge; | |
2496 r2 -= fontFudge; | |
2497 } | |
2498 else if ( this.labelRadius.end > node.labelRadius.end ) | |
2499 { | |
2500 r1 -= fontFudge; | |
2501 r2 += fontFudge; | |
2502 } | |
2503 else | |
2504 { | |
2505 r1 -= fontFudge; | |
2506 r2 -= fontFudge; | |
2507 } | |
2508 | |
2509 var r1s = r1 * r1; | |
2510 var r2s = r2 * r2; | |
2511 | |
2512 // distance between the centers of the two labels | |
2513 // | |
2514 var dist = Math.sqrt(r1s + r2s - 2 * r1 * r2 * Math.cos(a)); | |
2515 | |
2516 // angle at our label center between our radius and the line to the other label center | |
2517 // | |
2518 var b = Math.acos((r1s + dist * dist - r2s) / (2 * r1 * dist)); | |
2519 | |
2520 // distance from our label center to the intersection of the two tangents | |
2521 // | |
2522 var l1 = Math.sin(a + b - Math.PI / 2) * dist / Math.sin(Math.PI - a); | |
2523 | |
2524 // distance from other label center the the intersection of the two tangents | |
2525 // | |
2526 var l2 = Math.sin(Math.PI / 2 - b) * dist / Math.sin(Math.PI - a); | |
2527 | |
2528 l1 = Math.abs(l1) - .4 * fontSize; | |
2529 l2 = Math.abs(l2) - .4 * fontSize; | |
2530 /* | |
2531 // amount to shorten the distances because of the height of the font | |
2532 // | |
2533 var l3 = 0; | |
2534 var fontRadius = fontSize * .55; | |
2535 // | |
2536 if ( l1 < 0 || l2 < 0 ) | |
2537 { | |
2538 var l4 = fontRadius / Math.tan(a); | |
2539 l1 = Math.abs(l1); | |
2540 l2 = Math.abs(l2); | |
2541 | |
2542 l1 -= l4; | |
2543 l2 -= l4; | |
2544 } | |
2545 else | |
2546 { | |
2547 var c = Math.PI - a; | |
2548 | |
2549 l3 = fontRadius * Math.tan(c / 2); | |
2550 } | |
2551 */ | |
2552 if ( this.labelWidth.end / 2 > l1 && node.labelWidth.end / 2 > l2 ) | |
2553 { | |
2554 // shorten the farthest one from the intersection | |
2555 | |
2556 if ( l1 > l2 ) | |
2557 { | |
2558 this.restrictLabelWidth(2 * (l1));// - l3 - fontRadius)); | |
2559 } | |
2560 else | |
2561 { | |
2562 node.restrictLabelWidth(2 * (l2));// - l3 - fontRadius)); | |
2563 } | |
2564 }/* | |
2565 else if ( this.labelWidth.end / 2 > l1 + l3 && node.labelWidth.end / 2 > l2 - l3 ) | |
2566 { | |
2567 node.restrictLabelWidth(2 * (l2 - l3)); | |
2568 } | |
2569 else if ( this.labelWidth.end / 2 > l1 - l3 && node.labelWidth.end / 2 > l2 + l3 ) | |
2570 { | |
2571 this.restrictLabelWidth(2 * (l1 - l3)); | |
2572 }*/ | |
2573 } | |
2574 } | |
2575 | |
2576 this.setMagnitudes = function(baseMagnitude) | |
2577 { | |
2578 this.magnitude = this.getMagnitude(); | |
2579 this.baseMagnitude = baseMagnitude; | |
2580 | |
2581 for ( var i = 0; i < this.children.length; i++ ) | |
2582 { | |
2583 this.children[i].setMagnitudes(baseMagnitude); | |
2584 baseMagnitude += this.children[i].magnitude; | |
2585 } | |
2586 | |
2587 this.maxChildMagnitude = baseMagnitude; | |
2588 } | |
2589 | |
2590 this.setMaxDepths = function() | |
2591 { | |
2592 this.maxDepth = this.depth; | |
2593 this.maxDepthCollapsed = this.depthCollapsed; | |
2594 | |
2595 for ( i in this.children ) | |
2596 { | |
2597 var child = this.children[i]; | |
2598 | |
2599 child.setMaxDepths(); | |
2600 | |
2601 if ( child.maxDepth > this.maxDepth ) | |
2602 { | |
2603 this.maxDepth = child.maxDepth; | |
2604 } | |
2605 | |
2606 if | |
2607 ( | |
2608 child.maxDepthCollapsed > this.maxDepthCollapsed && | |
2609 (child.depth <= maxAbsoluteDepth || maxAbsoluteDepth == 0) | |
2610 ) | |
2611 { | |
2612 this.maxDepthCollapsed = child.maxDepthCollapsed; | |
2613 } | |
2614 } | |
2615 } | |
2616 | |
2617 this.setTargetLabelRadius = function() | |
2618 { | |
2619 var depth = this.getDepth() - selectedNode.getDepth() + 1; | |
2620 var index = depth - 2; | |
2621 var labelOffset = labelOffsets[index]; | |
2622 | |
2623 if ( this.radial ) | |
2624 { | |
2625 //this.labelRadius.setTarget((this.radiusInner.end + 1) / 2); | |
2626 var max = | |
2627 depth == maxDisplayDepth ? | |
2628 1 : | |
2629 compressedRadii[index + 1]; | |
2630 | |
2631 this.labelRadius.setTarget((compressedRadii[index] + max) / 2); | |
2632 } | |
2633 else | |
2634 { | |
2635 var radiusCenter; | |
2636 var width; | |
2637 | |
2638 if ( compress ) | |
2639 { | |
2640 if ( nLabelOffsets[index] > 1 ) | |
2641 { | |
2642 this.labelRadius.setTarget | |
2643 ( | |
2644 lerp | |
2645 ( | |
2646 labelOffset + .75, | |
2647 0, | |
2648 nLabelOffsets[index] + .5, | |
2649 compressedRadii[index], | |
2650 compressedRadii[index + 1] | |
2651 ) | |
2652 ); | |
2653 } | |
2654 else | |
2655 { | |
2656 this.labelRadius.setTarget((compressedRadii[index] + compressedRadii[index + 1]) / 2); | |
2657 } | |
2658 } | |
2659 else | |
2660 { | |
2661 radiusCenter = | |
2662 nodeRadius * (depth - 1) + | |
2663 nodeRadius / 2; | |
2664 width = nodeRadius; | |
2665 | |
2666 this.labelRadius.setTarget | |
2667 ( | |
2668 radiusCenter + width * ((labelOffset + 1) / (nLabelOffsets[index] + 1) - .5) | |
2669 ); | |
2670 } | |
2671 } | |
2672 | |
2673 if ( ! this.hide && ! this.keyed && nLabelOffsets[index] ) | |
2674 { | |
2675 // check last and first labels in each track for overlap | |
2676 | |
2677 for ( var i = 0; i < maxDisplayDepth - 1; i++ ) | |
2678 { | |
2679 for ( var j = 0; j <= nLabelOffsets[i]; j++ ) | |
2680 { | |
2681 var last = labelLastNodes[i][j]; | |
2682 var first = labelFirstNodes[i][j]; | |
2683 | |
2684 if ( last ) | |
2685 { | |
2686 if ( j == nLabelOffsets[i] ) | |
2687 { | |
2688 // last is radial | |
2689 this.setLabelWidth(last); | |
2690 } | |
2691 else | |
2692 { | |
2693 last.setLabelWidth(this); | |
2694 } | |
2695 } | |
2696 | |
2697 if ( first ) | |
2698 { | |
2699 if ( j == nLabelOffsets[i] ) | |
2700 { | |
2701 this.setLabelWidth(first); | |
2702 } | |
2703 else | |
2704 { | |
2705 first.setLabelWidth(this); | |
2706 } | |
2707 } | |
2708 } | |
2709 } | |
2710 | |
2711 if ( selectedNode.canDisplayLabelOther ) | |
2712 { | |
2713 this.setLabelWidth(selectedNode); // in case there is an 'other' label | |
2714 } | |
2715 | |
2716 if ( this.radial ) | |
2717 { | |
2718 // use the last 'track' of this depth for radial | |
2719 | |
2720 labelLastNodes[index][nLabelOffsets[index]] = this; | |
2721 | |
2722 if ( labelFirstNodes[index][nLabelOffsets[index]] == 0 ) | |
2723 { | |
2724 labelFirstNodes[index][nLabelOffsets[index]] = this; | |
2725 } | |
2726 } | |
2727 else | |
2728 { | |
2729 labelLastNodes[index][labelOffset] = this; | |
2730 | |
2731 // update offset | |
2732 | |
2733 labelOffsets[index] += 1; | |
2734 | |
2735 if ( labelOffsets[index] > nLabelOffsets[index] ) | |
2736 { | |
2737 labelOffsets[index] -= nLabelOffsets[index]; | |
2738 | |
2739 if ( !(nLabelOffsets[index] & 1) ) | |
2740 { | |
2741 labelOffsets[index]--; | |
2742 } | |
2743 } | |
2744 else if ( labelOffsets[index] == nLabelOffsets[index] ) | |
2745 { | |
2746 labelOffsets[index] -= nLabelOffsets[index]; | |
2747 | |
2748 if ( false && !(nLabelOffsets[index] & 1) ) | |
2749 { | |
2750 labelOffsets[index]++; | |
2751 } | |
2752 } | |
2753 | |
2754 if ( labelFirstNodes[index][labelOffset] == 0 ) | |
2755 { | |
2756 labelFirstNodes[index][labelOffset] = this; | |
2757 } | |
2758 } | |
2759 } | |
2760 else if ( this.hide ) | |
2761 { | |
2762 this.labelWidth.end = 0; | |
2763 } | |
2764 } | |
2765 | |
2766 this.setTargets = function() | |
2767 { | |
2768 if ( this == selectedNode ) | |
2769 { | |
2770 this.setTargetsSelected | |
2771 ( | |
2772 0, | |
2773 1, | |
2774 lightnessBase, | |
2775 false, | |
2776 false | |
2777 ); | |
2778 return; | |
2779 } | |
2780 | |
2781 var depthRelative = this.getDepth() - selectedNode.getDepth(); | |
2782 | |
2783 var parentOfSelected = selectedNode.hasParent(this); | |
2784 /* ( | |
2785 // ! this.getCollapse() && | |
2786 this.baseMagnitude <= selectedNode.baseMagnitude && | |
2787 this.baseMagnitude + this.magnitude >= | |
2788 selectedNode.baseMagnitude + selectedNode.magnitude | |
2789 ); | |
2790 */ | |
2791 if ( parentOfSelected ) | |
2792 { | |
2793 this.resetLabelWidth(); | |
2794 } | |
2795 else | |
2796 { | |
2797 //context.font = fontNormal; | |
2798 var dim = context.measureText(this.name); | |
2799 this.nameWidth = dim.width; | |
2800 //this.labelWidth.setTarget(this.labelWidth.end); | |
2801 this.labelWidth.setTarget(0); | |
2802 } | |
2803 | |
2804 // set angles | |
2805 // | |
2806 if ( this.baseMagnitude <= selectedNode.baseMagnitude ) | |
2807 { | |
2808 this.angleStart.setTarget(0); | |
2809 } | |
2810 else | |
2811 { | |
2812 this.angleStart.setTarget(Math.PI * 2); | |
2813 } | |
2814 // | |
2815 if | |
2816 ( | |
2817 parentOfSelected || | |
2818 this.baseMagnitude + this.magnitude >= | |
2819 selectedNode.baseMagnitude + selectedNode.magnitude | |
2820 ) | |
2821 { | |
2822 this.angleEnd.setTarget(Math.PI * 2); | |
2823 } | |
2824 else | |
2825 { | |
2826 this.angleEnd.setTarget(0); | |
2827 } | |
2828 | |
2829 // children | |
2830 // | |
2831 for ( var i = 0; i < this.children.length; i++ ) | |
2832 { | |
2833 this.children[i].setTargets(); | |
2834 } | |
2835 | |
2836 if ( this.getDepth() <= selectedNode.getDepth() ) | |
2837 { | |
2838 // collapse in | |
2839 | |
2840 this.radiusInner.setTarget(0); | |
2841 | |
2842 if ( parentOfSelected ) | |
2843 { | |
2844 this.labelRadius.setTarget | |
2845 ( | |
2846 (depthRelative) * | |
2847 historySpacingFactor * fontSize / gRadius | |
2848 ); | |
2849 //this.scale.setTarget(1 - (selectedNode.getDepth() - this.getDepth()) / 18); // TEMP | |
2850 } | |
2851 else | |
2852 { | |
2853 this.labelRadius.setTarget(0); | |
2854 //this.scale.setTarget(1); // TEMP | |
2855 } | |
2856 } | |
2857 else if ( depthRelative + 1 > maxDisplayDepth ) | |
2858 { | |
2859 // collapse out | |
2860 | |
2861 this.radiusInner.setTarget(1); | |
2862 this.labelRadius.setTarget(1); | |
2863 //this.scale.setTarget(1); // TEMP | |
2864 } | |
2865 else | |
2866 { | |
2867 // don't collapse | |
2868 | |
2869 if ( compress ) | |
2870 { | |
2871 this.radiusInner.setTarget(compressedRadii[depthRelative - 1]); | |
2872 } | |
2873 else | |
2874 { | |
2875 this.radiusInner.setTarget(nodeRadius * (depthRelative)); | |
2876 } | |
2877 | |
2878 //this.scale.setTarget(1); // TEMP | |
2879 | |
2880 if ( this == selectedNode ) | |
2881 { | |
2882 this.labelRadius.setTarget(0); | |
2883 } | |
2884 else | |
2885 { | |
2886 if ( compress ) | |
2887 { | |
2888 this.labelRadius.setTarget | |
2889 ( | |
2890 (compressedRadii[depthRelative - 1] + compressedRadii[depthRelative]) / 2 | |
2891 ); | |
2892 } | |
2893 else | |
2894 { | |
2895 this.labelRadius.setTarget(nodeRadius * (depthRelative) + nodeRadius / 2); | |
2896 } | |
2897 } | |
2898 } | |
2899 | |
2900 // this.r.start = this.r.end; | |
2901 // this.g.start = this.g.end; | |
2902 // this.b.start = this.b.end; | |
2903 | |
2904 this.r.setTarget(255); | |
2905 this.g.setTarget(255); | |
2906 this.b.setTarget(255); | |
2907 | |
2908 this.alphaLine.setTarget(0); | |
2909 this.alphaArc.setTarget(0); | |
2910 this.alphaWedge.setTarget(0); | |
2911 this.alphaPattern.setTarget(0); | |
2912 this.alphaOther.setTarget(0); | |
2913 | |
2914 if ( parentOfSelected && ! this.getCollapse() ) | |
2915 { | |
2916 var alpha = | |
2917 ( | |
2918 1 - | |
2919 (selectedNode.getDepth() - this.getDepth()) / | |
2920 (Math.floor((compress ? compressedRadii[0] : nodeRadius) * gRadius / (historySpacingFactor * fontSize) - .5) + 1) | |
2921 ); | |
2922 | |
2923 if ( alpha < 0 ) | |
2924 { | |
2925 alpha = 0; | |
2926 } | |
2927 | |
2928 this.alphaLabel.setTarget(alpha); | |
2929 this.radial = false; | |
2930 } | |
2931 else | |
2932 { | |
2933 this.alphaLabel.setTarget(0); | |
2934 } | |
2935 | |
2936 this.hideAlonePrev = this.hideAlone; | |
2937 this.hidePrev = this.hide; | |
2938 | |
2939 if ( parentOfSelected ) | |
2940 { | |
2941 this.hideAlone = false; | |
2942 this.hide = false; | |
2943 } | |
2944 | |
2945 if ( this.getParent() == selectedNode.getParent() ) | |
2946 { | |
2947 this.hiddenEnd = null; | |
2948 } | |
2949 | |
2950 this.radialPrev = this.radial; | |
2951 } | |
2952 | |
2953 this.setTargetsSelected = function(hueMin, hueMax, lightness, hide, nextSiblingHidden) | |
2954 { | |
2955 var collapse = this.getCollapse(); | |
2956 var depth = this.getDepth() - selectedNode.getDepth() + 1; | |
2957 var canDisplayChildLabels = false; | |
2958 var lastChild; | |
2959 | |
2960 if ( this.hasChildren() )//&& ! hide ) | |
2961 { | |
2962 lastChild = this.children[this.children.length - 1]; | |
2963 this.hideAlone = true; | |
2964 } | |
2965 else | |
2966 { | |
2967 this.hideAlone = false; | |
2968 } | |
2969 | |
2970 // set child wedges | |
2971 // | |
2972 for ( var i = 0; i < this.children.length; i++ ) | |
2973 { | |
2974 this.children[i].setTargetWedge(); | |
2975 | |
2976 if | |
2977 ( | |
2978 ! this.children[i].hide && | |
2979 ( collapse || depth < maxDisplayDepth ) && | |
2980 this.depth < maxAbsoluteDepth | |
2981 ) | |
2982 { | |
2983 canDisplayChildLabels = true; | |
2984 this.hideAlone = false; | |
2985 } | |
2986 } | |
2987 | |
2988 if ( this == selectedNode || lastChild && lastChild.angleEnd.end < this.angleEnd.end - .01) | |
2989 { | |
2990 this.hideAlone = false; | |
2991 } | |
2992 | |
2993 if ( this.hideAlonePrev == undefined ) | |
2994 { | |
2995 this.hideAlonePrev = this.hideAlone; | |
2996 } | |
2997 | |
2998 if ( this == selectedNode ) | |
2999 { | |
3000 var otherArc = | |
3001 this.children.length ? | |
3002 angleFactor * | |
3003 ( | |
3004 this.baseMagnitude + this.magnitude - | |
3005 lastChild.baseMagnitude - lastChild.magnitude | |
3006 ) | |
3007 : this.baseMagnitude + this.magnitude; | |
3008 this.canDisplayLabelOther = | |
3009 this.children.length ? | |
3010 otherArc * | |
3011 (this.children[0].radiusInner.end + 1) * gRadius >= | |
3012 minWidth() | |
3013 : true; | |
3014 | |
3015 this.keyUnclassified = false; | |
3016 | |
3017 if ( this.canDisplayLabelOther ) | |
3018 { | |
3019 this.angleOther = Math.PI * 2 - otherArc / 2; | |
3020 } | |
3021 else if ( otherArc > 0.0000000001 ) | |
3022 { | |
3023 this.keyUnclassified = true; | |
3024 keys++; | |
3025 } | |
3026 | |
3027 this.angleStart.setTarget(0); | |
3028 this.angleEnd.setTarget(Math.PI * 2); | |
3029 | |
3030 if ( this.children.length ) | |
3031 { | |
3032 this.radiusInner.setTarget(0); | |
3033 } | |
3034 else | |
3035 { | |
3036 this.radiusInner.setTarget(compressedRadii[0]); | |
3037 } | |
3038 | |
3039 this.hidePrev = this.hide; | |
3040 this.hide = false; | |
3041 this.hideAlonePrev = this.hideAlone; | |
3042 this.hideAlone = false; | |
3043 this.keyed = false; | |
3044 } | |
3045 | |
3046 if ( hueMax - hueMin > 1 / 12 ) | |
3047 { | |
3048 hueMax = hueMin + 1 / 12; | |
3049 } | |
3050 | |
3051 // set lightness | |
3052 // | |
3053 if ( ! ( hide || this.hideAlone ) ) | |
3054 { | |
3055 if ( useHue() ) | |
3056 { | |
3057 lightness = (lightnessBase + lightnessMax) / 2; | |
3058 } | |
3059 else | |
3060 { | |
3061 lightness = lightnessBase + (depth - 1) * lightnessFactor; | |
3062 | |
3063 if ( lightness > lightnessMax ) | |
3064 { | |
3065 lightness = lightnessMax; | |
3066 } | |
3067 } | |
3068 } | |
3069 | |
3070 if ( hide ) | |
3071 { | |
3072 this.hide = true; | |
3073 } | |
3074 | |
3075 if ( this.hidePrev == undefined ) | |
3076 { | |
3077 this.hidePrev = this.hide; | |
3078 } | |
3079 | |
3080 var hiddenStart = -1; | |
3081 var hiddenHueNumer = 0; | |
3082 var hiddenHueDenom = 0; | |
3083 var i = 0; | |
3084 | |
3085 if ( ! this.hide ) | |
3086 { | |
3087 this.hiddenEnd = null; | |
3088 } | |
3089 | |
3090 while ( true ) | |
3091 { | |
3092 if ( ! this.hideAlone && ! hide && ( i == this.children.length || ! this.children[i].hide ) ) | |
3093 { | |
3094 // reached a non-hidden child or the end; set targets for | |
3095 // previous group of hidden children (if any) using their | |
3096 // average hue | |
3097 | |
3098 if ( hiddenStart != -1 ) | |
3099 { | |
3100 var hiddenHue = hiddenHueDenom ? hiddenHueNumer / hiddenHueDenom : hueMin; | |
3101 | |
3102 for ( var j = hiddenStart; j < i; j++ ) | |
3103 { | |
3104 this.children[j].setTargetsSelected | |
3105 ( | |
3106 hiddenHue, | |
3107 null, | |
3108 lightness, | |
3109 false, | |
3110 j < i - 1 | |
3111 ); | |
3112 | |
3113 this.children[j].hiddenEnd = null; | |
3114 } | |
3115 | |
3116 this.children[hiddenStart].hiddenEnd = i - 1; | |
3117 } | |
3118 } | |
3119 | |
3120 if ( i == this.children.length ) | |
3121 { | |
3122 break; | |
3123 } | |
3124 | |
3125 var child = this.children[i]; | |
3126 var childHueMin; | |
3127 var childHueMax; | |
3128 | |
3129 if ( this.magnitude > 0 && ! this.hide && ! this.hideAlone ) | |
3130 { | |
3131 if ( useHue() ) | |
3132 { | |
3133 childHueMin = child.hues[currentDataset]; | |
3134 } | |
3135 else if ( this == selectedNode ) | |
3136 { | |
3137 var min = 0.0; | |
3138 var max = 1.0; | |
3139 | |
3140 if ( this.children.length > 6 ) | |
3141 { | |
3142 childHueMin = lerp((1 - Math.pow(1 - i / this.children.length, 1.4)) * .95, 0, 1, min, max); | |
3143 childHueMax = lerp((1 - Math.pow(1 - (i + .55) / this.children.length, 1.4)) * .95, 0, 1, min, max); | |
3144 } | |
3145 else | |
3146 { | |
3147 childHueMin = lerp(i / this.children.length, 0, 1, min, max); | |
3148 childHueMax = lerp((i + .55) / this.children.length, 0, 1, min, max); | |
3149 } | |
3150 } | |
3151 else | |
3152 { | |
3153 childHueMin = lerp | |
3154 ( | |
3155 child.baseMagnitude, | |
3156 this.baseMagnitude, | |
3157 this.baseMagnitude + this.magnitude, | |
3158 hueMin, | |
3159 hueMax | |
3160 ); | |
3161 childHueMax = lerp | |
3162 ( | |
3163 child.baseMagnitude + child.magnitude * .99, | |
3164 this.baseMagnitude, | |
3165 this.baseMagnitude + this.magnitude, | |
3166 hueMin, | |
3167 hueMax | |
3168 ); | |
3169 } | |
3170 } | |
3171 else | |
3172 { | |
3173 childHueMin = hueMin; | |
3174 childHueMax = hueMax; | |
3175 } | |
3176 | |
3177 if ( ! this.hideAlone && ! hide && ! this.hide && child.hide ) | |
3178 { | |
3179 if ( hiddenStart == -1 ) | |
3180 { | |
3181 hiddenStart = i; | |
3182 } | |
3183 | |
3184 if ( useHue() ) | |
3185 { | |
3186 hiddenHueNumer += childHueMin * child.magnitude; | |
3187 hiddenHueDenom += child.magnitude; | |
3188 } | |
3189 else | |
3190 { | |
3191 hiddenHueNumer += childHueMin; | |
3192 hiddenHueDenom++; | |
3193 } | |
3194 } | |
3195 else | |
3196 { | |
3197 hiddenStart = -1; | |
3198 | |
3199 this.children[i].setTargetsSelected | |
3200 ( | |
3201 childHueMin, | |
3202 childHueMax, | |
3203 lightness, | |
3204 hide || this.keyed || this.hideAlone || this.hide && ! collapse, | |
3205 false | |
3206 ); | |
3207 } | |
3208 | |
3209 i++; | |
3210 } | |
3211 | |
3212 if ( this.hue && this.magnitude ) | |
3213 { | |
3214 this.hue.setTarget(this.hues[currentDataset]); | |
3215 | |
3216 if ( this.attributes[magnitudeIndex][lastDataset] == 0 ) | |
3217 { | |
3218 this.hue.start = this.hue.end; | |
3219 } | |
3220 } | |
3221 | |
3222 this.radialPrev = this.radial; | |
3223 | |
3224 if ( this == selectedNode ) | |
3225 { | |
3226 this.resetLabelWidth(); | |
3227 this.labelWidth.setTarget(this.nameWidth * labelWidthFudge); | |
3228 this.alphaWedge.setTarget(0); | |
3229 this.alphaLabel.setTarget(1); | |
3230 this.alphaOther.setTarget(1); | |
3231 this.alphaArc.setTarget(0); | |
3232 this.alphaLine.setTarget(0); | |
3233 this.alphaPattern.setTarget(0); | |
3234 this.r.setTarget(255); | |
3235 this.g.setTarget(255); | |
3236 this.b.setTarget(255); | |
3237 this.radial = false; | |
3238 this.labelRadius.setTarget(0); | |
3239 } | |
3240 else | |
3241 { | |
3242 var rgb = hslToRgb | |
3243 ( | |
3244 hueMin, | |
3245 saturation, | |
3246 lightness | |
3247 ); | |
3248 | |
3249 this.r.setTarget(rgb.r); | |
3250 this.g.setTarget(rgb.g); | |
3251 this.b.setTarget(rgb.b); | |
3252 this.alphaOther.setTarget(0); | |
3253 | |
3254 this.alphaWedge.setTarget(1); | |
3255 | |
3256 if ( this.hide || this.hideAlone ) | |
3257 { | |
3258 this.alphaPattern.setTarget(1); | |
3259 } | |
3260 else | |
3261 { | |
3262 this.alphaPattern.setTarget(0); | |
3263 } | |
3264 | |
3265 // set radial | |
3266 // | |
3267 if ( ! ( hide || this.hide ) )//&& ! this.keyed ) | |
3268 { | |
3269 if ( this.hideAlone ) | |
3270 { | |
3271 this.radial = true; | |
3272 } | |
3273 else if ( false && canDisplayChildLabels ) | |
3274 { | |
3275 this.radial = false; | |
3276 } | |
3277 else | |
3278 { | |
3279 this.radial = true; | |
3280 | |
3281 if ( this.hasChildren() && depth < maxDisplayDepth ) | |
3282 { | |
3283 var lastChild = this.children[this.children.length - 1]; | |
3284 | |
3285 if | |
3286 ( | |
3287 lastChild.angleEnd.end == this.angleEnd.end || | |
3288 ( | |
3289 (this.angleStart.end + this.angleEnd.end) / 2 - | |
3290 lastChild.angleEnd.end | |
3291 ) * (this.radiusInner.end + 1) * gRadius * 2 < | |
3292 minWidth() | |
3293 ) | |
3294 { | |
3295 this.radial = false; | |
3296 } | |
3297 } | |
3298 } | |
3299 } | |
3300 | |
3301 // set alphaLabel | |
3302 // | |
3303 if | |
3304 ( | |
3305 collapse || | |
3306 hide || | |
3307 this.hide || | |
3308 this.keyed || | |
3309 depth > maxDisplayDepth || | |
3310 ! this.canDisplayDepth() | |
3311 ) | |
3312 { | |
3313 this.alphaLabel.setTarget(0); | |
3314 } | |
3315 else | |
3316 { | |
3317 if | |
3318 ( | |
3319 (this.radial || nLabelOffsets[depth - 2]) | |
3320 ) | |
3321 { | |
3322 this.alphaLabel.setTarget(1); | |
3323 } | |
3324 else | |
3325 { | |
3326 this.alphaLabel.setTarget(0); | |
3327 | |
3328 if ( this.radialPrev ) | |
3329 { | |
3330 this.alphaLabel.start = 0; | |
3331 } | |
3332 } | |
3333 } | |
3334 | |
3335 // set alphaArc | |
3336 // | |
3337 if | |
3338 ( | |
3339 collapse || | |
3340 hide || | |
3341 depth > maxDisplayDepth || | |
3342 ! this.canDisplayDepth() | |
3343 ) | |
3344 { | |
3345 this.alphaArc.setTarget(0); | |
3346 } | |
3347 else | |
3348 { | |
3349 this.alphaArc.setTarget(1); | |
3350 } | |
3351 | |
3352 // set alphaLine | |
3353 // | |
3354 if | |
3355 ( | |
3356 hide || | |
3357 this.hide && nextSiblingHidden || | |
3358 depth > maxDisplayDepth || | |
3359 ! this.canDisplayDepth() | |
3360 ) | |
3361 { | |
3362 this.alphaLine.setTarget(0); | |
3363 } | |
3364 else | |
3365 { | |
3366 this.alphaLine.setTarget(1); | |
3367 } | |
3368 | |
3369 //if ( ! this.radial ) | |
3370 { | |
3371 this.resetLabelWidth(); | |
3372 } | |
3373 | |
3374 // set labelRadius target | |
3375 // | |
3376 if ( collapse ) | |
3377 { | |
3378 this.labelRadius.setTarget(this.radiusInner.end); | |
3379 } | |
3380 else | |
3381 { | |
3382 if ( depth > maxDisplayDepth || ! this.canDisplayDepth() ) | |
3383 { | |
3384 this.labelRadius.setTarget(1); | |
3385 } | |
3386 else | |
3387 { | |
3388 this.setTargetLabelRadius(); | |
3389 } | |
3390 } | |
3391 } | |
3392 } | |
3393 | |
3394 this.setTargetWedge = function() | |
3395 { | |
3396 var depth = this.getDepth() - selectedNode.getDepth() + 1; | |
3397 | |
3398 // set angles | |
3399 // | |
3400 var baseMagnitudeRelative = this.baseMagnitude - selectedNode.baseMagnitude; | |
3401 // | |
3402 this.angleStart.setTarget(baseMagnitudeRelative * angleFactor); | |
3403 this.angleEnd.setTarget((baseMagnitudeRelative + this.magnitude) * angleFactor); | |
3404 | |
3405 // set radiusInner | |
3406 // | |
3407 if ( depth > maxDisplayDepth || ! this.canDisplayDepth() ) | |
3408 { | |
3409 this.radiusInner.setTarget(1); | |
3410 } | |
3411 else | |
3412 { | |
3413 if ( compress ) | |
3414 { | |
3415 this.radiusInner.setTarget(compressedRadii[depth - 2]); | |
3416 } | |
3417 else | |
3418 { | |
3419 this.radiusInner.setTarget(nodeRadius * (depth - 1)); | |
3420 } | |
3421 } | |
3422 | |
3423 if ( this.hide != undefined ) | |
3424 { | |
3425 this.hidePrev = this.hide; | |
3426 } | |
3427 | |
3428 if ( this.hideAlone != undefined ) | |
3429 { | |
3430 this.hideAlonePrev = this.hideAlone; | |
3431 } | |
3432 | |
3433 // set hide | |
3434 // | |
3435 if | |
3436 ( | |
3437 (this.angleEnd.end - this.angleStart.end) * | |
3438 (this.radiusInner.end * gRadius + gRadius) < | |
3439 minWidth() | |
3440 ) | |
3441 { | |
3442 if ( depth == 2 && ! this.getCollapse() && this.depth <= maxAbsoluteDepth ) | |
3443 { | |
3444 this.keyed = true; | |
3445 keys++; | |
3446 this.hide = false; | |
3447 | |
3448 var percentage = this.getPercentage(); | |
3449 this.keyLabel = this.name + ' ' + percentage + '%'; | |
3450 var dim = context.measureText(this.keyLabel); | |
3451 this.keyNameWidth = dim.width; | |
3452 } | |
3453 else | |
3454 { | |
3455 this.keyed = false; | |
3456 this.hide = depth > 2; | |
3457 } | |
3458 } | |
3459 else | |
3460 { | |
3461 this.keyed = false; | |
3462 this.hide = false; | |
3463 } | |
3464 } | |
3465 | |
3466 this.shortenLabel = function() | |
3467 { | |
3468 var label = this.name; | |
3469 | |
3470 var labelWidth = this.nameWidth; | |
3471 var maxWidth = this.labelWidth.current(); | |
3472 var minEndLength = 0; | |
3473 | |
3474 if ( labelWidth > maxWidth && label.length > minEndLength * 2 ) | |
3475 { | |
3476 var endLength = | |
3477 Math.floor((label.length - 1) * maxWidth / labelWidth / 2); | |
3478 | |
3479 if ( endLength < minEndLength ) | |
3480 { | |
3481 endLength = minEndLength; | |
3482 } | |
3483 | |
3484 return ( | |
3485 label.substring(0, endLength) + | |
3486 '...' + | |
3487 label.substring(label.length - endLength)); | |
3488 } | |
3489 else | |
3490 { | |
3491 return label; | |
3492 } | |
3493 } | |
3494 | |
3495 /* this.shouldAddSearchResultsString = function() | |
3496 { | |
3497 if ( this.isSearchResult ) | |
3498 { | |
3499 return this.searchResults > 1; | |
3500 } | |
3501 else | |
3502 { | |
3503 return this.searchResults > 0; | |
3504 } | |
3505 } | |
3506 */ | |
3507 this.sort = function() | |
3508 { | |
3509 this.children.sort(function(a, b){return b.getMagnitude() - a.getMagnitude()}); | |
3510 | |
3511 for (var i = 0; i < this.children.length; i++) | |
3512 { | |
3513 this.children[i].sort(); | |
3514 } | |
3515 } | |
3516 } | |
3517 | |
3518 var options; | |
3519 | |
3520 function addOptionElement(position, innerHTML, title) | |
3521 { | |
3522 var div = document.createElement("div"); | |
3523 // div.style.position = 'absolute'; | |
3524 // div.style.top = position + 'px'; | |
3525 div.innerHTML = innerHTML; | |
3526 // div.style.display = 'block'; | |
3527 div.style.padding = '2px'; | |
3528 | |
3529 if ( title ) | |
3530 { | |
3531 div.title = title; | |
3532 } | |
3533 | |
3534 options.appendChild(div); | |
3535 var height = 0;//div.clientHeight; | |
3536 return position + height; | |
3537 } | |
3538 | |
3539 function addOptionElements(hueName, hueDefault) | |
3540 { | |
3541 options = document.createElement('div'); | |
3542 options.style.position = 'absolute'; | |
3543 options.style.top = '0px'; | |
3544 options.addEventListener('mousedown', function(e) {mouseClick(e)}, false); | |
3545 // options.onmouseup = function(e) {mouseUp(e)} | |
3546 document.body.appendChild(options); | |
3547 | |
3548 document.body.style.font = '11px sans-serif'; | |
3549 var position = 5; | |
3550 | |
3551 details = document.createElement('div'); | |
3552 details.style.position = 'absolute'; | |
3553 details.style.top = '1%'; | |
3554 details.style.right = '2%'; | |
3555 details.style.textAlign = 'right'; | |
3556 document.body.insertBefore(details, canvas); | |
3557 // <div id="details" style="position:absolute;top:1%;right:2%;text-align:right;"> | |
3558 | |
3559 details.innerHTML = '\ | |
3560 <span id="detailsName" style="font-weight:bold"></span> \ | |
3561 <input type="button" id="detailsExpand" onclick="expand(focusNode);"\ | |
3562 value="↔" title="Expand this wedge to become the new focus of the chart"/><br/>\ | |
3563 <div id="detailsInfo" style="float:right"></div>'; | |
3564 | |
3565 keyControl = document.createElement('input'); | |
3566 keyControl.type = 'button'; | |
3567 keyControl.value = showKeys ? 'x' : '…'; | |
3568 keyControl.style.position = ''; | |
3569 keyControl.style.position = 'fixed'; | |
3570 keyControl.style.visibility = 'hidden'; | |
3571 | |
3572 document.body.insertBefore(keyControl, canvas); | |
3573 | |
3574 var logoElement = document.getElementById('logo'); | |
3575 | |
3576 if ( logoElement ) | |
3577 { | |
3578 logoImage = logoElement.src; | |
3579 } | |
3580 else | |
3581 { | |
3582 logoImage = 'http://marbl.github.io/Krona/img/logo-med.png'; | |
3583 } | |
3584 | |
3585 // document.getElementById('options').style.fontSize = '9pt'; | |
3586 position = addOptionElement | |
3587 ( | |
3588 position, | |
3589 '<a style="margin:2px" target="_blank" href="https://github.com/marbl/Krona/wiki"><img style="vertical-align:middle;width:108px;height:30px;" src="' + logoImage + '" alt="Logo of Krona"/></a><input type="button" id="back" value="←" title="Go back (Shortcut: ←)"/>\ | |
3590 <input type="button" id="forward" value="→" title="Go forward (Shortcut: →)"/> \ | |
3591 Search: <input type="text" id="search"/>\ | |
3592 <input id="searchClear" type="button" value="x" onclick="clearSearch()"/> \ | |
3593 <span id="searchResults"></span>' | |
3594 ); | |
3595 | |
3596 if ( datasets > 1 ) | |
3597 { | |
3598 var size = datasets < datasetSelectSize ? datasets : datasetSelectSize; | |
3599 | |
3600 var select = | |
3601 '<table style="border-collapse:collapse;padding:0px"><tr><td style="padding:0px">' + | |
3602 '<select id="datasets" style="min-width:100px" size="' + size + '" onchange="onDatasetChange()">'; | |
3603 | |
3604 for ( var i = 0; i < datasetNames.length; i++ ) | |
3605 { | |
3606 select += '<option>' + datasetNames[i] + '</option>'; | |
3607 } | |
3608 | |
3609 select += | |
3610 '</select></td><td style="vertical-align:top;padding:1px;">' + | |
3611 '<input style="display:block" title="Previous dataset (Shortcut: ↑)" id="prevDataset" type="button" value="↑" onclick="prevDataset()" disabled="true"/>' + | |
3612 '<input title="Next dataset (Shortcut: ↓)" id="nextDataset" type="button" value="↓" onclick="nextDataset()"/><br/></td>' + | |
3613 '<td style="padding-top:1px;vertical-align:top"><input title="Switch to the last dataset that was viewed (Shortcut: TAB)" id="lastDataset" type="button" style="font:11px Times new roman" value="last" onclick="selectLastDataset()"/></td></tr></table>'; | |
3614 | |
3615 position = addOptionElement(position + 5, select); | |
3616 | |
3617 datasetDropDown = document.getElementById('datasets'); | |
3618 datasetButtonLast = document.getElementById('lastDataset'); | |
3619 datasetButtonPrev = document.getElementById('prevDataset'); | |
3620 datasetButtonNext = document.getElementById('nextDataset'); | |
3621 | |
3622 position += datasetDropDown.clientHeight; | |
3623 } | |
3624 | |
3625 position = addOptionElement | |
3626 ( | |
3627 position + 5, | |
3628 '<input type="button" id="maxAbsoluteDepthDecrease" value="-"/>\ | |
3629 <span id="maxAbsoluteDepth"></span>\ | |
3630 <input type="button" id="maxAbsoluteDepthIncrease" value="+"/> Max depth', | |
3631 'Maximum depth to display, counted from the top level \ | |
3632 and including collapsed wedges.' | |
3633 ); | |
3634 | |
3635 position = addOptionElement | |
3636 ( | |
3637 position, | |
3638 '<input type="button" id="fontSizeDecrease" value="-"/>\ | |
3639 <span id="fontSize"></span>\ | |
3640 <input type="button" id="fontSizeIncrease" value="+"/> Font size' | |
3641 ); | |
3642 | |
3643 position = addOptionElement | |
3644 ( | |
3645 position, | |
3646 '<input type="button" id="radiusDecrease" value="-"/>\ | |
3647 <input type="button" id="radiusIncrease" value="+"/> Chart size' | |
3648 ); | |
3649 | |
3650 if ( hueName ) | |
3651 { | |
3652 hueDisplayName = attributes[attributeIndex(hueName)].displayName; | |
3653 | |
3654 position = addOptionElement | |
3655 ( | |
3656 position + 5, | |
3657 '<input type="checkbox" id="useHue" style="float:left" ' + | |
3658 '/><div>Color by<br/>' + hueDisplayName + | |
3659 '</div>' | |
3660 ); | |
3661 | |
3662 useHueCheckBox = document.getElementById('useHue'); | |
3663 useHueCheckBox.checked = hueDefault; | |
3664 useHueCheckBox.onclick = handleResize; | |
3665 useHueCheckBox.onmousedown = suppressEvent; | |
3666 } | |
3667 /* | |
3668 position = addOptionElement | |
3669 ( | |
3670 position + 5, | |
3671 ' <input type="checkbox" id="shorten" checked="checked" />Shorten labels</div>', | |
3672 'Prevent labels from overlapping by shortening them' | |
3673 ); | |
3674 | |
3675 position = addOptionElement | |
3676 ( | |
3677 position, | |
3678 ' <input type="checkbox" id="compress" checked="checked" />Compress', | |
3679 'Compress wedges if needed to show the entire depth' | |
3680 ); | |
3681 */ | |
3682 position = addOptionElement | |
3683 ( | |
3684 position, | |
3685 '<input type="checkbox" id="collapse" checked="checked" />Collapse', | |
3686 'Collapse wedges that are redundant (entirely composed of another wedge)' | |
3687 ); | |
3688 | |
3689 position = addOptionElement | |
3690 ( | |
3691 position + 5, | |
3692 '<input type="button" id="snapshot" value="Snapshot"/>', | |
3693 'Render the current view as SVG (Scalable Vector Graphics), a publication-\ | |
3694 quality format that can be printed and saved (see Help for browser compatibility)' | |
3695 ); | |
3696 | |
3697 position = addOptionElement | |
3698 ( | |
3699 position + 5, | |
3700 '<input type="button" id="linkButton" value="Link"/>\ | |
3701 <input type="text" size="30" id="linkText"/>', | |
3702 'Show a link to this view that can be copied for bookmarking or sharing' | |
3703 ); | |
3704 | |
3705 position = addOptionElement | |
3706 ( | |
3707 position + 5, | |
3708 '<input type="button" id="help" value="?"\ | |
3709 onclick="window.open(\'https://github.com/marbl/Krona/wiki/Browsing%20Krona%20charts\', \'help\')"/>', | |
3710 'Help' | |
3711 ); | |
3712 } | |
3713 | |
3714 function arrow(angleStart, angleEnd, radiusInner) | |
3715 { | |
3716 if ( context.globalAlpha == 0 ) | |
3717 { | |
3718 return; | |
3719 } | |
3720 | |
3721 var angleCenter = (angleStart + angleEnd) / 2; | |
3722 var radiusArrowInner = radiusInner - gRadius / 10;//nodeRadius * gRadius; | |
3723 var radiusArrowOuter = gRadius * 1.1;//(1 + nodeRadius); | |
3724 var radiusArrowCenter = (radiusArrowInner + radiusArrowOuter) / 2; | |
3725 var pointLength = (radiusArrowOuter - radiusArrowInner) / 5; | |
3726 | |
3727 context.fillStyle = highlightFill; | |
3728 context.lineWidth = highlightLineWidth; | |
3729 | |
3730 // First, mask out the first half of the arrow. This will prevent the tips | |
3731 // from superimposing if the arrow goes most of the way around the circle. | |
3732 // Masking is done by setting the clipping region to the inverse of the | |
3733 // half-arrow, which is defined by cutting the half-arrow out of a large | |
3734 // rectangle | |
3735 // | |
3736 context.beginPath(); | |
3737 context.arc(0, 0, radiusInner, angleCenter, angleEnd, false); | |
3738 context.lineTo | |
3739 ( | |
3740 radiusArrowInner * Math.cos(angleEnd), | |
3741 radiusArrowInner * Math.sin(angleEnd) | |
3742 ); | |
3743 context.lineTo | |
3744 ( | |
3745 radiusArrowCenter * Math.cos(angleEnd) - pointLength * Math.sin(angleEnd), | |
3746 radiusArrowCenter * Math.sin(angleEnd) + pointLength * Math.cos(angleEnd) | |
3747 ); | |
3748 context.lineTo | |
3749 ( | |
3750 radiusArrowOuter * Math.cos(angleEnd), | |
3751 radiusArrowOuter * Math.sin(angleEnd) | |
3752 ); | |
3753 context.arc(0, 0, gRadius, angleEnd, angleCenter, true); | |
3754 context.closePath(); | |
3755 context.moveTo(-imageWidth, -imageHeight); | |
3756 context.lineTo(imageWidth, -imageHeight); | |
3757 context.lineTo(imageWidth, imageHeight); | |
3758 context.lineTo(-imageWidth, imageHeight); | |
3759 context.closePath(); | |
3760 context.save(); | |
3761 context.clip(); | |
3762 | |
3763 // Next, draw the other half-arrow with the first half masked out | |
3764 // | |
3765 context.beginPath(); | |
3766 context.arc(0, 0, radiusInner, angleCenter, angleStart, true); | |
3767 context.lineTo | |
3768 ( | |
3769 radiusArrowInner * Math.cos(angleStart), | |
3770 radiusArrowInner * Math.sin(angleStart) | |
3771 ); | |
3772 context.lineTo | |
3773 ( | |
3774 radiusArrowCenter * Math.cos(angleStart) + pointLength * Math.sin(angleStart), | |
3775 radiusArrowCenter * Math.sin(angleStart) - pointLength * Math.cos(angleStart) | |
3776 ); | |
3777 context.lineTo | |
3778 ( | |
3779 radiusArrowOuter * Math.cos(angleStart), | |
3780 radiusArrowOuter * Math.sin(angleStart) | |
3781 ); | |
3782 context.arc(0, 0, gRadius, angleStart, angleCenter, false); | |
3783 context.fill(); | |
3784 context.stroke(); | |
3785 | |
3786 // Finally, remove the clipping region and draw the first half-arrow. This | |
3787 // half is extended slightly to fill the seam. | |
3788 // | |
3789 context.restore(); | |
3790 context.beginPath(); | |
3791 context.arc(0, 0, radiusInner, angleCenter - 2 / (2 * Math.PI * radiusInner), angleEnd, false); | |
3792 context.lineTo | |
3793 ( | |
3794 radiusArrowInner * Math.cos(angleEnd), | |
3795 radiusArrowInner * Math.sin(angleEnd) | |
3796 ); | |
3797 context.lineTo | |
3798 ( | |
3799 radiusArrowCenter * Math.cos(angleEnd) - pointLength * Math.sin(angleEnd), | |
3800 radiusArrowCenter * Math.sin(angleEnd) + pointLength * Math.cos(angleEnd) | |
3801 ); | |
3802 context.lineTo | |
3803 ( | |
3804 radiusArrowOuter * Math.cos(angleEnd), | |
3805 radiusArrowOuter * Math.sin(angleEnd) | |
3806 ); | |
3807 context.arc(0, 0, gRadius, angleEnd, angleCenter - 2 / (2 * Math.PI * gRadius), true); | |
3808 context.fill(); | |
3809 context.stroke(); | |
3810 } | |
3811 | |
3812 function attributeIndex(aname) | |
3813 { | |
3814 for ( var i = 0 ; i < attributes.length; i++ ) | |
3815 { | |
3816 if ( aname == attributes[i].name ) | |
3817 { | |
3818 return i; | |
3819 } | |
3820 } | |
3821 | |
3822 return null; | |
3823 } | |
3824 | |
3825 function checkHighlight() | |
3826 { | |
3827 var lastHighlightedNode = highlightedNode; | |
3828 var lastHighlightingHidden = highlightingHidden; | |
3829 | |
3830 highlightedNode = selectedNode; | |
3831 resetKeyOffset(); | |
3832 | |
3833 if ( progress == 1 ) | |
3834 { | |
3835 selectedNode.checkHighlight(); | |
3836 if ( selectedNode.getParent() ) | |
3837 { | |
3838 selectedNode.getParent().checkHighlightCenter(); | |
3839 } | |
3840 | |
3841 focusNode.checkHighlightMap(); | |
3842 } | |
3843 | |
3844 if ( highlightedNode != selectedNode ) | |
3845 { | |
3846 if ( highlightedNode == focusNode ) | |
3847 { | |
3848 // canvas.style.display='none'; | |
3849 // window.resizeBy(1,0); | |
3850 // canvas.style.cursor='ew-resize'; | |
3851 // window.resizeBy(-1,0); | |
3852 // canvas.style.display='inline'; | |
3853 } | |
3854 else | |
3855 { | |
3856 // canvas.style.cursor='pointer'; | |
3857 } | |
3858 } | |
3859 else | |
3860 { | |
3861 // canvas.style.cursor='auto'; | |
3862 } | |
3863 | |
3864 if | |
3865 ( | |
3866 ( | |
3867 true || | |
3868 highlightedNode != lastHighlightedNode || | |
3869 highlightingHidden != highlightingHiddenLast | |
3870 ) && | |
3871 progress == 1 | |
3872 ) | |
3873 { | |
3874 draw(); // TODO: handle in update() | |
3875 } | |
3876 } | |
3877 | |
3878 function checkSelectedCollapse() | |
3879 { | |
3880 var newNode = selectedNode; | |
3881 | |
3882 while ( newNode.getCollapse() ) | |
3883 { | |
3884 newNode = newNode.children[0]; | |
3885 } | |
3886 | |
3887 if ( newNode.children.length == 0 && newNode.getParent() ) | |
3888 { | |
3889 newNode = newNode.getParent(); | |
3890 } | |
3891 | |
3892 if ( newNode != selectedNode ) | |
3893 { | |
3894 selectNode(newNode); | |
3895 } | |
3896 } | |
3897 | |
3898 function clearSearch() | |
3899 { | |
3900 if ( search.value != '' ) | |
3901 { | |
3902 search.value = ''; | |
3903 onSearchChange(); | |
3904 } | |
3905 } | |
3906 | |
3907 function createSVG() | |
3908 { | |
3909 svgNS = "http://www.w3.org/2000/svg"; | |
3910 var SVG = {}; | |
3911 SVG.xlinkns = "http://www.w3.org/1999/xlink"; | |
3912 | |
3913 var newSVG = document.createElementNS(svgNS, "svg:svg"); | |
3914 | |
3915 newSVG.setAttribute("id", "canvas"); | |
3916 // How big is the canvas in pixels | |
3917 newSVG.setAttribute("width", '100%'); | |
3918 newSVG.setAttribute("height", '100%'); | |
3919 // Set the coordinates used by drawings in the canvas | |
3920 // newSVG.setAttribute("viewBox", "0 0 " + imageWidth + " " + imageHeight); | |
3921 // Define the XLink namespace that SVG uses | |
3922 newSVG.setAttributeNS | |
3923 ( | |
3924 "http://www.w3.org/2000/xmlns/", | |
3925 "xmlns:xlink", | |
3926 SVG.xlinkns | |
3927 ); | |
3928 | |
3929 return newSVG; | |
3930 } | |
3931 | |
3932 function degrees(radians) | |
3933 { | |
3934 return radians * 180 / Math.PI; | |
3935 } | |
3936 | |
3937 function draw() | |
3938 { | |
3939 tweenFrames++; | |
3940 //resize(); | |
3941 // context.fillRect(0, 0, imageWidth, imageHeight); | |
3942 context.clearRect(0, 0, imageWidth, imageHeight); | |
3943 | |
3944 context.font = fontNormal; | |
3945 context.textBaseline = 'middle'; | |
3946 | |
3947 //context.strokeStyle = 'rgba(0, 0, 0, 0.3)'; | |
3948 context.translate(centerX, centerY); | |
3949 | |
3950 resetKeyOffset(); | |
3951 | |
3952 head.draw(false, false); // draw pie slices | |
3953 head.draw(true, false); // draw labels | |
3954 | |
3955 var pathRoot = selectedNode; | |
3956 | |
3957 if ( focusNode != 0 && focusNode != selectedNode ) | |
3958 { | |
3959 context.globalAlpha = 1; | |
3960 focusNode.drawHighlight(true); | |
3961 pathRoot = focusNode; | |
3962 } | |
3963 | |
3964 if | |
3965 ( | |
3966 highlightedNode && | |
3967 highlightedNode.getDepth() >= selectedNode.getDepth() && | |
3968 highlightedNode != focusNode | |
3969 ) | |
3970 { | |
3971 if | |
3972 ( | |
3973 progress == 1 && | |
3974 highlightedNode != selectedNode && | |
3975 ( | |
3976 highlightedNode != focusNode || | |
3977 focusNode.children.length > 0 | |
3978 ) | |
3979 ) | |
3980 { | |
3981 context.globalAlpha = 1; | |
3982 highlightedNode.drawHighlight(true); | |
3983 } | |
3984 | |
3985 //pathRoot = highlightedNode; | |
3986 } | |
3987 else if | |
3988 ( | |
3989 progress == 1 && | |
3990 highlightedNode.getDepth() < selectedNode.getDepth() | |
3991 ) | |
3992 { | |
3993 context.globalAlpha = 1; | |
3994 highlightedNode.drawHighlightCenter(); | |
3995 } | |
3996 | |
3997 if ( quickLook && false) // TEMP | |
3998 { | |
3999 context.globalAlpha = 1 - progress / 2; | |
4000 selectedNode.drawHighlight(true); | |
4001 } | |
4002 else if ( progress < 1 )//&& zoomOut() ) | |
4003 { | |
4004 if ( !zoomOut)//() ) | |
4005 { | |
4006 context.globalAlpha = selectedNode.alphaLine.current(); | |
4007 selectedNode.drawHighlight(true); | |
4008 } | |
4009 else if ( selectedNodeLast ) | |
4010 { | |
4011 context.globalAlpha = 1 - 4 * Math.pow(progress - .5, 2); | |
4012 selectedNodeLast.drawHighlight(false); | |
4013 } | |
4014 } | |
4015 | |
4016 drawDatasetName(); | |
4017 | |
4018 //drawHistory(); | |
4019 | |
4020 context.translate(-centerX, -centerY); | |
4021 context.globalAlpha = 1; | |
4022 | |
4023 mapRadius = | |
4024 (imageHeight / 2 - details.clientHeight - details.offsetTop) / | |
4025 (pathRoot.getDepth() - 1) * 3 / 4 / 2; | |
4026 | |
4027 if ( mapRadius > maxMapRadius ) | |
4028 { | |
4029 mapRadius = maxMapRadius; | |
4030 } | |
4031 | |
4032 mapBuffer = mapRadius / 2; | |
4033 | |
4034 //context.font = fontNormal; | |
4035 pathRoot.drawMap(pathRoot); | |
4036 | |
4037 if ( hueDisplayName && useHue() ) | |
4038 { | |
4039 drawLegend(); | |
4040 } | |
4041 } | |
4042 | |
4043 function drawBubble(angle, radius, width, radial, flip) | |
4044 { | |
4045 var height = fontSize * 2; | |
4046 var x; | |
4047 var y; | |
4048 | |
4049 width = width + fontSize; | |
4050 | |
4051 if ( radial ) | |
4052 { | |
4053 y = -fontSize; | |
4054 | |
4055 if ( flip ) | |
4056 { | |
4057 x = radius - width + fontSize / 2; | |
4058 } | |
4059 else | |
4060 { | |
4061 x = radius - fontSize / 2; | |
4062 } | |
4063 } | |
4064 else | |
4065 { | |
4066 x = -width / 2; | |
4067 y = -radius - fontSize; | |
4068 } | |
4069 | |
4070 if ( snapshotMode ) | |
4071 { | |
4072 drawBubbleSVG(x + centerX, y + centerY, width, height, fontSize, angle); | |
4073 } | |
4074 else | |
4075 { | |
4076 drawBubbleCanvas(x, y, width, height, fontSize, angle); | |
4077 } | |
4078 } | |
4079 | |
4080 function drawBubbleCanvas(x, y, width, height, radius, rotation) | |
4081 { | |
4082 context.strokeStyle = 'black'; | |
4083 context.lineWidth = highlightLineWidth; | |
4084 context.fillStyle = 'rgba(255, 255, 255, .75)'; | |
4085 context.rotate(rotation); | |
4086 roundedRectangle(x, y, width, fontSize * 2, fontSize); | |
4087 context.fill(); | |
4088 context.stroke(); | |
4089 context.rotate(-rotation); | |
4090 } | |
4091 | |
4092 function drawBubbleSVG(x, y, width, height, radius, rotation) | |
4093 { | |
4094 svg += | |
4095 '<rect x="' + x + '" y="' + y + | |
4096 '" width="' + width + | |
4097 '" height="' + height + | |
4098 '" rx="' + radius + | |
4099 '" ry="' + radius + | |
4100 '" fill="rgba(255, 255, 255, .75)' + | |
4101 '" class="highlight" ' + | |
4102 'transform="rotate(' + | |
4103 degrees(rotation) + ',' + centerX + ',' + centerY + | |
4104 ')"/>'; | |
4105 } | |
4106 | |
4107 function drawDatasetName() | |
4108 { | |
4109 var alpha = datasetAlpha.current(); | |
4110 | |
4111 if ( alpha > 0 ) | |
4112 { | |
4113 var radius = gRadius * compressedRadii[0] / -2; | |
4114 | |
4115 if ( alpha > 1 ) | |
4116 { | |
4117 alpha = 1; | |
4118 } | |
4119 | |
4120 context.globalAlpha = alpha; | |
4121 | |
4122 drawBubble(0, -radius, datasetWidths[currentDataset], false, false); | |
4123 drawText(datasetNames[currentDataset], 0, radius, 0, 'center', true); | |
4124 } | |
4125 } | |
4126 | |
4127 function drawHistory() | |
4128 { | |
4129 var alpha = 1; | |
4130 context.textAlign = 'center'; | |
4131 | |
4132 for ( var i = 0; i < nodeHistoryPosition && alpha > 0; i++ ) | |
4133 { | |
4134 | |
4135 context.globalAlpha = alpha - historyAlphaDelta * tweenFactor; | |
4136 context.fillText | |
4137 ( | |
4138 nodeHistory[nodeHistoryPosition - i - 1].name, | |
4139 0, | |
4140 (i + tweenFactor) * historySpacingFactor * fontSize - 1 | |
4141 ); | |
4142 | |
4143 if ( alpha > 0 ) | |
4144 { | |
4145 alpha -= historyAlphaDelta; | |
4146 } | |
4147 } | |
4148 | |
4149 context.globalAlpha = 1; | |
4150 } | |
4151 | |
4152 function drawLegend() | |
4153 { | |
4154 var left = imageWidth * .01; | |
4155 var width = imageHeight * .0265; | |
4156 var height = imageHeight * .15; | |
4157 var top = imageHeight - fontSize * 3.5 - height; | |
4158 var textLeft = left + width + fontSize / 2; | |
4159 | |
4160 context.fillStyle = 'black'; | |
4161 context.textAlign = 'start'; | |
4162 context.font = fontNormal; | |
4163 // context.fillText(valueStartText, textLeft, top + height); | |
4164 // context.fillText(valueEndText, textLeft, top); | |
4165 context.fillText(hueDisplayName, left, imageHeight - fontSize * 1.5); | |
4166 | |
4167 var gradient = context.createLinearGradient(0, top + height, 0, top); | |
4168 | |
4169 for ( var i = 0; i < hueStopPositions.length; i++ ) | |
4170 { | |
4171 gradient.addColorStop(hueStopPositions[i], hueStopHsl[i]); | |
4172 | |
4173 var textY = top + (1 - hueStopPositions[i]) * height; | |
4174 | |
4175 if | |
4176 ( | |
4177 i == 0 || | |
4178 i == hueStopPositions.length - 1 || | |
4179 textY > top + fontSize && textY < top + height - fontSize | |
4180 ) | |
4181 { | |
4182 context.fillText(hueStopText[i], textLeft, textY); | |
4183 } | |
4184 } | |
4185 | |
4186 context.fillStyle = gradient; | |
4187 context.fillRect(left, top, width, height); | |
4188 context.lineWidth = thinLineWidth; | |
4189 context.strokeRect(left, top, width, height); | |
4190 } | |
4191 | |
4192 function drawLegendSVG() | |
4193 { | |
4194 var left = imageWidth * .01; | |
4195 var width = imageHeight * .0265; | |
4196 var height = imageHeight * .15; | |
4197 var top = imageHeight - fontSize * 3.5 - height; | |
4198 var textLeft = left + width + fontSize / 2; | |
4199 | |
4200 var text = ''; | |
4201 | |
4202 text += svgText(hueDisplayName, left, imageHeight - fontSize * 1.5); | |
4203 | |
4204 var svgtest = '<linearGradient id="gradient" x1="0%" y1="100%" x2="0%" y2="0%">'; | |
4205 | |
4206 for ( var i = 0; i < hueStopPositions.length; i++ ) | |
4207 { | |
4208 svgtest += | |
4209 '<stop offset="' + round(hueStopPositions[i] * 100) + | |
4210 '%" style="stop-color:' + hueStopHsl[i] + '"/>'; | |
4211 | |
4212 var textY = top + (1 - hueStopPositions[i]) * height; | |
4213 | |
4214 if | |
4215 ( | |
4216 i == 0 || | |
4217 i == hueStopPositions.length - 1 || | |
4218 textY > top + fontSize && textY < top + height - fontSize | |
4219 ) | |
4220 { | |
4221 text += svgText(hueStopText[i], textLeft, textY); | |
4222 } | |
4223 } | |
4224 | |
4225 svgtest += '</linearGradient>'; | |
4226 //alert(svgtest); | |
4227 svg += svgtest; | |
4228 svg += | |
4229 '<rect style="fill:url(#gradient)" x="' + left + '" y="' + top + | |
4230 '" width="' + width + '" height="' + height + '"/>'; | |
4231 | |
4232 svg += text; | |
4233 } | |
4234 | |
4235 function drawSearchHighlights(label, bubbleX, bubbleY, rotation, center) | |
4236 { | |
4237 var index = -1; | |
4238 var labelLength = label.length; | |
4239 | |
4240 bubbleX -= fontSize / 4; | |
4241 | |
4242 do | |
4243 { | |
4244 index = label.toLowerCase().indexOf(search.value.toLowerCase(), index + 1); | |
4245 | |
4246 if ( index != -1 && index < labelLength ) | |
4247 { | |
4248 var dim = context.measureText(label.substr(0, index)); | |
4249 var x = bubbleX + dim.width; | |
4250 | |
4251 dim = context.measureText(label.substr(index, search.value.length)); | |
4252 | |
4253 var y = bubbleY - fontSize * 3 / 4; | |
4254 var width = dim.width + fontSize / 2; | |
4255 var height = fontSize * 3 / 2; | |
4256 var radius = fontSize / 2; | |
4257 | |
4258 if ( snapshotMode ) | |
4259 { | |
4260 if ( center ) | |
4261 { | |
4262 x += centerX; | |
4263 y += centerY; | |
4264 } | |
4265 | |
4266 svg += | |
4267 '<rect x="' + x + '" y="' + y + | |
4268 '" width="' + width + | |
4269 '" height="' + height + | |
4270 '" rx="' + radius + | |
4271 '" ry="' + radius + | |
4272 '" class="searchHighlight' + | |
4273 '" transform="rotate(' + | |
4274 degrees(rotation) + ',' + centerX + ',' + centerY + | |
4275 ')"/>'; | |
4276 } | |
4277 else | |
4278 { | |
4279 context.fillStyle = 'rgb(255, 255, 100)'; | |
4280 context.rotate(rotation); | |
4281 roundedRectangle(x, y, width, height, radius); | |
4282 context.fill(); | |
4283 context.rotate(-rotation); | |
4284 } | |
4285 } | |
4286 } | |
4287 while ( index != -1 && index < labelLength ); | |
4288 } | |
4289 | |
4290 function drawText(text, x, y, angle, anchor, bold, color) | |
4291 { | |
4292 if ( color == undefined ) | |
4293 { | |
4294 color = 'black'; | |
4295 } | |
4296 | |
4297 if ( snapshotMode ) | |
4298 { | |
4299 svg += | |
4300 '<text x="' + (centerX + x) + '" y="' + (centerY + y) + | |
4301 '" text-anchor="' + anchor + '" style="font-color:' + color + ';font-weight:' + (bold ? 'bold' : 'normal') + | |
4302 '" transform="rotate(' + degrees(angle) + ',' + centerX + ',' + centerY + ')">' + | |
4303 text + '</text>'; | |
4304 } | |
4305 else | |
4306 { | |
4307 context.fillStyle = color; | |
4308 context.textAlign = anchor; | |
4309 context.font = bold ? fontBold : fontNormal; | |
4310 context.rotate(angle); | |
4311 context.fillText(text, x, y); | |
4312 context.rotate(-angle); | |
4313 } | |
4314 } | |
4315 | |
4316 function drawTextPolar | |
4317 ( | |
4318 text, | |
4319 innerText, | |
4320 angle, | |
4321 radius, | |
4322 radial, | |
4323 bubble, | |
4324 bold, | |
4325 searchResult, | |
4326 searchResults | |
4327 ) | |
4328 { | |
4329 var anchor; | |
4330 var textX; | |
4331 var textY; | |
4332 var spacer; | |
4333 var totalText = text; | |
4334 var flip; | |
4335 | |
4336 if ( snapshotMode ) | |
4337 { | |
4338 spacer = '   '; | |
4339 } | |
4340 else | |
4341 { | |
4342 spacer = ' '; | |
4343 } | |
4344 | |
4345 if ( radial ) | |
4346 { | |
4347 flip = angle < 3 * Math.PI / 2; | |
4348 | |
4349 if ( flip ) | |
4350 { | |
4351 angle -= Math.PI; | |
4352 radius = -radius; | |
4353 anchor = 'end'; | |
4354 | |
4355 if ( innerText ) | |
4356 { | |
4357 totalText = text + spacer + innerText; | |
4358 } | |
4359 } | |
4360 else | |
4361 { | |
4362 anchor = 'start'; | |
4363 | |
4364 if ( innerText ) | |
4365 { | |
4366 totalText = innerText + spacer + text; | |
4367 } | |
4368 } | |
4369 | |
4370 textX = radius; | |
4371 textY = 0; | |
4372 } | |
4373 else | |
4374 { | |
4375 flip = angle < Math.PI || angle > 2 * Math.PI; | |
4376 var label; | |
4377 | |
4378 anchor = snapshotMode ? 'middle' : 'center'; | |
4379 | |
4380 if ( flip ) | |
4381 { | |
4382 angle -= Math.PI; | |
4383 radius = -radius; | |
4384 } | |
4385 | |
4386 angle += Math.PI / 2; | |
4387 textX = 0; | |
4388 textY = -radius; | |
4389 } | |
4390 | |
4391 if ( bubble ) | |
4392 { | |
4393 var textActual = totalText; | |
4394 | |
4395 if ( innerText && snapshotMode ) | |
4396 { | |
4397 if ( flip ) | |
4398 { | |
4399 textActual = text + ' ' + innerText; | |
4400 } | |
4401 else | |
4402 { | |
4403 textActual = innerText + ' ' + text; | |
4404 } | |
4405 } | |
4406 | |
4407 if ( searchResults ) | |
4408 { | |
4409 textActual = textActual + searchResultString(searchResults); | |
4410 } | |
4411 | |
4412 var textWidth = measureText(textActual, bold); | |
4413 | |
4414 var x = textX; | |
4415 | |
4416 if ( anchor == 'end' ) | |
4417 { | |
4418 x -= textWidth; | |
4419 } | |
4420 else if ( anchor != 'start' ) | |
4421 { | |
4422 // centered | |
4423 x -= textWidth / 2; | |
4424 } | |
4425 | |
4426 drawBubble(angle, radius, textWidth, radial, flip); | |
4427 | |
4428 if ( searchResult ) | |
4429 { | |
4430 drawSearchHighlights | |
4431 ( | |
4432 textActual, | |
4433 x, | |
4434 textY, | |
4435 angle, | |
4436 true | |
4437 ) | |
4438 } | |
4439 } | |
4440 | |
4441 if ( searchResults ) | |
4442 { | |
4443 totalText = totalText + searchResultString(searchResults); | |
4444 } | |
4445 | |
4446 drawText(totalText, textX, textY, angle, anchor, bold); | |
4447 | |
4448 return flip; | |
4449 } | |
4450 | |
4451 function drawTick(start, length, angle) | |
4452 { | |
4453 if ( snapshotMode ) | |
4454 { | |
4455 svg += | |
4456 '<line x1="' + (centerX + start) + | |
4457 '" y1="' + centerY + | |
4458 '" x2="' + (centerX + start + length) + | |
4459 '" y2="' + centerY + | |
4460 '" class="tick" transform="rotate(' + | |
4461 degrees(angle) + ',' + centerX + ',' + centerY + | |
4462 ')"/>'; | |
4463 } | |
4464 else | |
4465 { | |
4466 context.rotate(angle); | |
4467 context.beginPath(); | |
4468 context.moveTo(start, 0); | |
4469 context.lineTo(start + length, 0); | |
4470 context.lineWidth = thinLineWidth * 2; | |
4471 context.stroke(); | |
4472 context.rotate(-angle); | |
4473 } | |
4474 } | |
4475 | |
4476 function drawWedge | |
4477 ( | |
4478 angleStart, | |
4479 angleEnd, | |
4480 radiusInner, | |
4481 radiusOuter, | |
4482 color, | |
4483 patternAlpha, | |
4484 highlight | |
4485 ) | |
4486 { | |
4487 if ( context.globalAlpha == 0 ) | |
4488 { | |
4489 return; | |
4490 } | |
4491 | |
4492 if ( snapshotMode ) | |
4493 { | |
4494 if ( angleEnd == angleStart + Math.PI * 2 ) | |
4495 { | |
4496 // fudge to prevent overlap, which causes arc ambiguity | |
4497 // | |
4498 angleEnd -= .1 / gRadius; | |
4499 } | |
4500 | |
4501 var longArc = angleEnd - angleStart > Math.PI ? 1 : 0; | |
4502 | |
4503 var x1 = centerX + radiusInner * Math.cos(angleStart); | |
4504 var y1 = centerY + radiusInner * Math.sin(angleStart); | |
4505 | |
4506 var x2 = centerX + gRadius * Math.cos(angleStart); | |
4507 var y2 = centerY + gRadius * Math.sin(angleStart); | |
4508 | |
4509 var x3 = centerX + gRadius * Math.cos(angleEnd); | |
4510 var y3 = centerY + gRadius * Math.sin(angleEnd); | |
4511 | |
4512 var x4 = centerX + radiusInner * Math.cos(angleEnd); | |
4513 var y4 = centerY + radiusInner * Math.sin(angleEnd); | |
4514 | |
4515 var dArray = | |
4516 [ | |
4517 " M ", x1, ",", y1, | |
4518 " L ", x2, ",", y2, | |
4519 " A ", gRadius, ",", gRadius, " 0 ", longArc, ",1 ", x3, ",", y3, | |
4520 " L ", x4, ",", y4, | |
4521 " A ", radiusInner, ",", radiusInner, " 0 ", longArc, " 0 ", x1, ",", y1, | |
4522 " Z " | |
4523 ]; | |
4524 | |
4525 svg += | |
4526 '<path class="'+ (highlight ? 'highlight' : 'wedge') + '" fill="' + color + | |
4527 '" d="' + dArray.join('') + '"/>'; | |
4528 | |
4529 if ( patternAlpha > 0 ) | |
4530 { | |
4531 svg += | |
4532 '<path class="wedge" fill="url(#hiddenPattern)" d="' + | |
4533 dArray.join('') + '"/>'; | |
4534 } | |
4535 } | |
4536 else | |
4537 { | |
4538 // fudge to prevent seams during animation | |
4539 // | |
4540 angleEnd += 1 / gRadius; | |
4541 | |
4542 context.fillStyle = color; | |
4543 context.beginPath(); | |
4544 context.arc(0, 0, radiusInner, angleStart, angleEnd, false); | |
4545 context.arc(0, 0, radiusOuter, angleEnd, angleStart, true); | |
4546 context.closePath(); | |
4547 context.fill(); | |
4548 | |
4549 if ( patternAlpha > 0 ) | |
4550 { | |
4551 context.save(); | |
4552 context.clip(); | |
4553 context.globalAlpha = patternAlpha; | |
4554 context.fillStyle = hiddenPattern; | |
4555 context.fill(); | |
4556 context.restore(); | |
4557 } | |
4558 | |
4559 if ( highlight ) | |
4560 { | |
4561 context.lineWidth = highlight ? highlightLineWidth : thinLineWidth; | |
4562 context.strokeStyle = 'black'; | |
4563 context.stroke(); | |
4564 } | |
4565 } | |
4566 } | |
4567 | |
4568 function expand(node) | |
4569 { | |
4570 selectNode(node); | |
4571 updateView(); | |
4572 } | |
4573 | |
4574 function focusLost() | |
4575 { | |
4576 mouseX = -1; | |
4577 mouseY = -1; | |
4578 checkHighlight(); | |
4579 document.body.style.cursor = 'auto'; | |
4580 } | |
4581 | |
4582 function fontSizeDecrease() | |
4583 { | |
4584 if ( fontSize > 1 ) | |
4585 { | |
4586 fontSize--; | |
4587 updateViewNeeded = true; | |
4588 } | |
4589 } | |
4590 | |
4591 function fontSizeIncrease() | |
4592 { | |
4593 fontSize++; | |
4594 updateViewNeeded = true; | |
4595 } | |
4596 | |
4597 function getGetString(name, value, bool) | |
4598 { | |
4599 return name + '=' + (bool ? value ? 'true' : 'false' : value); | |
4600 } | |
4601 | |
4602 function hideLink() | |
4603 { | |
4604 hide(linkText); | |
4605 show(linkButton); | |
4606 } | |
4607 | |
4608 function show(object) | |
4609 { | |
4610 object.style.display = 'inline'; | |
4611 } | |
4612 | |
4613 function hide(object) | |
4614 { | |
4615 object.style.display = 'none'; | |
4616 } | |
4617 | |
4618 function showLink() | |
4619 { | |
4620 var urlHalves = String(document.location).split('?'); | |
4621 var newGetVariables = new Array(); | |
4622 | |
4623 newGetVariables.push | |
4624 ( | |
4625 getGetString('dataset', currentDataset, false), | |
4626 getGetString('node', selectedNode.id, false), | |
4627 getGetString('collapse', collapse, true), | |
4628 getGetString('color', useHue(), true), | |
4629 getGetString('depth', maxAbsoluteDepth - 1, false), | |
4630 getGetString('font', fontSize, false), | |
4631 getGetString('key', showKeys, true) | |
4632 ); | |
4633 | |
4634 hide(linkButton); | |
4635 show(linkText); | |
4636 linkText.value = urlHalves[0] + '?' + getVariables.concat(newGetVariables).join('&'); | |
4637 //linkText.disabled = false; | |
4638 linkText.focus(); | |
4639 linkText.select(); | |
4640 //linkText.disabled = true; | |
4641 // document.location = urlHalves[0] + '?' + getVariables.join('&'); | |
4642 } | |
4643 | |
4644 function getFirstChild(element) | |
4645 { | |
4646 element = element.firstChild; | |
4647 | |
4648 if ( element && element.nodeType != 1 ) | |
4649 { | |
4650 element = getNextSibling(element); | |
4651 } | |
4652 | |
4653 return element; | |
4654 } | |
4655 | |
4656 function getNextSibling(element) | |
4657 { | |
4658 do | |
4659 { | |
4660 element = element.nextSibling; | |
4661 } | |
4662 while ( element && element.nodeType != 1 ); | |
4663 | |
4664 return element; | |
4665 } | |
4666 | |
4667 function getPercentage(fraction) | |
4668 { | |
4669 return round(fraction * 100); | |
4670 } | |
4671 | |
4672 function hslText(hue) | |
4673 { | |
4674 if ( 1 || snapshotMode ) | |
4675 { | |
4676 // Safari doesn't seem to allow hsl() in SVG | |
4677 | |
4678 var rgb = hslToRgb(hue, saturation, (lightnessBase + lightnessMax) / 2); | |
4679 | |
4680 return rgbText(rgb.r, rgb.g, rgb.b); | |
4681 } | |
4682 else | |
4683 { | |
4684 var hslArray = | |
4685 [ | |
4686 'hsl(', | |
4687 Math.floor(hue * 360), | |
4688 ',', | |
4689 Math.floor(saturation * 100), | |
4690 '%,', | |
4691 Math.floor((lightnessBase + lightnessMax) * 50), | |
4692 '%)' | |
4693 ]; | |
4694 | |
4695 return hslArray.join(''); | |
4696 } | |
4697 } | |
4698 | |
4699 function hslToRgb(h, s, l) | |
4700 { | |
4701 var m1, m2; | |
4702 var r, g, b; | |
4703 | |
4704 if (s == 0) | |
4705 { | |
4706 r = g = b = Math.floor((l * 255)); | |
4707 } | |
4708 else | |
4709 { | |
4710 if (l <= 0.5) | |
4711 { | |
4712 m2 = l * (s + 1); | |
4713 } | |
4714 else | |
4715 { | |
4716 m2 = l + s - l * s; | |
4717 } | |
4718 | |
4719 m1 = l * 2 - m2; | |
4720 | |
4721 r = Math.floor(hueToRgb(m1, m2, h + 1 / 3)); | |
4722 g = Math.floor(hueToRgb(m1, m2, h)); | |
4723 b = Math.floor(hueToRgb(m1, m2, h - 1/3)); | |
4724 } | |
4725 | |
4726 return {r: r, g: g, b: b}; | |
4727 } | |
4728 | |
4729 function hueToRgb(m1, m2, hue) | |
4730 { | |
4731 var v; | |
4732 | |
4733 while (hue < 0) | |
4734 { | |
4735 hue += 1; | |
4736 } | |
4737 | |
4738 while (hue > 1) | |
4739 { | |
4740 hue -= 1; | |
4741 } | |
4742 | |
4743 if (6 * hue < 1) | |
4744 v = m1 + (m2 - m1) * hue * 6; | |
4745 else if (2 * hue < 1) | |
4746 v = m2; | |
4747 else if (3 * hue < 2) | |
4748 v = m1 + (m2 - m1) * (2/3 - hue) * 6; | |
4749 else | |
4750 v = m1; | |
4751 | |
4752 return 255 * v; | |
4753 } | |
4754 | |
4755 function interpolateHue(hueStart, hueEnd, valueStart, valueEnd) | |
4756 { | |
4757 // since the gradient will be RGB based, we need to add stops to hit all the | |
4758 // colors in the hue spectrum | |
4759 | |
4760 hueStopPositions = new Array(); | |
4761 hueStopHsl = new Array(); | |
4762 hueStopText = new Array(); | |
4763 | |
4764 hueStopPositions.push(0); | |
4765 hueStopHsl.push(hslText(hueStart)); | |
4766 hueStopText.push(round(valueStart)); | |
4767 | |
4768 for | |
4769 ( | |
4770 var i = (hueStart > hueEnd ? 5 / 6 : 1 / 6); | |
4771 (hueStart > hueEnd ? i > 0 : i < 1); | |
4772 i += (hueStart > hueEnd ? -1 : 1) / 6 | |
4773 ) | |
4774 { | |
4775 if | |
4776 ( | |
4777 hueStart > hueEnd ? | |
4778 i > hueEnd && i < hueStart : | |
4779 i > hueStart && i < hueEnd | |
4780 ) | |
4781 { | |
4782 hueStopPositions.push(lerp(i, hueStart, hueEnd, 0, 1)); | |
4783 hueStopHsl.push(hslText(i)); | |
4784 hueStopText.push(round(lerp | |
4785 ( | |
4786 i, | |
4787 hueStart, | |
4788 hueEnd, | |
4789 valueStart, | |
4790 valueEnd | |
4791 ))); | |
4792 } | |
4793 } | |
4794 | |
4795 hueStopPositions.push(1); | |
4796 hueStopHsl.push(hslText(hueEnd)); | |
4797 hueStopText.push(round(valueEnd)); | |
4798 } | |
4799 | |
4800 function keyLineAngle(angle, keyAngle, bendRadius, keyX, keyY, pointsX, pointsY) | |
4801 { | |
4802 if ( angle < Math.PI / 2 && keyY < bendRadius * Math.sin(angle) | |
4803 || angle > Math.PI / 2 && keyY < bendRadius) | |
4804 { | |
4805 return Math.asin(keyY / bendRadius); | |
4806 } | |
4807 else | |
4808 { | |
4809 // find the angle of the normal to a tangent line that goes to | |
4810 // the label | |
4811 | |
4812 var textDist = Math.sqrt | |
4813 ( | |
4814 Math.pow(keyX, 2) + | |
4815 Math.pow(keyY, 2) | |
4816 ); | |
4817 | |
4818 var tanAngle = Math.acos(bendRadius / textDist) + keyAngle; | |
4819 | |
4820 if ( angle < tanAngle || angle < Math.PI / 2 )//|| labelLeft < centerX ) | |
4821 { | |
4822 // angle doesn't reach far enough for tangent; collapse and | |
4823 // connect directly to label | |
4824 | |
4825 if ( keyY / Math.tan(angle) > 0 ) | |
4826 { | |
4827 pointsX.push(keyY / Math.tan(angle)); | |
4828 pointsY.push(keyY); | |
4829 } | |
4830 else | |
4831 { | |
4832 pointsX.push(bendRadius * Math.cos(angle)); | |
4833 pointsY.push(bendRadius * Math.sin(angle)); | |
4834 } | |
4835 | |
4836 return angle; | |
4837 } | |
4838 else | |
4839 { | |
4840 return tanAngle; | |
4841 } | |
4842 } | |
4843 } | |
4844 | |
4845 function keyOffset() | |
4846 { | |
4847 return imageHeight - (keys - currentKey + 1) * (keySize + keyBuffer) + keyBuffer - margin; | |
4848 } | |
4849 | |
4850 function lerp(value, fromStart, fromEnd, toStart, toEnd) | |
4851 { | |
4852 return (value - fromStart) * | |
4853 (toEnd - toStart) / | |
4854 (fromEnd - fromStart) + | |
4855 toStart; | |
4856 } | |
4857 | |
4858 function createCanvas() | |
4859 { | |
4860 canvas = document.createElement('canvas'); | |
4861 document.body.appendChild(canvas); | |
4862 context = canvas.getContext('2d'); | |
4863 } | |
4864 | |
4865 function load() | |
4866 { | |
4867 document.body.style.overflow = "hidden"; | |
4868 document.body.style.margin = 0; | |
4869 | |
4870 createCanvas(); | |
4871 | |
4872 if ( context == undefined ) | |
4873 { | |
4874 document.body.innerHTML = '\ | |
4875 <br/>This browser does not support HTML5 (see \ | |
4876 <a href="https://github.com/marbl/Krona/wiki/Browser%20support">Browser support</a>).\ | |
4877 '; | |
4878 return; | |
4879 } | |
4880 | |
4881 if ( typeof context.fillText != 'function' ) | |
4882 { | |
4883 document.body.innerHTML = '\ | |
4884 <br/>This browser does not support HTML5 canvas text (see \ | |
4885 <a href="https://github.com/marbl/Krona/wiki/Browser%20support">Browser support</a>).\ | |
4886 '; | |
4887 return; | |
4888 } | |
4889 | |
4890 resize(); | |
4891 | |
4892 var kronaElement = document.getElementsByTagName('krona')[0]; | |
4893 | |
4894 var magnitudeName; | |
4895 var hueName; | |
4896 var hueDefault; | |
4897 var hueStart; | |
4898 var hueEnd; | |
4899 var valueStart; | |
4900 var valueEnd; | |
4901 | |
4902 if ( kronaElement.getAttribute('collapse') != undefined ) | |
4903 { | |
4904 collapse = kronaElement.getAttribute('collapse') == 'true'; | |
4905 } | |
4906 | |
4907 if ( kronaElement.getAttribute('key') != undefined ) | |
4908 { | |
4909 showKeys = kronaElement.getAttribute('key') == 'true'; | |
4910 } | |
4911 | |
4912 for | |
4913 ( | |
4914 var element = getFirstChild(kronaElement); | |
4915 element; | |
4916 element = getNextSibling(element) | |
4917 ) | |
4918 { | |
4919 switch ( element.tagName.toLowerCase() ) | |
4920 { | |
4921 case 'attributes': | |
4922 magnitudeName = element.getAttribute('magnitude'); | |
4923 // | |
4924 for | |
4925 ( | |
4926 var attributeElement = getFirstChild(element); | |
4927 attributeElement; | |
4928 attributeElement = getNextSibling(attributeElement) | |
4929 ) | |
4930 { | |
4931 var tag = attributeElement.tagName.toLowerCase(); | |
4932 | |
4933 if ( tag == 'attribute' ) | |
4934 { | |
4935 var attribute = new Attribute(); | |
4936 attribute.name = attributeElement.firstChild.nodeValue.toLowerCase(); | |
4937 attribute.displayName = attributeElement.getAttribute('display'); | |
4938 | |
4939 if ( attributeElement.getAttribute('hrefBase') ) | |
4940 { | |
4941 attribute.hrefBase = attributeElement.getAttribute('hrefBase'); | |
4942 } | |
4943 | |
4944 if ( attributeElement.getAttribute('target') ) | |
4945 { | |
4946 attribute.target = attributeElement.getAttribute('target'); | |
4947 } | |
4948 | |
4949 if ( attribute.name == magnitudeName ) | |
4950 { | |
4951 magnitudeIndex = attributes.length; | |
4952 } | |
4953 | |
4954 if ( attributeElement.getAttribute('listAll') ) | |
4955 { | |
4956 attribute.listAll = attributeElement.getAttribute('listAll').toLowerCase(); | |
4957 } | |
4958 else if ( attributeElement.getAttribute('listNode') ) | |
4959 { | |
4960 attribute.listNode = attributeElement.getAttribute('listNode').toLowerCase(); | |
4961 } | |
4962 else if ( attributeElement.getAttribute('dataAll') ) | |
4963 { | |
4964 attribute.dataAll = attributeElement.getAttribute('dataAll').toLowerCase(); | |
4965 } | |
4966 else if ( attributeElement.getAttribute('dataNode') ) | |
4967 { | |
4968 attribute.dataNode = attributeElement.getAttribute('dataNode').toLowerCase(); | |
4969 } | |
4970 | |
4971 if ( attributeElement.getAttribute('postUrl') ) | |
4972 { | |
4973 attribute.postUrl = attributeElement.getAttribute('postUrl'); | |
4974 } | |
4975 | |
4976 if ( attributeElement.getAttribute('postVar') ) | |
4977 { | |
4978 attribute.postVar = attributeElement.getAttribute('postVar'); | |
4979 } | |
4980 | |
4981 if ( attributeElement.getAttribute('mono') ) | |
4982 { | |
4983 attribute.mono = true; | |
4984 } | |
4985 | |
4986 attributes.push(attribute); | |
4987 } | |
4988 else if ( tag == 'list' ) | |
4989 { | |
4990 var attribute = new Attribute(); | |
4991 | |
4992 attribute.name = attributeElement.firstChild.nodeValue; | |
4993 attribute.list = true; | |
4994 attributes.push(attribute); | |
4995 } | |
4996 else if ( tag == 'data' ) | |
4997 { | |
4998 var attribute = new Attribute(); | |
4999 | |
5000 attribute.name = attributeElement.firstChild.nodeValue; | |
5001 attribute.data = true; | |
5002 attributes.push(attribute); | |
5003 | |
5004 var enableScript = document.createElement('script'); | |
5005 var date = new Date(); | |
5006 enableScript.src = | |
5007 attributeElement.getAttribute('enable') + '?' + | |
5008 date.getTime(); | |
5009 document.body.appendChild(enableScript); | |
5010 } | |
5011 } | |
5012 break; | |
5013 | |
5014 case 'color': | |
5015 hueName = element.getAttribute('attribute'); | |
5016 hueStart = Number(element.getAttribute('hueStart')) / 360; | |
5017 hueEnd = Number(element.getAttribute('hueEnd')) / 360; | |
5018 valueStart = Number(element.getAttribute('valueStart')); | |
5019 valueEnd = Number(element.getAttribute('valueEnd')); | |
5020 // | |
5021 interpolateHue(hueStart, hueEnd, valueStart, valueEnd); | |
5022 // | |
5023 if ( element.getAttribute('default') == 'true' ) | |
5024 { | |
5025 hueDefault = true; | |
5026 } | |
5027 break; | |
5028 | |
5029 case 'datasets': | |
5030 datasetNames = new Array(); | |
5031 // | |
5032 for ( j = getFirstChild(element); j; j = getNextSibling(j) ) | |
5033 { | |
5034 datasetNames.push(j.firstChild.nodeValue); | |
5035 } | |
5036 datasets = datasetNames.length; | |
5037 break; | |
5038 | |
5039 case 'node': | |
5040 head = loadTreeDOM | |
5041 ( | |
5042 element, | |
5043 magnitudeName, | |
5044 hueName, | |
5045 hueStart, | |
5046 hueEnd, | |
5047 valueStart, | |
5048 valueEnd | |
5049 ); | |
5050 break; | |
5051 } | |
5052 } | |
5053 | |
5054 // get GET options | |
5055 // | |
5056 var urlHalves = String(document.location).split('?'); | |
5057 var datasetDefault = 0; | |
5058 var maxDepthDefault; | |
5059 var nodeDefault = 0; | |
5060 // | |
5061 if ( urlHalves[1] ) | |
5062 { | |
5063 var vars = urlHalves[1].split('&'); | |
5064 | |
5065 for ( i = 0; i < vars.length; i++ ) | |
5066 { | |
5067 var pair = vars[i].split('='); | |
5068 | |
5069 switch ( pair[0] ) | |
5070 { | |
5071 case 'collapse': | |
5072 collapse = pair[1] == 'true'; | |
5073 break; | |
5074 | |
5075 case 'color': | |
5076 hueDefault = pair[1] == 'true'; | |
5077 break; | |
5078 | |
5079 case 'dataset': | |
5080 datasetDefault = Number(pair[1]); | |
5081 break; | |
5082 | |
5083 case 'depth': | |
5084 maxDepthDefault = Number(pair[1]) + 1; | |
5085 break; | |
5086 | |
5087 case 'key': | |
5088 showKeys = pair[1] == 'true'; | |
5089 break; | |
5090 | |
5091 case 'font': | |
5092 fontSize = Number(pair[1]); | |
5093 break; | |
5094 | |
5095 case 'node': | |
5096 nodeDefault = Number(pair[1]); | |
5097 break; | |
5098 | |
5099 default: | |
5100 getVariables.push(pair[0] + '=' + pair[1]); | |
5101 break; | |
5102 } | |
5103 } | |
5104 } | |
5105 | |
5106 addOptionElements(hueName, hueDefault); | |
5107 setCallBacks(); | |
5108 | |
5109 head.sort(); | |
5110 maxAbsoluteDepth = 0; | |
5111 selectDataset(datasetDefault); | |
5112 | |
5113 if ( maxDepthDefault && maxDepthDefault < head.maxDepth ) | |
5114 { | |
5115 maxAbsoluteDepth = maxDepthDefault; | |
5116 } | |
5117 else | |
5118 { | |
5119 maxAbsoluteDepth = head.maxDepth; | |
5120 } | |
5121 | |
5122 selectNode(nodes[nodeDefault]); | |
5123 | |
5124 setInterval(update, 20); | |
5125 | |
5126 window.onresize = handleResize; | |
5127 updateMaxAbsoluteDepth(); | |
5128 updateViewNeeded = true; | |
5129 } | |
5130 | |
5131 function loadTreeDOM | |
5132 ( | |
5133 domNode, | |
5134 magnitudeName, | |
5135 hueName, | |
5136 hueStart, | |
5137 hueEnd, | |
5138 valueStart, | |
5139 valueEnd | |
5140 ) | |
5141 { | |
5142 var newNode = new Node(); | |
5143 | |
5144 newNode.name = domNode.getAttribute('name'); | |
5145 | |
5146 if ( domNode.getAttribute('href') ) | |
5147 { | |
5148 newNode.href = domNode.getAttribute('href'); | |
5149 } | |
5150 | |
5151 if ( hueName ) | |
5152 { | |
5153 newNode.hues = new Array(); | |
5154 } | |
5155 | |
5156 for ( var i = getFirstChild(domNode); i; i = getNextSibling(i) ) | |
5157 { | |
5158 switch ( i.tagName.toLowerCase() ) | |
5159 { | |
5160 case 'node': | |
5161 var newChild = loadTreeDOM | |
5162 ( | |
5163 i, | |
5164 magnitudeName, | |
5165 hueName, | |
5166 hueStart, | |
5167 hueEnd, | |
5168 valueStart, | |
5169 valueEnd | |
5170 ); | |
5171 newChild.parent = newNode; | |
5172 newNode.children.push(newChild); | |
5173 break; | |
5174 | |
5175 default: | |
5176 var attributeName = i.tagName.toLowerCase(); | |
5177 var index = attributeIndex(attributeName); | |
5178 // | |
5179 newNode.attributes[index] = new Array(); | |
5180 // | |
5181 for ( var j = getFirstChild(i); j; j = getNextSibling(j) ) | |
5182 { | |
5183 if ( attributes[index] == undefined ) | |
5184 { | |
5185 var x = 5; | |
5186 } | |
5187 if ( attributes[index].list ) | |
5188 { | |
5189 newNode.attributes[index].push(new Array()); | |
5190 | |
5191 for ( var k = getFirstChild(j); k; k = getNextSibling(k) ) | |
5192 { | |
5193 newNode.attributes[index][newNode.attributes[index].length - 1].push(k.firstChild.nodeValue); | |
5194 } | |
5195 } | |
5196 else | |
5197 { | |
5198 var value = j.firstChild ? j.firstChild.nodeValue : ''; | |
5199 | |
5200 if ( j.getAttribute('href') ) | |
5201 { | |
5202 var target; | |
5203 | |
5204 if ( attributes[index].target ) | |
5205 { | |
5206 target = ' target="' + attributes[index].target + '"'; | |
5207 } | |
5208 | |
5209 value = '<a href="' + attributes[index].hrefBase + j.getAttribute('href') + '"' + target + '>' + value + '</a>'; | |
5210 } | |
5211 | |
5212 newNode.attributes[index].push(value); | |
5213 } | |
5214 } | |
5215 // | |
5216 if ( attributeName == magnitudeName || attributeName == hueName ) | |
5217 { | |
5218 for ( j = 0; j < datasets; j++ ) | |
5219 { | |
5220 var value = newNode.attributes[index][j] == undefined ? 0 : Number(newNode.attributes[index][j]); | |
5221 | |
5222 newNode.attributes[index][j] = value; | |
5223 | |
5224 if ( attributeName == hueName ) | |
5225 { | |
5226 var hue = lerp | |
5227 ( | |
5228 value, | |
5229 valueStart, | |
5230 valueEnd, | |
5231 hueStart, | |
5232 hueEnd | |
5233 ); | |
5234 | |
5235 if ( hue < hueStart == hueStart < hueEnd ) | |
5236 { | |
5237 hue = hueStart; | |
5238 } | |
5239 else if ( hue > hueEnd == hueStart < hueEnd ) | |
5240 { | |
5241 hue = hueEnd; | |
5242 } | |
5243 | |
5244 newNode.hues[j] = hue; | |
5245 } | |
5246 } | |
5247 | |
5248 if ( attributeName == hueName ) | |
5249 { | |
5250 newNode.hue = new Tween(newNode.hues[0], newNode.hues[0]); | |
5251 } | |
5252 } | |
5253 break; | |
5254 } | |
5255 } | |
5256 | |
5257 return newNode; | |
5258 } | |
5259 | |
5260 function maxAbsoluteDepthDecrease() | |
5261 { | |
5262 if ( maxAbsoluteDepth > 2 ) | |
5263 { | |
5264 maxAbsoluteDepth--; | |
5265 head.setMaxDepths(); | |
5266 handleResize(); | |
5267 } | |
5268 } | |
5269 | |
5270 function maxAbsoluteDepthIncrease() | |
5271 { | |
5272 if ( maxAbsoluteDepth < head.maxDepth ) | |
5273 { | |
5274 maxAbsoluteDepth++; | |
5275 head.setMaxDepths(); | |
5276 handleResize(); | |
5277 } | |
5278 } | |
5279 | |
5280 function measureText(text, bold) | |
5281 { | |
5282 context.font = bold ? fontBold : fontNormal; | |
5283 var dim = context.measureText(text); | |
5284 return dim.width; | |
5285 } | |
5286 | |
5287 function min(a, b) | |
5288 { | |
5289 return a < b ? a : b; | |
5290 } | |
5291 | |
5292 function minWidth() | |
5293 { | |
5294 // Min wedge width (at center) for displaying a node (or for displaying a | |
5295 // label if it's at the highest level being viewed, multiplied by 2 to make | |
5296 // further calculations simpler | |
5297 | |
5298 return (fontSize * 2.3); | |
5299 } | |
5300 | |
5301 function mouseMove(e) | |
5302 { | |
5303 mouseX = e.pageX; | |
5304 mouseY = e.pageY - headerHeight; | |
5305 mouseXRel = (mouseX - centerX) * backingScale() | |
5306 mouseYRel = (mouseY - centerY) * backingScale() | |
5307 | |
5308 if ( head && ! quickLook ) | |
5309 { | |
5310 checkHighlight(); | |
5311 } | |
5312 } | |
5313 | |
5314 function mouseClick(e) | |
5315 { | |
5316 if ( highlightedNode == focusNode && focusNode != selectedNode || selectedNode.hasParent(highlightedNode) ) | |
5317 { | |
5318 if ( highlightedNode.hasChildren() ) | |
5319 { | |
5320 expand(highlightedNode); | |
5321 } | |
5322 } | |
5323 else if ( progress == 1 )//( highlightedNode != selectedNode ) | |
5324 { | |
5325 setFocus(highlightedNode); | |
5326 // document.body.style.cursor='ew-resize'; | |
5327 draw(); | |
5328 checkHighlight(); | |
5329 var date = new Date(); | |
5330 mouseDownTime = date.getTime(); | |
5331 mouseDown = true; | |
5332 } | |
5333 } | |
5334 | |
5335 function mouseUp(e) | |
5336 { | |
5337 if ( quickLook ) | |
5338 { | |
5339 navigateBack(); | |
5340 quickLook = false; | |
5341 } | |
5342 | |
5343 mouseDown = false; | |
5344 } | |
5345 | |
5346 function navigateBack() | |
5347 { | |
5348 if ( nodeHistoryPosition > 0 ) | |
5349 { | |
5350 nodeHistory[nodeHistoryPosition] = selectedNode; | |
5351 nodeHistoryPosition--; | |
5352 | |
5353 if ( nodeHistory[nodeHistoryPosition].collapse ) | |
5354 { | |
5355 collapseCheckBox.checked = collapse = false; | |
5356 } | |
5357 | |
5358 setSelectedNode(nodeHistory[nodeHistoryPosition]); | |
5359 updateDatasetButtons(); | |
5360 updateView(); | |
5361 } | |
5362 } | |
5363 | |
5364 function navigateUp() | |
5365 { | |
5366 if ( selectedNode.getParent() ) | |
5367 { | |
5368 selectNode(selectedNode.getParent()); | |
5369 updateView(); | |
5370 } | |
5371 } | |
5372 | |
5373 function navigateForward() | |
5374 { | |
5375 if ( nodeHistoryPosition < nodeHistory.length - 1 ) | |
5376 { | |
5377 nodeHistoryPosition++; | |
5378 var newNode = nodeHistory[nodeHistoryPosition]; | |
5379 | |
5380 if ( newNode.collapse ) | |
5381 { | |
5382 collapseCheckBox.checked = collapse = false; | |
5383 } | |
5384 | |
5385 if ( nodeHistoryPosition == nodeHistory.length - 1 ) | |
5386 { | |
5387 // this will ensure the forward button is disabled | |
5388 | |
5389 nodeHistory.length = nodeHistoryPosition; | |
5390 } | |
5391 | |
5392 setSelectedNode(newNode); | |
5393 updateDatasetButtons(); | |
5394 updateView(); | |
5395 } | |
5396 } | |
5397 | |
5398 function nextDataset() | |
5399 { | |
5400 var newDataset = currentDataset; | |
5401 | |
5402 do | |
5403 { | |
5404 if ( newDataset == datasets - 1 ) | |
5405 { | |
5406 newDataset = 0; | |
5407 } | |
5408 else | |
5409 { | |
5410 newDataset++; | |
5411 } | |
5412 } | |
5413 while ( datasetDropDown.options[newDataset].disabled ) | |
5414 | |
5415 selectDataset(newDataset); | |
5416 } | |
5417 | |
5418 function onDatasetChange() | |
5419 { | |
5420 selectDataset(datasetDropDown.selectedIndex); | |
5421 } | |
5422 | |
5423 function onKeyDown(event) | |
5424 { | |
5425 if | |
5426 ( | |
5427 event.keyCode == 37 && | |
5428 document.activeElement.id != 'search' && | |
5429 document.activeElement.id != 'linkText' | |
5430 ) | |
5431 { | |
5432 navigateBack(); | |
5433 event.preventDefault(); | |
5434 } | |
5435 else if | |
5436 ( | |
5437 event.keyCode == 39 && | |
5438 document.activeElement.id != 'search' && | |
5439 document.activeElement.id != 'linkText' | |
5440 ) | |
5441 { | |
5442 navigateForward(); | |
5443 event.preventDefault(); | |
5444 } | |
5445 else if ( event.keyCode == 38 && datasets > 1 ) | |
5446 { | |
5447 prevDataset(); | |
5448 | |
5449 //if ( document.activeElement.id == 'datasets' ) | |
5450 { | |
5451 event.preventDefault(); | |
5452 } | |
5453 } | |
5454 else if ( event.keyCode == 40 && datasets > 1 ) | |
5455 { | |
5456 nextDataset(); | |
5457 | |
5458 //if ( document.activeElement.id == 'datasets' ) | |
5459 { | |
5460 event.preventDefault(); | |
5461 } | |
5462 } | |
5463 else if ( event.keyCode == 9 && datasets > 1 ) | |
5464 { | |
5465 selectLastDataset(); | |
5466 event.preventDefault(); | |
5467 } | |
5468 else if ( event.keyCode == 83 ) | |
5469 { | |
5470 progress += .2; | |
5471 } | |
5472 else if ( event.keyCode == 66 ) | |
5473 { | |
5474 progress -= .2; | |
5475 } | |
5476 else if ( event.keyCode == 70 ) | |
5477 { | |
5478 progress = 1; | |
5479 } | |
5480 } | |
5481 | |
5482 function onKeyPress(event) | |
5483 { | |
5484 if ( event.keyCode == 38 && datasets > 1 ) | |
5485 { | |
5486 // prevDataset(); | |
5487 | |
5488 //if ( document.activeElement.id == 'datasets' ) | |
5489 { | |
5490 event.preventDefault(); | |
5491 } | |
5492 } | |
5493 else if ( event.keyCode == 40 && datasets > 1 ) | |
5494 { | |
5495 // nextDataset(); | |
5496 | |
5497 //if ( document.activeElement.id == 'datasets' ) | |
5498 { | |
5499 event.preventDefault(); | |
5500 } | |
5501 } | |
5502 } | |
5503 | |
5504 function onKeyUp(event) | |
5505 { | |
5506 if ( event.keyCode == 27 && document.activeElement.id == 'search' ) | |
5507 { | |
5508 search.value = ''; | |
5509 onSearchChange(); | |
5510 } | |
5511 else if ( event.keyCode == 38 && datasets > 1 ) | |
5512 { | |
5513 // prevDataset(); | |
5514 | |
5515 //if ( document.activeElement.id == 'datasets' ) | |
5516 { | |
5517 event.preventDefault(); | |
5518 } | |
5519 } | |
5520 else if ( event.keyCode == 40 && datasets > 1 ) | |
5521 { | |
5522 // nextDataset(); | |
5523 | |
5524 //if ( document.activeElement.id == 'datasets' ) | |
5525 { | |
5526 event.preventDefault(); | |
5527 } | |
5528 } | |
5529 } | |
5530 | |
5531 function onSearchChange() | |
5532 { | |
5533 nSearchResults = 0; | |
5534 head.search(); | |
5535 | |
5536 if ( search.value == '' ) | |
5537 { | |
5538 searchResults.innerHTML = ''; | |
5539 } | |
5540 else | |
5541 { | |
5542 searchResults.innerHTML = nSearchResults + ' results'; | |
5543 } | |
5544 | |
5545 setFocus(selectedNode); | |
5546 draw(); | |
5547 } | |
5548 | |
5549 function post(url, variable, value, postWindow) | |
5550 { | |
5551 var form = document.createElement('form'); | |
5552 var input = document.createElement('input'); | |
5553 var inputDataset = document.createElement('input'); | |
5554 | |
5555 form.appendChild(input); | |
5556 form.appendChild(inputDataset); | |
5557 | |
5558 form.method = "POST"; | |
5559 form.action = url; | |
5560 | |
5561 if ( postWindow == undefined ) | |
5562 { | |
5563 form.target = '_blank'; | |
5564 postWindow = window; | |
5565 } | |
5566 | |
5567 input.type = 'hidden'; | |
5568 input.name = variable; | |
5569 input.value = value; | |
5570 | |
5571 inputDataset.type = 'hidden'; | |
5572 inputDataset.name = 'dataset'; | |
5573 inputDataset.value = currentDataset; | |
5574 | |
5575 postWindow.document.body.appendChild(form); | |
5576 form.submit(); | |
5577 } | |
5578 | |
5579 function prevDataset() | |
5580 { | |
5581 var newDataset = currentDataset; | |
5582 | |
5583 do | |
5584 { | |
5585 if ( newDataset == 0 ) | |
5586 { | |
5587 newDataset = datasets - 1; | |
5588 } | |
5589 else | |
5590 { | |
5591 newDataset--; | |
5592 } | |
5593 } | |
5594 while ( datasetDropDown.options[newDataset].disabled ); | |
5595 | |
5596 selectDataset(newDataset); | |
5597 } | |
5598 | |
5599 function radiusDecrease() | |
5600 { | |
5601 if ( bufferFactor < .309 ) | |
5602 { | |
5603 bufferFactor += .03; | |
5604 updateViewNeeded = true; | |
5605 } | |
5606 } | |
5607 | |
5608 function radiusIncrease() | |
5609 { | |
5610 if ( bufferFactor > .041 ) | |
5611 { | |
5612 bufferFactor -= .03; | |
5613 updateViewNeeded = true; | |
5614 } | |
5615 } | |
5616 | |
5617 function resetKeyOffset() | |
5618 { | |
5619 currentKey = 1; | |
5620 keyMinTextLeft = centerX + gRadius + buffer - buffer / (keys + 1) / 2 + fontSize / 2; | |
5621 keyMinAngle = 0; | |
5622 } | |
5623 | |
5624 function rgbText(r, g, b) | |
5625 { | |
5626 var rgbArray = | |
5627 [ | |
5628 "rgb(", | |
5629 Math.floor(r), | |
5630 ",", | |
5631 Math.floor(g), | |
5632 ",", | |
5633 Math.floor(b), | |
5634 ")" | |
5635 ]; | |
5636 | |
5637 return rgbArray.join(''); | |
5638 } | |
5639 | |
5640 function round(number) | |
5641 { | |
5642 if ( number >= 1 || number <= -1 ) | |
5643 { | |
5644 return number.toFixed(0); | |
5645 } | |
5646 else | |
5647 { | |
5648 return number.toPrecision(1); | |
5649 } | |
5650 } | |
5651 | |
5652 function roundedRectangle(x, y, width, height, radius) | |
5653 { | |
5654 if ( radius * 2 > width ) | |
5655 { | |
5656 radius = width / 2; | |
5657 } | |
5658 | |
5659 if ( radius * 2 > height ) | |
5660 { | |
5661 radius = height / 2; | |
5662 } | |
5663 | |
5664 context.beginPath(); | |
5665 context.arc(x + radius, y + radius, radius, Math.PI, Math.PI * 3 / 2, false); | |
5666 context.lineTo(x + width - radius, y); | |
5667 context.arc(x + width - radius, y + radius, radius, Math.PI * 3 / 2, Math.PI * 2, false); | |
5668 context.lineTo(x + width, y + height - radius); | |
5669 context.arc(x + width - radius, y + height - radius, radius, 0, Math.PI / 2, false); | |
5670 context.lineTo(x + radius, y + height); | |
5671 context.arc(x + radius, y + height - radius, radius, Math.PI / 2, Math.PI, false); | |
5672 context.lineTo(x, y + radius); | |
5673 } | |
5674 | |
5675 function passClick(e) | |
5676 { | |
5677 mouseClick(e); | |
5678 } | |
5679 | |
5680 function searchResultString(results) | |
5681 { | |
5682 var searchResults = this.searchResults; | |
5683 | |
5684 if ( this.isSearchResult ) | |
5685 { | |
5686 // don't count ourselves | |
5687 searchResults--; | |
5688 } | |
5689 | |
5690 return ' - ' + results + (results > 1 ? ' results' : ' result'); | |
5691 } | |
5692 | |
5693 function setCallBacks() | |
5694 { | |
5695 canvas.onselectstart = function(){return false;} // prevent unwanted highlighting | |
5696 options.onselectstart = function(){return false;} // prevent unwanted highlighting | |
5697 document.onmousemove = mouseMove; | |
5698 window.onblur = focusLost; | |
5699 window.onmouseout = focusLost; | |
5700 document.onkeyup = onKeyUp; | |
5701 document.onkeydown = onKeyDown; | |
5702 canvas.onmousedown = mouseClick; | |
5703 document.onmouseup = mouseUp; | |
5704 keyControl.onclick = toggleKeys; | |
5705 collapseCheckBox = document.getElementById('collapse'); | |
5706 collapseCheckBox.checked = collapse; | |
5707 collapseCheckBox.onclick = handleResize; | |
5708 collapseCheckBox.onmousedown = suppressEvent; | |
5709 maxAbsoluteDepthText = document.getElementById('maxAbsoluteDepth'); | |
5710 maxAbsoluteDepthButtonDecrease = document.getElementById('maxAbsoluteDepthDecrease'); | |
5711 maxAbsoluteDepthButtonIncrease = document.getElementById('maxAbsoluteDepthIncrease'); | |
5712 maxAbsoluteDepthButtonDecrease.onclick = maxAbsoluteDepthDecrease; | |
5713 maxAbsoluteDepthButtonIncrease.onclick = maxAbsoluteDepthIncrease; | |
5714 maxAbsoluteDepthButtonDecrease.onmousedown = suppressEvent; | |
5715 maxAbsoluteDepthButtonIncrease.onmousedown = suppressEvent; | |
5716 fontSizeText = document.getElementById('fontSize'); | |
5717 fontSizeButtonDecrease = document.getElementById('fontSizeDecrease'); | |
5718 fontSizeButtonIncrease = document.getElementById('fontSizeIncrease'); | |
5719 fontSizeButtonDecrease.onclick = fontSizeDecrease; | |
5720 fontSizeButtonIncrease.onclick = fontSizeIncrease; | |
5721 fontSizeButtonDecrease.onmousedown = suppressEvent; | |
5722 fontSizeButtonIncrease.onmousedown = suppressEvent; | |
5723 radiusButtonDecrease = document.getElementById('radiusDecrease'); | |
5724 radiusButtonIncrease = document.getElementById('radiusIncrease'); | |
5725 radiusButtonDecrease.onclick = radiusDecrease; | |
5726 radiusButtonIncrease.onclick = radiusIncrease; | |
5727 radiusButtonDecrease.onmousedown = suppressEvent; | |
5728 radiusButtonIncrease.onmousedown = suppressEvent; | |
5729 maxAbsoluteDepth = 0; | |
5730 backButton = document.getElementById('back'); | |
5731 backButton.onclick = navigateBack; | |
5732 backButton.onmousedown = suppressEvent; | |
5733 forwardButton = document.getElementById('forward'); | |
5734 forwardButton.onclick = navigateForward; | |
5735 forwardButton.onmousedown = suppressEvent; | |
5736 snapshotButton = document.getElementById('snapshot'); | |
5737 snapshotButton.onclick = snapshot; | |
5738 snapshotButton.onmousedown = suppressEvent; | |
5739 detailsName = document.getElementById('detailsName'); | |
5740 detailsExpand = document.getElementById('detailsExpand'); | |
5741 detailsInfo = document.getElementById('detailsInfo'); | |
5742 search = document.getElementById('search'); | |
5743 search.onkeyup = onSearchChange; | |
5744 search.onmousedown = suppressEvent; | |
5745 searchResults = document.getElementById('searchResults'); | |
5746 useHueDiv = document.getElementById('useHueDiv'); | |
5747 linkButton = document.getElementById('linkButton'); | |
5748 linkButton.onclick = showLink; | |
5749 linkButton.onmousedown = suppressEvent; | |
5750 linkText = document.getElementById('linkText'); | |
5751 linkText.onblur = hideLink; | |
5752 linkText.onmousedown = suppressEvent; | |
5753 hide(linkText); | |
5754 var helpButton = document.getElementById('help'); | |
5755 helpButton.onmousedown = suppressEvent; | |
5756 var searchClear = document.getElementById('searchClear'); | |
5757 searchClear.onmousedown = suppressEvent; | |
5758 if ( datasets > 1 ) | |
5759 { | |
5760 datasetDropDown.onmousedown = suppressEvent; | |
5761 var prevDatasetButton = document.getElementById('prevDataset'); | |
5762 prevDatasetButton.onmousedown = suppressEvent; | |
5763 var nextDatasetButton = document.getElementById('nextDataset'); | |
5764 nextDatasetButton.onmousedown = suppressEvent; | |
5765 var lastDatasetButton = document.getElementById('lastDataset'); | |
5766 lastDatasetButton.onmousedown = suppressEvent; | |
5767 } | |
5768 | |
5769 image = document.getElementById('hiddenImage'); | |
5770 image.onload = function() | |
5771 { | |
5772 hiddenPattern = context.createPattern(image, 'repeat'); | |
5773 } | |
5774 | |
5775 var loadingImageElement = document.getElementById('loadingImage'); | |
5776 | |
5777 if ( loadingImageElement ) | |
5778 { | |
5779 loadingImage = loadingImageElement.src; | |
5780 } | |
5781 } | |
5782 | |
5783 function selectDataset(newDataset) | |
5784 { | |
5785 lastDataset = currentDataset; | |
5786 currentDataset = newDataset | |
5787 if ( datasets > 1 ) | |
5788 { | |
5789 datasetDropDown.selectedIndex = currentDataset; | |
5790 updateDatasetButtons(); | |
5791 datasetAlpha.start = 1.5; | |
5792 datasetChanged = true; | |
5793 } | |
5794 head.setMagnitudes(0); | |
5795 head.setDepth(1, 1); | |
5796 head.setMaxDepths(); | |
5797 handleResize(); | |
5798 } | |
5799 | |
5800 function selectLastDataset() | |
5801 { | |
5802 selectDataset(lastDataset); | |
5803 handleResize(); | |
5804 } | |
5805 | |
5806 function selectNode(newNode) | |
5807 { | |
5808 if ( selectedNode != newNode ) | |
5809 { | |
5810 // truncate history at current location to create a new branch | |
5811 // | |
5812 nodeHistory.length = nodeHistoryPosition; | |
5813 | |
5814 if ( selectedNode != 0 ) | |
5815 { | |
5816 nodeHistory.push(selectedNode); | |
5817 nodeHistoryPosition++; | |
5818 } | |
5819 | |
5820 setSelectedNode(newNode); | |
5821 //updateView(); | |
5822 } | |
5823 | |
5824 updateDatasetButtons(); | |
5825 } | |
5826 | |
5827 function setFocus(node) | |
5828 { | |
5829 if ( node == focusNode ) | |
5830 { | |
5831 // return; | |
5832 } | |
5833 | |
5834 focusNode = node; | |
5835 | |
5836 if ( node.href ) | |
5837 { | |
5838 detailsName.innerHTML = | |
5839 '<a target="_blank" href="' + node.href + '">' + node.name + '</a>'; | |
5840 } | |
5841 else | |
5842 { | |
5843 detailsName.innerHTML = node.name; | |
5844 } | |
5845 | |
5846 var table = '<table>'; | |
5847 //TODO: use CSS margins instead of an additional column | |
5848 table += '<tr><td></td><td></td></tr>'; | |
5849 | |
5850 for ( var i = 0; i < node.attributes.length; i++ ) | |
5851 { | |
5852 if ( attributes[i].displayName && node.attributes[i] != undefined ) | |
5853 { | |
5854 var index = node.attributes[i].length == 1 && attributes[i].mono ? 0 : currentDataset; | |
5855 | |
5856 if ( typeof node.attributes[i][currentDataset] == 'number' || node.attributes[i][index] != undefined && node.attributes[i][currentDataset] != '' ) | |
5857 { | |
5858 var value = node.attributes[i][index]; | |
5859 | |
5860 if ( attributes[i].listNode != undefined ) | |
5861 { | |
5862 value = | |
5863 '<a href="" onclick="showList(' + | |
5864 attributeIndex(attributes[i].listNode) + ',' + i + | |
5865 ',false);return false;" title="Show list">' + | |
5866 value + '</a>'; | |
5867 } | |
5868 else if ( attributes[i].listAll != undefined ) | |
5869 { | |
5870 value = | |
5871 '<a href="" onclick="showList(' + | |
5872 attributeIndex(attributes[i].listAll) + ',' + i + | |
5873 ',true);return false;" title="Show list">' + | |
5874 value + '</a>'; | |
5875 } | |
5876 else if ( attributes[i].dataNode != undefined && dataEnabled ) | |
5877 { | |
5878 value = | |
5879 '<a href="" onclick="showData(' + | |
5880 attributeIndex(attributes[i].dataNode) + ',' + i + | |
5881 ',false);return false;" title="Show data">' + | |
5882 value + '</a>'; | |
5883 } | |
5884 else if ( attributes[i].dataAll != undefined && dataEnabled ) | |
5885 { | |
5886 value = | |
5887 '<a href="" onclick="showData(' + | |
5888 attributeIndex(attributes[i].dataAll) + ',' + i + | |
5889 ',true);return false;" title="Show data">' + | |
5890 value + '</a>'; | |
5891 } | |
5892 | |
5893 table += | |
5894 '<tr><td><strong>' + attributes[i].displayName + ':</strong></td><td>' + | |
5895 value + '</td></tr>'; | |
5896 } | |
5897 } | |
5898 } | |
5899 | |
5900 table += '</table>'; | |
5901 detailsInfo.innerHTML = table; | |
5902 | |
5903 detailsExpand.disabled = !focusNode.hasChildren() || focusNode == selectedNode; | |
5904 } | |
5905 | |
5906 function setSelectedNode(newNode) | |
5907 { | |
5908 if ( selectedNode && selectedNode.hasParent(newNode) ) | |
5909 { | |
5910 zoomOut = true; | |
5911 } | |
5912 else | |
5913 { | |
5914 zoomOut = false; | |
5915 } | |
5916 | |
5917 selectedNodeLast = selectedNode; | |
5918 selectedNode = newNode; | |
5919 | |
5920 //if ( focusNode != selectedNode ) | |
5921 { | |
5922 setFocus(selectedNode); | |
5923 } | |
5924 } | |
5925 | |
5926 function waitForData(dataWindow, target, title, time, postUrl, postVar) | |
5927 { | |
5928 if ( nodeData.length == target ) | |
5929 { | |
5930 if ( postUrl != undefined ) | |
5931 { | |
5932 for ( var i = 0; i < nodeData.length; i++ ) | |
5933 { | |
5934 nodeData[i] = nodeData[i].replace(/\n/g, ','); | |
5935 } | |
5936 | |
5937 var postString = nodeData.join(''); | |
5938 postString = postString.slice(0, -1); | |
5939 | |
5940 dataWindow.document.body.removeChild(dataWindow.document.getElementById('loading')); | |
5941 document.body.removeChild(document.getElementById('data')); | |
5942 | |
5943 post(postUrl, postVar, postString, dataWindow); | |
5944 } | |
5945 else | |
5946 { | |
5947 //dataWindow.document.body.removeChild(dataWindow.document.getElementById('loading')); | |
5948 //document.body.removeChild(document.getElementById('data')); | |
5949 | |
5950 dataWindow.document.open(); | |
5951 dataWindow.document.write('<pre>' + nodeData.join('') + '</pre>'); | |
5952 dataWindow.document.close(); | |
5953 } | |
5954 | |
5955 dataWindow.document.title = title; // replace after document.write() | |
5956 } | |
5957 else | |
5958 { | |
5959 var date = new Date(); | |
5960 | |
5961 if ( date.getTime() - time > 10000 ) | |
5962 { | |
5963 dataWindow.document.body.removeChild(dataWindow.document.getElementById('loading')); | |
5964 document.body.removeChild(document.getElementById('data')); | |
5965 dataWindow.document.body.innerHTML = | |
5966 'Timed out loading supplemental files for:<br/>' + document.location; | |
5967 } | |
5968 else | |
5969 { | |
5970 setTimeout(function() {waitForData(dataWindow, target, title, time, postUrl, postVar);}, 100); | |
5971 } | |
5972 } | |
5973 } | |
5974 | |
5975 function data(newData) | |
5976 { | |
5977 nodeData.push(newData); | |
5978 } | |
5979 | |
5980 function enableData() | |
5981 { | |
5982 dataEnabled = true; | |
5983 } | |
5984 | |
5985 function showData(indexData, indexAttribute, summary) | |
5986 { | |
5987 var dataWindow = window.open('', '_blank'); | |
5988 var title = 'Krona - ' + attributes[indexAttribute].displayName + ' - ' + focusNode.name; | |
5989 dataWindow.document.title = title; | |
5990 | |
5991 nodeData = new Array(); | |
5992 | |
5993 if ( dataWindow && dataWindow.document && dataWindow.document.body != null ) | |
5994 { | |
5995 //var loadImage = document.createElement('img'); | |
5996 //loadImage.src = "file://localhost/Users/ondovb/Krona/KronaTools/img/loading.gif"; | |
5997 //loadImage.id = "loading"; | |
5998 //loadImage.alt = "Loading..."; | |
5999 //dataWindow.document.body.appendChild(loadImage); | |
6000 dataWindow.document.body.innerHTML = | |
6001 '<img id="loading" src="' + loadingImage + '" alt="Loading..."></img>'; | |
6002 } | |
6003 | |
6004 var scripts = document.createElement('div'); | |
6005 scripts.id = 'data'; | |
6006 document.body.appendChild(scripts); | |
6007 | |
6008 var files = focusNode.getData(indexData, summary); | |
6009 | |
6010 var date = new Date(); | |
6011 var time = date.getTime(); | |
6012 | |
6013 for ( var i = 0; i < files.length; i++ ) | |
6014 { | |
6015 var script = document.createElement('script'); | |
6016 script.src = files[i] + '?' + time; | |
6017 scripts.appendChild(script); | |
6018 } | |
6019 | |
6020 waitForData(dataWindow, files.length, title, time, attributes[indexAttribute].postUrl, attributes[indexAttribute].postVar); | |
6021 | |
6022 return false; | |
6023 } | |
6024 | |
6025 function showList(indexList, indexAttribute, summary) | |
6026 { | |
6027 var list = focusNode.getList(indexList, summary); | |
6028 | |
6029 if ( attributes[indexAttribute].postUrl != undefined ) | |
6030 { | |
6031 post(attributes[indexAttribute].postUrl, attributes[indexAttribute].postVar, list.join(',')); | |
6032 } | |
6033 else | |
6034 { | |
6035 var dataWindow = window.open('', '_blank'); | |
6036 | |
6037 if ( true || navigator.appName == 'Microsoft Internet Explorer' ) // :( | |
6038 { | |
6039 dataWindow.document.open(); | |
6040 dataWindow.document.write('<pre>' + list.join('\n') + '</pre>'); | |
6041 dataWindow.document.close(); | |
6042 } | |
6043 else | |
6044 { | |
6045 var pre = document.createElement('pre'); | |
6046 dataWindow.document.body.appendChild(pre); | |
6047 pre.innerHTML = list; | |
6048 } | |
6049 | |
6050 dataWindow.document.title = 'Krona - ' + attributes[indexAttribute].displayName + ' - ' + focusNode.name; | |
6051 } | |
6052 } | |
6053 | |
6054 function snapshot() | |
6055 { | |
6056 svg = svgHeader(); | |
6057 | |
6058 resetKeyOffset(); | |
6059 | |
6060 snapshotMode = true; | |
6061 | |
6062 selectedNode.draw(false, true); | |
6063 selectedNode.draw(true, true); | |
6064 | |
6065 if ( focusNode != 0 && focusNode != selectedNode ) | |
6066 { | |
6067 context.globalAlpha = 1; | |
6068 focusNode.drawHighlight(true); | |
6069 } | |
6070 | |
6071 if ( hueDisplayName && useHue() ) | |
6072 { | |
6073 drawLegendSVG(); | |
6074 } | |
6075 | |
6076 snapshotMode = false; | |
6077 | |
6078 svg += svgFooter(); | |
6079 | |
6080 var snapshotWindow = window.open('', '_blank', '', 'replace=false'); | |
6081 snapshotWindow.document.write('<html><body><a href="data:image/svg+xml,' + encodeURIComponent(svg) + '" download="snapshot.svg">Download Snapshot</a></html></body>'); | |
6082 snapshotWindow.document.write(svg); | |
6083 } | |
6084 | |
6085 function save() | |
6086 { | |
6087 alert(document.body.innerHTML); | |
6088 } | |
6089 | |
6090 function spacer() | |
6091 { | |
6092 if ( snapshotMode ) | |
6093 { | |
6094 return '   '; | |
6095 } | |
6096 else | |
6097 { | |
6098 return ' '; | |
6099 } | |
6100 } | |
6101 | |
6102 function suppressEvent(e) | |
6103 { | |
6104 e.cancelBubble = true; | |
6105 if (e.stopPropagation) e.stopPropagation(); | |
6106 } | |
6107 | |
6108 function svgFooter() | |
6109 { | |
6110 return '</svg>'; | |
6111 } | |
6112 | |
6113 function svgHeader() | |
6114 { | |
6115 var patternWidth = fontSize * .6;//radius / 50; | |
6116 | |
6117 return '\ | |
6118 <?xml version="1.0" standalone="no"?>\ | |
6119 <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" \ | |
6120 "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">\ | |
6121 <svg width="' + imageWidth + '" height="' + imageHeight + '" version="1.1"\ | |
6122 xmlns="http://www.w3.org/2000/svg">\ | |
6123 <title>Krona (snapshot) - ' + | |
6124 (datasets > 1 ? datasetNames[currentDataset] + ' - ' : '') + selectedNode.name + | |
6125 '</title>\ | |
6126 <defs>\ | |
6127 <style type="text/css">\ | |
6128 text {font-size: ' + fontSize + 'px; font-family: ' + fontFamily + '; dominant-baseline:central}\ | |
6129 path {stroke-width:' + thinLineWidth * fontSize / 12 + ';}\ | |
6130 path.wedge {stroke:none}\ | |
6131 path.line {fill:none;stroke:black;}\ | |
6132 line {stroke:black;stroke-width:' + thinLineWidth * fontSize / 12 + ';}\ | |
6133 line.tick {stroke-width:' + thinLineWidth * fontSize / 6 + ';}\ | |
6134 line.pattern {stroke-width:' + thinLineWidth * fontSize / 18 + ';}\ | |
6135 circle {fill:none;stroke:black;stroke-width:' + thinLineWidth * fontSize / 12 + ';}\ | |
6136 rect {stroke:black;stroke-width:' + thinLineWidth * fontSize / 12 + ';}\ | |
6137 .highlight {stroke:black;stroke-width:'+ highlightLineWidth * fontSize / 12 + ';}\ | |
6138 .searchHighlight {fill:rgb(255, 255, 100);stroke:none;}\ | |
6139 </style>\ | |
6140 <pattern id="hiddenPattern" patternUnits="userSpaceOnUse" \ | |
6141 x="0" y="0" width="' + patternWidth + '" height="' + patternWidth + '">\ | |
6142 <line class="pattern" x1="0" y1="0" x2="' + patternWidth / 2 + '" y2="' + patternWidth / 2 + '"/>\ | |
6143 <line class="pattern" x1="' + patternWidth / 2 + '" y1="' + patternWidth + | |
6144 '" x2="' + patternWidth + '" y2="' + patternWidth / 2 + '"/>\ | |
6145 </pattern>\ | |
6146 </defs>\ | |
6147 '; | |
6148 } | |
6149 | |
6150 function svgText(text, x, y, anchor, bold, color) | |
6151 { | |
6152 if ( typeof(anchor) == 'undefined' ) | |
6153 { | |
6154 anchor = 'start'; | |
6155 } | |
6156 | |
6157 if ( color == undefined ) | |
6158 { | |
6159 color = 'black'; | |
6160 } | |
6161 | |
6162 return '<text x="' + x + '" y="' + y + | |
6163 '" style="font-color:' + color + ';font-weight:' + (bold ? 'bold' : 'normal') + | |
6164 '" text-anchor="' + anchor + '">' + text + '</text>'; | |
6165 } | |
6166 | |
6167 function toggleKeys() | |
6168 { | |
6169 if ( showKeys ) | |
6170 { | |
6171 keyControl.value = '…'; | |
6172 showKeys = false; | |
6173 } | |
6174 else | |
6175 { | |
6176 keyControl.value = 'x'; | |
6177 showKeys = true; | |
6178 } | |
6179 | |
6180 updateKeyControl(); | |
6181 | |
6182 if ( progress == 1 ) | |
6183 { | |
6184 draw(); | |
6185 } | |
6186 } | |
6187 | |
6188 function update() | |
6189 { | |
6190 if ( ! head ) | |
6191 { | |
6192 return; | |
6193 } | |
6194 | |
6195 if ( mouseDown && focusNode != selectedNode ) | |
6196 { | |
6197 var date = new Date(); | |
6198 | |
6199 if ( date.getTime() - mouseDownTime > quickLookHoldLength ) | |
6200 { | |
6201 if ( focusNode.hasChildren() ) | |
6202 { | |
6203 expand(focusNode); | |
6204 quickLook = true; | |
6205 } | |
6206 } | |
6207 } | |
6208 | |
6209 if ( updateViewNeeded ) | |
6210 { | |
6211 resize(); | |
6212 mouseX = -1; | |
6213 mouseY = -1; | |
6214 | |
6215 collapse = collapseCheckBox.checked; | |
6216 compress = true;//compressCheckBox.checked; | |
6217 shorten = true;//shortenCheckBox.checked; | |
6218 | |
6219 checkSelectedCollapse(); | |
6220 updateMaxAbsoluteDepth(); | |
6221 | |
6222 if ( focusNode.getCollapse() || focusNode.depth > maxAbsoluteDepth ) | |
6223 { | |
6224 setFocus(selectedNode); | |
6225 } | |
6226 else | |
6227 { | |
6228 setFocus(focusNode); | |
6229 } | |
6230 | |
6231 updateView(); | |
6232 | |
6233 updateViewNeeded = false; | |
6234 } | |
6235 | |
6236 var date = new Date(); | |
6237 progress = (date.getTime() - tweenStartTime) / tweenLength; | |
6238 // progress += .01; | |
6239 | |
6240 if ( progress >= 1 ) | |
6241 { | |
6242 progress = 1; | |
6243 } | |
6244 | |
6245 if ( progress != progressLast ) | |
6246 { | |
6247 tweenFactor =// progress; | |
6248 (1 / (1 + Math.exp(-tweenCurvature * (progress - .5))) - .5) / | |
6249 (tweenMax - .5) / 2 + .5; | |
6250 | |
6251 if ( progress == 1 ) | |
6252 { | |
6253 snapshotButton.disabled = false; | |
6254 zoomOut = false; | |
6255 | |
6256 //updateKeyControl(); | |
6257 | |
6258 if ( ! quickLook ) | |
6259 { | |
6260 //checkHighlight(); | |
6261 } | |
6262 | |
6263 | |
6264 if ( fpsDisplay ) | |
6265 { | |
6266 fpsDisplay.innerHTML = 'fps: ' + Math.round(tweenFrames * 1000 / tweenLength); | |
6267 } | |
6268 } | |
6269 | |
6270 draw(); | |
6271 } | |
6272 | |
6273 progressLast = progress; | |
6274 } | |
6275 | |
6276 function updateDatasetButtons() | |
6277 { | |
6278 if ( datasets == 1 ) | |
6279 { | |
6280 return; | |
6281 } | |
6282 | |
6283 var node = selectedNode ? selectedNode : head; | |
6284 | |
6285 datasetButtonLast.disabled = | |
6286 node.attributes[magnitudeIndex][lastDataset] == 0; | |
6287 | |
6288 datasetButtonPrev.disabled = true; | |
6289 datasetButtonNext.disabled = true; | |
6290 | |
6291 for ( var i = 0; i < datasets; i++ ) | |
6292 { | |
6293 var disable = node.attributes[magnitudeIndex][i] == 0; | |
6294 | |
6295 datasetDropDown.options[i].disabled = disable; | |
6296 | |
6297 if ( ! disable ) | |
6298 { | |
6299 if ( i != currentDataset ) | |
6300 { | |
6301 datasetButtonPrev.disabled = false; | |
6302 datasetButtonNext.disabled = false; | |
6303 } | |
6304 } | |
6305 } | |
6306 } | |
6307 | |
6308 function updateDatasetWidths() | |
6309 { | |
6310 if ( datasets > 1 ) | |
6311 { | |
6312 for ( var i = 0; i < datasets; i++ ) | |
6313 { | |
6314 context.font = fontBold; | |
6315 var dim = context.measureText(datasetNames[i]); | |
6316 datasetWidths[i] = dim.width; | |
6317 } | |
6318 } | |
6319 } | |
6320 | |
6321 function updateKeyControl() | |
6322 { | |
6323 if ( keys == 0 )//|| progress != 1 ) | |
6324 { | |
6325 keyControl.style.visibility = 'hidden'; | |
6326 } | |
6327 else | |
6328 { | |
6329 keyControl.style.visibility = 'visible'; | |
6330 keyControl.style.right = margin + 'px'; | |
6331 | |
6332 if ( showKeys ) | |
6333 { | |
6334 keyControl.style.top = | |
6335 imageHeight - | |
6336 ( | |
6337 keys * (keySize + keyBuffer) - | |
6338 keyBuffer + | |
6339 margin + | |
6340 keyControl.clientHeight * 1.5 | |
6341 ) + 'px'; | |
6342 } | |
6343 else | |
6344 { | |
6345 keyControl.style.top = | |
6346 (imageHeight - margin - keyControl.clientHeight) + 'px'; | |
6347 } | |
6348 } | |
6349 } | |
6350 | |
6351 function updateView() | |
6352 { | |
6353 if ( selectedNode.depth > maxAbsoluteDepth - 1 ) | |
6354 { | |
6355 maxAbsoluteDepth = selectedNode.depth + 1; | |
6356 } | |
6357 | |
6358 highlightedNode = selectedNode; | |
6359 | |
6360 angleFactor = 2 * Math.PI / (selectedNode.magnitude); | |
6361 | |
6362 maxPossibleDepth = Math.floor(gRadius / (fontSize * minRingWidthFactor)); | |
6363 | |
6364 if ( maxPossibleDepth < 4 ) | |
6365 { | |
6366 maxPossibleDepth = 4; | |
6367 } | |
6368 | |
6369 var minRadiusInner = fontSize * 8 / gRadius; | |
6370 var minRadiusFirst = fontSize * 6 / gRadius; | |
6371 var minRadiusOuter = fontSize * 5 / gRadius; | |
6372 | |
6373 if ( .25 < minRadiusInner ) | |
6374 { | |
6375 minRadiusInner = .25; | |
6376 } | |
6377 | |
6378 if ( .15 < minRadiusFirst ) | |
6379 { | |
6380 minRadiusFirst = .15; | |
6381 } | |
6382 | |
6383 if ( .15 < minRadiusOuter ) | |
6384 { | |
6385 minRadiusOuter = .15; | |
6386 } | |
6387 | |
6388 // visibility of nodes depends on the depth they are displayed at, | |
6389 // so we need to set the max depth assuming they can all be displayed | |
6390 // and iterate it down based on the deepest child node we can display | |
6391 // | |
6392 var maxDepth; | |
6393 var newMaxDepth = selectedNode.getMaxDepth() - selectedNode.getDepth() + 1; | |
6394 // | |
6395 do | |
6396 { | |
6397 maxDepth = newMaxDepth; | |
6398 | |
6399 if ( ! compress && maxDepth > maxPossibleDepth ) | |
6400 { | |
6401 maxDepth = maxPossibleDepth; | |
6402 } | |
6403 | |
6404 if ( compress ) | |
6405 { | |
6406 compressedRadii = new Array(maxDepth); | |
6407 | |
6408 compressedRadii[0] = minRadiusInner; | |
6409 | |
6410 var offset = 0; | |
6411 | |
6412 while | |
6413 ( | |
6414 lerp | |
6415 ( | |
6416 Math.atan(offset + 2), | |
6417 Math.atan(offset + 1), | |
6418 Math.atan(maxDepth + offset - 1), | |
6419 minRadiusInner, | |
6420 1 - minRadiusOuter | |
6421 ) - minRadiusInner > minRadiusFirst && | |
6422 offset < 10 | |
6423 ) | |
6424 { | |
6425 offset++; | |
6426 } | |
6427 | |
6428 offset--; | |
6429 | |
6430 for ( var i = 1; i < maxDepth; i++ ) | |
6431 { | |
6432 compressedRadii[i] = lerp | |
6433 ( | |
6434 Math.atan(i + offset), | |
6435 Math.atan(offset), | |
6436 Math.atan(maxDepth + offset - 1), | |
6437 minRadiusInner, | |
6438 1 - minRadiusOuter | |
6439 ) | |
6440 } | |
6441 } | |
6442 else | |
6443 { | |
6444 nodeRadius = 1 / maxDepth; | |
6445 } | |
6446 | |
6447 newMaxDepth = selectedNode.maxVisibleDepth(maxDepth); | |
6448 | |
6449 if ( compress ) | |
6450 { | |
6451 if ( newMaxDepth <= maxPossibleDepth ) | |
6452 { | |
6453 // compress | |
6454 } | |
6455 } | |
6456 else | |
6457 { | |
6458 if ( newMaxDepth > maxPossibleDepth ) | |
6459 { | |
6460 newMaxDepth = maxPossibleDepth; | |
6461 } | |
6462 } | |
6463 } | |
6464 while ( newMaxDepth < maxDepth ); | |
6465 | |
6466 maxDisplayDepth = maxDepth; | |
6467 | |
6468 lightnessFactor = (lightnessMax - lightnessBase) / (maxDepth > 8 ? 8 : maxDepth); | |
6469 keys = 0; | |
6470 | |
6471 nLabelOffsets = new Array(maxDisplayDepth - 1); | |
6472 labelOffsets = new Array(maxDisplayDepth - 1); | |
6473 labelLastNodes = new Array(maxDisplayDepth - 1); | |
6474 labelFirstNodes = new Array(maxDisplayDepth - 1); | |
6475 | |
6476 for ( var i = 0; i < maxDisplayDepth - 1; i++ ) | |
6477 { | |
6478 if ( compress ) | |
6479 { | |
6480 if ( i == maxDisplayDepth - 1 ) | |
6481 { | |
6482 nLabelOffsets[i] = 0; | |
6483 } | |
6484 else | |
6485 { | |
6486 var width = | |
6487 (compressedRadii[i + 1] - compressedRadii[i]) * | |
6488 gRadius; | |
6489 | |
6490 nLabelOffsets[i] = Math.floor(width / fontSize / 1.2); | |
6491 | |
6492 if ( nLabelOffsets[i] > 2 ) | |
6493 { | |
6494 nLabelOffsets[i] = min | |
6495 ( | |
6496 Math.floor(width / fontSize / 1.75), | |
6497 5 | |
6498 ); | |
6499 } | |
6500 } | |
6501 } | |
6502 else | |
6503 { | |
6504 nLabelOffsets[i] = Math.max | |
6505 ( | |
6506 Math.floor(Math.sqrt((nodeRadius * gRadius / fontSize)) * 1.5), | |
6507 3 | |
6508 ); | |
6509 } | |
6510 | |
6511 labelOffsets[i] = Math.floor((nLabelOffsets[i] - 1) / 2); | |
6512 labelLastNodes[i] = new Array(nLabelOffsets[i] + 1); | |
6513 labelFirstNodes[i] = new Array(nLabelOffsets[i] + 1); | |
6514 | |
6515 for ( var j = 0; j <= nLabelOffsets[i]; j++ ) | |
6516 { | |
6517 // these arrays will allow nodes with neighboring labels to link to | |
6518 // each other to determine max label length | |
6519 | |
6520 labelLastNodes[i][j] = 0; | |
6521 labelFirstNodes[i][j] = 0; | |
6522 } | |
6523 } | |
6524 | |
6525 fontSizeText.innerHTML = fontSize; | |
6526 fontNormal = fontSize + 'px ' + fontFamily; | |
6527 context.font = fontNormal; | |
6528 fontBold = 'bold ' + fontSize + 'px ' + fontFamily; | |
6529 tickLength = fontSize * .7; | |
6530 | |
6531 head.setTargets(0); | |
6532 | |
6533 keySize = ((imageHeight - margin * 3) * 1 / 2) / keys * 3 / 4; | |
6534 | |
6535 if ( keySize > fontSize * maxKeySizeFactor ) | |
6536 { | |
6537 keySize = fontSize * maxKeySizeFactor; | |
6538 } | |
6539 | |
6540 keyBuffer = keySize / 3; | |
6541 | |
6542 fontSizeLast = fontSize; | |
6543 | |
6544 if ( datasetChanged ) | |
6545 { | |
6546 datasetChanged = false; | |
6547 } | |
6548 else | |
6549 { | |
6550 datasetAlpha.start = 0; | |
6551 } | |
6552 | |
6553 var date = new Date(); | |
6554 tweenStartTime = date.getTime(); | |
6555 progress = 0; | |
6556 tweenFrames = 0; | |
6557 | |
6558 updateKeyControl(); | |
6559 updateDatasetWidths(); | |
6560 | |
6561 document.title = 'Krona - ' + selectedNode.name; | |
6562 updateNavigationButtons(); | |
6563 snapshotButton.disabled = true; | |
6564 | |
6565 maxAbsoluteDepthText.innerHTML = maxAbsoluteDepth - 1; | |
6566 | |
6567 maxAbsoluteDepthButtonDecrease.disabled = (maxAbsoluteDepth == 2); | |
6568 maxAbsoluteDepthButtonIncrease.disabled = (maxAbsoluteDepth == head.maxDepth); | |
6569 | |
6570 if ( collapse != collapseLast && search.value != '' ) | |
6571 { | |
6572 onSearchChange(); | |
6573 collapseLast = collapse; | |
6574 } | |
6575 } | |
6576 | |
6577 function updateMaxAbsoluteDepth() | |
6578 { | |
6579 while ( maxAbsoluteDepth > 1 && selectedNode.depth > maxAbsoluteDepth - 1 ) | |
6580 { | |
6581 selectedNode = selectedNode.getParent(); | |
6582 } | |
6583 } | |
6584 | |
6585 function updateNavigationButtons() | |
6586 { | |
6587 backButton.disabled = (nodeHistoryPosition == 0); | |
6588 // upButton.disabled = (selectedNode.getParent() == 0); | |
6589 forwardButton.disabled = (nodeHistoryPosition == nodeHistory.length); | |
6590 } | |
6591 | |
6592 function useHue() | |
6593 { | |
6594 return useHueCheckBox && useHueCheckBox.checked; | |
6595 } | |
6596 /* | |
6597 function zoomOut() | |
6598 { | |
6599 return ( | |
6600 selectedNodeLast != 0 && | |
6601 selectedNodeLast.getDepth() < selectedNode.getDepth()); | |
6602 } | |
6603 */ | |
6604 </script> | |
6605 </head> | |
6606 <body> | |
6607 <img id="hiddenImage" src="" style="display:none" alt="Hidden Image"/> | |
6608 <img id="loadingImage" src="" style="display:none" alt="Loading Indicator"/> | |
6609 <img id="logo" src="" style="display:none" alt="Logo of Krona"/> | |
6610 <noscript>Javascript must be enabled to view this page.</noscript> | |
6611 <div style="display:none"> | |
6612 <krona collapse="true" key="true"> | |
6613 <attributes magnitude="count"> | |
6614 <list>members</list> | |
6615 <attribute display="Reads" listAll="members">count</attribute> | |
6616 <attribute display="Unassigned" listNode="members">unassigned</attribute> | |
6617 <attribute display="Rank" mono="true">rank</attribute> | |
6618 </attributes> | |
6619 <datasets> | |
6620 <dataset>0</dataset> | |
6621 </datasets> | |
6622 <node name="Root"> | |
6623 <count><val>100</val></count> | |
6624 <node name="Eukaryota"> | |
6625 <rank><val>superkingdom</val></rank> | |
6626 <count><val>94</val></count> | |
6627 <node name="Metazoa"> | |
6628 <rank><val>kingdom</val></rank> | |
6629 <count><val>94</val></count> | |
6630 <node name="Chordata"> | |
6631 <rank><val>phylum</val></rank> | |
6632 <count><val>94</val></count> | |
6633 <node name="Craniata"> | |
6634 <rank><val>subphylum</val></rank> | |
6635 <count><val>94</val></count> | |
6636 <node name="Gnathostomata"> | |
6637 <rank><val>superclass</val></rank> | |
6638 <count><val>94</val></count> | |
6639 <node name="Mammalia"> | |
6640 <rank><val>class</val></rank> | |
6641 <count><val>94</val></count> | |
6642 <node name="Euarchontoglires"> | |
6643 <count><val>94</val></count> | |
6644 <rank><val>superorder</val></rank> | |
6645 <node name="Rodentia"> | |
6646 <count><val>94</val></count> | |
6647 <rank><val>order</val></rank> | |
6648 <node name="Sciurognathi"> | |
6649 <rank><val>suborder</val></rank> | |
6650 <count><val>94</val></count> | |
6651 <node name="Muridae"> | |
6652 <rank><val>family</val></rank> | |
6653 <count><val>94</val></count> | |
6654 <node name="Murinae"> | |
6655 <rank><val>subfamily</val></rank> | |
6656 <count><val>94</val></count> | |
6657 <node name="Rattus"> | |
6658 <rank><val>genus</val></rank> | |
6659 <count><val>94</val></count> | |
6660 <members> | |
6661 <vals><val>IA_1-296315</val><val>IA_1-322295</val></vals> | |
6662 </members> | |
6663 <unassigned><val>2</val></unassigned> | |
6664 <node name="Rattus norvegicus"> | |
6665 <members> | |
6666 <vals><val>IA_1-144417</val><val>IA_1-278966</val><val>IA_1-314709</val><val>IA_1-324951</val><val>IA_1-27817</val><val>IA_1-95255</val><val>IA_1-104173</val><val>IA_1-135979</val><val>IA_1-139090</val><val>IA_1-139090</val><val>IA_1-139090</val><val>IA_1-144996</val><val>IA_1-160446</val><val>IA_1-160446</val><val>IA_1-160446</val><val>IA_1-160446</val><val>IA_1-160446</val><val>IA_1-160446</val><val>IA_1-160446</val><val>IA_1-160446</val><val>IA_1-161439</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-190855</val><val>IA_1-205154</val><val>IA_1-205154</val><val>IA_1-205154</val><val>IA_1-205154</val><val>IA_1-205154</val><val>IA_1-205154</val><val>IA_1-205154</val><val>IA_1-205154</val><val>IA_1-205154</val><val>IA_1-205154</val><val>IA_1-216231</val><val>IA_1-236286</val><val>IA_1-236286</val><val>IA_1-236286</val><val>IA_1-236286</val><val>IA_1-236286</val><val>IA_1-236286</val><val>IA_1-236286</val><val>IA_1-236286</val><val>IA_1-236286</val><val>IA_1-236286</val><val>IA_1-236286</val><val>IA_1-236286</val><val>IA_1-237681</val><val>IA_1-250166</val><val>IA_1-254274</val><val>IA_1-254274</val><val>IA_1-27817</val><val>IA_1-29000</val><val>IA_1-291427</val><val>IA_1-291427</val><val>IA_1-293054</val><val>IA_1-293054</val><val>IA_1-296315</val><val>IA_1-310974</val><val>IA_1-310974</val><val>IA_1-311282</val><val>IA_1-311282</val><val>IA_1-42600</val><val>IA_1-45102</val><val>IA_1-45102</val><val>IA_1-48105</val><val>IA_1-48105</val><val>IA_1-57254</val><val>IA_1-61975</val><val>IA_1-61975</val><val>IA_1-66943</val><val>IA_1-68288</val><val>IA_1-82334</val><val>IA_1-95526</val></vals> | |
6667 </members> | |
6668 <rank><val>species</val></rank> | |
6669 <count><val>92</val></count> | |
6670 </node> | |
6671 </node> | |
6672 </node> | |
6673 </node> | |
6674 </node> | |
6675 </node> | |
6676 </node> | |
6677 </node> | |
6678 </node> | |
6679 </node> | |
6680 </node> | |
6681 </node> | |
6682 </node> | |
6683 <node name="Bacteria"> | |
6684 <count><val>6</val></count> | |
6685 <rank><val>superkingdom</val></rank> | |
6686 <node name="Proteobacteria"> | |
6687 <count><val>6</val></count> | |
6688 <rank><val>phylum</val></rank> | |
6689 <node name="Gammaproteobacteria"> | |
6690 <rank><val>class</val></rank> | |
6691 <count><val>6</val></count> | |
6692 <node name="Enterobacteriales"> | |
6693 <rank><val>order</val></rank> | |
6694 <count><val>6</val></count> | |
6695 <node name="Enterobacteriaceae"> | |
6696 <rank><val>family</val></rank> | |
6697 <count><val>6</val></count> | |
6698 <node name="Shigella"> | |
6699 <count><val>6</val></count> | |
6700 <rank><val>genus</val></rank> | |
6701 <node name="Shigella flexneri"> | |
6702 <count><val>6</val></count> | |
6703 <rank><val>species</val></rank> | |
6704 <members> | |
6705 <vals><val>IA_1-79371</val><val>IA_1-84488</val><val>IA_1-270826</val><val>IA_1-285361</val><val>IA_1-93958</val><val>IA_1-99821</val></vals> | |
6706 </members> | |
6707 </node> | |
6708 </node> | |
6709 </node> | |
6710 </node> | |
6711 </node> | |
6712 </node> | |
6713 </node> | |
6714 </node> | |
6715 </krona> | |
6716 </div></body></html> |