本文转自;http://blog.csdn.net/michael_ouyang/article/details/56846185

loadmore

加载更多(分页加载)

当用户打开一个页面时,假设后台数据量庞大时,一次性地返回所有数据给客户端,页面的打开速度就会有所下降,而且用户只看上面的内容而不需要看后面的内容时,也浪费用户流量,基于优化的角度来考虑,后台不要一次性返回所有数据,当用户有需要再往下翻的时候,再加载更加数据出来。

业务需求:

列表滚动到底部时,继续往上拉,加载更多内容

必备参数:

(1)pageindex: 1 //第几次加载

(2)callbackcount: 15 //需要返回数据的个数

其他参数:

根据接口的所需参数

实现原理:

当第一次访问接口时,传递2个必备参数(第1次加载,需要返回数据的个数为15个),和其他参数(需要搜索的字符串)给后台,后台返回第一次数据过来。在请求成功的的回调函数中,判断返回的数据是否>0,是,则取出数据,渲染视图层,并把“上拉加载”显示在列表底部;否,则没有数据可取,并把“没有更多”显示在列表底部,同时把“上拉加载”隐藏掉。

当用户已经滚动到列表底部(这里使用到小程序提供的scroll-view组件的bindscrolltolower事件),触发bindscrolltolower事件,参数pageindex+1,再把2个必备参数(第2次加载,需要返回数据的个数为15个)和其他参数(需要搜索的字符串)给后台,后台把其余的数据返回给前台,前台在原来数据的基础上添加数据。

示例:

wxml:

  1. <view class="search">
  2. <view class="search-bar">
  3. <view class="search-wrap">
  4. <icon type="search" size="16" class="icon-search" />
  5. <input type="text" placeholder="请输入搜索内容" class="search-input" name="searchKeyword" bindinput="bindKeywordInput" value="{{searchKeyword}}" />
  6. </view>
  7. <view class="search-cancel" bindtap="keywordSearch">搜索</view>
  8. </view>
  9. <view class="search-result">
  10. <scroll-view scroll-y="true" bindscrolltolower="searchScrollLower">
  11. <view class="result-item" wx:for="{{searchSongList}}" wx:key="unique"  data-data="{{item}}" >
  12. <view class="icon{{item.isonly=='0' ? ' nocopyright' : ''}}"></view>
  13. <text class="title">{{item.songname}}</text>
  14. <view class="subtitle">
  15. <text wx:for="{{item.singer}}" wx:key="unique">{{item.name}}</text>
  16. </view>
  17. </view>
  18. <view class="loading" hidden="{{!searchLoading}}">正在载入更多...</view>
  19. <view class="loading complete" hidden="{{!searchLoadingComplete}}">已加载全部</view>
  20. </scroll-view>
  21. </view>
  22. </view>
<view class="search">
<view class="search-bar">
<view class="search-wrap">
<icon type="search" size="16" class="icon-search" />
<input type="text" placeholder="请输入搜索内容" class="search-input" name="searchKeyword" bindinput="bindKeywordInput" value="{{searchKeyword}}" />
</view>
<view class="search-cancel" bindtap="keywordSearch">搜索</view>
</view>
<view class="search-result">
<scroll-view scroll-y="true" bindscrolltolower="searchScrollLower">
<view class="result-item" wx:for="{{searchSongList}}" wx:key="unique" data-data="{{item}}" >
<view class="icon{{item.isonly=='0' ? ' nocopyright' : ''}}"></view>
<text class="title">{{item.songname}}</text>
<view class="subtitle">
<text wx:for="{{item.singer}}" wx:key="unique">{{item.name}}</text>
</view>
</view>
<view class="loading" hidden="{{!searchLoading}}">正在载入更多...</view>
<view class="loading complete" hidden="{{!searchLoadingComplete}}">已加载全部</view>
</scroll-view>
</view>
</view>

