【Win10】实现 ListViewBase 平滑滚动
首先解释下标题的 ListViewBase 是什么鬼。ListViewBase 我们可以查阅 MSDN 文档:https://msdn.microsoft.com/zh-cn/library/windows.ui.xaml.controls.listviewbase.aspx 得知,ListViewBase 是 ListView 和 GridView 的基类(ListView 和 GridView 则为常用的数据展示控件之一)。而本文的主要目的就是实现 ListView 和 GridView 的平滑滚动,因此我将标题写成“实现 ListViewBase 平滑滚动”而不是“实现 ListView 和 GridView 平滑滚动”(实际上本文适用于任何继承自 ListViewBase 的控件)。
首先我们先复习一下怎么滚动到 ListViewBase 的某一个 item。
在 ListViewBase 类中,有一个方法叫做 ScrollIntoView。这个方法有两个重载,我们看复杂一点,有两个参数的这个:
//
// 摘要:
// 滚动列表,以将指定数据项移入具有指定对齐方式的视图中。
//
// 参数:
// item:
// 要在视图中显示的数据项。
//
// alignment:
// 指定项是使用 Default 还是 Leading 对齐方式的枚举值。
[Overload("ScrollIntoViewWithAlignment")]
public void ScrollIntoView(System.Object item, ScrollIntoViewAlignment alignment);
第一个参数就是我们需要滚动到当前可视区域的 item,而第二个参数,Default 是指让其滚动到当前可视区域即可,Leading 则是指让其滚动到当前可视区域的顶部。
但是比较遗憾的是,这个方法一执行就(?)立马滚动到目标 item 了,完全不带一丁点动画效果(后文你会了解到内部执行仍需很少一段时间,尽管我们肉眼察觉不到)。在这个时代,没有一个好的 UI,怎么能吸引用户呢?因此我们就来研究并实现怎样能让 ListViewBase 平滑滚动到某个 item。
说起滚动的话,我们一定会想到 ScrollBar、ScrollViewer 这类的控件的。而幸运的是,ScrollViewer 有一个方法,叫 ChangeView 是带动画效果的(也可以选择不使用动画效果)。并且 ListView、GridView 内部都是有一个 ScrollViewer 的。那么我们自然而然就想到,是不是可以操作 ListViewBase 内部的这个 ScrollViewer 来实现平滑滚动。
先开始编写代码吧:
public static class ListViewBaseExtensions
{
public static void ScrollIntoViewSmoothly(this ListViewBase listViewBase, object item, ScrollIntoViewAlignment alignment)
{
if (listViewBase == null)
{
throw new ArgumentNullException(nameof(listViewBase));
} // GetFirstDescendantOfType 是 WinRTXamlToolkit 中的扩展方法,
// 寻找该控件在可视树上第一个符合类型的子元素。
ScrollViewer scrollViewer = listViewBase.GetFirstDescendantOfType<ScrollViewer>(); // 由于 ScrollViewer 肯定有,因此不做 null 检查判断了。 scrollViewer.ChangeView(targetHorizontalOffset, targetVerticalOffset, null);
}
}
然而问题来了,targetHorizontalOffset 和 targetVerticalOffset 我们是不知道的,也就是说,我们不知道目标 item 所在的位置。
尽管我们不知道,但是,ListViewBase 自身的 ScrollIntoView 方法它是知道的,那我们干脆就让它当个跑腿,先执行一次,然后就可以获取目标位置了。
public static class ListViewBaseExtensions
{
public static void ScrollIntoViewSmoothly(this ListViewBase listViewBase, object item, ScrollIntoViewAlignment alignment)
{
if (listViewBase == null)
{
throw new ArgumentNullException(nameof(listViewBase));
} // GetFirstDescendantOfType 是 WinRTXamlToolkit 中的扩展方法,
// 寻找该控件在可视树上第一个符合类型的子元素。
ScrollViewer scrollViewer = listViewBase.GetFirstDescendantOfType<ScrollViewer>(); // 由于 ScrollViewer 肯定有,因此不做 null 检查判断了。 // 记录初始位置,用于 ScrollIntoView 检测目标位置后复原。
double originHorizontalOffset = scrollViewer.HorizontalOffset;
double originVerticalOffset = scrollViewer.VerticalOffset; // 跑腿。
listViewBase.ScrollIntoView(item, alignment); // 获取目标位置。
double targetHorizontalOffset = scrollViewer.HorizontalOffset;
double targetVerticalOffset = scrollViewer.VerticalOffset; // scrollViewer.ChangeView(targetHorizontalOffset, targetVerticalOffset, null);
}
}
然而通过断点检查后,发现 targetHorizontalOffset 和 targetVerticalOffset 并没有发生变化。但是执行过后,ListViewBase 确实发生了滚动,因此我们质疑,是不是 ScrollIntoView 方法在控件内部是以一个异步的形式执行。
这个时候,我们还是想起近乎万能的 LayoutUpdated 事件吧。改写下代码。
public static class ListViewBaseExtensions
{
public static void ScrollIntoViewSmoothly(this ListViewBase listViewBase, object item, ScrollIntoViewAlignment alignment)
{
if (listViewBase == null)
{
throw new ArgumentNullException(nameof(listViewBase));
} // GetFirstDescendantOfType 是 WinRTXamlToolkit 中的扩展方法,
// 寻找该控件在可视树上第一个符合类型的子元素。
ScrollViewer scrollViewer = listViewBase.GetFirstDescendantOfType<ScrollViewer>(); // 由于 ScrollViewer 肯定有,因此不做 null 检查判断了。 // 记录初始位置,用于 ScrollIntoView 检测目标位置后复原。
double originHorizontalOffset = scrollViewer.HorizontalOffset;
double originVerticalOffset = scrollViewer.VerticalOffset; EventHandler<object> layoutUpdatedHandler = null;
layoutUpdatedHandler = delegate
{
listViewBase.LayoutUpdated -= layoutUpdatedHandler; // 获取目标位置。
double targetHorizontalOffset = scrollViewer.HorizontalOffset;
double targetVerticalOffset = scrollViewer.VerticalOffset; // scrollViewer.ChangeView(targetHorizontalOffset, targetVerticalOffset, null);
};
listViewBase.LayoutUpdated += layoutUpdatedHandler; // 跑腿。
listViewBase.ScrollIntoView(item, alignment);
}
}
这次我们再断点后,发现能够获取目标位置了!!(所以我上面说“内部执行仍需很少一段时间,尽管我们肉眼察觉不到”)
接下来,由于跑腿是已经滚动目标位置了,因此我们需要复原到原来的位置,再滚动到目标位置以实现平滑滚动的动画效果。
public static void ScrollIntoViewSmoothly(this ListViewBase listViewBase, object item, ScrollIntoViewAlignment alignment)
{
if (listViewBase == null)
{
throw new ArgumentNullException(nameof(listViewBase));
} // GetFirstDescendantOfType 是 WinRTXamlToolkit 中的扩展方法,
// 寻找该控件在可视树上第一个符合类型的子元素。
ScrollViewer scrollViewer = listViewBase.GetFirstDescendantOfType<ScrollViewer>(); // 由于 ScrollViewer 肯定有,因此不做 null 检查判断了。 // 记录初始位置,用于 ScrollIntoView 检测目标位置后复原。
double originHorizontalOffset = scrollViewer.HorizontalOffset;
double originVerticalOffset = scrollViewer.VerticalOffset; EventHandler<object> layoutUpdatedHandler = null;
layoutUpdatedHandler = delegate
{
listViewBase.LayoutUpdated -= layoutUpdatedHandler; // 获取目标位置。
double targetHorizontalOffset = scrollViewer.HorizontalOffset;
double targetVerticalOffset = scrollViewer.VerticalOffset; // 复原位置,且不需要使用动画效果。
scrollViewer.ChangeView(originHorizontalOffset, originVerticalOffset, null, true); // 最终目的,带平滑滚动效果滚动到 item。
scrollViewer.ChangeView(targetHorizontalOffset, targetVerticalOffset, null);
};
listViewBase.LayoutUpdated += layoutUpdatedHandler; // 跑腿。
listViewBase.ScrollIntoView(item, alignment);
}
}
执行之后,然而我们发现还是直接滚动到目标,不带一丁点动画效果
。但是,有了上面 ScrollIntoView 的经验后,我们自然而然也可以质疑 ChangeView 方法是不是像 ScrollIntoView 一样,内部也是异步执行的。再改写下:
public static class ListViewBaseExtensions
{
public static void ScrollIntoViewSmoothly(this ListViewBase listViewBase, object item, ScrollIntoViewAlignment alignment)
{
if (listViewBase == null)
{
throw new ArgumentNullException(nameof(listViewBase));
} // GetFirstDescendantOfType 是 WinRTXamlToolkit 中的扩展方法,
// 寻找该控件在可视树上第一个符合类型的子元素。
ScrollViewer scrollViewer = listViewBase.GetFirstDescendantOfType<ScrollViewer>(); // 由于 ScrollViewer 肯定有,因此不做 null 检查判断了。 // 记录初始位置,用于 ScrollIntoView 检测目标位置后复原。
double originHorizontalOffset = scrollViewer.HorizontalOffset;
double originVerticalOffset = scrollViewer.VerticalOffset; EventHandler<object> layoutUpdatedHandler = null;
layoutUpdatedHandler = delegate
{
listViewBase.LayoutUpdated -= layoutUpdatedHandler; // 获取目标位置。
double targetHorizontalOffset = scrollViewer.HorizontalOffset;
double targetVerticalOffset = scrollViewer.VerticalOffset; EventHandler<ScrollViewerViewChangedEventArgs> scrollHandler = null;
scrollHandler = delegate
{
scrollViewer.ViewChanged -= scrollHandler; // 最终目的,带平滑滚动效果滚动到 item。
scrollViewer.ChangeView(targetHorizontalOffset, targetVerticalOffset, null);
};
scrollViewer.ViewChanged += scrollHandler; // 复原位置,且不需要使用动画效果。
scrollViewer.ChangeView(originHorizontalOffset, originVerticalOffset, null, true); };
listViewBase.LayoutUpdated += layoutUpdatedHandler; // 跑腿。
listViewBase.ScrollIntoView(item, alignment);
}
}
这次我们终于成功了!!!
效果:

