source: temp/branches/ec-cube-beta/html/test/kakinaka/js/treeview/treeview.js @ 11379

Revision 11379, 52.3 KB checked in by kaki, 18 years ago (diff)
  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
Line 
1/*                                                                                                                                                     
2Copyright (c) 2006, Yahoo! Inc. All rights reserved.
3Code licensed under the BSD License:
4http://developer.yahoo.net/yui/license.txt
5version: 0.11.3
6*/
7
8/**
9 * Contains the tree view state data and the root node.  This is an
10 * ordered tree; child nodes will be displayed in the order created, and
11 * there currently is no way to change this.
12 *
13 * @constructor
14 * @param {string|HTMLElement} id The id of the element, or the element
15 * itself that the tree will be inserted into.
16 */
17YAHOO.widget.TreeView = function(id) {
18    if (id) { this.init(id); }
19};
20
21/**
22 * Count of all nodes in all trees
23 * @type int
24 */
25YAHOO.widget.TreeView.nodeCount = 0;
26
27YAHOO.widget.TreeView.prototype = {
28
29    /**
30     * The id of tree container element
31     *
32     * @type String
33     */
34    id: null,
35
36    /**
37     * The host element for this tree
38     * @private
39     */
40    _el: null,
41
42     /**
43     * Flat collection of all nodes in this tree
44     *
45     * @type Node[]
46     * @private
47     */
48    _nodes: null,
49
50    /**
51     * We lock the tree control while waiting for the dynamic loader to return
52     *
53     * @type boolean
54     */
55    locked: false,
56
57    /**
58     * The animation to use for expanding children, if any
59     *
60     * @type string
61     * @private
62     */
63    _expandAnim: null,
64
65    /**
66     * The animation to use for collapsing children, if any
67     *
68     * @type string
69     * @private
70     */
71    _collapseAnim: null,
72
73    /**
74     * The current number of animations that are executing
75     *
76     * @type int
77     * @private
78     */
79    _animCount: 0,
80
81    /**
82     * The maximum number of animations to run at one time.
83     *
84     * @type int
85     */
86    maxAnim: 2,
87
88    /**
89     * Sets up the animation for expanding children
90     *
91     * @param {string} the type of animation (acceptable values defined in
92     * YAHOO.widget.TVAnim)
93     */
94    setExpandAnim: function(type) {
95        if (YAHOO.widget.TVAnim.isValid(type)) {
96            this._expandAnim = type;
97        }
98    },
99
100    /**
101     * Sets up the animation for collapsing children
102     *
103     * @param {string} the type of animation (acceptable values defined in
104     * YAHOO.widget.TVAnim)
105     */
106    setCollapseAnim: function(type) {
107        if (YAHOO.widget.TVAnim.isValid(type)) {
108            this._collapseAnim = type;
109        }
110    },
111
112    /**
113     * Perform the expand animation if configured, or just show the
114     * element if not configured or too many animations are in progress
115     *
116     * @param el {HTMLElement} the element to animate
117     * @return {boolean} true if animation could be invoked, false otherwise
118     */
119    animateExpand: function(el) {
120
121        if (this._expandAnim && this._animCount < this.maxAnim) {
122            // this.locked = true;
123            var tree = this;
124            var a = YAHOO.widget.TVAnim.getAnim(this._expandAnim, el,
125                            function() { tree.expandComplete(); });
126            if (a) {
127                ++this._animCount;
128                a.animate();
129            }
130
131            return true;
132        }
133
134        return false;
135    },
136
137    /**
138     * Perform the collapse animation if configured, or just show the
139     * element if not configured or too many animations are in progress
140     *
141     * @param el {HTMLElement} the element to animate
142     * @return {boolean} true if animation could be invoked, false otherwise
143     */
144    animateCollapse: function(el) {
145
146        if (this._collapseAnim && this._animCount < this.maxAnim) {
147            // this.locked = true;
148            var tree = this;
149            var a = YAHOO.widget.TVAnim.getAnim(this._collapseAnim, el,
150                            function() { tree.collapseComplete(); });
151            if (a) {
152                ++this._animCount;
153                a.animate();
154            }
155
156            return true;
157        }
158
159        return false;
160    },
161
162    /**
163     * Function executed when the expand animation completes
164     */
165    expandComplete: function() {
166        --this._animCount;
167        // this.locked = false;
168    },
169
170    /**
171     * Function executed when the collapse animation completes
172     */
173    collapseComplete: function() {
174        --this._animCount;
175        // this.locked = false;
176    },
177
178    /**
179     * Initializes the tree
180     *
181     * @parm {string|HTMLElement} id the id of the element that will hold the tree
182     * @private
183     */
184    init: function(id) {
185
186        this.id = id;
187
188        if ("string" !== typeof id) {
189            this._el = id;
190            this.id = this.generateId(id);
191        }
192
193        this._nodes = [];
194
195        // store a global reference
196        YAHOO.widget.TreeView.trees[this.id] = this;
197
198        // Set up the root node
199        this.root = new YAHOO.widget.RootNode(this);
200
201
202    },
203
204    /**
205     * Renders the tree boilerplate and visible nodes
206     */
207    draw: function() {
208        var html = this.root.getHtml();
209        this.getEl().innerHTML = html;
210        this.firstDraw = false;
211    },
212
213    /**
214     * Returns the tree's host element
215     * @return {HTMLElement} the host element
216     */
217    getEl: function() {
218        if (! this._el) {
219            this._el = document.getElementById(this.id);
220        }
221        return this._el;
222    },
223
224    /**
225     * Nodes register themselves with the tree instance when they are created.
226     *
227     * @param node {Node} the node to register
228     * @private
229     */
230    regNode: function(node) {
231        this._nodes[node.index] = node;
232    },
233
234    /**
235     * Returns the root node of this tree
236     *
237     * @return {Node} the root node
238     */
239    getRoot: function() {
240        return this.root;
241    },
242
243    /**
244     * Configures this tree to dynamically load all child data
245     *
246     * @param {function} fnDataLoader the function that will be called to get the data
247     * @param iconMode {int} configures the icon that is displayed when a dynamic
248     * load node is expanded the first time without children.  By default, the
249     * "collapse" icon will be used.  If set to 1, the leaf node icon will be
250     * displayed.
251     */
252    setDynamicLoad: function(fnDataLoader, iconMode) {
253        this.root.setDynamicLoad(fnDataLoader, iconMode);
254    },
255
256    /**
257     * Expands all child nodes.  Note: this conflicts with the "multiExpand"
258     * node property.  If expand all is called in a tree with nodes that
259     * do not allow multiple siblings to be displayed, only the last sibling
260     * will be expanded.
261     */
262    expandAll: function() {
263        if (!this.locked) {
264            this.root.expandAll();
265        }
266    },
267
268    /**
269     * Collapses all expanded child nodes in the entire tree.
270     */
271    collapseAll: function() {
272        if (!this.locked) {
273            this.root.collapseAll();
274        }
275    },
276
277    /**
278     * Returns a node in the tree that has the specified index (this index
279     * is created internally, so this function probably will only be used
280     * in html generated for a given node.)
281     *
282     * @param {int} nodeIndex the index of the node wanted
283     * @return {Node} the node with index=nodeIndex, null if no match
284     */
285    getNodeByIndex: function(nodeIndex) {
286        var n = this._nodes[nodeIndex];
287        return (n) ? n : null;
288    },
289
290    /**
291     * Returns a node that has a matching property and value in the data
292     * object that was passed into its constructor.
293     *
294     * @param {object} property the property to search (usually a string)
295     * @param {object} value the value we want to find (usuall an int or string)
296     * @return {Node} the matching node, null if no match
297     */
298    getNodeByProperty: function(property, value) {
299        for (var i in this._nodes) {
300            var n = this._nodes[i];
301            if (n.data && value == n.data[property]) {
302                return n;
303            }
304        }
305
306        return null;
307    },
308
309    /**
310     * Returns a collection of nodes that have a matching property
311     * and value in the data object that was passed into its constructor. 
312     *
313     * @param {object} property the property to search (usually a string)
314     * @param {object} value the value we want to find (usuall an int or string)
315     * @return {Array} the matching collection of nodes, null if no match
316     */
317    getNodesByProperty: function(property, value) {
318        var values = [];
319        for (var i in this._nodes) {
320            var n = this._nodes[i];
321            if (n.data && value == n.data[property]) {
322                values.push(n);
323            }
324        }
325
326        return (values.length) ? values : null;
327    },
328
329    /**
330     * Removes the node and its children, and optionally refreshes the
331     * branch of the tree that was affected.
332     * @param {Node} The node to remove
333     * @param {boolean} autoRefresh automatically refreshes branch if true
334     * @return {boolean} False is there was a problem, true otherwise.
335     */
336    removeNode: function(node, autoRefresh) {
337
338        // Don't delete the root node
339        if (node.isRoot()) {
340            return false;
341        }
342
343        // Get the branch that we may need to refresh
344        var p = node.parent;
345        if (p.parent) {
346            p = p.parent;
347        }
348
349        // Delete the node and its children
350        this._deleteNode(node);
351
352        // Refresh the parent of the parent
353        if (autoRefresh && p && p.childrenRendered) {
354            p.refresh();
355        }
356
357        return true;
358    },
359
360    /**
361     * Deletes this nodes child collection, recursively.  Also collapses
362     * the node, and resets the dynamic load flag.  The primary use for
363     * this method is to purge a node and allow it to fetch its data
364     * dynamically again.
365     * @param {Node} node the node to purge
366     */
367    removeChildren: function(node) {
368        while (node.children.length) {
369             this._deleteNode(node.children[0]);
370        }
371
372        node.childrenRendered = false;
373        node.dynamicLoadComplete = false;
374        // node.collapse();
375        node.expand();
376        node.collapse();
377    },
378
379    /**
380     * Deletes the node and recurses children
381     * @private
382     */
383    _deleteNode: function(node) {
384        // Remove all the child nodes first
385        this.removeChildren(node);
386
387        // Remove the node from the tree
388        this.popNode(node);
389    },
390
391    /**
392     * Removes the node from the tree, preserving the child collection
393     * to make it possible to insert the branch into another part of the
394     * tree, or another tree.
395     * @param {Node} the node to remove
396     */
397    popNode: function(node) {
398        var p = node.parent;
399
400        // Update the parent's collection of children
401        var a = [];
402
403        for (var i=0, len=p.children.length;i<len;++i) {
404            if (p.children[i] != node) {
405                a[a.length] = p.children[i];
406            }
407        }
408
409        p.children = a;
410
411        // reset the childrenRendered flag for the parent
412        p.childrenRendered = false;
413
414         // Update the sibling relationship
415        if (node.previousSibling) {
416            node.previousSibling.nextSibling = node.nextSibling;
417        }
418
419        if (node.nextSibling) {
420            node.nextSibling.previousSibling = node.previousSibling;
421        }
422
423        node.parent = null;
424        node.previousSibling = null;
425        node.nextSibling = null;
426        node.tree = null;
427
428        // Update the tree's node collection
429        delete this._nodes[node.index];
430    },
431
432    /**
433     * toString
434     * @return {string} string representation of the tree
435     */
436    toString: function() {
437        return "TreeView " + this.id;
438    },
439
440    /**
441     * private
442     */
443    generateId: function(el) {
444        var id = el.id;
445
446        if (!id) {
447            id = "yui-tv-auto-id-" + YAHOO.widget.TreeView.counter;
448            YAHOO.widget.TreeView.counter++;
449        }
450
451        return id;
452    },
453
454    /**
455     * Abstract method that is executed when a node is expanded
456     * @param node {Node} the node that was expanded
457     */
458    onExpand: function(node) { },
459
460    /**
461     * Abstract method that is executed when a node is collapsed
462     * @param node {Node} the node that was collapsed.
463     */
464    onCollapse: function(node) { }
465
466};
467
468/**
469 * Global cache of tree instances
470 *
471 * @type Array
472 * @private
473 */
474YAHOO.widget.TreeView.trees = [];
475
476/**
477 * @private
478 */
479YAHOO.widget.TreeView.counter = 0;
480
481/**
482 * Global method for getting a tree by its id.  Used in the generated
483 * tree html.
484 *
485 * @param treeId {String} the id of the tree instance
486 * @return {TreeView} the tree instance requested, null if not found.
487 */
488YAHOO.widget.TreeView.getTree = function(treeId) {
489    var t = YAHOO.widget.TreeView.trees[treeId];
490    return (t) ? t : null;
491};
492
493/**
494 * Global method for getting a node by its id.  Used in the generated
495 * tree html.
496 *
497 * @param treeId {String} the id of the tree instance
498 * @param nodeIndex {String} the index of the node to return
499 * @return {Node} the node instance requested, null if not found
500 */
501YAHOO.widget.TreeView.getNode = function(treeId, nodeIndex) {
502    var t = YAHOO.widget.TreeView.getTree(treeId);
503    return (t) ? t.getNodeByIndex(nodeIndex) : null;
504};
505
506/**
507 * Adds an event.  Replace with event manager when available
508 *
509 * @param el the elment to bind the handler to
510 * @param {string} sType the type of event handler
511 * @param {function} fn the callback to invoke
512 * @param {boolean} capture if true event is capture phase, bubble otherwise
513 */
514YAHOO.widget.TreeView.addHandler = function (el, sType, fn, capture) {
515    capture = (capture) ? true : false;
516    if (el.addEventListener) {
517        el.addEventListener(sType, fn, capture);
518    } else if (el.attachEvent) {
519        el.attachEvent("on" + sType, fn);
520    } else {
521        el["on" + sType] = fn;
522    }
523};
524
525/**
526 * Attempts to preload the images defined in the styles used to draw the tree by
527 * rendering off-screen elements that use the styles.
528 */
529YAHOO.widget.TreeView.preload = function(prefix) {
530    prefix = prefix || "ygtv";
531    var styles = ["tn","tm","tmh","tp","tph","ln","lm","lmh","lp","lph","loading"];
532
533    var sb = [];
534   
535    for (var i = 0; i < styles.length; ++i) {
536        sb[sb.length] = '<span class="' + prefix + styles[i] + '">&#160;</span>';
537    }
538
539    var f = document.createElement("DIV");
540    var s = f.style;
541    s.position = "absolute";
542    s.top = "-1000px";
543    s.left = "-1000px";
544    f.innerHTML = sb.join("");
545
546    document.body.appendChild(f);
547};
548
549YAHOO.widget.TreeView.addHandler(window,
550                "load", YAHOO.widget.TreeView.preload);
551
552/**
553 * The base class for all tree nodes.  The node's presentation and behavior in
554 * response to mouse events is handled in Node subclasses.
555 *
556 * @param oData {object} a string or object containing the data that will
557 * be used to render this node
558 * @param oParent {Node} this node's parent node
559 * @param expanded {boolean} the initial expanded/collapsed state
560 * @constructor
561 */
562YAHOO.widget.Node = function(oData, oParent, expanded) {
563    if (oData) { this.init(oData, oParent, expanded); }
564};
565
566YAHOO.widget.Node.prototype = {
567
568    /**
569     * The index for this instance obtained from global counter in YAHOO.widget.TreeView.
570     *
571     * @type int
572     */
573    index: 0,
574
575    /**
576     * This node's child node collection.
577     *
578     * @type Node[]
579     */
580    children: null,
581
582    /**
583     * Tree instance this node is part of
584     *
585     * @type TreeView
586     */
587    tree: null,
588
589    /**
590     * The data linked to this node.  This can be any object or primitive
591     * value, and the data can be used in getNodeHtml().
592     *
593     * @type object
594     */
595    data: null,
596
597    /**
598     * Parent node
599     *
600     * @type Node
601     */
602    parent: null,
603
604    /**
605     * The depth of this node.  We start at -1 for the root node.
606     *
607     * @type int
608     */
609    depth: -1,
610
611    /**
612     * The href for the node's label.  If one is not specified, the href will
613     * be set so that it toggles the node.
614     *
615     * @type string
616     */
617    href: null,
618
619    /**
620     * The label href target, defaults to current window
621     *
622     * @type string
623     */
624    target: "_self",
625
626    /**
627     * The node's expanded/collapsed state
628     *
629     * @type boolean
630     */
631    expanded: false,
632
633    /**
634     * Can multiple children be expanded at once?
635     *
636     * @type boolean
637     */
638    multiExpand: true,
639
640    /**
641     * Should we render children for a collapsed node?  It is possible that the
642     * implementer will want to render the hidden data...  @todo verify that we
643     * need this, and implement it if we do.
644     *
645     * @type boolean
646     */
647    renderHidden: false,
648
649    /**
650     * This flag is set to true when the html is generated for this node's
651     * children, and set to false when new children are added.
652     * @type boolean
653     */
654    childrenRendered: false,
655
656    /**
657     * Dynamically loaded nodes only fetch the data the first time they are
658     * expanded.  This flag is set to true once the data has been fetched.
659     * @type boolean
660     */
661    dynamicLoadComplete: false,
662
663    /**
664     * This node's previous sibling
665     *
666     * @type Node
667     */
668    previousSibling: null,
669
670    /**
671     * This node's next sibling
672     *
673     * @type Node
674     */
675    nextSibling: null,
676
677    /**
678     * We can set the node up to call an external method to get the child
679     * data dynamically.
680     *
681     * @type boolean
682     * @private
683     */
684    _dynLoad: false,
685
686    /**
687     * Function to execute when we need to get this node's child data.
688     *
689     * @type function
690     */
691    dataLoader: null,
692
693    /**
694     * This is true for dynamically loading nodes while waiting for the
695     * callback to return.
696     *
697     * @type boolean
698     */
699    isLoading: false,
700
701    /**
702     * The toggle/branch icon will not show if this is set to false.  This
703     * could be useful if the implementer wants to have the child contain
704     * extra info about the parent, rather than an actual node.
705     *
706     * @type boolean
707     */
708    hasIcon: true,
709
710    /**
711     * Used to configure what happens when a dynamic load node is expanded
712     * and we discover that it does not have children.  By default, it is
713     * treated as if it still could have children (plus/minus icon).  Set
714     * iconMode to have it display like a leaf node instead.
715     * @type int
716     */
717    iconMode: 0,
718
719    /**
720     * The node type
721     * @private
722     */
723    _type: "Node",
724
725    /*
726    spacerPath: "http://us.i1.yimg.com/us.yimg.com/i/space.gif",
727    expandedText: "Expanded",
728    collapsedText: "Collapsed",
729    loadingText: "Loading",
730    */
731
732    /**
733     * Initializes this node, gets some of the properties from the parent
734     *
735     * @param oData {object} a string or object containing the data that will
736     * be used to render this node
737     * @param oParent {Node} this node's parent node
738     * @param expanded {boolean} the initial expanded/collapsed state
739     */
740    init: function(oData, oParent, expanded, index) {
741        this.data       = oData;
742        this.children   = [];
743        this.index      = YAHOO.widget.TreeView.nodeCount;
744        ++YAHOO.widget.TreeView.nodeCount;
745        this.expanded   = expanded;
746       
747
748        // oParent should never be null except when we create the root node.
749        if (oParent) {
750            oParent.appendChild(this);
751        }
752    },
753
754    /**
755     * Certain properties for the node cannot be set until the parent
756     * is known. This is called after the node is inserted into a tree.
757     * the parent is also applied to this node's children in order to
758     * make it possible to move a branch from one tree to another.
759     * @param {Node} parentNode this node's parent node
760     * @return {boolean} true if the application was successful
761     */
762    applyParent: function(parentNode) {
763        if (!parentNode) {
764            return false;
765        }
766
767        this.tree   = parentNode.tree;
768        this.parent = parentNode;
769        this.depth  = parentNode.depth + 1;
770
771        if (! this.href) {
772            this.href = "javascript:" + this.getToggleLink();
773        }
774
775        if (! this.multiExpand) {
776            this.multiExpand = parentNode.multiExpand;
777        }
778
779        this.tree.regNode(this);
780        parentNode.childrenRendered = false;
781
782        // cascade update existing children
783        for (var i=0, len=this.children.length;i<len;++i) {
784            this.children[i].applyParent(this);
785        }
786
787        return true;
788    },
789
790    /**
791     * Appends a node to the child collection.
792     *
793     * @param childNode {Node} the new node
794     * @return {Node} the child node
795     * @private
796     */
797    appendChild: function(childNode) {
798        if (this.hasChildren()) {
799            var sib = this.children[this.children.length - 1];
800            sib.nextSibling = childNode;
801            childNode.previousSibling = sib;
802        }
803        this.children[this.children.length] = childNode;
804        childNode.applyParent(this);
805
806        return childNode;
807    },
808
809    /**
810     * Appends this node to the supplied node's child collection
811     * @param parentNode {Node} the node to append to.
812     * @return {Node} The appended node
813     */
814    appendTo: function(parentNode) {
815        return parentNode.appendChild(this);
816    },
817
818    /**
819    * Inserts this node before this supplied node
820    *
821    * @param node {Node} the node to insert this node before
822    * @return {Node} the inserted node
823    */
824    insertBefore: function(node) {
825        var p = node.parent;
826        if (p) {
827
828            if (this.tree) {
829                this.tree.popNode(this);
830            }
831
832            var refIndex = node.isChildOf(p);
833            p.children.splice(refIndex, 0, this);
834            if (node.previousSibling) {
835                node.previousSibling.nextSibling = this;
836            }
837            this.previousSibling = node.previousSibling;
838            this.nextSibling = node;
839            node.previousSibling = this;
840
841            this.applyParent(p);
842        }
843
844        return this;
845    },
846 
847    /**
848    * Inserts this node after the supplied node
849    *
850    * @param node {Node} the node to insert after
851    * @return {Node} the inserted node
852    */
853    insertAfter: function(node) {
854        var p = node.parent;
855        if (p) {
856
857            if (this.tree) {
858                this.tree.popNode(this);
859            }
860
861            var refIndex = node.isChildOf(p);
862
863            if (!node.nextSibling) {
864                return this.appendTo(p);
865            }
866
867            p.children.splice(refIndex + 1, 0, this);
868
869            node.nextSibling.previousSibling = this;
870            this.previousSibling = node;
871            this.nextSibling = node.nextSibling;
872            node.nextSibling = this;
873
874            this.applyParent(p);
875        }
876
877        return this;
878    },
879
880    /**
881    * Returns true if the Node is a child of supplied Node
882    *
883    * @param parentNode {Node} the Node to check
884    * @return {boolean} The node index if this Node is a child of
885    *                   supplied Node, else -1.
886    * @private
887    */
888    isChildOf: function(parentNode) {
889        if (parentNode && parentNode.children) {
890            for (var i=0, len=parentNode.children.length; i<len ; ++i) {
891                if (parentNode.children[i] === this) {
892                    return i;
893                }
894            }
895        }
896
897        return -1;
898    },
899
900    /**
901     * Returns a node array of this node's siblings, null if none.
902     *
903     * @return Node[]
904     */
905    getSiblings: function() {
906        return this.parent.children;
907    },
908
909    /**
910     * Shows this node's children
911     */
912    showChildren: function() {
913        if (!this.tree.animateExpand(this.getChildrenEl())) {
914            if (this.hasChildren()) {
915                this.getChildrenEl().style.display = "";
916            }
917        }
918    },
919
920    /**
921     * Hides this node's children
922     */
923    hideChildren: function() {
924
925        if (!this.tree.animateCollapse(this.getChildrenEl())) {
926            this.getChildrenEl().style.display = "none";
927        }
928    },
929
930    /**
931     * Returns the id for this node's container div
932     *
933     * @return {string} the element id
934     */
935    getElId: function() {
936        return "ygtv" + this.index;
937    },
938
939    /**
940     * Returns the id for this node's children div
941     *
942     * @return {string} the element id for this node's children div
943     */
944    getChildrenElId: function() {
945        return "ygtvc" + this.index;
946    },
947
948    /**
949     * Returns the id for this node's toggle element
950     *
951     * @return {string} the toggel element id
952     */
953    getToggleElId: function() {
954        return "ygtvt" + this.index;
955    },
956
957    /**
958     * Returns the id for this node's spacer image.  The spacer is positioned
959     * over the toggle and provides feedback for screen readers.
960     * @return {string} the id for the spacer image
961     */
962    /*
963    getSpacerId: function() {
964        return "ygtvspacer" + this.index;
965    },
966    */
967
968    /**
969     * Returns this node's container html element
970     * @return {HTMLElement} the container html element
971     */
972    getEl: function() {
973        return document.getElementById(this.getElId());
974    },
975
976    /**
977     * Returns the div that was generated for this node's children
978     * @return {HTMLElement} this node's children div
979     */
980    getChildrenEl: function() {
981        return document.getElementById(this.getChildrenElId());
982    },
983
984    /**
985     * Returns the element that is being used for this node's toggle.
986     * @return {HTMLElement} this node's toggle html element
987     */
988    getToggleEl: function() {
989        return document.getElementById(this.getToggleElId());
990    },
991
992    /**
993     * Returns the element that is being used for this node's spacer.
994     * @return {HTMLElement} this node's spacer html element
995     */
996    /*
997    getSpacer: function() {
998        return document.getElementById( this.getSpacerId() ) || {};
999    },
1000    */
1001
1002    /*
1003    getStateText: function() {
1004        if (this.isLoading) {
1005            return this.loadingText;
1006        } else if (this.hasChildren(true)) {
1007            if (this.expanded) {
1008                return this.expandedText;
1009            } else {
1010                return this.collapsedText;
1011            }
1012        } else {
1013            return "";
1014        }
1015    },
1016    */
1017
1018    /**
1019     * Generates the link that will invoke this node's toggle method
1020     * @return {string} the javascript url for toggling this node
1021     */
1022    getToggleLink: function() {
1023        return "YAHOO.widget.TreeView.getNode(\'" + this.tree.id + "\'," +
1024            this.index + ").toggle()";
1025    },
1026
1027    /**
1028     * Hides this nodes children (creating them if necessary), changes the
1029     * toggle style.
1030     */
1031    collapse: function() {
1032        // Only collapse if currently expanded
1033        if (!this.expanded) { return; }
1034
1035        // fire the collapse event handler
1036        var ret = this.tree.onCollapse(this);
1037
1038        if ("undefined" != typeof ret && !ret) {
1039            return;
1040        }
1041
1042        if (!this.getEl()) {
1043            this.expanded = false;
1044            return;
1045        }
1046
1047        // hide the child div
1048        this.hideChildren();
1049        this.expanded = false;
1050
1051        if (this.hasIcon) {
1052            this.getToggleEl().className = this.getStyle();
1053        }
1054
1055        // this.getSpacer().title = this.getStateText();
1056
1057    },
1058
1059    /**
1060     * Shows this nodes children (creating them if necessary), changes the
1061     * toggle style, and collapses its siblings if multiExpand is not set.
1062     */
1063    expand: function() {
1064        // Only expand if currently collapsed.
1065        if (this.expanded) { return; }
1066
1067        // fire the expand event handler
1068        var ret = this.tree.onExpand(this);
1069
1070        if ("undefined" != typeof ret && !ret) {
1071            return;
1072        }
1073
1074        if (!this.getEl()) {
1075            this.expanded = true;
1076            return;
1077        }
1078
1079        if (! this.childrenRendered) {
1080            this.getChildrenEl().innerHTML = this.renderChildren();
1081        } else {
1082        }
1083
1084        this.expanded = true;
1085        if (this.hasIcon) {
1086            this.getToggleEl().className = this.getStyle();
1087        }
1088
1089        // this.getSpacer().title = this.getStateText();
1090
1091        // We do an extra check for children here because the lazy
1092        // load feature can expose nodes that have no children.
1093
1094        // if (!this.hasChildren()) {
1095        if (this.isLoading) {
1096            this.expanded = false;
1097            return;
1098        }
1099
1100        if (! this.multiExpand) {
1101            var sibs = this.getSiblings();
1102            for (var i=0; i<sibs.length; ++i) {
1103                if (sibs[i] != this && sibs[i].expanded) {
1104                    sibs[i].collapse();
1105                }
1106            }
1107        }
1108
1109        this.showChildren();
1110    },
1111
1112    /**
1113     * Returns the css style name for the toggle
1114     *
1115     * @return {string} the css class for this node's toggle
1116     */
1117    getStyle: function() {
1118        if (this.isLoading) {
1119            return "ygtvloading";
1120        } else {
1121            // location top or bottom, middle nodes also get the top style
1122            var loc = (this.nextSibling) ? "t" : "l";
1123
1124            // type p=plus(expand), m=minus(collapase), n=none(no children)
1125            var type = "n";
1126            if (this.hasChildren(true) || (this.isDynamic() && !this.getIconMode())) {
1127            // if (this.hasChildren(true)) {
1128                type = (this.expanded) ? "m" : "p";
1129            }
1130
1131            return "ygtv" + loc + type;
1132        }
1133    },
1134
1135    /**
1136     * Returns the hover style for the icon
1137     * @return {string} the css class hover state
1138     */
1139    getHoverStyle: function() {
1140        var s = this.getStyle();
1141        if (this.hasChildren(true) && !this.isLoading) {
1142            s += "h";
1143        }
1144        return s;
1145    },
1146
1147    /**
1148     * Recursively expands all of this node's children.
1149     */
1150    expandAll: function() {
1151        for (var i=0;i<this.children.length;++i) {
1152            var c = this.children[i];
1153            if (c.isDynamic()) {
1154                alert("Not supported (lazy load + expand all)");
1155                break;
1156            } else if (! c.multiExpand) {
1157                alert("Not supported (no multi-expand + expand all)");
1158                break;
1159            } else {
1160                c.expand();
1161                c.expandAll();
1162            }
1163        }
1164    },
1165
1166    /**
1167     * Recursively collapses all of this node's children.
1168     */
1169    collapseAll: function() {
1170        for (var i=0;i<this.children.length;++i) {
1171            this.children[i].collapse();
1172            this.children[i].collapseAll();
1173        }
1174    },
1175
1176    /**
1177     * Configures this node for dynamically obtaining the child data
1178     * when the node is first expanded.  Calling it without the callback
1179     * will turn off dynamic load for the node.
1180     *
1181     * @param fmDataLoader {function} the function that will be used to get the data.
1182     * @param iconMode {int} configures the icon that is displayed when a dynamic
1183     * load node is expanded the first time without children.  By default, the
1184     * "collapse" icon will be used.  If set to 1, the leaf node icon will be
1185     * displayed.
1186     */
1187    setDynamicLoad: function(fnDataLoader, iconMode) {
1188        if (fnDataLoader) {
1189            this.dataLoader = fnDataLoader;
1190            this._dynLoad = true;
1191        } else {
1192            this.dataLoader = null;
1193            this._dynLoad = false;
1194        }
1195
1196        if (iconMode) {
1197            this.iconMode = iconMode;
1198        }
1199    },
1200
1201    /**
1202     * Evaluates if this node is the root node of the tree
1203     *
1204     * @return {boolean} true if this is the root node
1205     */
1206    isRoot: function() {
1207        return (this == this.tree.root);
1208    },
1209
1210    /**
1211     * Evaluates if this node's children should be loaded dynamically.  Looks for
1212     * the property both in this instance and the root node.  If the tree is
1213     * defined to load all children dynamically, the data callback function is
1214     * defined in the root node
1215     *
1216     * @return {boolean} true if this node's children are to be loaded dynamically
1217     */
1218    isDynamic: function() {
1219        var lazy = (!this.isRoot() && (this._dynLoad || this.tree.root._dynLoad));
1220        return lazy;
1221    },
1222
1223    getIconMode: function() {
1224        return (this.iconMode || this.tree.root.iconMode);
1225    },
1226
1227    /**
1228     * Checks if this node has children.  If this node is lazy-loading and the
1229     * children have not been rendered, we do not know whether or not there
1230     * are actual children.  In most cases, we need to assume that there are
1231     * children (for instance, the toggle needs to show the expandable
1232     * presentation state).  In other times we want to know if there are rendered
1233     * children.  For the latter, "checkForLazyLoad" should be false.
1234     *
1235     * @param checkForLazyLoad {boolean} should we check for unloaded children?
1236     * @return {boolean} true if this has children or if it might and we are
1237     * checking for this condition.
1238     */
1239    hasChildren: function(checkForLazyLoad) {
1240        return ( this.children.length > 0 ||
1241                (checkForLazyLoad && this.isDynamic() && !this.dynamicLoadComplete) );
1242    },
1243
1244    /**
1245     * Expands if node is collapsed, collapses otherwise.
1246     */
1247    toggle: function() {
1248        if (!this.tree.locked && ( this.hasChildren(true) || this.isDynamic()) ) {
1249            if (this.expanded) { this.collapse(); } else { this.expand(); }
1250        }
1251    },
1252
1253    /**
1254     * Returns the markup for this node and its children.
1255     *
1256     * @return {string} the markup for this node and its expanded children.
1257     */
1258    getHtml: function() {
1259        var sb = [];
1260        sb[sb.length] = '<div class="ygtvitem" id="' + this.getElId() + '">';
1261        sb[sb.length] = this.getNodeHtml();
1262        sb[sb.length] = this.getChildrenHtml();
1263        sb[sb.length] = '</div>';
1264        return sb.join("");
1265    },
1266
1267    /**
1268     * Called when first rendering the tree.  We always build the div that will
1269     * contain this nodes children, but we don't render the children themselves
1270     * unless this node is expanded.
1271     *
1272     * @return {string} the children container div html and any expanded children
1273     * @private
1274     */
1275    getChildrenHtml: function() {
1276
1277        var sb = [];
1278        sb[sb.length] = '<div class="ygtvchildren"';
1279        sb[sb.length] = ' id="' + this.getChildrenElId() + '"';
1280        if (!this.expanded) {
1281            sb[sb.length] = ' style="display:none;"';
1282        }
1283        sb[sb.length] = '>';
1284
1285        // Don't render the actual child node HTML unless this node is expanded.
1286        if ( (this.hasChildren(true) && this.expanded) ||
1287                (this.renderHidden && !this.isDynamic()) ) {
1288            sb[sb.length] = this.renderChildren();
1289        }
1290
1291        sb[sb.length] = '</div>';
1292
1293        return sb.join("");
1294    },
1295
1296    /**
1297     * Generates the markup for the child nodes.  This is not done until the node
1298     * is expanded.
1299     *
1300     * @return {string} the html for this node's children
1301     * @private
1302     */
1303    renderChildren: function() {
1304
1305
1306        var node = this;
1307
1308        if (this.isDynamic() && !this.dynamicLoadComplete) {
1309            this.isLoading = true;
1310            this.tree.locked = true;
1311
1312            if (this.dataLoader) {
1313
1314                setTimeout(
1315                    function() {
1316                        node.dataLoader(node,
1317                            function() {
1318                                node.loadComplete();
1319                            });
1320                    }, 10);
1321               
1322            } else if (this.tree.root.dataLoader) {
1323
1324                setTimeout(
1325                    function() {
1326                        node.tree.root.dataLoader(node,
1327                            function() {
1328                                node.loadComplete();
1329                            });
1330                    }, 10);
1331
1332            } else {
1333                return "Error: data loader not found or not specified.";
1334            }
1335
1336            return "";
1337
1338        } else {
1339            return this.completeRender();
1340        }
1341    },
1342
1343    /**
1344     * Called when we know we have all the child data.
1345     * @return {string} children html
1346     */
1347    completeRender: function() {
1348        var sb = [];
1349
1350        for (var i=0; i < this.children.length; ++i) {
1351            this.children[i].childrenRendered = false;
1352            sb[sb.length] = this.children[i].getHtml();
1353        }
1354       
1355        this.childrenRendered = true;
1356
1357        return sb.join("");
1358    },
1359
1360    /**
1361     * Load complete is the callback function we pass to the data provider
1362     * in dynamic load situations.
1363     */
1364    loadComplete: function() {
1365        this.getChildrenEl().innerHTML = this.completeRender();
1366        this.dynamicLoadComplete = true;
1367        this.isLoading = false;
1368        this.expand();
1369        this.tree.locked = false;
1370    },
1371
1372    /**
1373     * Returns this node's ancestor at the specified depth.
1374     *
1375     * @param {int} depth the depth of the ancestor.
1376     * @return {Node} the ancestor
1377     */
1378    getAncestor: function(depth) {
1379        if (depth >= this.depth || depth < 0)  {
1380            return null;
1381        }
1382
1383        var p = this.parent;
1384       
1385        while (p.depth > depth) {
1386            p = p.parent;
1387        }
1388
1389        return p;
1390    },
1391
1392    /**
1393     * Returns the css class for the spacer at the specified depth for
1394     * this node.  If this node's ancestor at the specified depth
1395     * has a next sibling the presentation is different than if it
1396     * does not have a next sibling
1397     *
1398     * @param {int} depth the depth of the ancestor.
1399     * @return {string} the css class for the spacer
1400     */
1401    getDepthStyle: function(depth) {
1402        return (this.getAncestor(depth).nextSibling) ?
1403            "ygtvdepthcell" : "ygtvblankdepthcell";
1404    },
1405
1406    /**
1407     * Get the markup for the node.  This is designed to be overrided so that we can
1408     * support different types of nodes.
1409     *
1410     * @return {string} The HTML that will render this node.
1411     */
1412    getNodeHtml: function() {
1413        return "";
1414    },
1415
1416    /**
1417     * Regenerates the html for this node and its children.  To be used when the
1418     * node is expanded and new children have been added.
1419     */
1420    refresh: function() {
1421        // this.loadComplete();
1422        this.getChildrenEl().innerHTML = this.completeRender();
1423
1424        if (this.hasIcon) {
1425            var el = this.getToggleEl();
1426            if (el) {
1427                el.className = this.getStyle();
1428            }
1429        }
1430    },
1431
1432    /**
1433     * toString
1434     * @return {string} string representation of the node
1435     */
1436    toString: function() {
1437        return "Node (" + this.index + ")";
1438    }
1439
1440};
1441
1442/**
1443 * A custom YAHOO.widget.Node that handles the unique nature of
1444 * the virtual, presentationless root node.
1445 *
1446 * @extends YAHOO.widget.Node
1447 * @constructor
1448 */
1449YAHOO.widget.RootNode = function(oTree) {
1450    // Initialize the node with null params.  The root node is a
1451    // special case where the node has no presentation.  So we have
1452    // to alter the standard properties a bit.
1453    this.init(null, null, true);
1454   
1455    /**
1456     * For the root node, we get the tree reference from as a param
1457     * to the constructor instead of from the parent element.
1458     *
1459     * @type TreeView
1460     */
1461    this.tree = oTree;
1462};
1463
1464YAHOO.widget.RootNode.prototype = new YAHOO.widget.Node();
1465
1466// overrides YAHOO.widget.Node
1467YAHOO.widget.RootNode.prototype.getNodeHtml = function() {
1468    return "";
1469};
1470
1471YAHOO.widget.RootNode.prototype.toString = function() {
1472    return "RootNode";
1473};
1474
1475YAHOO.widget.RootNode.prototype.loadComplete = function() {
1476    this.tree.draw();
1477};
1478/**
1479 * The default node presentation.  The first parameter should be
1480 * either a string that will be used as the node's label, or an object
1481 * that has a string propery called label.  By default, the clicking the
1482 * label will toggle the expanded/collapsed state of the node.  By
1483 * changing the href property of the instance, this behavior can be
1484 * changed so that the label will go to the specified href.
1485 *
1486 * @extends YAHOO.widget.Node
1487 * @constructor
1488 * @param oData {object} a string or object containing the data that will
1489 * be used to render this node
1490 * @param oParent {YAHOO.widget.Node} this node's parent node
1491 * @param expanded {boolean} the initial expanded/collapsed state
1492 */
1493YAHOO.widget.TextNode = function(oData, oParent, expanded) {
1494    // this.type = "TextNode";
1495
1496    if (oData) {
1497        this.init(oData, oParent, expanded);
1498        this.setUpLabel(oData);
1499    }
1500
1501    /**
1502     * @private
1503     */
1504};
1505
1506YAHOO.widget.TextNode.prototype = new YAHOO.widget.Node();
1507
1508/**
1509 * The CSS class for the label href.  Defaults to ygtvlabel, but can be
1510 * overridden to provide a custom presentation for a specific node.
1511 *
1512 * @type string
1513 */
1514YAHOO.widget.TextNode.prototype.labelStyle = "ygtvlabel";
1515
1516/**
1517 * The derived element id of the label for this node
1518 *
1519 * @type string
1520 */
1521YAHOO.widget.TextNode.prototype.labelElId = null;
1522
1523/**
1524 * The text for the label.  It is assumed that the oData parameter will
1525 * either be a string that will be used as the label, or an object that
1526 * has a property called "label" that we will use.
1527 *
1528 * @type string
1529 */
1530YAHOO.widget.TextNode.prototype.label = null;
1531
1532/**
1533 * Sets up the node label
1534 *
1535 * @param oData string containing the label, or an object with a label property
1536 */
1537YAHOO.widget.TextNode.prototype.setUpLabel = function(oData) {
1538    if (typeof oData == "string") {
1539        oData = { label: oData };
1540    }
1541    this.label = oData.label;
1542   
1543    // update the link
1544    if (oData.href) {
1545        this.href = oData.href;
1546    }
1547
1548    // set the target
1549    if (oData.target) {
1550        this.target = oData.target;
1551    }
1552
1553    if (oData.style) {
1554        this.labelStyle = oData.style;
1555    }
1556
1557    this.labelElId = "ygtvlabelel" + this.index;
1558};
1559
1560/**
1561 * Returns the label element
1562 *
1563 * @return {object} the element
1564 */
1565YAHOO.widget.TextNode.prototype.getLabelEl = function() {
1566    return document.getElementById(this.labelElId);
1567};
1568
1569// overrides YAHOO.widget.Node
1570YAHOO.widget.TextNode.prototype.getNodeHtml = function() {
1571    var sb = [];
1572
1573    sb[sb.length] = '<table border="0" cellpadding="0" cellspacing="0">';
1574    sb[sb.length] = '<tr>';
1575   
1576    for (i=0;i<this.depth;++i) {
1577        // sb[sb.length] = '<td class="ygtvdepthcell">&#160;</td>';
1578        sb[sb.length] = '<td class="' + this.getDepthStyle(i) + '">&#160;</td>';
1579    }
1580
1581    var getNode = 'YAHOO.widget.TreeView.getNode(\'' +
1582                    this.tree.id + '\',' + this.index + ')';
1583
1584    sb[sb.length] = '<td';
1585    // sb[sb.length] = ' onselectstart="return false"';
1586    sb[sb.length] = ' id="' + this.getToggleElId() + '"';
1587    sb[sb.length] = ' class="' + this.getStyle() + '"';
1588    if (this.hasChildren(true)) {
1589        sb[sb.length] = ' onmouseover="this.className=';
1590        sb[sb.length] = getNode + '.getHoverStyle()"';
1591        sb[sb.length] = ' onmouseout="this.className=';
1592        sb[sb.length] = getNode + '.getStyle()"';
1593    }
1594    sb[sb.length] = ' onclick="javascript:' + this.getToggleLink() + '">';
1595
1596    /*
1597    sb[sb.length] = '<img id="' + this.getSpacerId() + '"';
1598    sb[sb.length] = ' alt=""';
1599    sb[sb.length] = ' tabindex=0';
1600    sb[sb.length] = ' src="' + this.spacerPath + '"';
1601    sb[sb.length] = ' title="' + this.getStateText() + '"';
1602    sb[sb.length] = ' class="ygtvspacer"';
1603    // sb[sb.length] = ' onkeypress="return ' + getNode + '".onKeyPress()"';
1604    sb[sb.length] = ' />';
1605    */
1606
1607    sb[sb.length] = '&#160;';
1608
1609    sb[sb.length] = '</td>';
1610    sb[sb.length] = '<td>';
1611    sb[sb.length] = '<a';
1612    sb[sb.length] = ' id="' + this.labelElId + '"';
1613    sb[sb.length] = ' class="' + this.labelStyle + '"';
1614    sb[sb.length] = ' href="' + this.href + '"';
1615    sb[sb.length] = ' target="' + this.target + '"';
1616    sb[sb.length] = ' onclick="return ' + getNode + '.onLabelClick(' + getNode +')"';
1617    if (this.hasChildren(true)) {
1618        sb[sb.length] = ' onmouseover="document.getElementById(\'';
1619        sb[sb.length] = this.getToggleElId() + '\').className=';
1620        sb[sb.length] = getNode + '.getHoverStyle()"';
1621        sb[sb.length] = ' onmouseout="document.getElementById(\'';
1622        sb[sb.length] = this.getToggleElId() + '\').className=';
1623        sb[sb.length] = getNode + '.getStyle()"';
1624    }
1625    sb[sb.length] = ' >';
1626    sb[sb.length] = this.label;
1627    sb[sb.length] = '</a>';
1628    sb[sb.length] = '</td>';
1629    sb[sb.length] = '</tr>';
1630    sb[sb.length] = '</table>';
1631
1632    return sb.join("");
1633};
1634
1635/**
1636 * Executed when the label is clicked
1637 * @param me {Node} this node
1638 * @scope the anchor tag clicked
1639 * @return false to cancel the anchor click
1640 */
1641YAHOO.widget.TextNode.prototype.onLabelClick = function(me) {
1642    //return true;
1643};
1644
1645YAHOO.widget.TextNode.prototype.toString = function() {
1646    return "TextNode (" + this.index + ") " + this.label;
1647};
1648
1649/**
1650 * A menu-specific implementation that differs from TextNode in that only
1651 * one sibling can be expanded at a time.
1652 * @extends YAHOO.widget.TextNode
1653 * @constructor
1654 */
1655YAHOO.widget.MenuNode = function(oData, oParent, expanded) {
1656    if (oData) {
1657        this.init(oData, oParent, expanded);
1658        this.setUpLabel(oData);
1659    }
1660
1661    /**
1662     * Menus usually allow only one branch to be open at a time.
1663     * @type boolean
1664     */
1665    this.multiExpand = false;
1666
1667    /**
1668     * @private
1669     */
1670
1671};
1672
1673YAHOO.widget.MenuNode.prototype = new YAHOO.widget.TextNode();
1674
1675YAHOO.widget.MenuNode.prototype.toString = function() {
1676    return "MenuNode (" + this.index + ") " + this.label;
1677};
1678
1679/**
1680 * This implementation takes either a string or object for the
1681 * oData argument.  If is it a string, we will use it for the display
1682 * of this node (and it can contain any html code).  If the parameter
1683 * is an object, we look for a parameter called "html" that will be
1684 * used for this node's display.
1685 *
1686 * @extends YAHOO.widget.Node
1687 * @constructor
1688 * @param oData {object} a string or object containing the data that will
1689 * be used to render this node
1690 * @param oParent {YAHOO.widget.Node} this node's parent node
1691 * @param expanded {boolean} the initial expanded/collapsed state
1692 * @param hasIcon {boolean} specifies whether or not leaf nodes should
1693 * have an icon
1694 */
1695YAHOO.widget.HTMLNode = function(oData, oParent, expanded, hasIcon) {
1696    if (oData) {
1697        this.init(oData, oParent, expanded);
1698        this.initContent(oData, hasIcon);
1699    }
1700};
1701
1702YAHOO.widget.HTMLNode.prototype = new YAHOO.widget.Node();
1703
1704/**
1705 * The CSS class for the html content container.  Defaults to ygtvhtml, but
1706 * can be overridden to provide a custom presentation for a specific node.
1707 *
1708 * @type string
1709 */
1710YAHOO.widget.HTMLNode.prototype.contentStyle = "ygtvhtml";
1711
1712/**
1713 * The generated id that will contain the data passed in by the implementer.
1714 *
1715 * @type string
1716 */
1717YAHOO.widget.HTMLNode.prototype.contentElId = null;
1718
1719/**
1720 * The HTML content to use for this node's display
1721 *
1722 * @type string
1723 */
1724YAHOO.widget.HTMLNode.prototype.content = null;
1725
1726/**
1727 * Sets up the node label
1728 *
1729 * @param {object} An html string or object containing an html property
1730 * @param {boolean} hasIcon determines if the node will be rendered with an
1731 * icon or not
1732 */
1733YAHOO.widget.HTMLNode.prototype.initContent = function(oData, hasIcon) {
1734    if (typeof oData == "string") {
1735        oData = { html: oData };
1736    }
1737
1738    this.html = oData.html;
1739    this.contentElId = "ygtvcontentel" + this.index;
1740    this.hasIcon = hasIcon;
1741
1742    /**
1743     * @private
1744     */
1745};
1746
1747/**
1748 * Returns the outer html element for this node's content
1749 *
1750 * @return {HTMLElement} the element
1751 */
1752YAHOO.widget.HTMLNode.prototype.getContentEl = function() {
1753    return document.getElementById(this.contentElId);
1754};
1755
1756// overrides YAHOO.widget.Node
1757YAHOO.widget.HTMLNode.prototype.getNodeHtml = function() {
1758    var sb = [];
1759
1760    sb[sb.length] = '<table border="0" cellpadding="0" cellspacing="0">';
1761    sb[sb.length] = '<tr>';
1762   
1763    for (i=0;i<this.depth;++i) {
1764        sb[sb.length] = '<td class="' + this.getDepthStyle(i) + '">&#160;</td>';
1765    }
1766
1767    if (this.hasIcon) {
1768        sb[sb.length] = '<td';
1769        sb[sb.length] = ' id="' + this.getToggleElId() + '"';
1770        sb[sb.length] = ' class="' + this.getStyle() + '"';
1771        sb[sb.length] = ' onclick="javascript:' + this.getToggleLink() + '"';
1772        if (this.hasChildren(true)) {
1773            sb[sb.length] = ' onmouseover="this.className=';
1774            sb[sb.length] = 'YAHOO.widget.TreeView.getNode(\'';
1775            sb[sb.length] = this.tree.id + '\',' + this.index +  ').getHoverStyle()"';
1776            sb[sb.length] = ' onmouseout="this.className=';
1777            sb[sb.length] = 'YAHOO.widget.TreeView.getNode(\'';
1778            sb[sb.length] = this.tree.id + '\',' + this.index +  ').getStyle()"';
1779        }
1780        sb[sb.length] = '>&#160;</td>';
1781    }
1782
1783    sb[sb.length] = '<td';
1784    sb[sb.length] = ' id="' + this.contentElId + '"';
1785    sb[sb.length] = ' class="' + this.contentStyle + '"';
1786    sb[sb.length] = ' >';
1787    sb[sb.length] = this.html;
1788    sb[sb.length] = '</td>';
1789    sb[sb.length] = '</tr>';
1790    sb[sb.length] = '</table>';
1791
1792    return sb.join("");
1793};
1794
1795YAHOO.widget.HTMLNode.prototype.toString = function() {
1796    return "HTMLNode (" + this.index + ")";
1797};
1798
1799/**
1800 * A static factory class for tree view expand/collapse animations
1801 *
1802 * @constructor
1803 */
1804YAHOO.widget.TVAnim = function() {
1805    return {
1806        /**
1807         * Constant for the fade in animation
1808         *
1809         * @type string
1810         */
1811        FADE_IN: "TVFadeIn",
1812
1813        /**
1814         * Constant for the fade out animation
1815         *
1816         * @type string
1817         */
1818        FADE_OUT: "TVFadeOut",
1819
1820        /**
1821         * Returns a ygAnim instance of the given type
1822         *
1823         * @param type {string} the type of animation
1824         * @param el {HTMLElement} the element to element (probably the children div)
1825         * @param callback {function} function to invoke when the animation is done.
1826         * @return {YAHOO.util.Animation} the animation instance
1827         */
1828        getAnim: function(type, el, callback) {
1829            if (YAHOO.widget[type]) {
1830                return new YAHOO.widget[type](el, callback);
1831            } else {
1832                return null;
1833            }
1834        },
1835
1836        /**
1837         * Returns true if the specified animation class is available
1838         *
1839         * @param type {string} the type of animation
1840         * @return {boolean} true if valid, false if not
1841         */
1842        isValid: function(type) {
1843            return (YAHOO.widget[type]);
1844        }
1845    };
1846} ();
1847
1848/**
1849 * A 1/2 second fade-in animation.
1850 *
1851 * @constructor
1852 * @param el {HTMLElement} the element to animate
1853 * @param callback {function} function to invoke when the animation is finished
1854 */
1855YAHOO.widget.TVFadeIn = function(el, callback) {
1856    /**
1857     * The element to animate
1858     * @type HTMLElement
1859     */
1860    this.el = el;
1861
1862    /**
1863     * the callback to invoke when the animation is complete
1864     *
1865     * @type function
1866     */
1867    this.callback = callback;
1868
1869    /**
1870     * @private
1871     */
1872};
1873
1874/**
1875 * Performs the animation
1876 */
1877YAHOO.widget.TVFadeIn.prototype = {
1878    animate: function() {
1879        var tvanim = this;
1880
1881        var s = this.el.style;
1882        s.opacity = 0.1;
1883        s.filter = "alpha(opacity=10)";
1884        s.display = "";
1885
1886        // var dur = ( navigator.userAgent.match(/msie/gi) ) ? 0.05 : 0.4;
1887        var dur = 0.4;
1888        // var a = new ygAnim_Fade(this.el, dur, 1);
1889        // a.setStart(0.1);
1890        // a.onComplete = function() { tvanim.onComplete(); };
1891
1892        // var a = new YAHOO.util.Anim(this.el, 'opacity', 0.1, 1);
1893        var a = new YAHOO.util.Anim(this.el, {opacity: {from: 0.1, to: 1, unit:""}}, dur);
1894        a.onComplete.subscribe( function() { tvanim.onComplete(); } );
1895        a.animate();
1896    },
1897
1898    /**
1899     * Clean up and invoke callback
1900     */
1901    onComplete: function() {
1902        this.callback();
1903    },
1904
1905    toString: function() {
1906        return "TVFadeIn";
1907    }
1908};
1909
1910/**
1911 * A 1/2 second fade out animation.
1912 *
1913 * @constructor
1914 * @param el {HTMLElement} the element to animate
1915 * @param callback {Function} function to invoke when the animation is finished
1916 */
1917YAHOO.widget.TVFadeOut = function(el, callback) {
1918    /**
1919     * The element to animate
1920     * @type HTMLElement
1921     */
1922    this.el = el;
1923
1924    /**
1925     * the callback to invoke when the animation is complete
1926     *
1927     * @type function
1928     */
1929    this.callback = callback;
1930
1931    /**
1932     * @private
1933     */
1934};
1935
1936/**
1937 * Performs the animation
1938 */
1939YAHOO.widget.TVFadeOut.prototype = {
1940    animate: function() {
1941        var tvanim = this;
1942        // var dur = ( navigator.userAgent.match(/msie/gi) ) ? 0.05 : 0.4;
1943        var dur = 0.4;
1944        // var a = new ygAnim_Fade(this.el, dur, 0.1);
1945        // a.onComplete = function() { tvanim.onComplete(); };
1946
1947        // var a = new YAHOO.util.Anim(this.el, 'opacity', 1, 0.1);
1948        var a = new YAHOO.util.Anim(this.el, {opacity: {from: 1, to: 0.1, unit:""}}, dur);
1949        a.onComplete.subscribe( function() { tvanim.onComplete(); } );
1950        a.animate();
1951    },
1952
1953    /**
1954     * Clean up and invoke callback
1955     */
1956    onComplete: function() {
1957        var s = this.el.style;
1958        s.display = "none";
1959        // s.opacity = 1;
1960        s.filter = "alpha(opacity=100)";
1961        this.callback();
1962    },
1963
1964    toString: function() {
1965        return "TVFadeOut";
1966    }
1967};
1968
Note: See TracBrowser for help on using the repository browser.