js:

  1. var util = require('../../utils/util.js')
  2. Page({
  3. data: {
  4. searchKeyword: '',  //需要搜索的字符
  5. searchSongList: [], //放置返回数据的数组
  6. isFromSearch: true,   // 用于判断searchSongList数组是不是空数组,默认true,空的数组
  7. searchPageNum: 1,   // 设置加载的第几次,默认是第一次
  8. callbackcount: 15,      //返回数据的个数
  9. searchLoading: false, //"上拉加载"的变量,默认false,隐藏
  10. searchLoadingComplete: false  //“没有数据”的变量,默认false,隐藏
  11. },
  12. //输入框事件,每输入一个字符,就会触发一次
  13. bindKeywordInput: function(e){
  14. console.log("输入框事件")
  15. this.setData({
  16. searchKeyword: e.detail.value
  17. })
  18. },
  19. //搜索,访问网络
  20. fetchSearchList: function(){
  21. let that = this;
  22. let searchKeyword = that.data.searchKeyword,//输入框字符串作为参数
  23. searchPageNum = that.data.searchPageNum,//把第几次加载次数作为参数
  24. callbackcount =that.data.callbackcount; //返回数据的个数
  25. //访问网络
  26. util.getSearchMusic(searchKeyword, searchPageNum,callbackcount, function(data){
  27. console.log(data)
  28. //判断是否有数据,有则取数据
  29. if(data.data.song.curnum != 0){
  30. let searchList = [];
  31. //如果isFromSearch是true从data中取出数据,否则先从原来的数据继续添加
  32. that.data.isFromSearch ? searchList=data.data.song.list : searchList=that.data.searchSongList.concat(data.data.song.list)
  33. that.setData({
  34. searchSongList: searchList, //获取数据数组
  35. zhida: data.data.zhida, //存放歌手属性的对象
  36. searchLoading: true   //把"上拉加载"的变量设为false,显示
  37. });
  38. //没有数据了,把“没有数据”显示,把“上拉加载”隐藏
  39. }else{
  40. that.setData({
  41. searchLoadingComplete: true, //把“没有数据”设为true,显示
  42. searchLoading: false  //把"上拉加载"的变量设为false,隐藏
  43. });
  44. }
  45. })
  46. },
  47. //点击搜索按钮,触发事件
  48. keywordSearch: function(e){
  49. this.setData({
  50. searchPageNum: 1,   //第一次加载,设置1
  51. searchSongList:[],  //放置返回数据的数组,设为空
  52. isFromSearch: true,  //第一次加载,设置true
  53. searchLoading: true,  //把"上拉加载"的变量设为true,显示
  54. searchLoadingComplete:false //把“没有数据”设为false,隐藏
  55. })
  56. this.fetchSearchList();
  57. },
  58. //滚动到底部触发事件
  59. searchScrollLower: function(){
  60. let that = this;
  61. if(that.data.searchLoading && !that.data.searchLoadingComplete){
  62. that.setData({
  63. searchPageNum: that.data.searchPageNum+1,  //每次触发上拉事件,把searchPageNum+1
  64. isFromSearch: false  //触发到上拉事件,把isFromSearch设为为false
  65. });
  66. that.fetchSearchList();
  67. }
  68. }
  69. })
