(一). 说明
用Tree显示菜单及物品列表(从服务端获取数据)比较方便, 当前显示Tree 主要有两种方式:
1. 在Tree初始化时将数据全部一次性从服务端获取, 获取完数据后页面展开或收缩时就不再需要获取数据,
这样, 获取完数据使用时效率比较高, 但当树节点很多时, 在每次初始化时会有较大的延迟.
2. 初始化时只加载展开的节点, 当用户需要查看某个节点下的数据时, 再去取数据, 这样, 初始化时延迟会相
对减少, 但每次单击节点时要获取数据, 页面每次都要刷新, 所以也会产生延迟.
此事例用Ajax实现第二种方式, 每次只动态加载要展开的节点数据(闭合节点不展开时,则不获取其子节点的
数据), 另外加载节点时页面不会刷新.
(二). 运行示例图
(三). AjaxPro.NET简介
首先对AjaxPro.NET作一下介绍, AjaxPro.NET是一个优秀的Ajax框架, 在实际应用中只要添加其DLL
引用并进行简单的配置, 即可以非常方便的在客户端直接调用服务端方法, 来获取Tree节点.
(四).使用AjaxPro.NET预配置
1. 添加 AjaxPro.dll 文件的引用(示例代码中已经包含,直接COPY过来使用即可).
2. 在Web.config文件中添加以下配置,
1 < httpHandlers > 2 < add verb = " POST,GET " path = " ajaxpro/*.ashx " type = " AjaxPro.AjaxHandlerFactory, AjaxPro " /> 3 </ httpHandlers > 3. 在要使用AjaxPro.NET框架的页面 *.aspx.cs 的 Page_Load事件中加如下代码: AjaxPro.Utility.RegisterTypeForAjax( typeof (_Default)); 4. 经过以上三步骤后, 只要在后台服务端的方法前面增加属性[AjaxMethod]后: 1 [AjaxMethod()] // or [AjaxPro.AjaxMethod] 2 public ArrayList GetSearchItems( string strQuery ) 3 { 4 // 生成数据源 5 ArrayList items = new ArrayList(); 6 items.Add( " King " ); 7 items.Add( " Rose " ); 8 return items ; 9 } 10 就可以在客户端直接使用服务端方法, 非常方便, 客户端调用后台代码如下: var returnValue = 后台代码类名.GetSearchItems(参数);(五). 代码
1. 页面 Tree.aspx 代码:
1 <% @ Page Language = " C# " AutoEventWireup = " true " CodeFile = " Tree.aspx.cs " Inherits = " _Default " %> 2 3 <! DOCTYPE html PUBLIC " -//W3C//DTD XHTML 1.0 Transitional//EN " " http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd " > 4 5 < html xmlns = " http://www.w3.org/1999/xhtml " > 6 < head runat = " server " > 7 < title > Ajax Efficient Tree </ title > 8 < link type = " text/css " href = " css/tree.css " rel = " stylesheet " > 9 </ head > 10 < body > 11 < form id = " form1 " runat = " server " > 12 < div > 13 < asp:Panel ID = " Panel1 " runat = " server " Height = " 424px " Width = " 251px " > 14 < div id = " CategoryTree " class = " TreeMenu " ></ div > 15 </ asp:Panel > 16 < script language = " jscript " > 17 var tree = document.getElementById( " CategoryTree " ); 18 var root = document.createElement( " li " ); 19 root.id = " li_0 " ; 20 tree.appendChild( root ); 21 ExpandSubCategory( 0 ); 22 function ExpandSubCategory( categoryID ) 23 { 24 var liFather = document.getElementById( " li_ " + categoryID ); 25 if ( liFather.getElementsByTagName( " li " ).length > 0 ) 26 { 27 ChangeStatus( categoryID ); 28 return ; 29 } 30 liFather.className = " Opened " ; 31 SwitchNode( categoryID, true ); 32 33 // 仅获取当前节点的子Nodes 34 _Default.GetSubCategory( categoryID, GetSubCategory_callback ); 35 } 36 function SwitchNode( CategoryID, show ) 37 { 38 var li_father = document.getElementById( " li_ " + CategoryID); 39 if ( show ) 40 { 41 var ul = document.createElement( " ul " ); 42 ul.id = " ul_note_ " + CategoryID; 43 44 var note = document.createElement( " li " ); 45 note.className = " Child " ; 46 47 var img = document.createElement( " img " ); 48 img.className = " s " ; 49 img.src = " css/s.gif " ; 50 51 var a = document.createElement( " a " ); 52 a.href = " javascript:void(0); " ; 53 a.innerHTML = " Please waiting " ; 54 55 note.appendChild(img); 56 note.appendChild(a); 57 ul.appendChild(note); 58 li_father.appendChild(ul); 59 } 60 else 61 { 62 var ul = document.getElementById( " ul_note_ " + CategoryID ); 63 if ( ul ) 64 { 65 li_father.removeChild(ul); 66 } 67 } 68 } 69 function GetSubCategory_callback( response ) 70 { 71 var dt = response.value.Tables[ 0 ]; 72 if ( dt.Rows.length > 0 ) 73 { 74 var iCategoryID = dt.Rows[ 0 ].FatherID; 75 } 76 var li_father = document.getElementById( " li_ " + iCategoryID ); 77 var ul = document.createElement( " ul " ); 78 for ( var i = 0 ; i < dt.Rows.length; i ++ ) 79 { 80 if ( dt.Rows[i].IsChild == 1 ) 81 { 82 var li = document.createElement( " li " ); 83 li.className = " Child " ; 84 li.id = " li_ " + dt.Rows[i].CategoryID; 85 var img = document.createElement( " img " ); 86 img.id = dt.Rows[i].CategoryID; 87 img.className = " s " ; 88 img.src = " css/s.gif " ; 89 var a = document.createElement( " a " ); 90 a.href = " javascript:OpenDocument(' " + dt.Rows[i].CategoryID + " '); " ; 91 a.innerHTML = dt.Rows[i].CategoryName; 92 } 93 else 94 { 95 var li = document.createElement( " li " ); 96 li.className = " Closed " ; 97 li.id = " li_ " + dt.Rows[i].CategoryID; 98 var img = document.createElement( " img " ); 99 img.id = dt.Rows[i].CategoryID; 100 img.className = " s " ; 101 img.src = " css/s.gif " ; 102 img.onclick = function(){ ExpandSubCategory( this .id ); }; 103 img.alt = " Expand/collapse " ; 104 var a = document.createElement( " a " ); 105 a.href = " javascript:ExpandSubCategory(' " + dt.Rows[i].CategoryID + " '); " ; 106 a.innerHTML = dt.Rows[i].CategoryName; 107 } 108 li.appendChild(img); 109 li.appendChild(a); 110 ul.appendChild(li); 111 } 112 li_father.appendChild(ul); 113 SwitchNode( iCategoryID, false ); 114 } 115 116 // 单击叶节点时, 异步从服务端获取单个节点的数据. 117 function OpenDocument( CategoryID ) 118 { 119 _Default.GetNameByCategoryID( CategoryID, GetNameByCategoryID_callback ); 120 } 121 122 function GetNameByCategoryID_callback( response ) 123 { 124 alert( response.value ); 125 } 126 127 function ChangeStatus( CategoryID ) 128 { 129 var li_father = document.getElementById( " li_ " + CategoryID ); 130 if ( li_father.className == " Closed " ) 131 { 132 li_father.className = " Opened " ; 133 } 134 else 135 { 136 li_father.className = " Closed " ; 137 } 138 } 139 </ script > 140 </ div > 141 </ form > 142 </ body > 143 </ html >2. 页面后台文件 Tree.aspx.cs 代码:
1 using System; 2 using System.Data; 3 using System.Configuration; 4 using System.Web; 5 using System.Web.Security; 6 using System.Web.UI; 7 using System.Web.UI.WebControls; 8 using System.Web.UI.WebControls.WebParts; 9 using System.Web.UI.HtmlControls; 10 11 public partial class _Default : System.Web.UI.Page 12 { 13 // 此对象用于存放所有的节点数 14 public static DataSet dsAllNodes = new DataSet(); 15 16 protected void Page_Load( object sender, EventArgs e) 17 { 18 AjaxPro.Utility.RegisterTypeForAjax( typeof (_Default)); 19 CreateNodes(); 20 } 21 22 private DataTable CreateStructure() 23 { 24 DataTable dt = new DataTable(); 25 dt.Columns.Add( new DataColumn( " CategoryID " , typeof ( int ))); 26 dt.Columns.Add( new DataColumn( " CategoryName " , typeof ( string ))); 27 dt.Columns.Add( new DataColumn( " FatherID " , typeof ( string ))); 28 dt.Columns.Add( new DataColumn( " IsChild " , typeof ( bool ))); 29 return dt; 30 } 31 public void CreateNodes() 32 { 33 DataTable dt = this .CreateStructure(); 34 35 DataRow drNew = dt.NewRow(); 36 drNew[ " CategoryID " ] = 1 ; 37 drNew[ " CategoryName " ] = " 物品类别 " ; 38 drNew[ " FatherID " ] = 0 ; 39 dt.Rows.Add( drNew ); 40 41 drNew = dt.NewRow(); 42 drNew[ " CategoryID " ] = 2 ; 43 drNew[ " CategoryName " ] = " 水果 " ; 44 drNew[ " FatherID " ] = 1 ; 45 dt.Rows.Add( drNew ); 46 47 drNew = dt.NewRow(); 48 drNew[ " CategoryID " ] = 3 ; 49 drNew[ " CategoryName " ] = " 工具 " ; 50 drNew[ " FatherID " ] = 1 ; 51 dt.Rows.Add( drNew ); 52 53 drNew = dt.NewRow(); 54 drNew[ " CategoryID " ] = 4 ; 55 drNew[ " CategoryName " ] = " 萍果 " ; 56 drNew[ " FatherID " ] = 2 ; 57 dt.Rows.Add( drNew ); 58 59 drNew = dt.NewRow(); 60 drNew[ " CategoryID " ] = 5 ; 61 drNew[ " CategoryName " ] = " 香蕉 " ; 62 drNew[ " FatherID " ] = 2 ; 63 dt.Rows.Add( drNew ); 64 65 drNew = dt.NewRow(); 66 drNew[ " CategoryID " ] = 6 ; 67 drNew[ " CategoryName " ] = " 桔子 " ; 68 drNew[ " FatherID " ] = 2 ; 69 dt.Rows.Add( drNew ); 70 71 drNew = dt.NewRow(); 72 drNew[ " CategoryID " ] = 7 ; 73 drNew[ " CategoryName " ] = " 萝卜 " ; 74 drNew[ " FatherID " ] = 2 ; 75 dt.Rows.Add( drNew ); 76 77 drNew = dt.NewRow(); 78 drNew[ " CategoryID " ] = 8 ; 79 drNew[ " CategoryName " ] = " 钢笔 " ; 80 drNew[ " FatherID " ] = 3 ; 81 dt.Rows.Add( drNew ); 82 83 drNew = dt.NewRow(); 84 drNew[ " CategoryID " ] = 9 ; 85 drNew[ " CategoryName " ] = " 铅笔 " ; 86 drNew[ " FatherID " ] = 3 ; 87 dt.Rows.Add( drNew ); 88 89 drNew = dt.NewRow(); 90 drNew[ " CategoryID " ] = 10 ; 91 drNew[ " CategoryName " ] = " 尺子 " ; 92 drNew[ " FatherID " ] = 3 ; 93 dt.Rows.Add( drNew ); 94 95 drNew = dt.NewRow(); 96 drNew[ " CategoryID " ] = 11 ; 97 drNew[ " CategoryName " ] = " 橡皮 " ; 98 drNew[ " FatherID " ] = 3 ; 99 dt.Rows.Add( drNew ); 100 101 dsAllNodes.Tables.Add(dt); 102 } 103 104 [AjaxPro.AjaxMethod] 105 public DataSet GetSubCategory( int CategoryID) 106 { 107 DataSet ds = new DataSet(); 108 DataTable dt = this .CreateStructure(); 109 DataRow[] drSelect = dsAllNodes.Tables[ 0 ].Select( " FatherID= " + CategoryID.ToString()); 110 foreach (DataRow drTemp in drSelect) 111 { 112 DataRow dr = dt.NewRow(); 113 dr[ " CategoryID " ] = drTemp[ " CategoryID " ]; 114 dr[ " CategoryName " ] = drTemp[ " CategoryName " ]; 115 dr[ " FatherID " ] = drTemp[ " FatherID " ]; 116 dr[ " IsChild " ] = IsLeaf( int .Parse( drTemp[ " CategoryID " ].ToString() ) ); 117 dt.Rows.Add(dr); 118 } 119 ds.Tables.Add(dt); 120 return ds; 121 } 122 123 [AjaxPro.AjaxMethod] 124 public bool IsLeaf( int Category) 125 { 126 foreach (DataRow dr in dsAllNodes.Tables[ 0 ].Rows) 127 { 128 if (dr[ " FatherID " ] != null && int .Parse(dr[ " FatherID " ].ToString()) == Category) 129 { 130 return false ; 131 } 132 } 133 return true ; 134 } 135 136 [AjaxPro.AjaxMethod] 137 public string GetNameByCategoryID( string CategoryID ) 138 { 139 foreach ( DataRow dr in dsAllNodes.Tables[ 0 ].Rows ) 140 { 141 if ( dr[ " CategoryID " ].ToString() == CategoryID.ToString() ) 142 { 143 return dr[ " CategoryName " ].ToString(); 144 } 145 } 146 return "" ; 147 } 148 }(六). 示例代码下载:
http://www.cnitblog.com/Files/ChengKing/AjaxPro.net_EfficientTree.rar