WPF TreeView: почему виртуализация нарушает BringIntoView()?

Когда вы установите VirtualizingStackPanel.IsVirtualizing="True" на TreeView в WPF:

<Window x:Class="WpfApplication2.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="MainWindow" Height="350" Width="525">
    <Grid>
        <TreeView Name="MainTree" VirtualizingStackPanel.IsVirtualizing="True" VirtualizingStackPanel.VirtualizationMode="Recycling"/>
    </Grid>
</Window>

Следующий код позади с вызовом TreeViewItem.BringIntoView() не работает:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;

namespace WpfApplication2
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
            TreeViewItem lastTvi2 = null;
            for (int i = 0; i < 100; i++)
            {
                TreeViewItem tvi = new TreeViewItem();
                tvi.Header = "Hello";
                MainTree.Items.Add(tvi);
                for (int j = 0; j < 100; j++)
                {
                    TreeViewItem tvi2 = new TreeViewItem();
                    tvi2.Header = "World";
                    tvi.Items.Add(tvi2);
                    lastTvi2 = tvi2;
                }
            }
            lastTvi2.BringIntoView();
        }
    }
}

Тем не менее, когда вы удалите линию VirtualizingStackPanel.IsVirtualizing="True", это принесет lastTvi2 объект (последний элемент в дереве) в представлении. Почему это ломает TreeViewItem.BringIntoView() метод? Это нормальное поведение или ошибка.NET?

1 ответ

Я несколько раз сталкивался с этой проблемой и нашел решение, которое работает для меня, используя стандартный древовидный список и стандартный древовидный элемент.

<UserControl
...
<TreeView
    VirtualizingPanel.IsVirtualizing="True"
    VirtualizingPanel.VirtualizationMode="Recycling"/>
...
</UserControl>

public static class TreeViewHelper
{
    private static T FindVisualChild<T>(System.Windows.Media.Visual visual) where T : System.Windows.Media.Visual
    {
        for (int i = 0; i < VisualTreeHelper.GetChildrenCount(visual); i++)
        {
            System.Windows.Media.Visual child = (System.Windows.Media.Visual)VisualTreeHelper.GetChild(visual, i);
            if (child != null)
            {
                T correctlyTyped = child as T;
                if (correctlyTyped != null)
                {
                    return correctlyTyped;
                }

                T descendent = FindVisualChild<T>(child);
                if (descendent != null)
                {
                    return descendent;
                }
            }
        }

        return null;
    }

    public static void BringIntoView(TreeViewItem item)
    {
        ItemsControl parent = item.Parent as ItemsControl;

        if (parent != null)
        {
            System.Windows.Controls.VirtualizingStackPanel itemHost = FindVisualChild<System.Windows.Controls.VirtualizingStackPanel>(parent);

            if (itemHost != null)
            {
                itemHost.BringIndexIntoViewPublic(parent.Items.IndexOf(item));
                item.Focus();
            }
        }
    }
}
Другие вопросы по тегам