是否可以在WPF列表视图中实现平滑滚动?

时间:2009-06-23 16:54:45

标签: c# .net wpf listview scroll

是否可以在WPF listview中实现平滑滚动,就像它在Firefox中的工作方式一样? 当Firefox浏览器包含所有listview个项目并按住鼠标中键(但不是释放)并拖动它时,它应该平滑地滚动listview个项目。当你释放它应该停止。

看起来这在winforms中是不可能的,但我想知道它是否在WPF中可用?

5 个答案:

答案 0 :(得分:65)

您可以实现平滑滚动,但是会丢失项目虚拟化,所以基本上只有在列表中的元素很少时才应该使用此技术:

信息在这里:Smooth scrolling on listbox

  

您是否尝试过设置:

     
ScrollViewer.CanContentScroll="False"
     列表框上的

     

这种方式滚动是由面板而不是listBox处理的...如果你这样做会失去虚拟化,如果你有很多内容它可能会变慢。

答案 1 :(得分:10)

确实可以做你要问的事情,虽然它需要相当数量的自定义代码。

通常在WPF中,ScrollViewer使用所谓的逻辑滚动,这意味着它将逐项滚动而不是按偏移量滚动。其他答案涵盖了将逻辑滚动行为更改为物理滚动行为的一些方法。另一种方法是使用ScrollViwer和IScrollInfo公开的ScrollToVertialOffset和ScrollToHorizo​​ntalOffset方法。

要实现较大的部分,按下鼠标滚轮时滚动,我们需要使用MouseDown和MouseMove事件。

<ListView x:Name="uiListView"
          Mouse.MouseDown="OnListViewMouseDown"
          Mouse.MouseMove="OnListViewMouseMove"
          ScrollViewer.CanContentScroll="False">
    ....
</ListView>

在MouseDown中,我们将记录当前鼠标位置,我们将使用它作为相对点来确定我们滚动的方向。在鼠标移动中,我们将获得ListView的ScrollViwer组件,然后相应地滚动它。

private Point myMousePlacementPoint;

private void OnListViewMouseDown(object sender, MouseButtonEventArgs e)
{
    if (e.MiddleButton == MouseButtonState.Pressed)
    {
        myMousePlacementPoint = this.PointToScreen(Mouse.GetPosition(this));
    }
}

private void OnListViewMouseMove(object sender, MouseEventArgs e)
{
    ScrollViewer scrollViewer = ScrollHelper.GetScrollViewer(uiListView) as ScrollViewer;

    if (e.MiddleButton == MouseButtonState.Pressed)
    {
        var currentPoint = this.PointToScreen(Mouse.GetPosition(this));

        if (currentPoint.Y < myMousePlacementPoint.Y)
        {
            scrollViewer.ScrollToVerticalOffset(scrollViewer.VerticalOffset - 3);
        }
        else if (currentPoint.Y > myMousePlacementPoint.Y)
        {
            scrollViewer.ScrollToVerticalOffset(scrollViewer.VerticalOffset + 3);
        }

        if (currentPoint.X < myMousePlacementPoint.X)
        {
            scrollViewer.ScrollToHorizontalOffset(scrollViewer.HorizontalOffset - 3);
        }
        else if (currentPoint.X > myMousePlacementPoint.X)
        {
            scrollViewer.ScrollToHorizontalOffset(scrollViewer.HorizontalOffset + 3);
        }
    }
}

public static DependencyObject GetScrollViewer(DependencyObject o)
{
    // Return the DependencyObject if it is a ScrollViewer
    if (o is ScrollViewer)
    { return o; }

    for (int i = 0; i < VisualTreeHelper.GetChildrenCount(o); i++)
    {
        var child = VisualTreeHelper.GetChild(o, i);

        var result = GetScrollViewer(child);
        if (result == null)
        {
            continue;
        }
        else
        {
            return result;
        }
    }
    return null;
}

它缺少一些领域,因为它只是一个概念证明,但它绝对应该让你开始朝着正确的方向前进。要让鼠标从初始MouseDown点移开后不断滚动,滚动逻辑可以进入DispatcherTimer或类似的东西。

答案 2 :(得分:4)

如果您使用的是.NET 4.5(或者如果您愿意破解一下,则为4.0),那么答案是over here

答案 3 :(得分:3)

尝试在 ListView 上将 ScrollViewer.CanContentScroll 附加属性设置为 false 。但是像 Pop Catalin 所说的那样,你会失去项目虚拟化,这意味着列表中的所有项目都会被立即加载和填充,而不是当需要显示一组项目时 - 所以如果列表很大,它可能会导致一些内存和性能问题。

答案 4 :(得分:0)

尝试将listview的高度设置为自动并将其包装在滚动查看器中。

<ScrollViewer IsTabStop="True" VerticalScrollBarVisibility="Auto">
     <ListView></ListView>
</ScrollViewer>

别忘了提及ScrollViewer的高度 希望这会有所帮助......