var util = require('../../utils/util.js')
Page({
data: {
searchKeyword: '', //需要搜索的字符
searchSongList: [], //放置返回数据的数组
isFromSearch: true, // 用于判断searchSongList数组是不是空数组,默认true,空的数组
searchPageNum: 1, // 设置加载的第几次,默认是第一次
callbackcount: 15, //返回数据的个数
searchLoading: false, //"上拉加载"的变量,默认false,隐藏
searchLoadingComplete: false //“没有数据”的变量,默认false,隐藏
},
//输入框事件,每输入一个字符,就会触发一次
bindKeywordInput: function(e){
console.log("输入框事件")
this.setData({
searchKeyword: e.detail.value
})
},
//搜索,访问网络
fetchSearchList: function(){
let that = this;
let searchKeyword = that.data.searchKeyword,//输入框字符串作为参数
searchPageNum = that.data.searchPageNum,//把第几次加载次数作为参数
callbackcount =that.data.callbackcount; //返回数据的个数
//访问网络
util.getSearchMusic(searchKeyword, searchPageNum,callbackcount, function(data){
console.log(data)
//判断是否有数据,有则取数据
if(data.data.song.curnum != 0){
let searchList = [];
//如果isFromSearch是true从data中取出数据,否则先从原来的数据继续添加
that.data.isFromSearch ? searchList=data.data.song.list : searchList=that.data.searchSongList.concat(data.data.song.list)
that.setData({
searchSongList: searchList, //获取数据数组
zhida: data.data.zhida, //存放歌手属性的对象
searchLoading: true //把"上拉加载"的变量设为false,显示
});
//没有数据了,把“没有数据”显示,把“上拉加载”隐藏
}else{
that.setData({
searchLoadingComplete: true, //把“没有数据”设为true,显示
searchLoading: false //把"上拉加载"的变量设为false,隐藏
});
}
})
},
//点击搜索按钮,触发事件
keywordSearch: function(e){
this.setData({
searchPageNum: 1, //第一次加载,设置1
searchSongList:[], //放置返回数据的数组,设为空
isFromSearch: true, //第一次加载,设置true
searchLoading: true, //把"上拉加载"的变量设为true,显示
searchLoadingComplete:false //把“没有数据”设为false,隐藏
})
this.fetchSearchList();
},
//滚动到底部触发事件
searchScrollLower: function(){
let that = this;
if(that.data.searchLoading && !that.data.searchLoadingComplete){
that.setData({
searchPageNum: that.data.searchPageNum+1, //每次触发上拉事件,把searchPageNum+1
isFromSearch: false //触发到上拉事件,把isFromSearch设为为false
});
that.fetchSearchList();
}
}
})

util.js:

  1. function getSearchMusic(keyword, pageindex, callbackcount, callback){
  2. wx.request({
  3. url: 'https://c.y.qq.com/soso/fcgi-bin/search_for_qq_cp',
  4. data: {
  5. g_tk: 5381,
  6. uin: 0,
  7. format: 'json',
  8. inCharset: 'utf-8',
  9. outCharset: 'utf-8',
  10. notice: 0,
  11. platform: 'h5',
  12. needNewCode: 1,
  13. w: keyword,
  14. zhidaqu: 1,
  15. catZhida: 1,
  16. t: 0,
  17. flag: 1,
  18. ie: 'utf-8',
  19. sem: 1,
  20. aggr: 0,
  21. perpage: 20,
  22. n: callbackcount,  //返回数据的个数
  23. p: pageindex,
  24. remoteplace: 'txt.mqq.all',
  25. _: Date.now()
  26. },
  27. method: 'GET',
  28. header: {'content-Type': 'application/json'},
  29. success: function(res){
  30. if(res.statusCode == 200){
  31. callback(res.data);
  32. }
  33. }
  34. })
  35. }
  36. module.exports = {
  37. getSearchMusic: getSearchMusic
  38. }
function getSearchMusic(keyword, pageindex, callbackcount, callback){
wx.request({
url: 'https://c.y.qq.com/soso/fcgi-bin/search_for_qq_cp',
data: {
g_tk: 5381,
uin: 0,
format: 'json',
inCharset: 'utf-8',
outCharset: 'utf-8',
notice: 0,
platform: 'h5',
needNewCode: 1,
w: keyword,
zhidaqu: 1,
catZhida: 1,
t: 0,
flag: 1,
ie: 'utf-8',
sem: 1,
aggr: 0,
perpage: 20,
n: callbackcount, //返回数据的个数
p: pageindex,
remoteplace: 'txt.mqq.all',
_: Date.now()
},
method: 'GET',
header: {'content-Type': 'application/json'},
success: function(res){
if(res.statusCode == 200){
callback(res.data);
}
}
})
} module.exports = {
getSearchMusic: getSearchMusic
}

