源码 - mmrxia/js-pull-to-refresh GitHub Wiki
/* global define, Roll */ (function(window, document, Roll) { 'use strict'
var rAF = window.requestAnimationFrame || window.webkitRequestAnimationFrame || function(callback) {
setTimeout(callback, 17)
}
var TSF = Roll.utils.TSF
var moveTo = Roll.utils.moveTo
var IMG_LOADING = ''
var IMG_ARROW = ''
var IMG_FINISH = ''
var IMG_STYLE = 'width:24px;height:24px;display:block;'
Roll.prototype.pulldown = function(params) {
var me = this
var keys = Object.keys(params || {})
var boxDiv
var iconSpan
var textSpan
var loading
var rotating
var angle = 0
// 默认选项
var options = {
iconArrow: "<img style='" + IMG_STYLE + "' src='" + IMG_ARROW + "'>",
iconLoading: "<img style='" + IMG_STYLE + "' src='" + IMG_LOADING + "'>",
iconFinish: "<img style='" + IMG_STYLE + "' src='" + IMG_FINISH + "'>",
textPull: '下拉刷新',
textRelease: '释放刷新',
textLoading: '正在加载',
textFinish: '刷新完成',
spinning: true,
// 应网友要求添加一个选项控制loading时是否旋转icon
refresh: null// 刷新自定义执行的函数
}
for (var k in keys) {
options[keys[k]] = params[keys[k]]
}
// 先行加载图片,触发浏览器缓存起来避免刷新时再加载显得不流畅
document.createElement('div').innerHTML = options.iconArrow + options.iconLoading + options.iconFinish
// 创建下拉的div
boxDiv = document.createElement('div')
boxDiv.className = 'roll-plugin-pulldown'
boxDiv.style.cssText = 'position:absolute;top:-44px;width:100%;height:44px;line-height:44px;font-size:16px;text-align:center;'
iconSpan = document.createElement('span')
iconSpan.className = 'roll-plugin-pulldown-icon'
iconSpan.style.cssText = 'display:inline-block;width:24px;height:24px;position:absolute;top:10px;left:25%;'
iconSpan.innerHTML = options.iconArrow
boxDiv.appendChild(iconSpan)
textSpan = document.createElement('span')
textSpan.className = 'roll-plugin-pulldown-text'
textSpan.innerHTML = options.textPull
boxDiv.appendChild(textSpan)
me.wrapper.appendChild(boxDiv)
// 监听滑动事件
me.on('scroll', function() {
boxDiv.style[TSF] = me.scroller.style[TSF]
// 达到一定位置显示释放刷新
if (!loading) {
if (me.y > 44) {
iconSpan.style[TSF] = 'rotateZ(180deg)'
textSpan.innerHTML = options.textRelease
} else {
iconSpan.style[TSF] = 'rotateZ(0deg)'
textSpan.innerHTML = options.textPull
}
}
})
me.on('touchEnd', function() {
if (me.y >= 44) {
// 超过44px禁止回弹
me.y = 0
me.options.momentum = false
// 刷新
iconSpan.style[TSF] = 'rotateZ(0deg)'
iconSpan.innerHTML = options.iconLoading
textSpan.innerHTML = options.textLoading
setTimeout(function() {
me.scrollTo(0, 44, 200, true, doRefresh).minScrollY = 44
moveTo(boxDiv, 0, 44, 200)
me.options.momentum = true
}, 10)
} else if (!loading) {
moveTo(boxDiv, 0, 0, 200)
}
})
// 执行刷新
function doRefresh() {
if (!loading) {
loading = true
rotating = true
angle = 0
if (options.spinning) {
makeRotate()
}
setTimeout(function() {
if (typeof options.refresh === 'function') {
options.refresh(function() {
// 完成刷新
rotating = false
iconSpan.innerHTML = options.iconFinish
textSpan.innerHTML = options.textFinish
// 收起刷新栏
setTimeout(function() {
moveTo(boxDiv, 0, 0, 200)
me.scrollTo(0, 0, 200, true, function() {
loading = false
iconSpan.innerHTML = options.iconArrow
textSpan.innerHTML = options.textPull
}).minScrollY = 0
}, 500)
})
}
}, 200)
}
}
// 使iconSpan旋转下来
function makeRotate() {
angle = angle + 6 >= 360 ? 0 : angle + 6
iconSpan.style[TSF] = 'rotateZ(' + angle + 'deg)'
if (rotating) {
rAF(makeRotate)
} else {
iconSpan.style[TSF] = 'rotateZ(0deg)'
}
}
return me
}
Roll.prototype.pulldown.version = '{{version}}'
// CommonJS/AMD/CMD规范导出Roll
if (typeof module !== 'undefined' && module.exports) {
module.exports = Roll
}
if (typeof define === 'function') {
define(function() {
return Roll
})
}
} )(window, document, Roll)