问题描述
最近在项目中遇到一个Ext 异步树头痛的问题:树中的每个节点都有一个checkbox,当勾选父节点时,也需要将其所有的子节点都选中,但是问题是当前还没有加载这些子节点。知道可以通过对父节点加tree.on(checkedchange,function(node){node.expand(..递归.)});的形式逐个展开所有的子节点,但是怎样才能知道父节点下的所有子节点都被展开选中了呢? 因为要在这之后对该父节点和其所有子节点做一些操作。 tree.on('checkedchange',function(node,checked){ node.attributes.checked = checked; node.expand(true,true,function(){ node.eachChild(function(child){ child.getUI.checked = checked; child.attributes.checked = checked; child.toggle(); child.fireEvent('checkedchange',child,checked,tree); }); node.collapse(); }); },tree);这个操作可能是在用户选中了父节点的情况下马上就执行(从选中父节点到执行操作的时间很短,其所有子节点根本都还没有完全展开),这样就有可能会丢失一些还没有来得及展开完的子节点,所以需要一个判断所有节点都被异步加载完的事件......(父节点下所有子节点的节点个数先前是未知的)....希望大家各抒己见,帮帮解决下,谢谢!!本人才30分的积分,给出5分吧。 问题补充:谢谢牛人的回答。。这样递归的展开所有的子节点,但是还是不能具体的返回标志说已经展开完毕所有节点,我想利用这个标志判断是否已经加载完毕,能否说的更详细点。
解决方案
……我主要给你的是思路,程序什么的都是随手敲的,未经验证所以经常有些小错误,出错的地方是这里么?if(len<=0){taskDone(); // <--- 这里? 应该修改为 taskDone.call(this);return;}// 递归展开for(var i = 0, len = cs.length; i < len; i++) {cs[i].deepExpand(anim, taskDone, this);}逻辑是没问题,每展开一个子节点,都会调用taskDone回调。taskDone中有判断当前完成了多少个节点,如果根据计数判断都完成的就调用最终回调。
解决方案二:
可能是没考虑叶子节点的情况加上一句吧:if(len<=0){taskDone();return;}// 递归展开for(var i = 0, len = cs.length; i < len; i++) {cs[i].deepExpand(anim, taskDone, this);}
解决方案三:
你试试就知道了,这里是个递归调用,for循环中调用的是自身:deepExpand
解决方案四:
给你扩展的deepExpand不是有个回调参数吗?如果全部展开完毕就会调用这个回调换句话说,回调被调用时,该节点的所有子节点一定被全部展开完成
解决方案五:
哦。。。以上代码有个小错误,第4行调用expand首参数应该是false:this.expand(false, anim, function(){
解决方案六:
扩展一下树节点,增加一个deepExpand接口Ext.override(Ext.tree.TreeNode, {deepExpand : function(anim, callback, scope){// 先展开本节点this.expand(true, anim, function(){// 然后展开子节点var cs = this.childNodes, expanded = 0,len = cs.length,taskDone = function(){// 每展开成功一个子节点,计数+1expanded++;// 如果所有子节点都展开,调用最终回调if(expanded >= len){this.runCallback(callback, scope || this, [this]);}};// 递归展开for(var i = 0, len = cs.length; i < len; i++) {cs[i].deepExpand(anim, taskDone, this);}}, this);}});