wxss:

  1. page{
  2. display: flex;
  3. flex-direction: column;
  4. height: 100%;
  5. }
  6. /*搜索*/
  7. .search{
  8. flex: auto;
  9. display: flex;
  10. flex-direction: column;
  11. background: #fff;
  12. }
  13. .search-bar{
  14. flex: none;
  15. display: flex;
  16. align-items: center;
  17. justify-content: space-between;
  18. padding: 20rpx;
  19. background: #f4f4f4;
  20. }
  21. .search-wrap{
  22. position: relative;
  23. flex: auto;
  24. display: flex;
  25. align-items: center;
  26. height: 80rpx;
  27. padding: 0 20rpx;
  28. background: #fff;
  29. border-radius: 6rpx;
  30. }
  31. .search-wrap .icon-search{
  32. margin-right: 10rpx;
  33. }
  34. .search-wrap .search-input{
  35. flex: auto;
  36. font-size: 28rpx;
  37. }
  38. .search-cancel{
  39. padding: 0 20rpx;
  40. font-size: 28rpx;
  41. }
  42. /*搜索结果*/
  43. .search-result{
  44. flex: auto;
  45. position: relative;
  46. }
  47. .search-result scroll-view{
  48. position: absolute;
  49. bottom: 0;
  50. left: 0;
  51. right: 0;
  52. top: 0;
  53. }
  54. .result-item{
  55. position: relative;
  56. display: flex;
  57. flex-direction: column;
  58. padding: 20rpx 0 20rpx 110rpx;
  59. overflow: hidden;
  60. border-bottom: 2rpx solid #e5e5e5;
  61. }
  62. .result-item .media{
  63. position: absolute;
  64. left: 16rpx;
  65. top: 16rpx;
  66. width: 80rpx;
  67. height: 80rpx;
  68. border-radius: 999rpx;
  69. }
  70. .result-item .title,
  71. .result-item .subtitle{
  72. overflow: hidden;
  73. text-overflow: ellipsis;
  74. white-space: nowrap;
  75. line-height: 36rpx;
  76. }
  77. .result-item .title{
  78. margin-bottom: 4rpx;
  79. color: #000;
  80. }
  81. .result-item .subtitle{
  82. color: #808080;
  83. font-size: 24rpx;
  84. }
  85. .result-item:first-child .subtitle text{
  86. margin-right: 20rpx;
  87. }
  88. .result-item:not(:first-child) .subtitle text:not(:first-child):before{
  89. content: '/';
  90. margin: 0 8rpx;
  91. }
  92. .loading{
  93. padding: 10rpx;
  94. text-align: center;
  95. }
  96. .loading:before{
  97. display: inline-block;
  98. margin-right: 5rpx;
  99. vertical-align: middle;
  100. content: '';
  101. width: 40rpx;
  102. height: 40rpx;
  103. background: url(../../images/icon-loading.png) no-repeat;
  104. background-size: contain;
  105. animation: rotate 1s linear infinite;
  106. }
  107. .loading.complete:before{
  108. display: none;
  109. }
page{
display: flex;
flex-direction: column;
height: 100%;
} /*搜索*/
.search{
flex: auto;
display: flex;
flex-direction: column;
background: #fff;
}
.search-bar{
flex: none;
display: flex;
align-items: center;
justify-content: space-between;
padding: 20rpx;
background: #f4f4f4;
}
.search-wrap{
position: relative;
flex: auto;
display: flex;
align-items: center;
height: 80rpx;
padding: 0 20rpx;
background: #fff;
border-radius: 6rpx;
}
.search-wrap .icon-search{
margin-right: 10rpx;
}
.search-wrap .search-input{
flex: auto;
font-size: 28rpx;
}
.search-cancel{
padding: 0 20rpx;
font-size: 28rpx;
} /*搜索结果*/
.search-result{
flex: auto;
position: relative;
}
.search-result scroll-view{
position: absolute;
bottom: 0;
left: 0;
right: 0;
top: 0;
}
.result-item{
position: relative;
display: flex;
flex-direction: column;
padding: 20rpx 0 20rpx 110rpx;
overflow: hidden;
border-bottom: 2rpx solid #e5e5e5;
} .result-item .media{
position: absolute;
left: 16rpx;
top: 16rpx;
width: 80rpx;
height: 80rpx;
border-radius: 999rpx;
}
.result-item .title,
.result-item .subtitle{
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
line-height: 36rpx;
}
.result-item .title{
margin-bottom: 4rpx;
color: #000;
}
.result-item .subtitle{
color: #808080;
font-size: 24rpx;
}
.result-item:first-child .subtitle text{
margin-right: 20rpx;
}
.result-item:not(:first-child) .subtitle text:not(:first-child):before{
content: '/';
margin: 0 8rpx;
}
.loading{
padding: 10rpx;
text-align: center;
}
.loading:before{
display: inline-block;
margin-right: 5rpx;
vertical-align: middle;
content: '';
width: 40rpx;
height: 40rpx;
background: url(../../images/icon-loading.png) no-repeat;
background-size: contain;
animation: rotate 1s linear infinite;
}
.loading.complete:before{
display: none;
}

