最近在做移动端h5页面,所以分页什么的就不能按照传统pc端的分页器的思维去做了,这么小的屏幕去点击也不太方便一般来讲移动端都是上拉加载更多,符合正常使用习惯。
首先简单写一下模板部分的html代码,,很简单清晰的逻辑:
<template> <div class="loadmore"> <div class="loadmore__body"> <slot></slot> </div> <div class="loadmore__footer"> <span v-if="loading"> <i class="tc-loading"></i> <span>正在加载</span> </span> <span v-else-if="loadable">上拉加载更多</span> <span v-else>没有更多了</span> </div> </div> </template>
然后就是业务部分了
在动手写组件之前,先理清需求:
加载页面 -> 滑到底部 -> 上拉一定距离 -> 加载第二页 -> 继续前面步骤 -> 没有更多
这是一个用户交互逻辑,而我们需要将其映射为代码逻辑:
首屏自动加载第一页 -> 滑动到底部&&按下时候滑动距离Y轴有一定偏移量 -> 请求后端加载第二页 -> 根据返回字段判断是否还有下一页
有了代码逻辑,主干就出来了,加载和判断由事件来控制,而又作为一个vue组件,我们需要配合vue生命周期来挂载事件和销毁事件
export default { mounted() { // 确定容器 // 容器绑定事件 }, beforeDestory() { // 解绑事件 }, }
如果没有解绑的话,每次你加载组件,就会绑定一次事件…
然后我们需要一些核心事件回调方法来在合适的时间加载数据渲染页面, 回想一下,第一我们需要http获取数据的load函数,然后我们需要三个绑定事件的回调函数pointDown(), pointMove(), pointUp(),分别对应用户按下、移动、弹起手指操作:
export default { ··· methods:{ /** * 加载一组数据的方法 */ load() { // 设置options this.$axios.request(options).then((res) => { // 获取数据后的处理 }).catch((e) => { // 异常处理 }) }, /** * 鼠标按下事件处理函数 * @param {Object} e - 事件对象 */ pointerdown(e) { // 获取按下的位置 this.pageY = e.changedTouches ? e.changedTouches[0].pageY : e.pageY }, /** * 鼠标移动事件处理函数 * @param {Object} e - 事件对象 */ pointermove(e) { const container = this.$container const pageY = e.changedTouches ? e.changedTouches[0].pageY : e.pageY const moveY = pageY - this.pageY // 如果已经向下滚动到页面最底部 if (moveY < 0 && (container.scrollTop + Math.min( global.innerHeight, container.clientHeight, )) >= container.scrollHeight) { // 阻止原生的上拉拖动会露出页面底部空白区域的行为(主要针对iOS版微信) e.preventDefault() // 如果上拉距离超过50像素,则加载下一页 if (moveY < -50) { this.pageY = pageY this.load() } } }, /** * 鼠标松开事件处理函数 */ pointerup() { // 这边就是取消拖动状态,需要注意在拖动过程中不要再次触发一些事件回调,否侧乱套 this.dragging = false }, }, ··· }
基本上主干已经算完工了,一些props传入或者一些逻辑控制细节需要再额外添加,贴出整个组件的源码:
<template> <div class="loadmore"> <!-- <div class="loadmore__header"></div> --> <div class="loadmore__body"> <slot></slot> </div> <div class="loadmore__footer"> <span v-if="loading"> <i class="tc-loading"></i> <span>正在加载</span> </span> <span v-else-if="loadable">上拉加载更多</span> <span v-else>没有更多了</span> </div> </div> </template> <script type="text/babel"> import axios from 'axios' const CancelToken = axios.CancelToken export default { data() { return { /** * 总页数(由服务端返回) * @type {number} */ count: 0, /** * 是否正在拖拽中 * @type {boolean} */ dragging: false, /** * 已加载次数 * @type {number} */ times: 0, /** * 已开始记载 * @type {boolean} */ started: false, /** * 正在加载中 * @type {boolean} */ loading: false, } }, props: { /** * 初始化后自动开始加载数据 */ autoload: { type: Boolean, default: true, }, /** * 离组件最近的可滚动父级元素(用于监听事件及获取滚动条位置) */ container: { // Selector or Element default: 'body', }, /** * 禁用组件 */ disabled: { type: Boolean, default: false, }, /** * Axios请求参数配置对象 * {@link https://github.com/mzabriskie/axios#request-config} */ options: { type: Object, default: null, }, /** * 起始页码 */ page: { type: Number, default: 1, }, /** * 每页加载数据条数 */ rows: { type: Number, default: 10, }, /** * 数据加载请求地址 */ url: { type: String, default: '', }, }, computed: { /** * 是否可以加载 * @returns {boolean} 是与否 */ loadable() { return !this.disabled && (!this.started || (this.page + this.times) <= this.count) }, }, mounted() { let container = this.container if (container) { if (typeof container === 'string') { container = document.querySelector(container) } else if (!container.querySelector) { container = document.body } } if (!container) { container = document.body } this.$container = container this.onPointerDown = this.pointerdown.bind(this) this.onPointerMove = this.pointermove.bind(this) this.onPointerUp = this.pointerup.bind(this) if (global.PointerEvent) { container.addEventListener('pointerdown', this.onPointerDown, false) container.addEventListener('pointermove', this.onPointerMove, false) container.addEventListener('pointerup', this.onPointerUp, false) container.addEventListener('pointercancel', this.onPointerUp, false) } else { container.addEventListener('touchstart', this.onPointerDown, false) container.addEventListener('touchmove', this.onPointerMove, false) container.addEventListener('touchend', this.onPointerUp, false) container.addEventListener('touchcancel', this.onPointerUp, false) container.addEventListener('mousedown', this.onPointerDown, false) container.addEventListener('mousemove', this.onPointerMove, false) container.addEventListener('mouseup', this.onPointerUp, false) } if (this.autoload) { this.load() } }, // eslint-disable-next-line beforeDestroy() { const container = this.$container if (global.PointerEvent) { container.removeEventListener('pointerdown', this.onPointerDown, false) container.removeEventListener('pointermove', this.onPointerMove, false) container.removeEventListener('pointerup', this.onPointerUp, false) container.removeEventListener('pointercancel', this.onPointerUp, false) } else { container.removeEventListener('touchstart', this.onPointerDown, false) container.removeEventListener('touchmove', this.onPointerMove, false) container.removeEventListener('touchend', this.onPointerUp, false) container.removeEventListener('touchcancel', this.onPointerUp, false) container.removeEventListener('mousedown', this.onPointerDown, false) container.removeEventListener('mousemove', this.onPointerMove, false) container.removeEventListener('mouseup', this.onPointerUp, false) } if (this.loading && this.cancel) { this.cancel() } }, methods: { /** * 加载一组数据的方法 */ load() { if (this.disabled || this.loading) { return } this.started = true this.loading = true const params = { currentPage: this.page + this.times, pageSize: this.rows, } const options = Object.assign({}, this.options, { url: this.url, cancelToken: new CancelToken((cancel) => { this.cancel = cancel }), }) if (String(options.method).toUpperCase() === 'POST') { options.data = Object.assign({}, options.data, params) } else { options.params = Object.assign({}, options.params, params) } this.$axios.request(options).then((res) => { const data = res.result this.times += 1 this.loading = false this.count = data.pageCount this.$emit('success', data.list) this.$emit('complete') }).catch((e) => { this.loading = false this.$emit('error', e) this.$emit('complete') }) }, /** * 重置加载相关变量 */ reset() { this.count = 0 this.times = 0 this.started = false this.loading = false }, /** *重新开始加载 */ restart() { this.reset() this.load() }, /** * 鼠标按下事件处理函数 * @param {Object} e - 事件对象 */ pointerdown(e) { if (this.disabled || !this.loadable || this.loading) { return } this.dragging = true this.pageY = e.changedTouches ? e.changedTouches[0].pageY : e.pageY }, /** * 鼠标移动事件处理函数 * @param {Object} e - 事件对象 */ pointermove(e) { if (!this.dragging) { return } const container = this.$container const pageY = e.changedTouches ? e.changedTouches[0].pageY : e.pageY const moveY = pageY - this.pageY // 如果已经向下滚动到页面最底部 if (moveY < 0 && (container.scrollTop + Math.min( global.innerHeight, container.clientHeight, )) >= container.scrollHeight) { // 阻止原生的上拉拖动会露出页面底部空白区域的行为(主要针对iOS版微信) e.preventDefault() // 如果上拉距离超过50像素,则加载下一页 if (moveY < -50) { this.pageY = pageY this.load() } } }, /** * 鼠标松开事件处理函数 */ pointerup() { this.dragging = false }, }, } </script>
以上所述是小编给大家介绍的vue loadmore组件上拉加载更多功能示例代码,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对小牛知识库网站的支持!
本文向大家介绍Android RecyclerView添加上拉加载更多功能,包括了Android RecyclerView添加上拉加载更多功能的使用技巧和注意事项,需要的朋友参考一下 上一篇文章已经介绍了如何为RecyclerView添加FootView,在此基础上,要添加分页加载的功能其实已经很简单了。 上一篇文章地址:为RecyclerView添加FootView和HeadView 效果:(源
本文向大家介绍Android ListView实现上拉加载更多和下拉刷新功能,包括了Android ListView实现上拉加载更多和下拉刷新功能的使用技巧和注意事项,需要的朋友参考一下 本文实例为大家介绍了Android ListView下拉刷新功能的实现方法和功能,供大家参考,具体内容如下 1、ListView优化方式 界面缓存:ViewHolder+convertView 分页加载:上拉刷新
本文向大家介绍jQuery实现下拉加载功能实例代码,包括了jQuery实现下拉加载功能实例代码的使用技巧和注意事项,需要的朋友参考一下 废话不多说了,直接给大家贴代码了,具体代码如下所示: 以上代码是小编给大家分享的jQuery实现下拉加载功能实例代码,希望对大家有所帮助!
本文向大家介绍jQuery实现的上拉刷新功能组件示例,包括了jQuery实现的上拉刷新功能组件示例的使用技巧和注意事项,需要的朋友参考一下 本文实例讲述了jQuery实现的上拉刷新功能组件。分享给大家供大家参考,具体如下: 技术要点: 1、jQuery的插件写法 2、上拉刷新步骤分解 3、css样式 jQuery的插件写法: 上拉刷新步骤分解: 上拉刷新可以分解成三个部分:一是开始(start),
本文向大家介绍Android实现上拉加载更多以及下拉刷新功能(ListView),包括了Android实现上拉加载更多以及下拉刷新功能(ListView)的使用技巧和注意事项,需要的朋友参考一下 首先为大家介绍Andorid5.0原生下拉刷新简单实现。 先上效果图; 相对于上一个19.1.0版本中的横条效果好看了很多。使用起来也很简单。 几行代码就可以实现下拉刷新,效果也还不错,不用引入第三方ja
本文向大家介绍vue2.0 移动端实现下拉刷新和上拉加载更多的示例,包括了vue2.0 移动端实现下拉刷新和上拉加载更多的示例的使用技巧和注意事项,需要的朋友参考一下 本人正在基于 vue2.0 + webpack + es6 搭建前端架构,整理了部分插件,下面这个是下拉更新 上拉更多的,挺好用的,分享给大家。 直接上代码,不懂的多看几遍,下面我换会告诉大家如何使用。 把上面组件拷贝一下,存成后缀