comparison test-data/krona_test1.html @ 1:09552faff9c0 draft

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