运行:

微信小程序教程系列

相关连接:http://blog.csdn.net/michael_ouyang/article/details/54700871

基础篇

------------------------------------------------------------

微信开发者工具的快捷键

微信小程序的文件结构  —— 微信小程序教程系列(1)

微信小程序的生命周期实例演示 ——  微信小程序教程系列(2)

微信小程序的动态修改视图层的数据 ——  微信小程序教程系列(3)

微信小程序的新建页面 —— 微信小程序教程系列(4)

微信小程序的如何使用全局属性 ——  微信小程序教程系列(5)

微信小程序的页面跳转 —— 微信小程序教程系列(6)

微信小程序标题栏和导航栏的设置 ——  微信小程序教程系列(7)

微信小程序的作用域和模块化 —— 微信小程序教程系列(8)

微信小程序视图层的数据绑定 —— 微信小程序教程系列(9)

微信小程序视图层的条件渲染 —— 微信小程序教程系列(10)

微信小程序视图层的列表渲染 —— 微信小程序教程系列(11)

微信小程序视图层的模板 —— 微信小程序教程系列(12)

微信小程序之wxss  —— 微信小程序教程系列(13)

微信小程序的网络请求 —— 微信小程序教程系列(14)

微信小程序的百度地图获取地理位置 ——  微信小程序教程系列(15)

微信小程序使用百度api获取天气信息  —— 微信小程序教程系列(16)

微信小程序获取系统日期和时间 ——  微信小程序教程系列(17)

微信小程序之上拉加载和下拉刷新 ——  微信小程序教程系列(18)

微信小程序之组件 —— 微信小程序教程系列(19)

微信小程序之微信登陆 —— 微信小程序教程系列(20)

实战篇

------------------------------------------------------------

微信小程序之顶部导航栏实例 —— 微信小程序实战系列(1)

微信小程序之上拉加载(分页加载)实例  —— 微信小程序实战系列(2)

微信小程序之轮播图实例 —— 微信小程序实战系列(3)

微信小程序之仿android fragment之可滑动的底部导航栏实例  —— 微信小程序实战系列(4)

微信小程序之登录页实例 —— 微信小程序实战系列(5)

微信小程序之自定义toast实例 ——  微信小程序实战系列(6)

微信小程序之自定义抽屉菜单(从下拉出)实例  —— 微信小程序实战系列(7)

微信小程序之自定义模态弹窗(带动画)实例  —— 微信小程序实战系列(8)

电商篇

------------------------------------------------------------

微信小程序之侧栏分类  —— 微信小程序实战商城系列(1)

微信小程序之仿淘宝分类入口 —— 微信小程序实战商城系列(2)

微信小程序之购物数量加减 —— 微信小程序实战商城系列(3)

微信小程序之商品属性分类 —— 微信小程序实战商城系列(4)

微信小程序之购物车 —— 微信小程序实战商城系列(5)

未完待续。。。

更多小程序的教程:http://blog.csdn.net/column/details/14653.html

谢谢观看,不足之处,敬请指导

 

附:项目下载地址 http://download.csdn.net/detail/michael_ouyang/9773794

