之前项目中的列表是采用的IScroll,但是在使用IScroll有一个问题就是:当内容不足全屏的时候,是木有办法往下拉的,这样就达不到刷新的目的了。【这是本人工作中遇到的,具体例子具体分析,这里只作一个参考】

大致的例子是这样的:
<style>
* {
margin: 0;
padding: 0;
}
html,body,.container {
width: 100%;
height: 100%;
}
.container>ul>li {
padding: 15px 20px;
text-align: center;
border-bottom: 1px solid #ccc;
}
</style>
<div id="container" class="container">
<ul class="scroller">
<li>item1</li>
<li>item2</li>
<li>item3</li>
<li>item4</li>
<li>item5</li>
</ul>
</div>
<script src="https://cdn.bootcss.com/iScroll/5.2.0/iscroll.min.js"></script>
<script>
var myScroll = null;
function onLoad() {
myScroll = new IScroll('container');
}
window.addEventListener('DOMContentLoaded', onLoad, false);
</script>
那么,既然超过一屏是可以刷新的,那我们就来逛逛代码吧。在github上搜索iscroll,打开第一个,找到src下面的core.js。
1. 思路
首先既然要下拉,肯定会触发touchstart、touchmove以及touchend事件。搜索touchmove,很好,在_initEvents中的注册了这个事件。
_initEvents: function (remove) {
// ...
// 这里省略若干代码
if ( utils.hasTouch && !this.options.disableTouch ) {
eventType(this.wrapper, 'touchstart', this);
eventType(target, 'touchmove', this);
eventType(target, 'touchcancel', this);
eventType(target, 'touchend', this);
}
// ...
},
好吧,看到这里的时候,我表示懵了一下逼,这不就是个绑定事件么?this又是一个什么鬼,然后我去查了一下文档,发现了这么一个东西。文档地址
target.addEventListener(type, listener[, options]); target.addEventListener(type, listener[, useCapture]); target.addEventListener(type, listener[, useCapture, wantsUntrusted ]); // // Gecko/Mozilla only listener 当所监听的事件类型触发时,会接收到一个事件通知(实现了 Event 接口的对象)对象。listener 必须是一个实现了 EventListener 接口的对象,或者是一个函数
木有看错,listener是一个对象或者是一个函数。前提是这个对象实现了EventListener接口。我们接着往下看,发现了这么一个例子。
var Something = function(element) {
// |this| is a newly created object
this.name = 'Something Good';
this.handleEvent = function(event) {
console.log(this.name);
// 'Something Good', as this is bound to newly created object
switch(event.type) {
case 'click':
// some code here...
break;
case 'dblclick':
// some code here...
break;
}
};
// Note that the listeners in this case are |this|, not this.handleEvent
element.addEventListener('click', this, false);
element.addEventListener('dblclick', this, false);
// You can properly remove the listeners
element.removeEventListener('click', this, false);
element.removeEventListener('dblclick', this, false);
}
var s = new Something(document.body);
然后在去IScroll的源码去找,发现了同样的实现方式。在default文件夹中有一个handleEvent.js。
好了,这个梗先告一段落。还是继续看源码。在handleEvent.js中,有这么一段东西。
handleEvent: function (e) {
switch ( e.type ) {
case 'touchstart':
case 'pointerdown':
case 'MSPointerDown':
case 'mousedown':
this._start(e);
break;
case 'touchmove':
case 'pointermove':
case 'MSPointerMove':
case 'mousemove':
this._move(e);
break;
case 'touchend':
case 'pointerup':
case 'MSPointerUp':
case 'mouseup':
case 'touchcancel':
case 'pointercancel':
case 'MSPointerCancel':
case 'mousecancel':
this._end(e);
break;
// ...
}
}
};
发现在start/move/end分别调用了内部方法_start/_move/_end方法。去看看这三个方法,看其中可能会引起不会滑动的点。
在_start方法中,看到这样的几行代码,会不会是直接返回了呢?分析分析:
if ( !this.enabled || (this.initiated && utils.eventType[e.type] !== this.initiated) {
return;
}
// ...
var point = e.touches ? e.touches[0] : e,
pos;
this.initiated = utils.eventType[e.type];
this.moved = false;
initiated属性在最开始肯定是没有的,而enabled默认是true,所以在最开始执行这个方法的时候是不会返回的,而是会给initiated这个属性设置当前的eventType值,这个值会在_move方法中用到。重点来看看_move方法。
if ( !this.enabled || utils.eventType[e.type] !== this.initiated ) {
return;
}
首先来进行类型判断,因为在_start方法中已经定义了这个值,所以这里也不会返回。接着往下看:
if ( timestamp - this.endTime > 300 && (absDistX < 10 && absDistY < 10) ) {
return;
}
【实际上是两次click事件的模拟】如果两次滑动的时间大于了300ms,并且只要一个方向上的位移少于10像素,那么也是会返回的。那么会不会呢,打个断点测试一下就知道了。这里就不贴图了,实际中的测试结果是,每一次移动肯定是在300ms以内的,这里之所以判断300ms,主要是click事件执行会有一个300ms的延迟。而每一次移动,由于手指的触点比较大,还是会大于10像素的,即使两次不大于10像素,也是不影响的。所以这点不会返回。那么继续接着看:
// If you are scrolling in one direction lock the other
if ( !this.directionLocked && !this.options.freeScroll ) {
if ( absDistX > absDistY + this.options.directionLockThreshold ) {
this.directionLocked = 'h'; // lock horizontally
} else if ( absDistY >= absDistX + this.options.directionLockThreshold ) {
this.directionLocked = 'v'; // lock vertically
} else {
this.directionLocked = 'n'; // no lock
}
}
if ( this.directionLocked == 'h' ) {
if ( this.options.eventPassthrough == 'vertical' ) {
e.preventDefault();
} else if ( this.options.eventPassthrough == 'horizontal' ) {
this.initiated = false;
return;
}
deltaY = 0;
} else if ( this.directionLocked == 'v' ) {
if ( this.options.eventPassthrough == 'horizontal' ) {
e.preventDefault();
} else if ( this.options.eventPassthrough == 'vertical' ) {
this.initiated = false;
return;
}
deltaX = 0;
}
第一个条件判断只要是定义了这次滑动的方向是什么。h表示水平方向,v表示竖直方向。我们是要向下滑动,所以我们关注的是竖直方向。看第二个条件判断,如果是竖直方向,那么将水平方向的deltaX值变为0。这样做的目的是保持绝对的竖直方向。因为移动实际还是根据元素的位移值来的。当probe的版本为2以下的时候,是根据css3的transform属性来移动位移的,为3版本的时候是根据决定对位来移动的。所以这里只要不把我们的deltaY置为0就说明木有什么问题。继续往下看代码:
deltaX = this.hasHorizontalScroll ? deltaX : 0; deltaY = this.hasVerticalScroll ? deltaY : 0; newX = this.x + deltaX; newY = this.y + deltaY; // ... // 这里是移动 this._translate(newX, newY);
测试中发现,这个hasVerticalScroll一直是false,那么deltaY一直就是0,也就是移动了也白移动。找到问题原因。那么,这个hasVerticalScroll是从哪里来的?全局找呀找,在refresh中找到这样几行代码:
this.wrapperWidth = this.wrapper.clientWidth; this.wrapperHeight = this.wrapper.clientHeight; var rect = utils.getRect(this.scroller); /* REPLACE START: refresh */ this.scrollerWidth = rect.width; this.scrollerHeight = rect.height; this.maxScrollX = this.wrapperWidth - this.scrollerWidth; this.maxScrollY = this.wrapperHeight - this.scrollerHeight; /* REPLACE END: refresh */ this.hasHorizontalScroll = this.options.scrollX && this.maxScrollX < 0; this.hasVerticalScroll = this.options.scrollY && this.maxScrollY < 0;
refresh方法会在IScroll实例化的时候调用一次。粗略一看,scrollY内置为true,所以只有maxScrollY会大于0。往上看。this.wrapperHeight - this.scrollerHeight肯定是大于0的呀,这就是问题所在。
那么看看我们最开始代码,这里的wrapperHeight为文档高度,scrollerHeight为内容高度,所以wrapperHeight高度始终大于scrollHeight。但是,手机端页面夹杂的列表,一般都有头部、底部,而中间部分一般都会采用padding的形式来使得列表在全局滚动,这样就不需要每次都要特定地计算列表的高度。
2. 解决方案
针对以上问题,只要我们能够使内部的滚动部分高度大于容器高度,那么就能触发滚动。
2.1 粗略做法
可以设置一个min-height属性为900px(900只是一个示例,只要够大就可以),这样就可以保证可以滑动。
2.2 精准做法
计算当前的容器高度,然后比容器高度多一个像素即可。
以上这篇IScroll那些事_当内容不足时下拉刷新的解决方法就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持。
# iscroll
# 下拉刷新
# 基于iScroll实现下拉刷新和上滑加载效果
# H5基于iScroll实现下拉刷新和上拉加载更多
# iscroll-probe实现下拉刷新和下拉加载效果
# iOS下拉刷新 UIScrollVie异常闪动问题
# 基于iscroll.js实现下拉刷新和上拉加载效果
# 基于HTML5上使用iScroll实现下拉刷新
# 上拉加载更多
# iscroll碰到Select无法选择下拉刷新的解决办法
# iscroll.js的上拉下拉刷新时无法回弹的解决方法
# JQuery插件iScroll实现下拉刷新
# 滚动翻页特效
# iscroll实现下拉刷新功能
# 两次
# 是一个
# 第一个
# 就不
# 往下看
# 会不会
# 会在
# 发现了
# 实现了
# 给大家
# 或者是
# 文档
# 就可以
# 有一个
# 的是
# 几行
# 这是
# 有什么
# 一个函数
# 是个
相关文章:
C++中的Pimpl idiom是什么,有什么好处?(隐藏实现)
网站视频制作书签怎么做,ie浏览器怎么将网站固定在书签工具栏?
品牌网站制作公司有哪些,买正品品牌一般去哪个网站买?
深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?
道歉网站制作流程,世纪佳缘致歉小吴事件,相亲网站身份信息伪造该如何稽查?
湖北网站制作公司有哪些,湖北清能集团官网?
手机怎么制作网站教程步骤,手机怎么做自己的网页链接?
Python如何创建带属性的XML节点
网站建设制作、微信公众号,公明人民医院怎么在网上预约?
企业网站制作公司网页,推荐几家专业的天津网站制作公司?
Python多线程使用规范_线程安全解析【教程】
个人网站制作流程图片大全,个人网站如何注销?
专业网站设计制作公司,如何制作一个企业网站,建设网站的基本步骤有哪些?
济南专业网站制作公司,济南信息工程学校怎么样?
北京营销型网站制作公司,可以用python做一个营销推广网站吗?
韩国网站服务器搭建指南:VPS选购、域名解析与DNS配置推荐
百度网页制作网站有哪些,谁能告诉我百度网站是怎么联系?
网站制作的方法有哪些,如何将自己制作的网站发布到网上?
专业网站制作服务公司,有哪些网站可以免费发布招聘信息?
c# Task.Yield 的作用是什么 它和Task.Delay(1)有区别吗
如何通过VPS搭建网站快速盈利?
如何在万网ECS上快速搭建专属网站?
如何快速打造个性化非模板自助建站?
小程序网站制作需要准备什么资料,如何制作小程序?
陕西网站制作公司有哪些,陕西凌云电器有限公司官网?
香港服务器租用费用高吗?如何避免常见误区?
高端建站三要素:定制模板、企业官网与响应式设计优化
如何快速搭建个人网站并优化SEO?
如何在建站之星绑定自定义域名?
详解免费开源的.NET多类型文件解压缩组件SharpZipLib(.NET组件介绍之七)
如何通过虚拟主机快速搭建个人网站?
如何在云虚拟主机上快速搭建个人网站?
,如何利用word制作宣传手册?
外贸公司网站制作哪家好,maersk船公司官网?
头像制作网站在线制作软件,dw网页背景图像怎么设置?
制作网站建设的公司有哪些,网站建设比较好的公司都有哪些?
,柠檬视频怎样兑换vip?
如何通过商城免费建站系统源码自定义网站主题?
建站主机服务器选购指南:轻量应用与VPS配置解析
独立制作一个网站多少钱,建立网站需要花多少钱?
,巨量百应是干嘛的?
香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化
Thinkphp 中 distinct 的用法解析
香港服务器网站生成指南:免费资源整合与高速稳定配置方案
学生网站制作软件,一个12岁的学生写小说,应该去什么样的网站?
网站好制作吗知乎,网站开发好学吗?有什么技巧?
网站专业制作公司有哪些,做一个公司网站要多少钱?
音响网站制作视频教程,隆霸音响官方网站?
如何选择网络建站服务器?高效建站必看指南
如何在建站之星网店版论坛获取技术支持?
*请认真填写需求信息,我们会在24小时内与您取得联系。