WPF TreeView:为什么虚拟化会破坏BringIntoView()函数?

3
当您在WPF中的TreeView上设置VirtualizingStackPanel.IsVirtualizing="True"时:
<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的错误?


由于该项尚未生成,因此它无法工作,这是正常行为(据我所知,如果有错误请见谅 :)) https://social.msdn.microsoft.com/forums/vstudio/en-US/8a7cc873-ac8c-4c6e-a6a0-b3bd8559c417/bringintoview-virtualizing-treeview - WPFUser
1
由于您的StackPanel是虚拟化的,因此视图之外的项目“不存在”。这就是虚拟化的目的,并通过UI节省了大量性能。问题在于,您无法滚动到不存在的项目(视图之外的项目)。有一个解决方案,使用ItemGeneratedContainer或类似的东西。我正在为您查找并将其发布,如果我能找到的话。 - Noel Widmer
我明白了,就WPF而言,虚拟化基本上意味着“仅呈现可见内容,如果请求其他内容,则按需呈现该其他内容”。这很像Objective-C列表视图的默认行为。 - Alexandru
1个回答

2

我曾经遇到过这个问题,通过使用标准的treeview和标准的treeviewitem,我找到了一个适合我的解决方案。

<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();
            }
        }
    }
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接