[转]微信小程序之加载更多(分页加载)实例 —— 微信小程序实战系列(2)的更多相关文章

  1. 微信小程序云开发-列表数据分页加载显示

    一.准备工作 1.创建数据库nums,向数据库中导入108条数据 2.修改数据库表nums的权限 二.新建页面ListPaginated 1.wxml文件 <!-- 显示列表数据 --> ...

  2. RecyclerView 判断滑到底部 顶部 预加载 更多 分页 MD

    Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱 MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina ...

  3. mui 上拉加载 实现分页加载功能

    mui 上拉加载 实现分页加载功能,效果图: 分页功能(上拉加载): 1.引入需要的css.js文件 <link href="static/css/mui.css" rel= ...

  4. 微信小程序实战篇-下拉刷新与加载更多

    下拉刷新 实现下拉刷新目前能想到的有两种方式 1. 调用系统的API,系统有提供下拉刷新的API接口 2. 监听scroll-view,自定义下拉刷新,还记得scroll-view里面有一个binds ...

  5. js实现『加载更多』功能实例

    DEMO : 滚动加载示例 关于如何实现『加载更多』功能,网上有插件可用,例如比较著名的使用iscroll.js实现的上拉加载更多.下拉刷新功能. 但实际用起来却是很麻烦.由于是第三方插件,要按照对方 ...

  6. google官方的下拉刷新+自定义上拉加载更多

    转载请标注转载:http://blog.csdn.net/oqihaogongyuan/article/details/50949118 google官方的下拉刷新+自定义上拉加载更多 现在很多app ...

  7. MyRecycleView带有上拉加载更多

    package com.gan.myrecycleview; import android.content.Context; import android.support.v4.widget.Swip ...

  8. Android之RecyclerView轻松实现下拉刷新和加载更多

    今天研究了下RecyclerView的滑动事件,特别是下拉刷新和加载更多事件,在现在几乎所有的APP显示数据列表时都用到了.自定义RecyclerView下拉刷新和加载更多听上去很复杂,实际上并不难, ...

  9. 微信小程序开发之 下拉刷新,上拉加载更多

    本文记载了如何在微信小程序里面实现下拉刷新,上拉加载更多 先开看一下界面 大致如此的界面吧. 这个Demo使用了微信的几个Api和事件,我先列出来. 1.wx.request (获取远程服务器的数据, ...

随机推荐

  1. WebAPI开发中的定时处理

    https://blog.csdn.net/lordwish/article/details/77931897

  2. Linq to SQL 参考Demo

    LINQ to SQL语句()之Where Where操作 适用场景:实现过滤,查询等功能. 说明:与SQL命令中的Where作用相似,都是起到范围限定也就是过滤作用的,而判断条件就是它后面所接的子句 ...

  3. windows下MySQL的安装(非安装包)

    命令代码 "C:\Program Files\MySQL\MySQL Server 5.6\bin\mysqld.exe" --install MySQL56_3308 --def ...

  4. 【已解决】wepy中使用分包加载报错

    问题: "xxx.js 出现脚本错误后者未正确调用Page()"       最近看小程序启动时间(性能监控),启动时间比较长,所以考虑使用分包加载.       但在使用过程中遇 ...

  5. Swift 里字符串(八)UnicodeScalarView

    即以 Unicode Scarlar 的方式来查看字符串. /// let flag = "

  6. C#获取文件版本、文件大小等信息

    使用以下C#程序代码可以非常方便地获取Windows系统中任意一个文件(尤其是可执行文件)的文件版本.文件大小.版权.产品名称等信息.所获取到的信息类似于在Windows操作系统中右键点击该文件,然后 ...

  7. Android UiAutomator UiDevice API

    UiDevice为单例模式 1.获取设备 static UiDevice getInstance() This method is deprecated. Should use getInstance ...

  8. ajax--底层代码

    ajax:Asynchronous JavaScript And XML,异步的js与XML.ajax并不是一种新的编程语言,而是一种使用现有标准的新方法.ajax能够在不重载整个网页的情况下与服务器 ...

  9. 3. C++ POD类型

    POD全称Plain Old Data,通常用于说明1个类型的属性.通俗的讲,一个类或结构体通过二进制拷贝后还能保持其数据不变,那么它就是一个POD类型. C++11将POD划分为2个基本概念的合集, ...

  10. Opserver 初探三《服务器数据监控》

    用Opserver 怎么像zabbix一样监控服务器呢,查看github官方说明,Opserver可用于连接任何支持Bosun, Orion, or direct WMI监控数据. Opserver ...