jQuery树形菜单TreeView高性能应用(5)
01. function nodeclick(e) { 02. var path = $(this).attr("tpath");//获取节点路径 03. var et = e.target || e.srcElement;//获取事件源 04. var item = getItem(path);//根据path获取节点的数据 05. //debugger; 06. if (et.tagName == "IMG") { 07. // +号需要展开,处理加减号 08. if ($(et).hasClass("bbit-tree-elbow-plus") || $(et).hasClass("bbit-tree-elbow-end-plus")) { 09. var ul = $(this).next(); //"bbit-tree-node-ct" 10. if (ul.hasClass("bbit-tree-node-ct")) { 11. ul.show(); 12. } 13. else { 14. var deep = path.split(".").length; 15. if (item.complete) { 16. item.ChildNodes != null && asnybuild(item.ChildNodes, deep, path, ul, item); 17. } 18. else { 19. $(this).addClass("bbit-tree-node-loading"); 20. asnyloadc(ul, item, function(data) { 21. item.complete = true; 22. item.ChildNodes = data; 23. asnybuild(data, deep, path, ul, item); 24. }); 25. } 26. } 27. if ($(et).hasClass("bbit-tree-elbow-plus")) { 28. $(et).swapClass("bbit-tree-elbow-plus", "bbit-tree-elbow-minus"); 29. } 30. else { 31. $(et).swapClass("bbit-tree-elbow-end-plus", "bbit-tree-elbow-end-minus"); 32. } 33. $(this).swapClass("bbit-tree-node-collapsed", "bbit-tree-node-expanded"); 34. } 35. else if ($(et).hasClass("bbit-tree-elbow-minus") || $(et).hasClass("bbit-tree-elbow-end-minus")) { //- 号需要收缩 36. $(this).next().hide(); 37. if ($(et).hasClass("bbit-tree-elbow-minus")) { 38. $(et).swapClass("bbit-tree-elbow-minus", "bbit-tree-elbow-plus"); 39. } 40. else { 41. $(et).swapClass("bbit-tree-elbow-end-minus", "bbit-tree-elbow-end-plus"); 42. } 43. $(this).swapClass("bbit-tree-node-expanded", "bbit-tree-node-collapsed"); 44. } 45. else if ($(et).hasClass("bbit-tree-node-cb")) // 点击了Checkbox 46. { 47. var s = item.checkstate != 1 ? 1 : 0; 48. var r = true; 49. if (dfop.oncheckboxclick) { //触发配置的函数 50. r = dfop.oncheckboxclick.call(et, item, s); 51. } 52. if (r != false) {//如果返回值不为false,即checkbxo变化有效 53. if (dfop.cascadecheck) {//允许触发级联 54. //遍历 55. cascade(check, item, s);//则向下关联 56. //上溯 57. bubble(check, item, s); //向上关联 58. } 59. else { 60. check(item, s, 1);//否则只管自己 61. } 62. } 63. } 64. } 65. else {//点击到了其他地方 66. if (dfop.citem) { //上一个当前节点 67. $("#" + id + "_" + dfop.citem.id).removeClass("bbit-tree-selected"); 68. } 69. dfop.citem = item;//这次的当前节点 70. $(this).addClass("bbit-tree-selected"); 71. if (dfop.onnodeclick) { 72. dfop.onnodeclick.call(this, item); 73. } 74. } 75.}展开节点,异步请求的部分代码应该不是很复杂就不细诉了,关键来讲一下级联
级联有两个问题要处理,第一个是遍历子节点,第二个是上溯到祖节点,因为我们的数据结构这两个操作都显得非常简单
02.function cascade(fn, item, args) { 03. if (fn(item, args, 1) != false) { 04. if (item.ChildNodes != null && item.ChildNodes.length > 0) { 05. var cs = item.ChildNodes; 06. for (var i = 0, len = cs.length; i < len; i++) { 07. cascade(fn, cs[i], args); 08. } 09. } 10. } 11.} 12.//冒泡的祖先 13.function bubble(fn, item, args) { 14. var p = item.parent; 15. while (p) { 16. if (fn(p, args, 0) === false) { 17. break; 18. } 19. p = p.parent; 20. } 21.}找到节点的同时都会触发check这个回调函数,来判断当前节点的状态,详细请看下面代码中的注释部分应该是比较清晰,描写了这个过程
02. var pstate = item.checkstate; //当前状态 03. if (type == 1) { 04. item.checkstate = state; //如果是遍历子节点,父是什么子就是什么 05. } 06. else {// 上溯 ,这个就复杂一些了 07. var cs = item.ChildNodes; //获取当前节点的所有子节点 08. var l = cs.length; 09. var ch = true; //是否不是中间状态 半选 10. for (var i = 0; i < l; i++) { 11. if ((state == 1 && cs[i].checkstate != 1) || state == 0 && cs[i].checkstate != 0) { 12. ch = false; 13. break;//他的子节点只要有一个没选中,那么他就是半选 14. } 15. } 16. if (ch) { 17. item.checkstate = state;//不是半选,则子节点是什么他就是什么 18. } 19. else { 20. item.checkstate = 2; //半选 21. } 22. } 23. //change show 如果节点已输出,而其前后状态不一样,则变化checkbxo的显示 24. if (item.render && pstate != item.checkstate) { 25. var et = $("#" + id + "_" + item.id + "_cb"); 26. if (et.length == 1) { 27. et.attr("src", dfop.cbiconpath + dfop.icons[item.checkstate]); 28. } 29. } 30.}至此我们树的主体功能已经完全实现了。其他就是公开一些方法等,大家可详见代码,示例中公开了两个一个当前选中的所有节点,另外一个当前的节点。
大家可以通过以下网址查看文中的示例,selected拼错了,大家海涵! windows azure部署还是麻烦懒得修改了3500+节点一次加载,大家可以点击根节点的全选来看看速度
http://jscs.cloudapp.net/ControlsSample/BigTreeSample
异步加载,按需加载的情况也是非常常用的,使用的是SQL Azure服务器在美国ing,所以可能异步有点慢,本地数据源那是瞬间的
http://jscs.cloudapp.net/






