5

因此,在下面的示例代码中,我创建了一个 UserControl UserControldChild,它是主窗口 Window1.xaml 的子窗口。为什么该FindName()方法无法在下面的代码中找到“myButton”?

这一定与WPF XAML NameScopes有关,但我还没有找到关于 NameScope 如何工作的好的解释。有人可以启发我吗?

//(xml) Window1.xaml    
<Window x:Class="VisualTreeTestApplication.Window1"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:VisualTreeTestApp="clr-namespace:VisualTreeTestApplication"
    Title="Window1" Height="400" Width="400">
    <Grid>
        <VisualTreeTestApp:UserControlChild/>
    </Grid>
</Window>

//(c#) Window1.xaml.cs
namespace VisualTreeTestApplication
{
  /// <summary>
  /// Interaction logic for Window1.xaml
  /// </summary>
  public partial class Window1 : Window
  {
    public Window1()
    {
      InitializeComponent();
      Button btnTest = (Button)Application.Current.MainWindow.FindName("myButton");
      // btnTest is null!
    }
  }
}

下面的用户控件:

//(wpf) UserControlChild.xaml
<UserControl x:Class="VisualTreeTestApplication.UserControlChild"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    Height="300" Width="300">
    <Grid x:Name="myGrid">      
        <Button x:Name="myButton" Margin="20" >Button</Button>
    </Grid>
</UserControl>

//(c#) UserControlChild.xaml.cs (no changes)
namespace VisualTreeTestApplication
{
  /// <summary>
  /// Interaction logic for UserControlChild.xaml
  /// </summary>
  public partial class UserControlChild : UserControl
  {
    public UserControlChild()
    {
      InitializeComponent();
    }
  }
}

万一这没有得到正确回答,我找到了使用此处帖子中记录的FindName() 的替代方法。

4

1 回答 1

5

你是对的 - 这与 XAML 名称范围有关。

这在 XAML Namescopes page 的 Name related APIs 部分中记录(有些差)。

基本上,如果您有一个 FrameworkElement 或 FrameworkContentElement,它将定义自己的名称范围。如果对没有名称范围的类型调用 FindName(),WPF 会向上搜索树,直到找到确实定义了名称范围的元素,然后在该名称范围内进行搜索。

在您的情况下,它在 Window 的名称范围内进行搜索(它是一个 FrameworkContentElement,因此它定义了自己的范围)。它只搜索在该范围内定义的元素。

但是,在您的情况下,该按钮位于 UserControl 的名称范围内,因此 Window.FindName() 找不到它。没有自动搜索树较低级别的范围。

这是一件好事——你的“窗口”不应该知道或不想知道它正在使用的 UserControl 的内部细节。如果您需要 UserControl 中的属性,它们应该在 UserControl 级别公开 - 让控件管理自己的子控件。

于 2009-11-19T16:27:39.110 回答