最近小程序做到一个很常见的需求,订单列表功能。虽说看着很常见,做起来也发现了有一些坑。效果如图:
这里就是顶部tab页切换到不同状态的订单,下次切回来不刷新页面而且需要跳到上次浏览的地方,然后滑动每个页面对应的tab也能完成相应的切换。小程序的话很容易就想到swiper组件,然后滑动的当前swiper-item和tab的index对应起来就可以了。而且页面滑动页有对应的scroll-view组件使用,最开始确实是用的scroll-view,而且这个组件里面还有自带的bindscroll和srcollTop属性,第一版做完后发现小程序的scroll-view会导致自己的下拉刷新onPullDownRefresh失效,而且开始设想的每次记录当前tab的页面滚动高度然后赋值给scroll-view的srcollTop属性,从而达到再切换回来的时候滚动到上次的位置,貌似也是不行的,既然scroll-view达不到滚动到上次记录的位置,那就还是换成普通的view吧。
闲话到此,直接上代码,WXML代码:
搜索
{
{item.columnName}}
{
{orderItem.orderTimeStr}}
{
{orderItem.orderStateStr}}
{
{orderItem.prodName}}
{
{orderItem.prodSpec}}
{
{orderItem.manufacture}}
共{
{orderItem.prodNum}}件
¥ {
{orderItem.orderPriceStr}}
约返奖励金{
{orderItem.yjTotalReward}}元
约返奖励金{
{orderItem.yjTotalReward}}元,已领{
{orderItem.getReward}}元
已领奖励金{
{orderItem.getReward}}元
待领奖励金{
{orderItem.canGetReward}}元
确认收货
查看物流
再次购买
领奖励金
去支付
{
{endTip}}
您还没有相关订单
—— 没有更多了 ——
可以看到swiper由于有一个默认的高度,所以需要在每个页签加载的时候计算出当前页签页面的高度,所以就有了style="height:{ {orderList[curListId].swiperHeight}}rpx;",由于我们的小程序要做新老版本的兼容,所以又有了一个showOrderVer的判断,老版本的话就是直接web-view嵌的H5页面,新版本的话是小程序原生的。
JS代码:
// pages/ordes/ordes.js
import http from "../../util/http";
import {config} from "../../config/envConfig";
Page({
/**
* 页面的初始数据
*/
data: {
orderVersion:'',//登录获得的版本号
showOrderVer:'',//显示订单版本
imgUrl:config.imageUrl,
//订单状态 默认全部 , 待支付-11,待出库-0,待收货-5,已完成-6
tabNavList: [{
"columnName": '全部',
"searchState": ""
}, {
"columnName": '待支付',
"searchState": "11"
}, {
"columnName": '待出库',
"searchState": "0"
}, {
"columnName": '待收货',
"searchState": "5"
}, {
"columnName": '完成',
"searchState": "6"
}],
navPosition: [],
orderList: {}, //订单列表对象,存放5个状态的订单
curListId: '', //当前选中状态的id,默认全部 , 待支付-11,待出库-0,待收货-5,已完成-6
endTipHidden: false,
endTip: '正在加载',
current: 0,
scrollLeft: 0,
px2rpx: 2,
winWidth: 375,
winHeight:0,
scrollTop: 0,
loadFlag:false,
isShowconfirmReceipt:false,//确认收货弹出框
confirmReceiptContent:['请收到商品后,再确认收货'],
orderCode:'',//确认收货弹出框按钮需要的参数
isShowRewardPrice:false,//预估可领奖励金弹出框
rewardPriceContent:['1.因税率和药监特殊要求,订单可能会被拆单发货,奖励金按拆单后的包裹计算。','2.若订单商品出库被冲红,实际领取奖励金以出库商品为准。'],
isShowMyReward:false,//奖励金入账提示弹出框
myRewardContent:'',
},
onShow(){
if(this.data.showOrderVer!=this.data.orderVersion){
//获得订单版本
this.getOrderVer();
}
},
/**
* 生命周期函数--监听页面加载
*/
onLoad: function(options) {
getApp().getGlobalUserInfo(true).then(res=>{
this.TOKEN=res.token;
this.setData({
current:options.index || 0, //当前tab页的index
orderVersion:res.orderVersion,//登录信息里的版本号
});
//获得订单版本
this.getOrderVer();
}).catch(err=>{
console.error(err)
})
},
//获得订单显示版本号
getOrderVer(){
http.fetch(http.GET_ORDER_VER).then(res => {
if(res.status == 1){
this.setData({
showOrderVer:res.data.version
})
if(res.data.version == 2){
this.setData({
orderList:{}
});
this.getSystem();
this.getNav(this.data.current);
}else{
this.setData({
url:config.page+'/Orders?TOKEN='+this.TOKEN+'&t='+Math.random()
})
}
}
}).catch((err) => {
})
},
//刷新页面
refreshPage:function(){
this.setData({
orderList:{}
});
var cid = this.data.curListId;
this.getOrderList(cid);
},
/**
* 页面上拉触底事件的处理函数
*/
onReachBottom: function () {
this.loadMoreList();
},
//下拉刷新
onPullDownRefresh: function() {
wx.showNavigationBarLoading()
this.refreshPage();
setTimeout(() => {
wx.hideNavigationBarLoading()
wx.stopPullDownRefresh()
}, 2000);
},
//跳转搜索页面
gotoOrderSearch(){
wx.navigateTo({
url: `/pages/orders/orderSearch/orderSearch`
});
},
timer: null,
getSystem: function() {
var self = this;
wx.getSystemInfo({
success: function(res) {
self.setData({
winWidth: res.windowWidth,
winHeight:res.windowHeight,
px2rpx: 750 / res.windowWidth
})
},
});
},
//获取导航和节点
getNav: function(index) {
var self = this;
self.setData({
current: index
});
//获取导航的初始位置
const query = wx.createSelectorQuery()
query.selectAll('.toc').boundingClientRect();
query.exec(function(res) {
self.setData({
navPosition: res[0]
})
if (index >= 4) {
self.setData({
scrollLeft: res[0][index].left
})
}
})
self.getOrderList(self.data.tabNavList[index].searchState);
},
//滑到底部加载更多
loadMoreList: function() {
var orderList = this.data.orderList;
var cid = this.data.curListId;
if (orderList[cid].page < orderList[cid].pageCount && !this.data.loadFlag) {
this.getOrderList(cid);
}
},
//请求列表
getOrderList: function(cid) {
var self = this;
this.setData({
curListId: cid,
loadFlag:true
})
var orderList = this.data.orderList;
if (!orderList[cid]) {
http.fetch(http.ORDER_LIST + `?searchState=${cid}&page=1`).then(res => {
var obj = {};
obj.page = res.data.page;
obj.pageCount = Math.ceil(res.data.total / res.data.rows);
obj.total = res.data.total;
obj.data = res.data.orderList;
if(res.data.total){
obj.swiperHeight = res.data.total > res.data.rows * res.data.page ? res.data.rows * res.data.page * 400 : res.data.total * 400;
}else{
obj.swiperHeight = 2*self.data.winHeight-170;
}
if (res.data.page * res.data.rows >= res.data.total) {
obj.endTip = '没有更多了';
obj.endTipHidden = true;
} else {
obj.endTip = '正在加载';
}
orderList[cid] = obj;
self.setData({
loadFlag:false,
orderList
})
}).catch((err) => {
})
} else {
if (orderList[cid].page < orderList[cid].pageCount) {
http.fetch(http.ORDER_LIST + `?searchState=${cid}&page=${orderList[cid].page+1}`,{
urlType: 'wap'
}).then(res => {
var obj = {};
obj.page = res.data.page;
obj.pageCount = Math.ceil(res.data.total / res.data.rows);
obj.total = res.data.total;
obj.data = orderList[cid].data.concat(res.data.orderList);
if (res.data.page * res.data.rows >= res.data.total) {
obj.endTip = '没有更多了';
obj.endTipHidden = true;
} else {
obj.endTip = '正在加载';
}
obj.swiperHeight = res.data.total > res.data.rows * res.data.page ? res.data.rows * res.data.page * 400 : res.data.total * 400 ;
orderList[cid] = obj;
self.setData({
loadFlag:false,
orderList
})
}).catch((err) => {
})
}
}
},
//切换导航(包含滑动swiper和切换导航跳转)
switchNav: function(index) {
if (index && this.data.current == index) return;
console.log('switchtab');
var cid = this.data.tabNavList[index].searchState;
var self = this;
var scrollLeft = 0;
if (self.data.navPosition[index].right * self.data.px2rpx + 62 >= 750) {
scrollLeft = self.data.navPosition[index].left;
}
var orderList = self.data.orderList;
var scrollTop = 0;
if (orderList[cid] && orderList[cid].scrollTop) {
scrollTop = orderList[cid].scrollTop
}
if (!orderList[cid]) {
clearTimeout(self.timer);
self.timer = null;
self.timer = setTimeout(function() {
self.setData({
curListId: cid,
current: index,
scrollLeft,
scrollTop
})
self.getOrderList(cid);
}, 500);
} else {
self.setData({
curListId: cid,
current: index,
scrollLeft,
scrollTop
})
}
//滚动到滚动条的位置
this.gotoScrollTop();
},
//获取滚动条位置
onPageScroll(e){
var self = this;
setTimeout(function() {
var orderList = self.data.orderList;
if (orderList[self.data.curListId]) {
orderList[self.data.curListId].scrollTop = e.scrollTop;
self.setData({
orderList
})
}
}, 300);
},
//滚动到滚动条的位置
gotoScrollTop(){
var self = this;
var orderList = self.data.orderList;
if (orderList[self.data.curListId]) {
wx.pageScrollTo({
scrollTop:orderList[self.data.curListId].scrollTop,
duration: 300
});
}
},
//切换导航
changeTab: function(e) {
console.log('changetab');
var index = e.currentTarget.dataset.index;
this.switchNav(index);
},
//滑动swiper
bindchange: function(e) {
console.log('changeswiper');
var index = e.detail.current;
//加上这个避免swiper过程,swiper-item会发生滑动混乱,滑动过快就会一直在闪动,新的API属性,touch说明是用户接触滑动,而不是自动滑动
if (e.detail.source && e.detail.source =='touch'){
this.switchNav(index);
}
},
//预估可领奖励金弹出框
rewardPricePop(){
//隐藏底部tabbar
wx.hideTabBar();
this.setData({
isShowRewardPrice: true
});
},
//确认收货弹框
confirmReceiptPop(event){
//隐藏底部tabbar
wx.hideTabBar();
this.setData({
isShowconfirmReceipt: true,
orderCode:event.currentTarget.dataset.ordercode
});
},
//点击确认收货按钮
onConfirmReceipt(event){
var self =this;
wx.showLoading();
http.fetch(http.CONFIRM_RECEIPT,{
commitId:event.currentTarget.dataset.ordercode
}).then(res => {
wx.hideLoading();
if(res.status == 1){
wx.showToast({
title:"收货成功",
icon: 'success',//图标,支持"success"、"loading"
complete:function(){
var cid = self.data.curListId;
self.setData({
orderList:{}
});
self.getOrderList(cid);
//滚动到当前位置
self.gotoScrollTop();
}
})
}else{
wx.showToast({
title: res.msg,
icon: 'none'
})
}
}).catch((err) => {
wx.hideLoading();
})
},
//跳转订单详情页面
gotoOrderDetail(event){
wx.navigateTo({
url: `/pages/orders/orderDetail/orderDetail?commitId=${event.currentTarget.dataset.ordercode}`
});
},
//去支付
gotoPay(event){
wx.navigateTo({
url: `/pages/pay/payWap/payWap?orderCode=${event.currentTarget.dataset.ordercode}`
});
},
//再次购买
payAgain(event) {
wx.showLoading();
http.fetch(http.ADD_CARTFROM_ORDER,{
orderid: event.currentTarget.dataset.ordercode,
v:2
}).then(res => {
wx.hideLoading();
if(res.status == 1){
wx.switchTab({
url: '/pages/shopcart/shopcart',
})
}
}).catch((err) => {
wx.hideLoading();
})
},
//领奖励金
getRewardPrice(event) {
wx.showLoading();
http.fetch(http.GET_ORDER_BONUS,{
orderCode: event.currentTarget.dataset.ordercode
}).then(res => {
wx.hideLoading();
if(res.data.success){
if(res.data=={}){
wx.showToast({
title: '服务器繁忙,请稍后再试',
icon: 'none'
})
}else{
//隐藏底部tabbar
wx.hideTabBar();
this.setData({
isShowMyReward:true,
myRewardContent:['+ '+res.data.bonus+' 元','奖励金已入账!'],
orderList:{}
});
var cid = this.data.curListId;
this.getOrderList(cid);
}
}else{
wx.showToast({
title: res.msg,
icon: 'none'
})
}
}).catch((err) => {
wx.hideLoading();
})
},
//跳转我的奖励金页面
onConfirmMyReward(){
wx.navigateTo({
url: `/pages/mine/wallet/walletDetails/bonus/index`
});
},
//查看物流
gotoLogistics:function(event){
wx.navigateTo({
url: `/pages/orders/orderLogistics/orderLogistics?orderCode=${event.currentTarget.dataset.ordercode}`
})
},
})
然后滑动到上次切换tab的位置的话,小程序有一个onPageScroll的方法,可以记录当前滚动条的位置,然后用wx.pageScrollTo让页面滚动到记录的位置即可,滑动swiper同样可以和tab联动。
WXSS代码:
/* pages/ordes/ordes.wxss */
page{
background: #F2F2F2;
}
.header-fixed{
position: fixed;
top:0;
width:100%;
z-index: 10;
}
/* 搜索部分样式 */
.clearfix::after{
display: table;
content: '';
clear: both;
}
.header-search{
display: -webkit-flex; /* Safari */
display: flex;
align-items: center;
justify-content: space-between;
background: #fff;
}
.search-box{
width: 82%;
display: -webkit-flex; /* Safari */
display: flex;
align-items: center;
margin:16rpx 0 16rpx 30rpx;
height: 60rpx;
line-height: 60rpx;
background:#EDEDED;
border-radius:30rpx;
}
.search-icon{
margin-left: 26rpx;
}
.search-input{
display: inline-block;
width:86%;
font-size: 26rpx;
margin-left: 14rpx;
}
.search-btn{
font-size: 28rpx;
color: #333333;
margin-right: 32rpx;
}
.search-icon{
width: 20rpx;
height:20rpx;
}
/* tab页签部分样式 */
.tabNav-box {
margin-top:170rpx;
}
.scroll-bangdan {
width: 100%;
height: 70rpx;
position: relative;
top: 0;
z-index: 500;
}
.tabTitle {
width: 100%;
height: 70rpx;
background:#fff;
display: flex;
align-items: center;
justify-content: space-around;
}
.titleUnsel {
color: #666666;
font-size: 30rpx;
display: flex;
flex-direction: column;
align-items: center;
height: 70rpx;
justify-content: flex-end;
}
.titleSel{
color: #333333;
font-size: 30rpx;
display: flex;
flex-direction: column;
align-items: center;
font-weight: bold;
height: 70rpx;
justify-content: flex-end;
}
.headerLineSel {
background: #0095FF;
height: 6rpx;
width: 40rpx;
position: relative;
margin-top: 10rpx;
box-shadow:2rpx 2rpx 6rpx 0px rgba(81,231,253,0.3);
border-radius:4rpx;
}
.headerLineUnsel {
background: #fff;
height: 6rpx;
width: 40rpx;
position: relative;
margin-top: 10rpx;
box-shadow:none;
}
.tabSwiper {
width: 100%;
}
/* 订单列表部分样式 */
.orderItem {
height:380rpx;
margin:20rpx 20rpx 0 20rpx;
background-color: #fff;
border-radius: 10rpx;
}
.order-header{
display: -webkit-flex; /* Safari */
display: flex;
align-items: center;
justify-content: space-between;
padding:20rpx;
}
.orderTime{
font-size: 28rpx;
color: #3E3E3E;
padding-left: 10rpx;
font-weight: bold;
}
.orderState{
font-size: 26rpx;
color: #FF7800;
margin-right: 10rpx;
font-weight: bold;
}
.grey999{
color:#999;
}
.order-body{
background: #FAFAFA;
padding:0 20rpx;
}
.goods-content{
display: -webkit-flex; /* Safari */
display: flex;
align-items: center;
justify-content: space-between;
padding-top: 20rpx;
}
.goods-content.morePadding{
padding-top:40rpx;
}
.goodsImg{
width:120rpx;
height:120rpx;
margin-right: 10rpx;
background: url("");
background-size: 100%;
}
.singleGood,.multiGoods{
display: -webkit-flex; /* Safari */
display: flex;
align-items: center;
justify-content: space-between;
}
.goodsText{
margin-left: 10rpx;
}
.goodsName{
font-size: 24rpx;
color:#333333;
max-width:240rpx;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.goodsDesc{
font-size: 22rpx;
color: #888888;
max-width:240rpx;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.goodsCompany{
margin-left: 10rpx;
}
.numAndPrice{
display: -webkit-flex; /* Safari */
display: flex;
align-items: center;
}
.goodsNum{
color: #333333;
font-size: 24rpx;
}
.goodsPrice{
color:#333333;
font-size: 32rpx;
font-weight: bold;
margin-left: 10rpx;
}
.yen{
font-size: 26rpx;
}
.rewardPrice-content{
font-size: 22rpx;
padding-top:10rpx;
padding-bottom: 20rpx;
min-height: 10rpx;
}
.rewardPriceText{
display: inline-block;
color: #888;
max-width:400rpx;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
vertical-align: middle;
}
.rewardPriceTips{
width:26rpx;
height: 26rpx;
margin:0 10rpx 0 4rpx;
vertical-align: middle;
}
.rewardPrice{
display: inline-block;
color: #C48117;
vertical-align: middle;
}
.order-footer{
text-align: right;
padding:24rpx 20rpx;
}
.order-btn{
display: inline-block;
padding:10rpx 26rpx;
border:1rpx solid #ccc;
border-radius:70rpx;
font-size:24rpx;
color:#333;
margin-right: 10rpx;
}
.getRewardPrice{
color:#D58B17;
border:1rpx solid #D58B17;
}
.gotoPay-box{
color: #0095FF;
}
.gotoPay{
color: #0095FF;
border:1rpx solid #0095FF;
}
.m-end {
font-size: 24rpx;
padding: 30rpx;
line-height: 28rpx;
text-align: center;
color: #B2B2B2;
}
.no_order{
position: absolute;
left: 50%;
top: 40%;
transform: translate(-50%, -50%);
z-index: -1;
}
.no_order_img{
width:300rpx;
height:300rpx;
}
.no_order_text{
text-align: center;
font-size: 24rpx;
color:#999;
}
页面还有个待支付的倒计时组件,JS代码如下:
// components/counter/counter.js
Component({
/**
* 组件的属性列表
*/
properties: {
remainTime:String,//剩余时间
callback: String, // 回调
},
/**
* 组件的初始数据
*/
data: {
time: ''
},
ready () {
this.init();
},
countDownID:null,
lifetimes: {
// 生命周期函数,可以为函数,或一个在methods段中定义的方法名
attached: function () {
},
moved: function () {
//clearInterval(this.countDownID);
},
detached: function () {
clearInterval(this.countDownID);
},
},
pageLifetimes: {
// 组件所在页面的生命周期函数
show: function () {
this.endfn();
},
hide: function () {
clearInterval(this.countDownID);
},
resize: function () { },
},
/**
* 组件的方法列表
*/
methods: {
init(){
var self = this;
if(self.data.remainTime!='' && self.data.remainTime >= 0){
var remainTime = parseInt(self.data.remainTime)*1000;
self.countDownID = setInterval(function() {
remainTime -= 1000;
self.getFormat(remainTime);
//console.log(self.countDownID)
}, 1000);
}else{
clearInterval(this.countDownID);
}
},
getFormat (remainTime) {
var time = "";
var min = parseInt((remainTime / 1000 / 60) % 60);
var sec = parseInt((remainTime / 1000) % 60);
min = min > 9 ? min : "0" + min;
sec = sec > 9 ? sec : "0" + sec;
time ='还剩' + min + '分' + sec + '秒';
if(remainTime >= 0){
this.setData({
time: time
});
}else{
clearInterval(this.countDownID);
this.endfn();
return;
}
},
//倒计时结束的回调
endfn () {
this.triggerEvent('callback', {});
}
}
})
一定要注意的就是每次倒计时结束和跳出订单列表页的时候需要清除当前定时器。