【WP8】ScrollViewer滑动到底触发器(ListBox失效)
很多时候会有到底加载更多的需求,而ScrollViewer不支持继承,无法继承它进行扩展,只能通过触发器来控制到底的事件(当然,可以通过UserControl去扩展)
思路:定义一个Trigger,自定义依赖属性,绑定到该属性到ScrollViewer的VerticalOffset属性上,然后监听属性的变化,就能监控到滚动事件了,然后判断滚动的位置从而判断出是否到底
原理很简单,下面看实现
// *************************************************
//
// 作者:bomo
// 小组:WP开发组
// 创建日期:2014/7/9 0:10:32
// 版本号:V1.00
// 说明:
//
// *************************************************
//
// 修改历史:
// Date WhoChanges Made
// 2014/7/9 0:10:32 bomo Initial creation
//
// ************************************************* using System.Collections.Generic;
using System.Linq;
using System.Windows;
using System.Windows.Media; namespace XTuOne.Common.Helpers
{
/// <summary>
/// 扩展依赖属性类
/// </summary>
public static class DependencyObjectExtend
{
public static IEnumerable<DependencyObject> GetDescendant(this DependencyObject element)
{
var list = new List<DependencyObject>();
var count = VisualTreeHelper.GetChildrenCount(element);
for (int i = ; i < count; i++)
{
var child = VisualTreeHelper.GetChild(element, i);
list.Add(child);
list.AddRange(child.GetDescendant());
} return list;
} /// <summary>
/// 查找子孙节点中符合类型的首个节点
/// </summary>
public static T GetFirstDescendantOfType<T>(this DependencyObject start) where T : DependencyObject
{
return start.GetDescendantsOfType<T>().FirstOrDefault();
} /// <summary>
/// 查找子孙节点中符合类型的节点
/// </summary>
public static IEnumerable<T> GetDescendantsOfType<T>(this DependencyObject start) where T : DependencyObject
{
return start.GetDescendants().OfType<T>();
} /// <summary>
/// 获取所有的子孙阶段
/// </summary>
public static IEnumerable<DependencyObject> GetDescendants(this DependencyObject start)
{
if (start == null)
yield break; var queue = new Queue<DependencyObject>();
queue.Enqueue(start);
yield return start; while (queue.Count > )
{
var parent = queue.Dequeue();
var count2 = VisualTreeHelper.GetChildrenCount(parent); for (int i = ; i < count2; i++)
{
var child = VisualTreeHelper.GetChild(parent, i);
yield return child;
queue.Enqueue(child);
}
}
}
}
}
依赖属性扩展类
// *************************************************
//
// 作者:bomo
// 小组:WP开发组
// 创建日期:2014/7/11 11:45:14
// 版本号:V1.00
// 说明:
//
// *************************************************
//
// 修改历史:
// Date WhoChanges Made
// 2014/7/11 11:45:14 bomo Initial creation
//
// ************************************************* using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Interactivity;
using XTuOne.Common.Helpers; namespace XTuOne.Utility.Helpers
{
/// <summary>
/// ScrollViewer到底触发器
/// </summary>
public class ScrollViewerToBottomTrigger : TriggerBase<DependencyObject>
{
public static readonly DependencyProperty VerticalOffsetProperty = DependencyProperty.Register(
"VerticalOffset", typeof(double), typeof(ScrollViewerToBottomTrigger),
new PropertyMetadata(0.0, VerticalOffsetPropertyChanged)); private ScrollViewer scrollView; public double VerticalOffset
{
get { return (double)GetValue(VerticalOffsetProperty); }
set { SetValue(VerticalOffsetProperty, value); }
} public static void VerticalOffsetPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
var behavior = d as ScrollViewerToBottomTrigger;
if (behavior != null)
behavior.OnVerticalOffsetChanged();
} protected override void OnAttached()
{
base.OnAttached(); if (AssociatedObject is FrameworkElement)
{
(AssociatedObject as FrameworkElement).SizeChanged += control_SizeChanged;
}
} private void control_SizeChanged(object sender, SizeChangedEventArgs e)
{
if (!(AssociatedObject is FrameworkElement))
return; var scroll = AssociatedObject.GetFirstDescendantOfType<ScrollViewer>();
if (scroll != null)
{
AttachedScroll(scroll);
(AssociatedObject as FrameworkElement).SizeChanged -= control_SizeChanged;
}
} private void AttachedScroll(ScrollViewer scroll)
{
if (scroll == null)
return;
scrollView = scroll; var binding = new Binding { Source = scroll, Path = new PropertyPath("VerticalOffset") };
BindingOperations.SetBinding(this, VerticalOffsetProperty, binding);
} private void OnVerticalOffsetChanged()
{
var scroll = scrollView;
if (scroll == null)
return; if (scroll.ExtentHeight - scroll.VerticalOffset - scroll.ViewportHeight <= )
{
InvokeActions(null);
}
}
}
}
通过Trigger实现,由于Trigger是一个附加元素,可以附加到任何一个符合的元素上(子孙元素包含ScrollViewer的控件),同时触发的事件可以支持绑定
使用(使用CM进行绑定)
<ScrollViewer x:Name="ScrollViewer">
<i:Interaction.Triggers>
<utilityHelpers:ScrollViewerToBottomTrigger>
<micro:ActionMessage MethodName="LoadMessage"/>
</utilityHelpers:ScrollViewerToBottomTrigger>
</i:Interaction.Triggers>
</ScrollViewer>
一个触发器只能对应触发一个事件(用于绑定),如果需要多个扩展,比如:到顶触发,滑动触发等,就需要编写多个触发器,如果这样可以考虑在UserControl扩展
发现一个问题(上面触发器在ListBox失效):
ListBox中的ScrollViewer的ExtentHeight是数据项的个数,比如ListBox有20条数据,那么ExtentHeight就是20
ListBox中的VerticalOffset是表示当前位置,如果当前屏幕中显示的第一条是第12条数据,那么VerticalOffset就是12.xxx
ListBox中的ViewportHeight是根据第一屏的数据计算的,把List滑动到顶部,然后计算出页面内有几个项,这个值是假定List内所有的项的大小都是一样的,如果ListBox中的ItemTemplate的高度不是相同大小的,就会出现误差,导致下面三种情况都可能出现
ExtentHeight == VerticalOffset + ViewportHeight
ExtentHeight > VerticalOffset + ViewportHeight
ExtentHeight < VerticalOffset + ViewportHeight
如果要用在ListBox上的话,需要保证每一项的高度都相同,否则会有误差(有时提前触发,有时候不触发)
【WP8】ScrollViewer滑动到底触发器(ListBox失效)的更多相关文章
- pyqt5在textBrowser添加文本并自动滑动到底
pyqt5在textBrowser添加文本并自动滑动到底 说明: 1.按下按钮pushButton,把单行文本框lineEdit里的内容循环不断的添加到多行文本展示框textBrowser.2.必须要 ...
- Unity触发器有时失效的原因
unity里面的触发器有时候不起作用,我原以为是失效了.其实是这样的,所谓触发器就是被触发的物体,例如你子弹打小怪.如果把子弹设置成触发器那么是不成功的,因为子弹是主动的啊,那么把小怪设置成触发器了呢 ...
- 移动端ios上下滑动翻页事件失效
移动端开发过程中,在添加上下滑动事件时候,引入了最常用的移动端库zepto.js及其touch模块,有一种现象,安卓的手机没有问题,上下滑动翻页很正常 :但是到了ios上面,好啊,上下滑动会出现弹性滚 ...
- selenium+python关于页面滚动条滑动到底的问题总结
1.如果滚动条是针对整个HTML可以用如下方式: js = "var q=document.documentElement.scrollTop=10000" # documentE ...
- WPF ItemsControl 控件支持鼠标滚轮滑动
此文章意在解决在WPF中ItemsControl类型的集合控件支持鼠标滚轮操作,并可控制滚动的速度. 第一步:给ItemsControl添加滚轮事件. this.listBox.AddHandler( ...
- 【朝花夕拾】Android自定义View篇之(七)Android事件分发机制(下)滑动冲突解决方案总结
前言 转载请声明,转自[https://www.cnblogs.com/andy-songwei/p/11072989.html],谢谢! 前面两篇文章,花了很大篇幅讲解了Android的事件分发机制 ...
- 写一个js向左滑动删除 交互特效的插件——Html5 touchmove
需求描述 需要实现类似QQ中对联系人的操作:向左滑动,滑出删除按钮.滑动超过一半时松开则自动滑到底,不到一半时松开则返回原处. 纯js实现 使用了h5的touchmove等事件,以及用js动态改变cs ...
- [WPF系列]-ListBox
引言 本文就WPF中的ListBox常用项给以实例代码演示,包括隐蔽属性的设置,Style设置,以及ControlTemplate的自定义. Listbox平滑滚动 <ListBox Ite ...
- WPF的ScrollViewer鼠标的滚动
在C# 中,两个ScrollViewer嵌套在一起或者ScrollViewer里面嵌套一个ListBox.Listview(控件本身有scrollviewer)的时候,我们本想要的效果是鼠标滚动整个S ...
随机推荐
- C#学习笔记(28)——匿名委托和Lambda表达式
说明(2017-11-21 18:51:32): 1. 例子为求1~100的和,答案应该是5050(小学学算盘的时候,我爹就让我算,从1拨到100是多少呀?当时的我年幼无知,还不知道高斯小时候的故事, ...
- 基于jQuery和CSS3超酷Material Design风格滑动菜单导航特效
分享一款效果非常炫酷的谷歌 Material Design 风格jQuery和CSS3滑动选项卡特效.该选项卡特效集合了扁平风格设计和按钮点击波特效.是一款设计的非常不错的Material Desig ...
- java基础篇---枚举详解
在JDK1.5之前,JAVA可以有两种方式定义新类型:类和接口,对于大部分面向对象编程,有这两种似乎就足够了,但是在一些特殊情况就不合适.例如:想要定义一个Color类,它只能有Red,Green,B ...
- wcf消息契约
1.最多一个参数和一个返回值,返回值和参数的类型都是消息类型. 下面的代码为定义一个消息契约的实例 [MessageContract] public class MyMessage { ...
- JAVA读取MongoDB中的二进制图片并在jsp中显示
http://blog.csdn.net/u012138706/article/details/52180665
- 数据库like匹配的实现猜测
insert into test_fulltext values("王正科技全文") select * from test_fulltext where data like &qu ...
- Install Shield常用函数以及参数
nstall Shield函数库 1 库函数综述InstallShield包含300多个内部库函数,用户可在安装脚本中调用它们来创建程序组,操作文件夹,处理目录,监督安装状态,创建对话框,操作文件及 ...
- WPF中的命令简介
使用Prism委托命令Demo: WPF委托命令DelegateCommand的传参方式 在WPF中使用命令的步骤很简单 1.创建命令 2.绑定命令 3.设置命令源 4.设置命令目标 WPF中命令的核 ...
- android 技术点记录
Android Service完全解析,关于服务你所需知道的一切(上) http://blog.csdn.net/guolin_blog/article/details/11952435 androi ...
- Java编程的逻辑 (38) - 剖析ArrayList
本系列文章经补充和完善,已修订整理成书<Java编程的逻辑>,由机械工业出版社华章分社出版,于2018年1月上市热销,读者好评如潮!各大网店和书店有售,欢迎购买,京东自营链接:http:/ ...