【问题标题】:How do I access a control inside a XAML DataTemplate?如何访问 XAML DataTemplate 中的控件?
【发布时间】:2013-04-28 19:53:40
【问题描述】:

我有这个翻转视图:

<FlipView x:Name="models_list" SelectionChanged="selectionChanged">
 <FlipView.ItemTemplate>
          <DataTemplate>
                <Grid x:Name="cv">
                        <Image x:Name="img1" Source = "{Binding ModelImage}" Stretch="Fill" Tag="{Binding ModelTag}"/>
                </Grid>
           </DataTemplate>
  </FlipView.ItemTemplate>

我想找到当前选定索引的 img1。在搜索它时,我在这里的一些帖子上找到了这种方法:

private DependencyObject FindChildControl<T>(DependencyObject control, string ctrlName)
    {
        int childNumber = VisualTreeHelper.GetChildrenCount(control);
        for (int i = 0; i < childNumber; i++)
        {
            DependencyObject child = VisualTreeHelper.GetChild(control, i);
            FrameworkElement fe = child as FrameworkElement;
            // Not a framework element or is null
            if (fe == null) return null;

            if (child is T && fe.Name== ctrlName)
            {
                // Found the control so return
                return child;
            }
            else
            {
                // Not found it - search children
                DependencyObject nextLevel = FindChildControl<T>(child, ctrlName);
                if (nextLevel != null)
                    return nextLevel;
            }
        }
        return null;
    }

它返回给我 Flipview 的第一个索引上的图像,但我需要当前选定索引上的图像。我尝试编辑此方法,但找不到所需的控件。谁能帮帮我?

【问题讨论】:

  • 这通常是错误的方法。你想用那个图像做什么?请注意,FlipView 将其子项虚拟化,因此 Image 甚至可能不存在。
  • @FilipSkakun 我想获取该图像的屏幕坐标。我可以用其他方式吗?
  • 你需要屏幕坐标做什么?
  • @DamirArh 我想在另一个图像上覆盖这个图像,这就是为什么我需要这个图像的确切位置......

标签: c# windows-8 windows-store-apps winrt-xaml flipview


【解决方案1】:

访问 DataTemplate 中的元素的简单解决方案是将 DataTemplate 的内容包装在 UserControl 中,您可以在其中访问 ItemsControl 项中的所有 UI 元素。我认为 FlipView 通常会虚拟化其项目,因此即使您绑定了 100 个项目 - 只有 2-3 个项目实际上可能在 UI 中具有当前表示(其中 1-2 个是隐藏的),因此您必须记住,当您想要替换任何内容,并且仅在将项目加载到控件中时才实际进行更改。

如果您确实需要识别表示 ItemsSource 中项目的项目容器 - 您可以检查 FlipView 的 ItemContainerGenerator 属性及其 ContainerFromItem() 方法。

要获取项目的坐标,您可以使用 WinRT XAML 工具包中的 GetBoundingRect() 扩展方法。

但是,总体而言,根据您的评论,最好的方法实际上可能完全不同。如果您将 FlipView 绑定到源 - 您通常可以通过更改绑定的源集合项的属性来控制显示或覆盖的图像。

【讨论】:

  • 我通过在一个列表中获取 FlipView 中的所有图像解决了这个问题,然后我遍历该列表以找到我需要的图像。是的,你是对的,我一次只能访问 FlipView 的 3 个项目,但由于我需要当前关注的项目,所以它始终存在于这三个项目中。谢谢你的帮助
  • @Tehreem 你能提供我面临同样情况的解决方案吗?
  • 您可以使用VisualTreeHelper 遍历可视化树,但我听说这可能不是访问元素的最高效方式,因此将您的DataTemplate 包装在UserControl 中可能会更好。不过不管用什么都好。
【解决方案2】:

您遇到的问题是DataTemplate 正在重复,而内容是由FlipView 生成的。 Name 未公开,因为它会与生成的前一个兄弟姐妹(或将要生成的下一个兄弟姐妹)冲突。

因此,要在DataTemplate 中获取命名元素,您必须首先获取生成的项目,然后在生成的项目中搜索您想要的元素。请记住,XAML 中的逻辑树是您按名称访问事物的方式。生成的项目不在逻辑树中。相反,它们位于可视树中(所有控件都位于可视树中)。这意味着您必须在可视树中搜索要引用的控件。 VisualTreeHelper 让您可以做到这一点。

现在,该怎么做?

我为此写了一篇文章,因为这是一个反复出现的问题:http://blog.jerrynixon.com/2012/09/how-to-access-named-control-inside-xaml.html 但解决方案的核心是递归方法,看起来像这样:

