Leer en inglés

Compartir a través de


Cómo: Buscar elementos generados por un objeto DataTemplate

En este ejemplo se muestra cómo buscar elementos generados por DataTemplate.

Ejemplo

En este ejemplo, hay un elemento ListBox que está enlazado a algunos datos XML:

<ListBox Name="myListBox" ItemTemplate="{StaticResource myDataTemplate}"
         IsSynchronizedWithCurrentItem="True">
  <ListBox.ItemsSource>
    <Binding Source="{StaticResource InventoryData}" XPath="Books/Book"/>
  </ListBox.ItemsSource>
</ListBox>

ListBox usa el elemento DataTemplate siguiente:

<DataTemplate x:Key="myDataTemplate">
  <TextBlock Name="textBlock" FontSize="14" Foreground="Blue">
    <TextBlock.Text>
      <Binding XPath="Title"/>
    </TextBlock.Text>
  </TextBlock>
</DataTemplate>

Si desea recuperar el elemento TextBlock generado por DataTemplate de un determinado ListBoxItem, debe obtener ListBoxItem, buscar ContentPresenter dentro de ese ListBoxItem y, a continuación, llamar a FindName en el objeto DataTemplate establecido en ContentPresenter. En el ejemplo siguiente se muestra cómo se realizan esos pasos. Para fines de demostración, en este ejemplo se crea un cuadro de mensaje que muestra el contenido de texto del bloque de texto generado por DataTemplate.

// Getting the currently selected ListBoxItem
// Note that the ListBox must have
// IsSynchronizedWithCurrentItem set to True for this to work
ListBoxItem myListBoxItem =
    (ListBoxItem)(myListBox.ItemContainerGenerator.ContainerFromItem(myListBox.Items.CurrentItem));

// Getting the ContentPresenter of myListBoxItem
ContentPresenter myContentPresenter = FindVisualChild<ContentPresenter>(myListBoxItem);

// Finding textBlock from the DataTemplate that is set on that ContentPresenter
DataTemplate myDataTemplate = myContentPresenter.ContentTemplate;
TextBlock myTextBlock = (TextBlock)myDataTemplate.FindName("textBlock", myContentPresenter);

// Do something to the DataTemplate-generated TextBlock
MessageBox.Show("The text of the TextBlock of the selected list item: "
    + myTextBlock.Text);

A continuación se muestra la implementación de FindVisualChild, que usa los métodos VisualTreeHelper:

private childItem FindVisualChild<childItem>(DependencyObject obj)
    where childItem : DependencyObject
{
    for (int i = 0; i < VisualTreeHelper.GetChildrenCount(obj); i++)
    {
        DependencyObject child = VisualTreeHelper.GetChild(obj, i);
        if (child != null && child is childItem)
        {
            return (childItem)child;
        }
        else
        {
            childItem childOfChild = FindVisualChild<childItem>(child);
            if (childOfChild != null)
                return childOfChild;
        }
    }
    return null;
}

Vea también