對Treelist 的一些功能的實現
1.數據綁定最基本的兩個屬性:KeyFieldName和ParentFieldName。(這兩個屬性一設置就基本上可以實現分級了)
可以通過代碼的編寫實現,也可以直接在屬性里面直接實現。
這種數據庫設計是比較常見的,一般數據滿足樹形關系就可以這樣設計。綁定數據時,只需指定DataSource為對應 的DataTable,指定KeyFieldName為表主鍵字段,ParentFieldName為表指向主鍵的外鍵字段名。
private void BindData() { this.tlOffice.DataSource = dtOffice; tlOffice.KeyFieldName = "OfficeID"; //tlOffice.DataMember = "OfficeName"; tlOffice.Columns["OfficeName"].Caption = "局名稱"; tlOffice.ParentFieldName = "ParentOfficeID"; }
2 基本功能的實現
① . 選擇某一節點時,該節點的子節點全部選擇 取消某一節點時,該節點的子節點全部取消選擇
哪個節點引起行為的?節點是選中還是取消選中?由此確定方法的兩個參數:TreeListNode和CheckState。遍歷該節點及其子孫,並將其選中狀態設置為該節點的狀態即可。
/// <summary> /// 選擇某一節點時,該節點的子節點全部選擇 取消某一節點時,該節點的子節點全部取消選擇 /// </summary> /// <param name="node"></param> /// <param name="state"></param> private void SetCheckedChildNodes(TreeListNode node, CheckState check) { for (int i = 0; i < node.Nodes.Count; i++) { node.Nodes[i].CheckState = check; SetCheckedChildNodes(node.Nodes[i], check); } }
② .某節點的子節點全部選擇時,該節點選擇;某節點的子節點未全部選擇時,該節點不選擇
/// <summary> /// 某節點的子節點全部選擇時,該節點選擇 某節點的子節點未全部選擇時,該節點不選擇 /// </summary> /// <param name="node"></param> /// <param name="check"></param> private void SetCheckedParentNodes(TreeListNode node, CheckState check) { if (node.ParentNode != null) { CheckState parentCheckState = node.ParentNode.CheckState; CheckState nodeCheckState; for (int i = 0; i < node.ParentNode.Nodes.Count; i++) { nodeCheckState = (CheckState)node.ParentNode.Nodes[i].CheckState; if (!check.Equals(nodeCheckState))//只要任意一個與其選中狀態不一樣即父節點狀態不全選 { parentCheckState = CheckState.Unchecked; break; } parentCheckState = check;//否則(該節點的兄弟節點選中狀態都相同),則父節點選中狀態為該節點的選中狀態 } node.ParentNode.CheckState = parentCheckState; SetCheckedParentNodes(node.ParentNode, check);//遍歷上級節點 } }
上兩步寫好了,別忘了上面的兩個方法在TreeList_AfterCheckNode里面觸發:
private void tlOffice_AfterCheckNode(object sender, DevExpress.XtraTreeList.NodeEventArgs e) { SetCheckedChildNodes(e.Node, e.Node.CheckState); SetCheckedParentNodes(e.Node, e.Node.CheckState); }
3.獲取選中的復選框數據列表
private List<int> lstCheckedOfficeID = new List<int>();//選擇局ID集合 /// <summary> /// 獲取選擇狀態的數據主鍵ID集合 /// </summary> /// <param name="parentNode">父級節點</param> private void GetCheckedOfficeID(TreeListNode parentNode) { if (parentNode.Nodes.Count == 0) { return;//遞歸終止 } foreach (TreeListNode node in parentNode.Nodes) { if (node.CheckState == CheckState.Checked) { DataRowView drv = tlOffice.GetDataRecordByNode(node) as DataRowView;
//關鍵代碼,就是不知道是這樣獲取數據而糾結了很久(鬼知道可以轉換為DataRowView啊) if (drv != null) { int OfficeID = (int)drv["OfficeID"]; lstCheckedOfficeID.Add(OfficeID); } } GetCheckedOfficeID(node); } }
下面測試獲取主鍵列表:
private void btnCheck_Click(object sender, EventArgs e) { this.lstCheckedOfficeID.Clear(); if (tlOffice.Nodes.Count > 0) { foreach (TreeListNode root in tlOffice.Nodes) { GetCheckedOfficeID(root); } } string idStr = string.Empty; foreach (int id in lstCheckedOfficeID) { idStr += id + " "; } MessageBox.Show(idStr); }