溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

C# WPF 父控件通過(guò)使用可視化樹(shù)找到子控件的示例代碼

發(fā)布時(shí)間:2020-09-03 16:16:29 來(lái)源:腳本之家 閱讀:221 作者:hello黃先森 欄目:編程語(yǔ)言

在我們使用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,效果如下:

C# WPF 父控件通過(guò)使用可視化樹(shù)找到子控件的示例代碼

我們可以看到重寫數(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

C# WPF 父控件通過(guò)使用可視化樹(shù)找到子控件的示例代碼

C# WPF 父控件通過(guò)使用可視化樹(shù)找到子控件的示例代碼

我們發(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:

C# WPF 父控件通過(guò)使用可視化樹(shù)找到子控件的示例代碼

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)站的支持!

向AI問(wèn)一下細(xì)節(jié)

免責(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)容。

AI