scrolllintoview与嵌套的ScrollViewer

本文关键字:ScrollViewer 嵌套 scrolllintoview | 更新日期: 2023-09-27 18:03:50

在我的应用程序中,我有一个帆布包裹在一个ScrollViewer(帆布比屏幕尺寸大)。在这个画布上,我放置了其他控件。画布上的一个元素包含一个虚拟化的DataGrid(有2000多行……所以也涉及到滚动)。现在我有了一个函数,它根据row元素的一些值(自动触发)在DataGrid中选择一行。当选定行时,我调用

uxDataGrid.ScrollIntoView(uxDataGrid.SelectedItems[0]);

什么工作得很好。事实上,它正在向好的方向发展。我想要的是,DataGrid中的元素被选中,然后DataGrid应该滚动到正确的位置。但我的Canvas scrolviewer也会接收到这个请求并在那里滚动。

我已经尝试拦截ScrollChangedEvent并设置已处理的标志。但这并没有奏效。

问题:

1)如何限制ScrollIntoView调用对本地用户控件的影响。

2)如果这是不可能的,我怎么能自己做滚动?我需要计算滚动查看器的垂直偏移量,但由于它是虚拟化的,我不知道如何找到行高度?

有什么建议吗?

我添加了一个快速示例来演示主体设置。当点击其中一个按钮时,我希望只有DataGrid滚动,而不是包含画布的ScrollViewer。
<Window x:Class="ScrollTest.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">
<DockPanel>
    <StackPanel DockPanel.Dock="Bottom" Orientation="Horizontal">
        <Button Click="Button1_Click">Scroll to 1</Button>
        <Button Click="Button100_Click">Scroll to 100</Button>
        <Button Click="Button200_Click">Scroll to 200</Button>
    </StackPanel>
    <ScrollViewer>
        <Canvas Width="5000" Height="5000">
            <DataGrid Name="uxDataGrid" ItemsSource="{Binding TestItems}" Width="500" Height="500"></DataGrid>
        </Canvas>
    </ScrollViewer>
</DockPanel>

public class TestItem
{
    public TestItem(int id)
    {
        Property1 = id.ToString();
        Property2 = "B";
    }
    public string Property1 { get; set; }
    public string Property2 { get; set; }
}
/// <summary>
/// Interaktionslogik für MainWindow.xaml
/// </summary>
public partial class MainWindow : Window
{
    public IList<TestItem> TestItems { get; set; }
    public MainWindow()
    {
        TestItems = new List<TestItem>();
        for ( int i = 0; i < 300; i++ )
        {
            TestItems.Add(new TestItem(i));
        }
        InitializeComponent();
        DataContext = this;
    }

    private void Button1_Click( object sender, RoutedEventArgs e )
    {
        uxDataGrid.ScrollIntoView( TestItems[0]);
    }
    private void Button100_Click( object sender, RoutedEventArgs e )
    {
        uxDataGrid.ScrollIntoView( TestItems[99] );
    }
    private void Button200_Click( object sender, RoutedEventArgs e )
    {
        uxDataGrid.ScrollIntoView( TestItems[199] );
    }
}

scrolllintoview与嵌套的ScrollViewer

Ok…经过一番研究,我找到了正确的事件:它被称为RequestBringIntoViewEvent。当拦截那个时,它按预期工作:

public partial class MainWindow : Window
{
    public MainWindow()
    {
        ...
        uxDataGrid.AddHandler( RequestBringIntoViewEvent, new RoutedEventHandler( HandleRequestBringIntoViewEvent ) );
    }
    private static void HandleRequestBringIntoViewEvent( object sender, RoutedEventArgs e )
    {
        e.Handled = true;
    }
    ...
}