最后我们像 ListViewBase 的 ScrollIntoView 方法,加多个只有一个参数的重载吧。
最终代码:
public static class ListViewBaseExtensions
{
public static void ScrollIntoViewSmoothly(this ListViewBase listViewBase, object item)
{
ScrollIntoViewSmoothly(listViewBase, item, ScrollIntoViewAlignment.Default);
} public static void ScrollIntoViewSmoothly(this ListViewBase listViewBase, object item, ScrollIntoViewAlignment alignment)
{
if (listViewBase == null)
{
throw new ArgumentNullException(nameof(listViewBase));
} // GetFirstDescendantOfType 是 WinRTXamlToolkit 中的扩展方法,
// 寻找该控件在可视树上第一个符合类型的子元素。
ScrollViewer scrollViewer = listViewBase.GetFirstDescendantOfType<ScrollViewer>(); // 由于 ScrollViewer 肯定有,因此不做 null 检查判断了。 // 记录初始位置,用于 ScrollIntoView 检测目标位置后复原。
double originHorizontalOffset = scrollViewer.HorizontalOffset;
double originVerticalOffset = scrollViewer.VerticalOffset; EventHandler<object> layoutUpdatedHandler = null;
layoutUpdatedHandler = delegate
{
listViewBase.LayoutUpdated -= layoutUpdatedHandler; // 获取目标位置。
double targetHorizontalOffset = scrollViewer.HorizontalOffset;
double targetVerticalOffset = scrollViewer.VerticalOffset; EventHandler<ScrollViewerViewChangedEventArgs> scrollHandler = null;
scrollHandler = delegate
{
scrollViewer.ViewChanged -= scrollHandler; // 最终目的,带平滑滚动效果滚动到 item。
scrollViewer.ChangeView(targetHorizontalOffset, targetVerticalOffset, null);
};
scrollViewer.ViewChanged += scrollHandler; // 复原位置,且不需要使用动画效果。
scrollViewer.ChangeView(originHorizontalOffset, originVerticalOffset, null, true); };
listViewBase.LayoutUpdated += layoutUpdatedHandler; // 跑腿。
listViewBase.ScrollIntoView(item, alignment);
}
}
最后再附送上 Demo:ListViewBaseScrollSmoothlyDemo.zip
最后的最后,冰天雪地裸体 360 度跪求一份 UWP/WP8.1 相关的工作
。(长期有效)
【Win10】实现 ListViewBase 平滑滚动的更多相关文章
- js平滑滚动到顶部,底部,指定地方
[原文链接] 采用锚点进行页面中的跳转的确很方便,但是要想增加网页的效果,可以使用jquery中的animate,实现滚动的一个动作,慢慢的滚动到你想跳转到的位置,从而看起来会非常高大上. [示例演示 ...
- 【转】使用jquery animate创建平滑滚动效果
这篇文章主要介绍了使用jquery animate创建平滑滚动效果,效果可以滚动到顶部.到底部或页面中指定地方,生要的是非常平滑,很舒服,需要的朋友可以参考下 滚动到顶部: $('.scroll_to ...
- Windows 10 Edge浏览器、照片查看程序关闭“平滑滚动”
升级到10后,这两个常用软件的“平滑滚动”功能,个人感觉体验有点不好,特别是图片这个自带程序,看了几十张图后就有点头晕了,所以把它关闭为好: 控制面板\系统和安全\系统\高级系统设置\高级\性能\设置 ...
- Android游戏开发之主角的移动与地图的平滑滚动
人物移动地图的平滑滚动处理 玩过rpg游戏的朋友应该都知道RPG的游戏地图一般都比较大 今天我和大家分享一下在RPG游戏中如何来处理超出手机屏幕大小的游戏地图. 如图所示为程序效果动画图 地图滚动的原 ...
- JQuery简单实现锚点链接的平滑滚动
在平时的项目中,我们经常需要一些特效链接,如果使效果进一步加强,我们可以使点击锚点链接平滑滚动到锚点,下面就来给大家讲解下如何使用jQuery来实现. 一般使用锚点来跳转到页面指定位置的时候,会生 ...
- javaScript滚动新闻之上下左右平滑滚动
<!DOCTYPE HTML> <html> <head> <meta http-equiv="Content-Type" content ...
- 页面中的平滑滚动——smooth-scroll.js的使用
正常的本页面锚链接跳转的时候跟PPT似的,特别生硬,用户体验非常差. 这时候我们就可以借助smooth-scroll.js这个插件,来实现本页面的平滑的跳转. 1首先,导入必须的JS文件 <sc ...
- 仿微信未读RecyclerView平滑滚动定位效果
效果图有红点的地方表示有未读消息,依次双击首页图标定位,然后定位到某个未读在手动下滑一点距离在次点击定位效果 用过 RecyclerView 的人都知道,自带有几个滚动到item下标的方法,但是不靠谱 ...
- C#在Win10与非Win10 Windows系统鼠标滚动编程的一点区别。
C#在win10和非Win10上处理鼠标滚动有一些区别,建一个Form1,放置一个FlowLayoutPanel ,类型的Panel1 Panel.MouseWheel += PanelOnMouse ...
随机推荐
- go 第一个项目
官方下载go: https://golang.org/dl/ 安装完成后:cmd命令下:go go env:查看当前的环境配置:
- 序列比对之Biostrings包
基本概念 Biostrings包很重要的3个功能是进行Pairwise sequence alignment 和Multiple sequence alignment及 Pattern finding ...
- 大型运输行业实战_day03_2_使用ajax将请求页面与请求数据分离
1.引入jquery 1.添加jquery包 2.在要使用jquery的页面中引入jquery 引入jquery后必须检查是否引入正确,这里值得注意的是 springMVC默认情况先会拦截 js文件, ...
- php71 gdnz
更新yum库:yum updat yum install epel-release yum install -y gcc gcc-c++ autoconf libjpeg libjpeg-devel ...
- Redis常用数据类型及命令
Redis数据类型 Redis支持五种数据类型:string(字符串),hash(哈希),list(列表),set(集合)及zset(sorted set:有序集合). 注意: 命令的关键词,如set ...
- cubic与spline插值点处的区别
cubic与spline都是Matlab的三次样条插值法,但是它们在插值点处仍然有着很微妙的区别,这个区别说明不了两种方法的好坏,只能根据实际情况进行合理筛选.以一维插值为例: clc clear % ...
- nginx常用配置说明
nginx的主配置(nginx.conf)说明 #worker进程数量 worker_processes 1; #错误日志 error_log logs/error.log; #进程ID文件 pid ...
- Js中的this关键字(吉木自学)
研究生毕业答辩完,开始继续为转行努力.小白要奋斗了,加油.本文引自JS核心系列:浅谈函数的作用域. 在一个函数中,this总是指向当前函数的所有者对象,this总是在运行时才能确定其具体的指向, 也才 ...
- 原生JS 实现元素排序
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- 新电脑的操作系统win10的所有设置问题汇总
上来改的win7发现很多驱动没法装,装了也不能用,后来只能改win10了,另外win7的风扇声音也很大. 1.关闭win10自动更新.在服务里面禁用winupdate 2.注销改成了点头像,然后点注销 ...