[Ext扩展]QM.ui.TreeCombo:多功能下拉树列表,内含文档与示例

有句老话叫不重复造轮子,既然网上已经有下拉树的扩展,为什么还要再做一个呢?答案很简单,网上那些满足不了我的需要。简单来说,本扩展UI组件具备以下功能:

  • 宽度自适应 下拉列表宽度可根据树的大小进行自动调整
  • 延迟加载 默认采用,提高页面渲染速度
  • 自动寻路 下拉列表展开时会将选中树结点的按其路径展开
  • 键盘导航 支持使用上下左右回车ESC、TAB进行操作
  • 智能过滤 支持使用汉字和拼音首字母对树进行过滤
  • 默认值支持 可以为组件设置默认值,组件会自动找到对应结点
  • 忽略父节点 默认情况只有选择子节点才有效


示例放到Ext根路径下即可查看,内含详细使用说明,效果图如下:

Ext版本已测试3.1.1、3.2.0
IE6下页面显示问题解决方案:设置css如下

Css代码 复制代码
  1. body {   
  2.    font-size:13px;   
  3. }  
body {
   font-size:13px;
}

(这是IE的问题,不用怀疑)
,Firefox、Chrome测试正常
源代码:

Js代码 复制代码
  1. /**  
  2.  *  
  3.  * 支持功能:  
  4.  * 1.自动宽度和高度调整;弹出层会根据依据树的宽高调整自身宽高,注意:设置listWidth属性后弹出  
  5.  * 层宽度将固定,树面板在需要时会出现水平滚动条;设置minListWidth属性可限定弹出层最小宽度,  
  6.  * 默认为TreeCombo的宽度;  
  7.  * 2.自动寻找选中结点:弹出层展开后会根据当前值找到对应结点并根据其树中路径展开树,其余结点将  
  8.  * 收缩。  
  9.  * 3.程序赋值:通过setValue(String nodeId);方法可通过代码设置TreeCombo的Value值,此时程序会  
  10.  * 自动找到树节点id对应的树节点text并将其显示到输入域中,如果找不到对应树节点,输入域会显示  
  11.  * 配置项valueNotFoundText的值提示这是个程序错误!  
  12.  * 4.自动检测:输入域默认是可以编辑的,ComboBox失去焦点后自动检测当前输入域中内容是否有对应的树  
  13.  * 结点,有的话设置value值为对应结点,没有则自动退回到上一次失去焦点时的状态;  
  14.  * 5.键盘导航:down(选择上一节点)、up(选择下一结点)、enter(选中当前选择结点)、left(收  
  15.  * 缩当前结点)、right(展开当前结点)、esc(取消编辑)、tab(取消编辑并跳转到下一个输入域);  
  16.  * 6.忽略父节点:ignoreFolder可设置父节点不能被选中,此属性为true时用户点击父节点时面板不会  
  17.  * 收缩、过滤时也会忽略父节点;  
  18.  * 也不会被赋值,调用setValue方法时如果找到的是父节点输入域会显示配置项valueNotFoundText的值;  
  19.  * 7.拼音首字母过滤:用户对输入域编辑时树节点都会根据输入域中字母进行拼音首字母过滤,匹配结点  
  20.  * 将保留并展开,内置缓存处理以提高多次查询的检索速度,可与键盘导航功能同时使用,最大限度提高  
  21.  * 数据录入速度;  
  22.  * 8.延迟初始化:默认情况下lazyInit属性为true,此时tree会在TreeCombo获得焦点后才进行创建,这样做  
  23.  * 可以提高页面加载速度。  
  24.  * 9.可输入状态下输入域允许粘贴,失去焦点后组件会找到对应树节点并为值域(value)赋值,找不到显示组件  
  25.  * 的emptyText  
  26.  * 10.默认值支持:如果需要默认值可以将结点id赋给value,默认情况下因为使用延迟加载,默认值只有在组件  
  27.  * 获得焦点后才会通过树找到要结点text属性并显示到界面上,如果需要加载后就显示初始值可将lazyInit设置  
  28.  * 为false  
  29.  *   
  30.  * 注意事项:  
  31.  * 1.如果树结点不是由root一次性加载,功能479将无法适用,此时请配置TreeCombo的editable属性为false;  
  32.  * 2.不同与官方的ComboBox,此下拉框没有forceSelection配置项(该配置项为false时用户可自定义下  
  33.  * 拉框内容),因为我不知道允许用户自定义内容有什么意义;  
  34.  * 3.为ComboBox配置tree属性必须配置Object直接量对象(用{}来声明的对象)不能是new创建的实例,  
  35.  * TreeCombo会负责树对象的创建和销毁;  
  36.  * 4.树的根结点是不可见的,此时树在创建时就会执行root的expand方法开始加载其子结点,因而树的root  
  37.  * 结点必须以new Ext.tree.AsyncTreeNode方式进行声明,否则无法添加事件判断其加载完成状态。  
  38.  * 5.如果树节点过多,过滤功能将严重影响性能,此时可设置  
  39.  *   
  40.  * v1.1改动:  
  41.  * 1.添加一个配置项maxHeight用于控制列表的最大高度  
  42.  *   
  43.  * @author chemzqm@gmail.com  
  44.  * @version 1.0.0  
  45.  * @createTime 2010-04-24 16:40:30  
  46.  *   
  47.  */  
  48. Ext.ns('QM.ui');   
  49.   
  50. QM.ui.TreeCombo = Ext.extend(Ext.form.TriggerField, {   
  51.     shadow: 'sides',   
  52.     /**  
  53.      * @cfg minListWidth 弹出层最小宽度,必须大于ComboBox宽度,与listWidth同时使用时无效  
  54.      */  
  55.     /**  
  56.      * @cfg listWidth 弹出层固定宽度,设置后弹出层不根据树宽度进行调整,必须大于ComboBox宽度  
  57.      *  
  58.      */  
  59.     /**  
  60.      * @cfg hiddenName 隐藏表单域名,form方式提交时需要,负责把value传到后台  
  61.      */  
  62.     /**  
  63.      * @cfg listClass 阴影层样式  
  64.      */  
  65.   
  66.     listAlign: 'tl-bl?',   
  67.     queryDelay:300,//查询函数缓冲时间(缓冲时间内再次调用将取消上次调用)   
  68.     valueNotFoundText: '没有指定数据',   
  69.     triggerAction: 'all',//下拉按钮点击时查询条件'all'查询出所有数据 'query'根据输入项进行前端匹配   
  70.     ignoreFolder: true,//父节点不做为数据源   
  71.     lazyInit: true,//控件获得焦点时才会初始化下拉框包括树   
  72.     loadingText:'加载中...',   
  73.     emptyText:'请选择...',   
  74.     forceSelection: true,//输入框的值只能是列表存有的值   
  75.     enableQuery : true,   
  76.        
  77.        
  78.     initComponent: function(){   
  79.         this.hiddenPkgs = [];//隐藏的分支节点   
  80.         QM.ui.TreeCombo.superclass.initComponent.call(this);   
  81.         this.addEvents('expand''collapse','beforeselect','select');   
  82.     },   
  83.     onRender: function(ct, position){   
  84.         QM.ui.TreeCombo.superclass.onRender.call(this, ct, position);   
  85.         if(this.hiddenName){   
  86.             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName,   
  87.                     id: (this.hiddenId||this.hiddenName)}, 'before'true);   
  88.         }   
  89.         if (this.lazyInit) {   
  90.             this.on('focus'this.initList, this, {   
  91.                 single: true  
  92.             });   
  93.         }   
  94.         else {   
  95.             this.initList();   
  96.         }   
  97.     },   
  98.        
  99.     initEvents : function(){   
  100.         QM.ui.TreeCombo.superclass.initEvents.call(this);   
  101.         this.keyNav = new Ext.KeyNav(this.el, {   
  102.             "up" : this.onKeyDown,   
  103.             "down" : function(e){   
  104.                 if (!this.isExpanded()) {   
  105.                     this.onTriggerClick();   
  106.                 }   
  107.                 else {   
  108.                     this.onKeyDown(e);   
  109.                 }   
  110.             },   
  111.             "left":this.onKeyDown,   
  112.             "right":this.onKeyDown,   
  113.             "enter":function(){   
  114.                 var node = this.tree.selModel.getSelectedNode();   
  115.                 this.onTreeClick(node);   
  116.             },   
  117.             "esc" : function(e){   
  118.                 this.collapse();   
  119.             },   
  120.             "tab" : function(e){   
  121.                 this.collapse();   
  122.                 return true;   
  123.             },   
  124.             scope : this,   
  125.             forceKeyDown : true  
  126.         });   
  127.         this.dqTask = new Ext.util.DelayedTask(this.initQuery, this);   
  128.         if(!this.enableKeyEvents){   
  129.             this.mon(this.el, 'keyup'this.onKeyUp, this);   
  130.         }   
  131.     },   
  132.     //上下左右回车让TreeSelectionModel来辅助实现   
  133.     onKeyDown:function(e){   
  134.         var sm = this.tree.getSelectionModel();   
  135.         if(sm){   
  136.             sm.onKeyDown(e);   
  137.         }   
  138.         this.el.focus();   
  139.     },   
  140.     initQuery:function(){   
  141.         this.doQuery(this.getRawValue());   
  142.     },   
  143.     onKeyUp : function(e){   
  144.         var k = e.getKey();   
  145.         if(this.editable !== false && this.readOnly !== true && (k == e.BACKSPACE || !e.isSpecialKey())){   
  146.             this.dqTask.delay(this.queryDelay);   
  147.         }   
  148.         Ext.form.ComboBox.superclass.onKeyUp.call(this, e);   
  149.     },   
  150.     initList: function(){   
  151.         if (!this.list) {   
  152.             var cls = 'x-combo-list',    
  153.             listParent = Ext.getDom(this.getListParent() || Ext.getBody()),    
  154.             zindex = parseInt(Ext.fly(listParent).getStyle('z-index'), 10);    
  155.             if (this.ownerCt && !zindex) {//找到父容器定义的z-index   
  156.                 this.findParentBy(function(ct){   
  157.                     zindex = parseInt(ct.getPositionEl().getStyle('z-index'), 10);   
  158.                     return !!zindex;   
  159.                 });   
  160.             }   
  161.             this.list = new Ext.Layer({   
  162.                 parentEl: listParent,   
  163.                 shadow: this.shadow,   
  164.                 cls: [cls, this.listClass].join(' '),   
  165.                 constrain: false,   
  166.                 zindex: (zindex || 12000) + 5   
  167.             });   
  168.             if(!this.minListWidth){   
  169.                 this.minListWidth = this.wrap.getWidth();   
  170.             }   
  171.             this.list.setStyle('width'this.minListWidth);   
  172.             this.list.setStyle('height''auto');   
  173.             this.list.swallowEvent('mousewheel');   
  174.             this.innerList = this.list.createChild({   
  175.                 cls: cls + '-inner'  
  176.             });   
  177.             this.initInner();   
  178.         }   
  179.     },   
  180.     initInner: function(){   
  181.         Ext.apply(this.tree, {   
  182.             applyTo: this.innerList,   
  183.             border: false,   
  184.             rootVisible: false,   
  185.             autoScroll: true  
  186.         });   
  187.         var root = this.tree.root;   
  188.         if(root instanceof Ext.tree.AsyncTreeNode){   
  189.             root.on('beforeload',this.onBeforeRootLoad,this,{single:true});        
  190.             root.on('load',this.onRootLoad,this,{single:true});   
  191.         }   
  192.         this.tree = Ext.create(this.tree, 'treepanel');   
  193.         this.tree.on({//加载完毕后再给树添加监听   
  194.             scope: this,   
  195.             expandnode: this.onTreeResize,   
  196.             collapsenode: this.onTreeResize,   
  197.             click: this.onTreeClick   
  198.         });   
  199.         if(this.editable){   
  200.             this.filter = new QM.ux.TreeFilter(this.tree,{   
  201.                 ignoreFolder:this.ignoreFolder,   
  202.                 clearAction:'collapse'  
  203.             });   
  204.         }   
  205.         if(this.value) {   
  206.             this.setValue(this.value);   
  207.         }   
  208.     },   
  209.     //@private   
  210.     onRootLoad:function(){   
  211.         this.isLoading = false;   
  212.         if (this.value) {   
  213.             this.setValue(this.value);   
  214.         }   
  215.         this.innerList.child('.loading-indicator').remove();   
  216.         if(this.isExpanded()){   
  217.             this.onLoad();   
  218.         }   
  219.     },   
  220.     //@private   
  221.     onTreeResize:function(){   
  222.         if(this.isExpanded()&&this.isQuerying!==true){   
  223.             this.restrict();   
  224.             this.el.focus();   
  225.         }   
  226.     },   
  227.     //@private   
  228.     onBeforeRootLoad : function(){   
  229.         this.isLoading = true;   
  230.         this.innerList.insertFirst({   
  231.              tag:'div',   
  232.              cls:'loading-indicator',   
  233.             html:this.loadingText   
  234.         });   
  235.     },   
  236.     //@private   
  237.     onLoad: function(){   
  238.         if (!this.hasFocus) {   
  239.             return;   
  240.         }      
  241.         this.expand();          
  242.         if(!this.selectByNode(this.value, true)){          
  243.             this.selectByNode(this.tree.root.firstChild, true);//没有的话选中第一个结点   
  244.         }   
  245.         if (this.editable) {   
  246.             this.el.focus();   
  247.         }   
  248.     },   
  249.     isExpanded: function(){   
  250.         return this.list && this.list.isVisible();   
  251.     },   
  252.     expand: function(){   
  253.         if (this.isExpanded() || !this.hasFocus) {   
  254.             return;   
  255.         }           
  256.         this.list.show();   
  257.         this.mon(Ext.getDoc(), {   
  258.             scope: this,   
  259.             mousewheel: this.collapseIf,   
  260.             mousedown: this.collapseIf   
  261.         });   
  262.         this.fireEvent('expand'this);   
  263.     },   
  264.     collapseIf: function(e){   
  265.         if (!e.within(this.wrap) && !e.within(this.list)) {   
  266.             this.collapse();   
  267.         }   
  268.     },   
  269.     //@public   
  270.     collapse: function(){   
  271.         if (!this.isExpanded()) {   
  272.             return;   
  273.         }   
  274.         this.list.hide();   
  275.         Ext.getDoc().un('mousewheel'this.collapseIf, this);   
  276.         Ext.getDoc().un('mousedown'this.collapseIf, this);   
  277.         this.fireEvent('collapse'this);   
  278.     },   
  279.     //重置弹出层宽度   
  280.     restrict: function(){   
  281.         this.innerList.dom.style.width = '10px';//外层挤压,值太小会被Chrome忽略   
  282.         var body = this.tree.body.dom;   
  283.             wpad = this.list.getFrameWidth('lr'),//边宽   
  284.               wa = Math.max(body.clientWidth, body.offsetWidth, body.scrollWidth),    
  285.                w = Math.max(wa, this.minListWidth - wpad);   
  286.                w = this.listWidth ? this.listWidth : w;   
  287.               lh = this.list.getHeight();   
  288.                h = (this.maxHeight&&this.maxHeight<lh)?this.maxHeight:lh;//获取高度   
  289.         this.list.setHeight(h);   
  290.         this.innerList.setHeight(h);   
  291.         this.list.setWidth(w + wpad);   
  292.         this.innerList.setWidth(w);   
  293.         this.list.alignTo.apply(this.list, [this.el].concat(this.listAlign));   
  294.         return;   
  295.            
  296.     },   
  297.     getListParent: function(){   
  298.         return document.body;   
  299.     },   
  300.     onTriggerClick: function(){   
  301.         if (this.readOnly || this.disabled) {   
  302.             return;   
  303.         }   
  304.         if (this.isExpanded()) {   
  305.             this.collapse();   
  306.             this.el.focus();   
  307.         }   
  308.         else {   
  309.             this.onFocus({});   
  310.             if(this.filter){   
  311.                 this.filter.clear();           
  312.             }   
  313.             this.onLoad();         
  314.         }   
  315.     },   
  316.     doQuery: function(q){   
  317.         q = Ext.isEmpty(q) ? '' : q;   
  318.         if (!this.isLoading && this.filter) {   
  319.             this.filter.filter(q);   
  320.         }   
  321.         if(this.filter.isCleared()){   
  322.             this.tree.root.firstChild.select();   
  323.         }   
  324.         else if(this.filter.hasMatch()){   
  325.             this.filter.matches[0].select();   
  326.         }   
  327.         this.el.focus();   
  328.         this.expand();   
  329.         this.restrict();   
  330.     },   
  331.     onTreeClick: function(node){   
  332.         if(this.fireEvent('beforeselect'this, node) !== false){   
  333.             if (this.ignoreFolder && !node.leaf)    
  334.                 return;   
  335.             this.setValue(node);   
  336.             this.collapse();   
  337.             this.fireEvent('select'this, node);   
  338.         }   
  339.     },   
  340.     //@public 确保树已加载所需结点再调用此方法,如果传的是id但是找不到结点value域将置空   
  341.     setValue: function(node){//根据TreeNode的id或者TreeNode对象设置值,显示TreeNode的text属性   
  342.         if (!this.tree.rendered||this.isLoading) {   
  343.             return null;   
  344.         }   
  345.         if(typeof node == 'string'){   
  346.             node = this.tree.getNodeById(node);   
  347.         }          
  348.         var text;   
  349.         if(!node||(this.ignoreFolder && !node.leaf)){   
  350.             text = this.valueNotFoundText;   
  351.         } else {   
  352.             text = node.text;   
  353.         }   
  354.         QM.ui.TreeCombo.superclass.setValue.call(this, text);   
  355.         if(this.hiddenField){   
  356.             this.hiddenField.value = node.id;   
  357.         }   
  358.         this.lastSelectionText = text;   
  359.         this.value = node?node.id:'';      
  360.         return this;   
  361.     },   
  362.     getValue : function(){   
  363.        return Ext.isDefined(this.value) ? this.value : '';   
  364.     },   
  365.     clearValue : function(){   
  366.         if(this.hiddenField){   
  367.             this.hiddenField.value = '';   
  368.         }   
  369.         this.setRawValue('');   
  370.         this.lastSelectionText = '';   
  371.         this.value = '';   
  372.     },   
  373.     // private   
  374.     validateBlur : function(){   
  375.         return !this.list || !this.list.isVisible();   
  376.     },   
  377.     //检查输入值是不是列表里有的,有的话设置对应value   
  378.     beforeBlur : function(){   
  379.         var val = this.getRawValue();   
  380.         node = this.tree.root.findChild('text',val,true);         
  381.         if(!node){   
  382.             if(val.length > 0 && val != this.emptyText){   
  383.                 this.el.dom.value = Ext.value(this.lastSelectionText, '');//值空或是上一次输入   
  384.             }else{//输入域清空,所有值清空   
  385.                 this.clearValue();   
  386.             }   
  387.         }else if(node){   
  388.             this.setValue(node);   
  389.         }   
  390.     },   
  391.     //根据id值或node对象选择到相应node并显示出来,scrollIntoView是否需要滚动   
  392.     selectByNode : function(node, scrollIntoView){   
  393.         if(!Ext.isEmpty(node, true)){   
  394.             if (typeof node == 'string') {   
  395.                 node = this.tree.getNodeById(node);   
  396.             }   
  397.             if(node){   
  398.                 this.tree.collapseAll();   
  399.                 this.tree.expandPath(node.getPath());//只展开选中结点   
  400.                 node.select();   
  401.                 if(scrollIntoView===true)   
  402.                     node.ensureVisible();   
  403.                 return true;   
  404.             }   
  405.         }   
  406.         return false;   
  407.     },   
  408.     // private   
  409.     postBlur  : function(){   
  410.         QM.ui.TreeCombo.superclass.postBlur.call(this);   
  411.         this.collapse();   
  412.         this.inKeyMode = false;   
  413.     },   
  414.     onDestroy: function(){   
  415.         if (this.dqTask){   
  416.             this.dqTask.cancel();   
  417.             this.dqTask = null;   
  418.         }          
  419.         Ext.destroy(this.tree,this.list,this.filter);   
  420.         QM.ui.TreeCombo.superclass.onDestroy.call(this);   
  421.     }   
  422. });   
  423.   
  424. Ext.reg('treecombo', QM.ui.TreeCombo);  
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值