您好,登錄后才能下訂單哦!
在我們使用WPF設(shè)計(jì)前臺(tái)界面時(shí),經(jīng)常會(huì)重寫數(shù)據(jù)模板,或者把控件放到數(shù)據(jù)模板里。但是一旦將控件放到數(shù)據(jù)模板中,在后臺(tái)就沒(méi)有辦法通過(guò)控件的名字來(lái)獲取它了,更沒(méi)辦法對(duì)它進(jìn)行操作(例如,隱藏,改變控件的某個(gè)值)。
如果你是比我還白的小白,對(duì)我剛剛陳述的東西不清楚,接下來(lái)我簡(jiǎn)單說(shuō)一下什么是把控件放在數(shù)據(jù)模板中,怎么樣的情況沒(méi)法后臺(tái)通過(guò)名字來(lái)獲取控件,如果讀者對(duì)于數(shù)據(jù)模板這些事兒已經(jīng)清楚了,或者只關(guān)心如何使用可視化樹(shù)可以將這部分跳過(guò)哈。
先上代碼介紹一下什么是數(shù)據(jù)模板以WPF中ListBox控件為例:
<ListBox Name="ListBox_1" HorizontalAlignment="Left" Height="299" Margin="10,10,0,0" VerticalAlignment="Top" Width="497" MouseDoubleClick="ListBox_1_OnMouseDoubleClick"> <ListBox.ItemTemplate> <DataTemplate> <Button Name="Button_1" Content="666"></Button> </DataTemplate> </ListBox.ItemTemplate> </ListBox>
我在后臺(tái)設(shè)置了顯示了8行item,效果如下:
我們可以看到重寫數(shù)據(jù)模板實(shí)現(xiàn)的效果是在ListBox的每一項(xiàng)Item都是一個(gè)Button,這里介紹的只是一些簡(jiǎn)單應(yīng)用例子,重寫模板是很強(qiáng)大的。因?yàn)槿绻玫娇梢暬瘶?shù)多半是因?yàn)槭褂昧藬?shù)據(jù)模板在后臺(tái)用名字無(wú)法找到相應(yīng)控件了,所以在此簡(jiǎn)單介紹一下,方便理解。
接下來(lái)我們?cè)诤笈_(tái)嘗試通過(guò)控件的名字來(lái)找到我們的ListBox和Button
我們發(fā)現(xiàn)通過(guò)控件的名字可以找到ListBox但是通過(guò)button的名字卻無(wú)法找到button,這就是數(shù)據(jù)模板搞的鬼。
但是沒(méi)有關(guān)系,我們可以通過(guò)可視化樹(shù)從ListBox里找到它的子控件我們想要的這個(gè)Button。
重點(diǎn)來(lái)了,先上代碼,可視化樹(shù)通過(guò)父控件找到它的子控件:
List<T> FindVisualChild<T>(DependencyObject obj) where T : DependencyObject { try { List<T> list = new List<T>(); for (int i = 0; i < VisualTreeHelper.GetChildrenCount(obj); i++) { DependencyObject child = VisualTreeHelper.GetChild(obj, i); if (child is T) { list.Add((T)child); List<T> childOfChildren = FindVisualChild<T>(child); if (childOfChildren != null) { list.AddRange(childOfChildren); } } else { List<T> childOfChildren = FindVisualChild<T>(child); if (childOfChildren != null) { list.AddRange(childOfChildren); } } } return list; } catch (Exception) { //MessageBox.Show(ee.Message); return null; } }
先將上面的方法復(fù)制到你的項(xiàng)目當(dāng)中,此時(shí)對(duì)于可視化樹(shù)的應(yīng)用已經(jīng)完成一半了。
接下來(lái)上代碼,通過(guò)可視化樹(shù)雙擊ListBox的ltem把對(duì)應(yīng)的button的Content值從666改成777:
private void ListBox_1_OnMouseDoubleClick(object sender, MouseButtonEventArgs e) { ListBoxItem myListBoxItem = (ListBoxItem)ListBox_1.ItemContainerGenerator.ContainerFromItem(ListBox_1.SelectedItem); List<Button> btnList = FindVisualChild<Button>(myListBoxItem); foreach (var item in btnList) { item.Content = "777"; } }
效果就是雙擊哪個(gè)item哪個(gè)item中的button從666變成了777。
我們通過(guò)父控件找到了里面的子控件button,我們便可以對(duì)它進(jìn)行任何操作(和用名字找到是一樣的)。
以上關(guān)于可視化樹(shù)的代碼可以應(yīng)用于ListBox,DataGrid,ListView,TreeView,對(duì)于“.ItemContainerGenerator.ContainerFromItem”這段代碼的含義我暫時(shí)不是很理解,歡迎指教和交流。
通過(guò)以上的例子相信讀者已經(jīng)可以使用可視化樹(shù)找到相應(yīng)的控件了,但在我的開(kāi)發(fā)過(guò)程中曾遇到過(guò)一些問(wèn)題,和對(duì)于使用可視化樹(shù)的一點(diǎn)小建議。
1.如果你在使用可視化樹(shù)執(zhí)行“ListBoxItem myListBoxItem = (ListBoxItem)ListBox_1.ItemContainerGenerator.ContainerFromItem(ListBox_1.SelectedItem);”這句返回值是空(實(shí)際上不是空),可能是因?yàn)榻缑鏇](méi)有初始化完畢,我的理解是,在前臺(tái)這個(gè)控件還沒(méi)生成完畢,或者是你修改了值但前臺(tái)還沒(méi)有修改,可以加上這句:
控件名.UpdateLayout();
之后在使用可視化樹(shù),這一條的說(shuō)法和形容可能有點(diǎn)不嚴(yán)謹(jǐn),歡迎指正交流。
2.可視化樹(shù)使用的是遞歸的方法,所以它的效率不是很高,如果在程序中大量使用可視化樹(shù),會(huì)使得程序變慢的。
3.調(diào)用可視化樹(shù)返回的列表如果沒(méi)有找到相應(yīng)的控件或是異常便會(huì)返回空值,所以建議在你遍歷可視化樹(shù)返回的列表時(shí),請(qǐng)先判斷否非為空。
補(bǔ)充:WPF查找子控件和父控件方法
一、查找某種類型的子控件,并返回一個(gè)List集合 public List<T> GetChildObjects<T>(DependencyObject obj, Type typename) where T : FrameworkElement { DependencyObject child = null; List<T> childList = new List<T>(); for (int i = 0; i <= VisualTreeHelper.GetChildrenCount(obj) - 1; i++) { child = VisualTreeHelper.GetChild(obj, i); if (child is T && (((T)child).GetType() == typename)) { childList.Add((T)child); } childList.AddRange(GetChildObjects<T>(child,typename)); } return childList; } 調(diào)用: List<Button> listButtons = GetChildObjects<Button>(parentPanel, typeof(Button)); //parentPanel就是xaml里定義的控件的x:name 二、通過(guò)名稱查找子控件,并返回一個(gè)List集合 public List<T> GetChildObjects<T>(DependencyObject obj, string name) where T : FrameworkElement { DependencyObject child = null; List<T> childList = new List<T>(); for (int i = 0; i <= VisualTreeHelper.GetChildrenCount(obj) - 1; i++) { child = VisualTreeHelper.GetChild(obj, i); if (child is T && (((T)child).GetType() == name |string.IsNullOrEmpty(name))) { childList.Add((T)child); } childList.AddRange(GetChildObjects<T>(child,name)); } return childList; } 調(diào)用: List<Button> listButtons = GetChildObjects<Button>(parentPanel, "button1"); 三、通過(guò)名稱查找某子控件: public T GetChildObject<T>(DependencyObject obj, string name) where T : FrameworkElement { DependencyObject child = null; T grandChild = null; for (int i = 0; i <= VisualTreeHelper.GetChildrenCount(obj) - 1; i++) { child = VisualTreeHelper.GetChild(obj, i); if (child is T && (((T)child).Name == name | string.IsNullOrEmpty(name))) { return (T)child; } else { grandChild = GetChildObject<T>(child, name); if (grandChild != null) return grandChild; } } returnnull; } 調(diào)用: StackPanel sp = GetChildObject<StackPanel>(this.LayoutRoot, "spDemoPanel"); 四、通過(guò)名稱查找父控件 public T GetParentObject<T>(DependencyObject obj, string name) where T : FrameworkElement { DependencyObject parent = VisualTreeHelper.GetParent(obj); while (parent != null) { if (parent is T && (((T)parent).Name == name | string.IsNullOrEmpty(name))) { return (T)parent; } parent = VisualTreeHelper.GetParent(parent); } returnnull; } 調(diào)用: Grid layoutGrid = VTHelper.GetParentObject<Grid>(this.spDemoPanel, "LayoutRoot");
總結(jié)
以上所述是小編給大家介紹的C# WPF 父控件通過(guò)使用可視化樹(shù)找到子控件,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)億速云網(wǎng)站的支持!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。