Pop*_*lin 68
您可以实现平滑滚动,但丢失项目虚拟化,所以基本上只有在列表中包含少量元素时才应使用此技术:
信息在这里:在列表框上平滑滚动
你试过设置:
Run Code Online (Sandbox Code Playgroud)ScrollViewer.CanContentScroll="False"在列表框上?
这种方式滚动是由面板而不是listBox处理的...如果你这样做会失去虚拟化,所以如果你有很多内容它可能会变慢.
rmo*_*ore 11
确实有可能做你要求的,虽然它需要相当数量的自定义代码.
通常在WPF中,ScrollViewer使用所谓的逻辑滚动,这意味着它将逐项滚动而不是按偏移量滚动.其他答案涵盖了将逻辑滚动行为更改为物理滚动行为的一些方法.另一种方法是使用ScrollViwer和IScrollInfo公开的ScrollToVertialOffset和ScrollToHorizontalOffset方法.
要实现较大的部分,按下鼠标滚轮时滚动,我们将需要使用MouseDown和MouseMove事件.
<ListView x:Name="uiListView"
Mouse.MouseDown="OnListViewMouseDown"
Mouse.MouseMove="OnListViewMouseMove"
ScrollViewer.CanContentScroll="False">
....
</ListView>
Run Code Online (Sandbox Code Playgroud)
在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;
}
Run Code Online (Sandbox Code Playgroud)
它缺少一些领域,因为它只是一个概念证明,但它绝对应该让你开始朝着正确的方向前进.要在鼠标从初始MouseDown点移开后不断滚动,滚动逻辑可以进入DispatcherTimer或类似的东西.
小智 11
我知道这篇文章已经有 13 年历史了,但这仍然是人们想做的事情。在较新版本的 .Net 中,您可以VirtualizingPanel.ScrollUnit="Pixel"
这样设置,这样您就不会丢失虚拟化,并且可以按像素滚动而不是按项目滚动。
尝试在ListView上将ScrollViewer.CanContentScroll附加属性设置为false。但是就像Pop Catalin所说的那样,你失去了项目虚拟化,这意味着列表中的所有项目都会立即加载和填充,而不是在需要显示一组项目时 - 所以如果列表很大,它可能会导致一些内存和性能问题。
| 归档时间: |
|
| 查看次数: |
44921 次 |
| 最近记录: |