caiyunyi/pages/ownWater/waterOrderList.vue
ylj20011123 b59c3ed264 update
2026-03-03 15:05:27 +08:00

622 lines
19 KiB
Vue

<template>
<view class="main">
<!-- 顶部固定导航 -->
<view class="header">
<!-- <view class="headerTop" :style="{ height: menu.height + 'px' }">
<view class="backBtn" @click="handleBack">
<image class="backIcon" src="/static/images/home/backArrowblack.svg" />
</view>
<view class="pageName">我的订单</view>
<view class="placeholder"></view>
</view>
<view class="filterSection">
<view class="searchBox">
<image class="searchIcon" src="/static/images/home/search_icon.svg" />
<input class="searchInput" placeholder="请输入商品名称关键字" v-model="searchText" confirm-type="search"
@confirm="handleSearch" />
</view>
<view class="filterBtn" @click="handleShowFilter">
<image class="filterIcon" src="/static/images/home/select_icon.svg" />
<text>筛选</text>
</view>
</view> -->
<!-- Tabs 切换 -->
<view class="tabs">
<view v-for="(item, index) in tabList" :key="index" class="tabItem"
:class="{ active: currentTab === item.value }" @click="handleTabChange(item.value)">
{{ item.label }}
</view>
</view>
<!-- 数据汇总统计 -->
<view class="summaryLine" v-if="orderList.length > 0">
共 <text class="highlight">{{ totalCount }}</text> 笔订单,
合计 <text class="highlight">{{ totalAmount }}</text> 元
</view>
</view>
<!-- 订单列表内容区 -->
<scroll-view class="listContainer" scroll-y @scrolltolower="handleLoadMore" :style="{ height: listHeight }">
<view class="orderList" v-if="orderList.length > 0">
<view class="orderItem" v-for="(order, index) in orderList" :key="index" @click="handleGoDetail(order)">
<view class="orderItemHeader">
<view class="orderTime">
{{ order.SALEBILL_STATE <= 1010 ? '期望送达: ' : '' }}{{ order.RESERVATION_DATE ||
order.ORDER_DATE }} </view>
<view class="orderStatus">{{ getStatusText(order.SALEBILL_STATE) }}</view>
</view>
<view class="orderItemContent">
<scroll-view class="imgScroll" scroll-x>
<view class="imgList">
<view class="imgWrapper" v-for="(img, idx) in order.IMAGELIST" :key="idx">
<image class="productImg"
:src="img.IMAGE_URL || 'https://eshangtech.com/ShopICO/no-picture.png'"
mode="aspectFill" />
<view class="productName">{{ img.IMAGE_TITLE }}</view>
</view>
</view>
</scroll-view>
<view class="orderStats">
<view class="amountBox">
<text class="unit">¥</text>
<text class="amount">{{ order.ORDER_AMOUNT }}</text>
</view>
<view class="count">{{ order.TOTAL_COUNT }}</view>
</view>
</view>
<view class="orderItemFooter">
<view class="descText" v-if="order.SALEBILL_DESC">{{ order.SALEBILL_DESC }}</view>
<view class="actionBtns">
<view class="btn confirm" @click.stop="handleConfirmReceipt(order)">验收确认</view>
</view>
</view>
</view>
<view class="noMore" v-if="isOver">我是有底线的</view>
</view>
<!-- 空状态 -->
<view class="emptyBox" v-else>
<view class="emptyText">暂无订单记录</view>
</view>
<view style="height: 40rpx;"></view>
</scroll-view>
<!-- 筛选弹窗 -->
<uni-popup ref="filterPopup" type="bottom">
<view class="filterPopupBox">
<view class="popHeader">筛选时间</view>
<view class="timeOptions">
<view class="timeItem" :class="{ active: timeIndex === item.value }"
v-for="(item, index) in timeList" :key="index" @click="timeIndex = item.value">
{{ item.label }}
</view>
</view>
<view class="confirmBtn" @click="handleFilterConfirm">确认</view>
</view>
</uni-popup>
</view>
</template>
<script>
import { mapGetters } from "vuex";
export default {
data() {
return {
menu: {},
searchText: "",
currentTab: 1,
tabList: [
{ label: '全部订单', value: 1 },
{ label: '待发货', value: 2 },
{ label: '已发货', value: 3 },
{ label: '已接收', value: 4 }
],
timeIndex: '',
timeList: [
{ label: '一周内', value: 'week' },
{ label: '近一月', value: 'month' },
{ label: '近三月', value: 'threeMouth' }
],
orderList: [],
totalCount: 0,
totalAmount: "0.00",
pageIndex: 1,
pageSize: 10,
isOver: false,
isLoading: false,
listHeight: "0px",
// 状态映射:参考 buyOrder
statusQueryMap: {
1: '',
2: '1010,3000',
3: '3050',
4: '4000,5000'
}
};
},
computed: {
...mapGetters({
user: "user",
}),
},
onLoad(options) {
this.menu = uni.getMenuButtonBoundingClientRect();
if (options.status) {
// 根据传入的状态码字符串匹配相应的 Tab
if (options.status === '1090,2000') {
this.currentTab = 2; // 待发货
} else if (options.status === '2010') {
this.currentTab = 3; // 已发货
} else if (options.status === '3000,4000') {
this.currentTab = 4; // 已接收
} else {
this.currentTab = 1; // 全部
}
}
this.calculateListHeight();
this.handleRefresh();
},
methods: {
calculateListHeight() {
// 简单计算:页面高度 - 头部固定高度
// 头部固定高度约为 menu.bottom + 搜索栏(88rpx) + Tabs(88rpx) + Summary(72rpx) + 间距
uni.getSystemInfo({
success: (res) => {
const headerHeight = this.menu.bottom + uni.upx2px(88 + 88 + 72 + 40);
this.listHeight = (res.windowHeight - headerHeight) + 'px';
}
});
},
handleBack() {
uni.navigateBack();
},
handleRefresh() {
this.pageIndex = 1;
this.isOver = false;
this.orderList = [];
this.fetchOrders();
},
handleLoadMore() {
if (this.isOver || this.isLoading) return;
this.pageIndex++;
this.fetchOrders();
},
handleSearch() {
this.handleRefresh();
},
handleTabChange(val) {
this.currentTab = val;
this.handleRefresh();
},
handleShowFilter() {
this.$refs.filterPopup.open();
},
handleFilterConfirm() {
this.$refs.filterPopup.close();
this.handleRefresh();
},
getStatusText(state) {
if (state === 1090) {
return '待发货'
} else if (state === 2000) {
return '备货中'
} else if (state === 2010) {
return '已发货'
} else if (state === 3000) {
return '已接收'
} else if (state === 4000) {
return '已完成'
}
},
async fetchOrders() {
// 这里预留后端接口调用
// 目前使用 mock 数据演示布局
this.isLoading = true;
uni.showLoading({ title: "加载中..." });
const res = await this.$api.$samemberGet('/EShangApiMain/WeChatMall/GetMallOrderList', {
WeChat_AppId: "wxee018fb96955552a",
MembershipId: this.user.MEMBERSHIP_ID,
orderState: this.currentTab === 2 ? '1090,2000' : this.currentTab === 3 ? '2010' : this.currentTab === 4 ? '3000,4000' : this.currentTab === 1 ? '1090,2000,2010,3000,4000' : ''
})
let mockList = res.Result_Data.List || [];
if (this.pageIndex >= 3) {
this.isOver = true;
}
this.orderList = this.pageIndex === 1 ? mockList : [...this.orderList, ...mockList];
this.totalCount = this.orderList.length;
this.totalAmount = this.orderList.reduce((sum, item) => sum + parseFloat(item.ORDER_AMOUNT), 0).toFixed(2);
this.isLoading = false;
uni.hideLoading();
},
handleGoDetail(order) {
uni.navigateTo({
url: `/pages/ownWater/waterOrderDetail?id=${order.PURCHASE_ID || order.SALEBILL_ID}`
});
},
// 验收确认
handleConfirmReceipt(order) {
uni.showModal({
title: '验收确认',
content: '确认已收到该订单的商品吗?',
confirmColor: '#F83D3D',
success: (res) => {
if (res.confirm) {
// TODO: 调用后端验收确认接口
uni.showToast({ title: '已确认验收', icon: 'success' });
}
}
});
}
}
};
</script>
<style lang="less" scoped>
.main {
background: #f2f4f5;
min-height: 100vh;
display: flex;
flex-direction: column;
}
.header {
background: #fff;
flex-shrink: 0;
z-index: 100;
box-shadow: 0 2rpx 10rpx rgba(0, 0, 0, 0.02);
.headerTop {
display: flex;
align-items: center;
padding: 0 30rpx;
position: relative;
.backBtn {
width: 80rpx;
height: 100%;
display: flex;
align-items: center;
.backIcon {
width: 44rpx;
height: 44rpx;
}
}
.pageName {
flex: 1;
text-align: center;
font-size: 32rpx;
font-weight: bold;
color: #020E1A;
}
.placeholder {
width: 80rpx;
}
}
.filterSection {
display: flex;
padding: 20rpx 30rpx;
align-items: center;
.searchBox {
flex: 1;
height: 72rpx;
background: #F2F4F5;
border-radius: 36rpx;
display: flex;
align-items: center;
padding: 0 24rpx;
margin-right: 20rpx;
.searchIcon {
width: 32rpx;
height: 32rpx;
margin-right: 12rpx;
}
.searchInput {
flex: 1;
font-size: 28rpx;
color: #333;
}
}
.filterBtn {
display: flex;
align-items: center;
font-size: 28rpx;
color: #6C737A;
.filterIcon {
width: 32rpx;
height: 32rpx;
margin-right: 8rpx;
}
}
}
.tabs {
display: flex;
height: 88rpx;
border-bottom: 2rpx solid #F2F4F5;
.tabItem {
flex: 1;
display: flex;
align-items: center;
justify-content: center;
font-size: 28rpx;
color: #6C737A;
position: relative;
&.active {
color: #F83D3D;
font-weight: bold;
&::after {
content: '';
position: absolute;
bottom: 0;
width: 48rpx;
height: 6rpx;
background: #F83D3D;
border-radius: 4rpx;
}
}
}
}
.summaryLine {
height: 72rpx;
padding: 0 30rpx;
display: flex;
align-items: center;
font-size: 26rpx;
color: #6C737A;
.highlight {
color: #020E1A;
font-weight: bold;
margin: 0 6rpx;
}
}
}
.listContainer {
flex: 1;
width: 100%;
}
.orderList {
padding: 24rpx;
.orderItem {
background: #fff;
border-radius: 16rpx;
padding: 24rpx;
margin-bottom: 24rpx;
box-shadow: 0 2rpx 12rpx rgba(0, 0, 0, 0.03);
.orderItemHeader {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 20rpx;
.orderTime {
font-size: 24rpx;
color: #fa5151;
}
.orderStatus {
font-size: 28rpx;
color: #F83D3D;
font-weight: 500;
}
}
.orderItemContent {
display: flex;
align-items: center;
margin-bottom: 20rpx;
position: relative;
.imgScroll {
flex: 1;
white-space: nowrap;
margin-right: 20rpx;
.imgList {
display: flex;
.imgWrapper {
display: flex;
flex-direction: column;
align-items: center;
margin-right: 16rpx;
.productImg {
width: 120rpx;
height: 120rpx;
border-radius: 8rpx;
background: #f8f8f8;
}
.productName {
font-size: 20rpx;
color: #666;
width: 120rpx;
overflow: hidden;
text-overflow: ellipsis;
text-align: center;
margin-top: 4rpx;
}
}
}
}
.orderStats {
text-align: right;
background: linear-gradient(to right, rgba(255, 255, 255, 0), #fff 20%);
padding-left: 40rpx;
.amountBox {
color: #020E1A;
font-weight: bold;
margin-bottom: 4rpx;
.unit {
font-size: 24rpx;
}
.amount {
font-size: 36rpx;
font-family: 'DIN Alternate';
}
}
.count {
font-size: 24rpx;
color: #9FA3A8;
}
}
}
.orderItemFooter {
border-top: 1rpx solid #F2F4F5;
padding-top: 20rpx;
display: flex;
justify-content: space-between;
align-items: center;
.descText {
font-size: 24rpx;
color: #FF6219;
flex: 1;
margin-right: 20rpx;
}
.actionBtns {
display: flex;
margin-left: auto;
.btn {
height: 60rpx;
padding: 0 24rpx;
border-radius: 30rpx;
font-size: 26rpx;
display: flex;
align-items: center;
justify-content: center;
margin-left: 16rpx;
&.normal {
border: 1rpx solid #D9DBDD;
color: #6C737A;
}
&.active {
border: 1rpx solid #F83D3D;
color: #F83D3D;
}
&.confirm {
background: #F2F5F6;
color: #E84D38;
font-weight: 500;
padding: 0 32rpx;
border-radius: 4rpx;
}
}
}
}
}
}
.emptyBox {
padding-top: 200rpx;
display: flex;
flex-direction: column;
align-items: center;
.emptyImg {
width: 240rpx;
height: 240rpx;
opacity: 0.6;
}
.emptyText {
font-size: 28rpx;
color: #999;
margin-top: 20rpx;
}
}
.noMore {
text-align: center;
font-size: 24rpx;
color: #ccc;
padding: 20rpx 0;
}
.filterPopupBox {
background: #fff;
border-radius: 24rpx 24rpx 0 0;
padding: 40rpx 30rpx calc(40rpx + env(safe-area-inset-bottom));
.popHeader {
font-size: 32rpx;
font-weight: bold;
margin-bottom: 40rpx;
text-align: center;
}
.timeOptions {
display: flex;
justify-content: space-between;
margin-bottom: 60rpx;
.timeItem {
width: 210rpx;
height: 72rpx;
background: #F2F4F5;
border-radius: 8rpx;
display: flex;
align-items: center;
justify-content: center;
font-size: 28rpx;
color: #333;
&.active {
background: rgba(248, 61, 61, 0.1);
color: #F83D3D;
border: 1rpx solid #F83D3D;
}
}
}
.confirmBtn {
height: 88rpx;
background: #F83D3D;
color: #fff;
border-radius: 44rpx;
display: flex;
align-items: center;
justify-content: center;
font-size: 30rpx;
font-weight: bold;
}
}
</style>