3

私のアプリケーションでは、ScrollViewer でラップされた Canvas があります (Canvas は画面サイズよりも大きいです)。このキャンバスには、他のコントロールを配置しました。キャンバス上の要素の 1 つには、仮想化された DataGrid が含まれています (2000 行以上あるため、スクロールも含まれます)。これで、行要素のいくつかの値に基づいて DataGrid 内の行を選択する関数ができました (自動的にトリガーされます)。行が選択されたら、私は呼び出します

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

完全に正常に機能しているもの。実際のところ、それは順調に進んでいます。私が望むのは、DataGrid の要素が選択され、DataGrid が正しい位置にスクロールすることです。しかし、私の Canvas scrollviewer もどういうわけかこのリクエストを受け取り、そこでスクロールしています。

ScrollChangedEvent をインターセプトして、handled フラグを設定しようとしました。しかし、それは機能していません。

質問:

1) ローカル ユーザー コントロールへの ScrollIntoView 呼び出しの効果を制限するにはどうすればよいですか。

2) それができない場合、自分でスクロールするにはどうすればよいですか? スクロールビューアの垂直オフセットを計算する必要がありますが、仮想化されているため、行の高さを調べる方法がわかりませんか?

助言がありますか?

プリンシパルのセットアップを示す簡単なサンプルを追加しました。ボタンの 1 つを押すと、キャンバスを含む ScrollViewer ではなく、DataGrid のみをスクロールします。

<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] );
    }
}
4

1 に答える 1

6

わかりました...いくつかの調査の後、私は正しいイベントを見つけました.それは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;
    }
    ...
}
于 2011-07-29T13:00:37.263 に答える