TreeNode.js
上传用户:dawnssy
上传日期:2022-08-06
资源大小:9345k
文件大小:19k
源码类别:

JavaScript

开发平台:

JavaScript

  1. /*!  * Ext JS Library 3.1.0  * Copyright(c) 2006-2009 Ext JS, LLC  * licensing@extjs.com  * http://www.extjs.com/license  */ /**
  2.  * @class Ext.tree.TreeNode
  3.  * @extends Ext.data.Node
  4.  * @cfg {String} text The text for this node
  5.  * @cfg {Boolean} expanded true to start the node expanded
  6.  * @cfg {Boolean} allowDrag False to make this node undraggable if {@link #draggable} = true (defaults to true)
  7.  * @cfg {Boolean} allowDrop False if this node cannot have child nodes dropped on it (defaults to true)
  8.  * @cfg {Boolean} disabled true to start the node disabled
  9.  * @cfg {String} icon The path to an icon for the node. The preferred way to do this
  10.  * is to use the cls or iconCls attributes and add the icon via a CSS background image.
  11.  * @cfg {String} cls A css class to be added to the node
  12.  * @cfg {String} iconCls A css class to be added to the nodes icon element for applying css background images
  13.  * @cfg {String} href URL of the link used for the node (defaults to #)
  14.  * @cfg {String} hrefTarget target frame for the link
  15.  * @cfg {Boolean} hidden True to render hidden. (Defaults to false).
  16.  * @cfg {String} qtip An Ext QuickTip for the node
  17.  * @cfg {Boolean} expandable If set to true, the node will always show a plus/minus icon, even when empty
  18.  * @cfg {String} qtipCfg An Ext QuickTip config for the node (used instead of qtip)
  19.  * @cfg {Boolean} singleClickExpand True for single click expand on this node
  20.  * @cfg {Function} uiProvider A UI <b>class</b> to use for this node (defaults to Ext.tree.TreeNodeUI)
  21.  * @cfg {Boolean} checked True to render a checked checkbox for this node, false to render an unchecked checkbox
  22.  * (defaults to undefined with no checkbox rendered)
  23.  * @cfg {Boolean} draggable True to make this node draggable (defaults to false)
  24.  * @cfg {Boolean} isTarget False to not allow this node to act as a drop target (defaults to true)
  25.  * @cfg {Boolean} allowChildren False to not allow this node to have child nodes (defaults to true)
  26.  * @cfg {Boolean} editable False to not allow this node to be edited by an {@link Ext.tree.TreeEditor} (defaults to true)
  27.  * @constructor
  28.  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node
  29.  */
  30. Ext.tree.TreeNode = function(attributes){
  31.     attributes = attributes || {};
  32.     if(Ext.isString(attributes)){
  33.         attributes = {text: attributes};
  34.     }
  35.     this.childrenRendered = false;
  36.     this.rendered = false;
  37.     Ext.tree.TreeNode.superclass.constructor.call(this, attributes);
  38.     this.expanded = attributes.expanded === true;
  39.     this.isTarget = attributes.isTarget !== false;
  40.     this.draggable = attributes.draggable !== false && attributes.allowDrag !== false;
  41.     this.allowChildren = attributes.allowChildren !== false && attributes.allowDrop !== false;
  42.     /**
  43.      * Read-only. The text for this node. To change it use <code>{@link #setText}</code>.
  44.      * @type String
  45.      */
  46.     this.text = attributes.text;
  47.     /**
  48.      * True if this node is disabled.
  49.      * @type Boolean
  50.      */
  51.     this.disabled = attributes.disabled === true;
  52.     /**
  53.      * True if this node is hidden.
  54.      * @type Boolean
  55.      */
  56.     this.hidden = attributes.hidden === true;
  57.     this.addEvents(
  58.         /**
  59.         * @event textchange
  60.         * Fires when the text for this node is changed
  61.         * @param {Node} this This node
  62.         * @param {String} text The new text
  63.         * @param {String} oldText The old text
  64.         */
  65.         'textchange',
  66.         /**
  67.         * @event beforeexpand
  68.         * Fires before this node is expanded, return false to cancel.
  69.         * @param {Node} this This node
  70.         * @param {Boolean} deep
  71.         * @param {Boolean} anim
  72.         */
  73.         'beforeexpand',
  74.         /**
  75.         * @event beforecollapse
  76.         * Fires before this node is collapsed, return false to cancel.
  77.         * @param {Node} this This node
  78.         * @param {Boolean} deep
  79.         * @param {Boolean} anim
  80.         */
  81.         'beforecollapse',
  82.         /**
  83.         * @event expand
  84.         * Fires when this node is expanded
  85.         * @param {Node} this This node
  86.         */
  87.         'expand',
  88.         /**
  89.         * @event disabledchange
  90.         * Fires when the disabled status of this node changes
  91.         * @param {Node} this This node
  92.         * @param {Boolean} disabled
  93.         */
  94.         'disabledchange',
  95.         /**
  96.         * @event collapse
  97.         * Fires when this node is collapsed
  98.         * @param {Node} this This node
  99.         */
  100.         'collapse',
  101.         /**
  102.         * @event beforeclick
  103.         * Fires before click processing. Return false to cancel the default action.
  104.         * @param {Node} this This node
  105.         * @param {Ext.EventObject} e The event object
  106.         */
  107.         'beforeclick',
  108.         /**
  109.         * @event click
  110.         * Fires when this node is clicked
  111.         * @param {Node} this This node
  112.         * @param {Ext.EventObject} e The event object
  113.         */
  114.         'click',
  115.         /**
  116.         * @event checkchange
  117.         * Fires when a node with a checkbox's checked property changes
  118.         * @param {Node} this This node
  119.         * @param {Boolean} checked
  120.         */
  121.         'checkchange',
  122.         /**
  123.         * @event beforedblclick
  124.         * Fires before double click processing. Return false to cancel the default action.
  125.         * @param {Node} this This node
  126.         * @param {Ext.EventObject} e The event object
  127.         */
  128.         'beforedblclick',
  129.         /**
  130.         * @event dblclick
  131.         * Fires when this node is double clicked
  132.         * @param {Node} this This node
  133.         * @param {Ext.EventObject} e The event object
  134.         */
  135.         'dblclick',
  136.         /**
  137.         * @event contextmenu
  138.         * Fires when this node is right clicked
  139.         * @param {Node} this This node
  140.         * @param {Ext.EventObject} e The event object
  141.         */
  142.         'contextmenu',
  143.         /**
  144.         * @event beforechildrenrendered
  145.         * Fires right before the child nodes for this node are rendered
  146.         * @param {Node} this This node
  147.         */
  148.         'beforechildrenrendered'
  149.     );
  150.     var uiClass = this.attributes.uiProvider || this.defaultUI || Ext.tree.TreeNodeUI;
  151.     /**
  152.      * Read-only. The UI for this node
  153.      * @type TreeNodeUI
  154.      */
  155.     this.ui = new uiClass(this);
  156. };
  157. Ext.extend(Ext.tree.TreeNode, Ext.data.Node, {
  158.     preventHScroll : true,
  159.     /**
  160.      * Returns true if this node is expanded
  161.      * @return {Boolean}
  162.      */
  163.     isExpanded : function(){
  164.         return this.expanded;
  165.     },
  166. /**
  167.  * Returns the UI object for this node.
  168.  * @return {TreeNodeUI} The object which is providing the user interface for this tree
  169.  * node. Unless otherwise specified in the {@link #uiProvider}, this will be an instance
  170.  * of {@link Ext.tree.TreeNodeUI}
  171.  */
  172.     getUI : function(){
  173.         return this.ui;
  174.     },
  175.     getLoader : function(){
  176.         var owner;
  177.         return this.loader || ((owner = this.getOwnerTree()) && owner.loader ? owner.loader : (this.loader = new Ext.tree.TreeLoader()));
  178.     },
  179.     // private override
  180.     setFirstChild : function(node){
  181.         var of = this.firstChild;
  182.         Ext.tree.TreeNode.superclass.setFirstChild.call(this, node);
  183.         if(this.childrenRendered && of && node != of){
  184.             of.renderIndent(true, true);
  185.         }
  186.         if(this.rendered){
  187.             this.renderIndent(true, true);
  188.         }
  189.     },
  190.     // private override
  191.     setLastChild : function(node){
  192.         var ol = this.lastChild;
  193.         Ext.tree.TreeNode.superclass.setLastChild.call(this, node);
  194.         if(this.childrenRendered && ol && node != ol){
  195.             ol.renderIndent(true, true);
  196.         }
  197.         if(this.rendered){
  198.             this.renderIndent(true, true);
  199.         }
  200.     },
  201.     // these methods are overridden to provide lazy rendering support
  202.     // private override
  203.     appendChild : function(n){
  204.         if(!n.render && !Ext.isArray(n)){
  205.             n = this.getLoader().createNode(n);
  206.         }
  207.         var node = Ext.tree.TreeNode.superclass.appendChild.call(this, n);
  208.         if(node && this.childrenRendered){
  209.             node.render();
  210.         }
  211.         this.ui.updateExpandIcon();
  212.         return node;
  213.     },
  214.     // private override
  215.     removeChild : function(node, destroy){
  216.         this.ownerTree.getSelectionModel().unselect(node);
  217.         Ext.tree.TreeNode.superclass.removeChild.apply(this, arguments);
  218.         // if it's been rendered remove dom node
  219.         if(node.ui.rendered){
  220.             node.ui.remove();
  221.         }
  222.         if(this.childNodes.length < 1){
  223.             this.collapse(false, false);
  224.         }else{
  225.             this.ui.updateExpandIcon();
  226.         }
  227.         if(!this.firstChild && !this.isHiddenRoot()) {
  228.             this.childrenRendered = false;
  229.         }
  230.         return node;
  231.     },
  232.     // private override
  233.     insertBefore : function(node, refNode){
  234.         if(!node.render){
  235.             node = this.getLoader().createNode(node);
  236.         }
  237.         var newNode = Ext.tree.TreeNode.superclass.insertBefore.call(this, node, refNode);
  238.         if(newNode && refNode && this.childrenRendered){
  239.             node.render();
  240.         }
  241.         this.ui.updateExpandIcon();
  242.         return newNode;
  243.     },
  244.     /**
  245.      * Sets the text for this node
  246.      * @param {String} text
  247.      */
  248.     setText : function(text){
  249.         var oldText = this.text;
  250.         this.text = this.attributes.text = text;
  251.         if(this.rendered){ // event without subscribing
  252.             this.ui.onTextChange(this, text, oldText);
  253.         }
  254.         this.fireEvent('textchange', this, text, oldText);
  255.     },
  256.     /**
  257.      * Triggers selection of this node
  258.      */
  259.     select : function(){
  260.         var t = this.getOwnerTree();
  261.         if(t){
  262.             t.getSelectionModel().select(this);
  263.         }
  264.     },
  265.     /**
  266.      * Triggers deselection of this node
  267.      * @param {Boolean} silent (optional) True to stop selection change events from firing.
  268.      */
  269.     unselect : function(silent){
  270.         var t = this.getOwnerTree();
  271.         if(t){
  272.             t.getSelectionModel().unselect(this, silent);
  273.         }
  274.     },
  275.     /**
  276.      * Returns true if this node is selected
  277.      * @return {Boolean}
  278.      */
  279.     isSelected : function(){
  280.         var t = this.getOwnerTree();
  281.         return t ? t.getSelectionModel().isSelected(this) : false;
  282.     },
  283.     /**
  284.      * Expand this node.
  285.      * @param {Boolean} deep (optional) True to expand all children as well
  286.      * @param {Boolean} anim (optional) false to cancel the default animation
  287.      * @param {Function} callback (optional) A callback to be called when
  288.      * expanding this node completes (does not wait for deep expand to complete).
  289.      * Called with 1 parameter, this node.
  290.      * @param {Object} scope (optional) The scope (<code>this</code> reference) in which the callback is executed. Defaults to this TreeNode.
  291.      */
  292.     expand : function(deep, anim, callback, scope){
  293.         if(!this.expanded){
  294.             if(this.fireEvent('beforeexpand', this, deep, anim) === false){
  295.                 return;
  296.             }
  297.             if(!this.childrenRendered){
  298.                 this.renderChildren();
  299.             }
  300.             this.expanded = true;
  301.             if(!this.isHiddenRoot() && (this.getOwnerTree().animate && anim !== false) || anim){
  302.                 this.ui.animExpand(function(){
  303.                     this.fireEvent('expand', this);
  304.                     this.runCallback(callback, scope || this, [this]);
  305.                     if(deep === true){
  306.                         this.expandChildNodes(true);
  307.                     }
  308.                 }.createDelegate(this));
  309.                 return;
  310.             }else{
  311.                 this.ui.expand();
  312.                 this.fireEvent('expand', this);
  313.                 this.runCallback(callback, scope || this, [this]);
  314.             }
  315.         }else{
  316.            this.runCallback(callback, scope || this, [this]);
  317.         }
  318.         if(deep === true){
  319.             this.expandChildNodes(true);
  320.         }
  321.     },
  322.     runCallback : function(cb, scope, args){
  323.         if(Ext.isFunction(cb)){
  324.             cb.apply(scope, args);
  325.         }
  326.     },
  327.     isHiddenRoot : function(){
  328.         return this.isRoot && !this.getOwnerTree().rootVisible;
  329.     },
  330.     /**
  331.      * Collapse this node.
  332.      * @param {Boolean} deep (optional) True to collapse all children as well
  333.      * @param {Boolean} anim (optional) false to cancel the default animation
  334.      * @param {Function} callback (optional) A callback to be called when
  335.      * expanding this node completes (does not wait for deep expand to complete).
  336.      * Called with 1 parameter, this node.
  337.      * @param {Object} scope (optional) The scope (<code>this</code> reference) in which the callback is executed. Defaults to this TreeNode.
  338.      */
  339.     collapse : function(deep, anim, callback, scope){
  340.         if(this.expanded && !this.isHiddenRoot()){
  341.             if(this.fireEvent('beforecollapse', this, deep, anim) === false){
  342.                 return;
  343.             }
  344.             this.expanded = false;
  345.             if((this.getOwnerTree().animate && anim !== false) || anim){
  346.                 this.ui.animCollapse(function(){
  347.                     this.fireEvent('collapse', this);
  348.                     this.runCallback(callback, scope || this, [this]);
  349.                     if(deep === true){
  350.                         this.collapseChildNodes(true);
  351.                     }
  352.                 }.createDelegate(this));
  353.                 return;
  354.             }else{
  355.                 this.ui.collapse();
  356.                 this.fireEvent('collapse', this);
  357.                 this.runCallback(callback, scope || this, [this]);
  358.             }
  359.         }else if(!this.expanded){
  360.             this.runCallback(callback, scope || this, [this]);
  361.         }
  362.         if(deep === true){
  363.             var cs = this.childNodes;
  364.             for(var i = 0, len = cs.length; i < len; i++) {
  365.              cs[i].collapse(true, false);
  366.             }
  367.         }
  368.     },
  369.     // private
  370.     delayedExpand : function(delay){
  371.         if(!this.expandProcId){
  372.             this.expandProcId = this.expand.defer(delay, this);
  373.         }
  374.     },
  375.     // private
  376.     cancelExpand : function(){
  377.         if(this.expandProcId){
  378.             clearTimeout(this.expandProcId);
  379.         }
  380.         this.expandProcId = false;
  381.     },
  382.     /**
  383.      * Toggles expanded/collapsed state of the node
  384.      */
  385.     toggle : function(){
  386.         if(this.expanded){
  387.             this.collapse();
  388.         }else{
  389.             this.expand();
  390.         }
  391.     },
  392.     /**
  393.      * Ensures all parent nodes are expanded, and if necessary, scrolls
  394.      * the node into view.
  395.      * @param {Function} callback (optional) A function to call when the node has been made visible.
  396.      * @param {Object} scope (optional) The scope (<code>this</code> reference) in which the callback is executed. Defaults to this TreeNode.
  397.      */
  398.     ensureVisible : function(callback, scope){
  399.         var tree = this.getOwnerTree();
  400.         tree.expandPath(this.parentNode ? this.parentNode.getPath() : this.getPath(), false, function(){
  401.             var node = tree.getNodeById(this.id);  // Somehow if we don't do this, we lose changes that happened to node in the meantime
  402.             tree.getTreeEl().scrollChildIntoView(node.ui.anchor);
  403.             this.runCallback(callback, scope || this, [this]);
  404.         }.createDelegate(this));
  405.     },
  406.     /**
  407.      * Expand all child nodes
  408.      * @param {Boolean} deep (optional) true if the child nodes should also expand their child nodes
  409.      */
  410.     expandChildNodes : function(deep){
  411.         var cs = this.childNodes;
  412.         for(var i = 0, len = cs.length; i < len; i++) {
  413.          cs[i].expand(deep);
  414.         }
  415.     },
  416.     /**
  417.      * Collapse all child nodes
  418.      * @param {Boolean} deep (optional) true if the child nodes should also collapse their child nodes
  419.      */
  420.     collapseChildNodes : function(deep){
  421.         var cs = this.childNodes;
  422.         for(var i = 0, len = cs.length; i < len; i++) {
  423.          cs[i].collapse(deep);
  424.         }
  425.     },
  426.     /**
  427.      * Disables this node
  428.      */
  429.     disable : function(){
  430.         this.disabled = true;
  431.         this.unselect();
  432.         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
  433.             this.ui.onDisableChange(this, true);
  434.         }
  435.         this.fireEvent('disabledchange', this, true);
  436.     },
  437.     /**
  438.      * Enables this node
  439.      */
  440.     enable : function(){
  441.         this.disabled = false;
  442.         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
  443.             this.ui.onDisableChange(this, false);
  444.         }
  445.         this.fireEvent('disabledchange', this, false);
  446.     },
  447.     // private
  448.     renderChildren : function(suppressEvent){
  449.         if(suppressEvent !== false){
  450.             this.fireEvent('beforechildrenrendered', this);
  451.         }
  452.         var cs = this.childNodes;
  453.         for(var i = 0, len = cs.length; i < len; i++){
  454.             cs[i].render(true);
  455.         }
  456.         this.childrenRendered = true;
  457.     },
  458.     // private
  459.     sort : function(fn, scope){
  460.         Ext.tree.TreeNode.superclass.sort.apply(this, arguments);
  461.         if(this.childrenRendered){
  462.             var cs = this.childNodes;
  463.             for(var i = 0, len = cs.length; i < len; i++){
  464.                 cs[i].render(true);
  465.             }
  466.         }
  467.     },
  468.     // private
  469.     render : function(bulkRender){
  470.         this.ui.render(bulkRender);
  471.         if(!this.rendered){
  472.             // make sure it is registered
  473.             this.getOwnerTree().registerNode(this);
  474.             this.rendered = true;
  475.             if(this.expanded){
  476.                 this.expanded = false;
  477.                 this.expand(false, false);
  478.             }
  479.         }
  480.     },
  481.     // private
  482.     renderIndent : function(deep, refresh){
  483.         if(refresh){
  484.             this.ui.childIndent = null;
  485.         }
  486.         this.ui.renderIndent();
  487.         if(deep === true && this.childrenRendered){
  488.             var cs = this.childNodes;
  489.             for(var i = 0, len = cs.length; i < len; i++){
  490.                 cs[i].renderIndent(true, refresh);
  491.             }
  492.         }
  493.     },
  494.     beginUpdate : function(){
  495.         this.childrenRendered = false;
  496.     },
  497.     endUpdate : function(){
  498.         if(this.expanded && this.rendered){
  499.             this.renderChildren();
  500.         }
  501.     },
  502.     destroy : function(){
  503.         this.unselect(true);
  504.         Ext.tree.TreeNode.superclass.destroy.call(this);
  505.         Ext.destroy(this.ui, this.loader);
  506.         this.ui = this.loader = null;
  507.     },
  508.     // private
  509.     onIdChange : function(id){
  510.         this.ui.onIdChange(id);
  511.     }
  512. });
  513. Ext.tree.TreePanel.nodeTypes.node = Ext.tree.TreeNode;