public void TestFirstName()
{
    foreach (var item in MyFlipView.Items)
    {
        var _Container = MyFlipView.ItemContainerGenerator
            .ContainerFromItem(item);
        var _Children = AllChildren(_Container);

        var _FirstName = _Children
            // only interested in TextBoxes
            .OfType<TextBox>()
            // only interested in FirstName
            .First(x => x.Name.Equals("FirstName"));

        // test & set color
        _FirstName.Background = 
            (string.IsNullOrWhiteSpace(_FirstName.Text))
            ? new SolidColorBrush(Colors.Red)
            : new SolidColorBrush(Colors.White);
    }
}

public List<Control> AllChildren(DependencyObject parent)
{
    var _List = new List<Control>();
    for (int i = 0; i < VisualTreeHelper.GetChildrenCount(parent); i++)
    {
        var _Child = VisualTreeHelper.GetChild(parent, i);
        if (_Child is Control)
            _List.Add(_Child as Control);
        _List.AddRange(AllChildren(_Child));
    }
    return _List;
}

这里的关键问题是像这样的方法获取所有子控件,然后在生成的子控件列表中搜索您想要的特定控件。有意义吗?

现在回答你的问题!

因为你特别想要当前选中的项目,你可以像这样简单地更新代码:

if (MyFlipView.SelectedItem == null)
    return;
var _Container = MyFlipView.ItemContainerGenerator
    .ContainerFromItem(MyFlipView.SelectedItem);
// then the same as above...

【讨论】:

  • Jerry - 这似乎只有在你有一个 SelectedItem 时才有效。您将如何通过在模板项边界之外单击按钮来定位元素?见stackoverflow.com/questions/23645331/…
  • @JerryNixon-MSFT 我尝试在 windows phone 8.1 应用程序中使用您的代码,但它找不到对 VisualTreeHelperClass 的引用。但是当我在对象浏览器中搜索它时,我可以在 System.Windows.Media 和 Windows.UI.Xaml.Media 中找到它。 OfType&lt;TextBox&gt;() 也给出错误 Systems.Generic.Connections.List 不包含 OfType() 的定义。 structure of my FlipView
  • 使用此代码(删除 ItemContainerGenerator,因为它现在已过时),我在 AllChildren 的 GetChildrenCount 处收到“参数不正确”。代码几乎相同,但我正在为 Windows 10 开发,而不是 8.1
【解决方案3】:

如果您只想要项目点击时的屏幕坐标...您可以在图像上注册一个点击处理程序,然后使用 senderimg.transform tovisual(null) 这为您提供了一个通用转换,您可以从中获取当前点坐标。

【讨论】:

    【解决方案4】:

    也许更通用的方法可能是这样的:

    private List<Control> AllChildren(DependencyObject parent)
    {
        var _List = new List<Control>();
        for (int i = 0; i < VisualTreeHelper.GetChildrenCount(parent); i++)
        {
           var _Child = VisualTreeHelper.GetChild(parent, i);
           if (_Child is Control)
           {
            _List.Add(_Child as Control);
           }
          _List.AddRange(AllChildren(_Child));
        }
        return _List;
     } 
    
    
    private T FindControl<T> (DependencyObject parentContainer, string controlName)
    {            
            var childControls = AllChildren(parentContainer);
            var control = childControls.OfType<Control>().Where(x => x.Name.Equals(controlName)).Cast<T>().First();           
            return control;
    }
    

    您可以像这样调用 FindControl:

    var parentContainer = this.flipView.ItemContainerGenerator.ContainerFromItem(this.flipView.SelectedItem);
    var myImage = FindControl<Image>(parentContainer, "img1");
    
    //suppose you want to change the visibility
    myImage.Visibility = Windows.UI.Xaml.Visibility.Collapsed;         
    

    【讨论】:

      【解决方案5】:

      我整天都在苦苦挣扎,似乎必须加载(渲染)控件才能从 DataTemplate 中获取它的子控件。这意味着您不能在加载、初始化的窗口上使用此代码或任何代码(用于相同目的)... 您可以在控件初始化后使用它,例如在 SelectedItem 上。

      我建议使用转换器并在转换器中定义请求的操作,而不是直接控制访问。

      【讨论】:

        【解决方案6】:

        因此,如果您通过绑定分配 Source,为什么您不对 rest 做同样的事情: &lt;FlipView SelectedIndex="{Binding SIndex}" SelectedItem="{Binding SItem}" SelectedValue="{Binding SValue}"/&gt;

        您必须首先为该属性准备位置。它可能是从INotifyPropertyChanged派生的类。

        聘请 MVVM 为您工作,大部分时间都在 XAML 中。当然一开始它似乎需要更多的工作,但是随着使用 MVVM 的更复杂的项目将是连贯和灵活的。

        【讨论】:

          猜你喜欢
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 2021-12-12
          • 2014-12-04
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          相关资源
          最近更新 更多