web-retail-h5/pages/refund/detail.vue

487 lines
11 KiB
Vue
Raw Normal View History

2025-07-07 16:56:07 +08:00
<template>
<view class="refund-container">
<!-- 导航栏 -->
<u-navbar
title="退款列表"
:border-bottom="false"
:background="{ backgroundColor: '#005bac' }"
title-color="#fff"
/>
<view class="tips">
<u-icon name="info-circle" color="#005bac" size="20"></u-icon>
<text class="tips-text">如有疑问请联系客服</text>
</view>
<!-- <view class="date-filter">
<view class="date-picker-container">
<view class="date-input-wrapper" @click="showStartDatePicker = true">
<text>{{ startDate || '开始时间' }}</text>
</view>
<text class="separator"></text>
<view class="date-input-wrapper" @click="showEndDatePicker = true">
<text>{{ endDate || '结束时间' }}</text>
</view>
</view>
<button class="search-button" @click="handleSearch">
<u-icon name="search" color="#ffffff" size="20" />
</button>
</view> -->
<!-- 空状态 -->
<u-empty
mode="data"
v-if="refundList.length === 0 && !loading"
text="暂无退款记录"
></u-empty>
2025-07-07 16:56:07 +08:00
<!-- 退款列表 -->
<view class="refund-list">
<u-list @scrolltolower="loadMore" :show-scrollbar="false">
<u-list-item v-for="(item, index) in refundList" :key="index">
<view class="refund-item">
<view class="refund-header">
<view class="order-info">
<text class="order-number"
>订单编号{{ item.businessCode }}</text
>
<text class="member-code">会员编号{{ item.memberCode }}</text>
</view>
<view
class="refund-status"
:class="getStatusClass(item.refundStatus)"
>
{{ getStatusText(item.refundStatus) }}
</view>
</view>
<view class="refund-content">
<view class="refund-amount">
<text class="amount-label">退款金额</text>
<text class="amount-value"
>¥{{ stateFormat(item.refundMoney) }}</text
>
</view>
<view class="refund-time">
<text class="time-label">退款时间</text>
<text class="time-value">{{
formatDate(item.finishTime)
}}</text>
</view>
</view>
</view>
</u-list-item>
</u-list>
<!-- 加载更多 -->
<u-loadmore :status="loadStatus" />
</view>
<u-datetime-picker
:show="showStartDatePicker"
v-model="startDate"
mode="date"
@confirm="onStartDateConfirm"
@cancel="showStartDatePicker = false"
style="flex: 0"
></u-datetime-picker>
<u-datetime-picker
:show="showEndDatePicker"
v-model="endDate"
mode="date"
@confirm="onEndDateConfirm"
@cancel="showEndDatePicker = false"
style="flex: 0"
></u-datetime-picker>
</view>
</template>
<script>
import { getRefundList } from '@/config/mine'
import dayjs from '../../uni_modules/uview-ui/libs/util/dayjs.js'
export default {
name: 'RefundDetail',
data() {
return {
// 筛选条件
startDate: '',
endDate: '',
showStartDatePicker: false,
showEndDatePicker: false,
maxDate: new Date().getTime(),
// 列表数据
refundList: [],
loading: false,
loadStatus: 'loadmore',
// 分页参数
currentPage: 1,
pageSize: 10,
hasMore: true,
// 状态映射
statusMap: {
0: '退款申请中',
1: '退款中',
2: '已退款',
3: '退款失败',
},
}
},
onLoad() {
this.loadRefundList()
},
onPullDownRefresh() {
this.refreshList()
},
methods: {
// 加载退款列表
async loadRefundList(isLoadMore = false) {
if (this.loading) return
this.loading = true
this.loadStatus = 'loading'
try {
const params = {
pageNum: this.currentPage,
pageSize: this.pageSize,
startDate: this.startDate,
endDate: this.endDate,
}
const res = await getRefundList(params)
if (res.code === 200) {
const newList = res.rows || []
if (isLoadMore) {
this.refundList = [...this.refundList, ...newList]
} else {
this.refundList = newList
}
// 判断是否还有更多数据
this.hasMore = newList.length === this.pageSize
this.loadStatus = this.hasMore ? 'loadmore' : 'nomore'
if (this.hasMore) {
this.currentPage++
}
} else {
this.$u.toast(res.message || '加载失败')
this.loadStatus = 'loadmore'
}
} catch (error) {
console.error('加载退款列表失败:', error)
this.$u.toast('加载失败,请重试')
this.loadStatus = 'loadmore'
} finally {
this.loading = false
uni.stopPullDownRefresh()
}
},
stateFormat(val) {
if (val) {
return Number(val)
.toFixed(2)
.replace(/(\d)(?=(\d{3})+\.)/g, ($0, $1) => {
return $1 + ','
})
.replace(/\.$/, '')
}
},
// 加载更多
loadMore() {
if (this.hasMore && !this.loading) {
this.loadRefundList(true)
}
},
// 刷新列表
refreshList() {
this.currentPage = 1
this.hasMore = true
this.loadRefundList()
},
// 搜索
handleSearch() {
if (this.startDate && this.endDate && this.startDate > this.endDate) {
this.$u.toast('开始日期不能大于结束日期')
return
}
this.refreshList()
},
// 重置
handleReset() {
this.startDate = ''
this.endDate = ''
this.refreshList()
},
onStartDateConfirm(e) {
this.startDate = this.formatDate(new Date(e.value))
this.startDateValue = e.value
this.showStartDatePicker = false
},
onEndDateConfirm(e) {
this.endDate = this.formatDate(new Date(e.value))
this.endDateValue = e.value
this.showEndDatePicker = false
},
// 格式化显示日期
formatDate(dateString) {
if (!dateString) return ''
return dayjs(dateString).format('YYYY-MM-DD HH:mm:ss')
},
// 获取状态文本
getStatusText(status) {
console.log(status)
return this.statusMap[status] || '未知状态'
},
// 获取状态样式类
getStatusClass(status) {
const classMap = {
0: 'status-pending',
1: 'status-failed',
2: 'status-success',
3: 'status-cancelled',
}
return classMap[status] || 'status-default'
},
},
}
</script>
<style lang="scss" scoped>
.refund-container {
min-height: 100vh;
background-color: #f5f5f5;
}
.tips {
display: flex;
align-items: center;
justify-content: center;
background-color: #f8f9fa;
margin: 16rpx 30rpx;
padding: 12rpx 20rpx;
border-radius: 4rpx;
border-left: 3rpx solid #005bac;
.tips-text {
font-size: 22rpx;
color: #005bac;
font-weight: 400;
margin-left: 8rpx;
line-height: 1.5;
}
}
.filter-section {
background-color: #fff;
padding: 20rpx 30rpx;
margin-bottom: 20rpx;
.date-filter {
display: flex;
align-items: center;
margin-bottom: 20rpx;
.date-item {
flex: 1;
display: flex;
align-items: center;
justify-content: space-between;
padding: 20rpx;
border: 1px solid #e4e4e4;
border-radius: 8rpx;
background-color: #fafafa;
.date-label {
font-size: 28rpx;
color: #666;
}
.date-value {
font-size: 28rpx;
color: #333;
margin-left: 10rpx;
}
}
.date-separator {
padding: 0 20rpx;
font-size: 28rpx;
color: #666;
}
}
.filter-actions {
display: flex;
justify-content: center;
gap: 20rpx;
}
}
.refund-list {
padding: 10rpx 30rpx 0;
}
.refund-item {
background-color: #fff;
border-radius: 12rpx;
padding: 30rpx;
margin-bottom: 20rpx;
box-shadow: 0 2rpx 8rpx rgba(0, 0, 0, 0.1);
.refund-header {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 20rpx;
.order-info {
flex: 1;
.order-number {
display: block;
font-size: 28rpx;
color: #333;
font-weight: 500;
margin-bottom: 8rpx;
}
.member-code {
font-size: 24rpx;
color: #666;
}
}
.refund-status {
padding: 8rpx 16rpx;
border-radius: 20rpx;
font-size: 24rpx;
font-weight: 500;
&.status-pending {
background-color: #fff7e6;
color: #fa8c16;
}
&.status-success {
background-color: #f6ffed;
color: #52c41a;
}
&.status-failed {
background-color: #fff2f0;
color: #ff4d4f;
}
&.status-cancelled {
background-color: #f5f5f5;
color: #8c8c8c;
}
&.status-default {
background-color: #f5f5f5;
color: #666;
}
}
}
.refund-content {
display: flex;
justify-content: space-between;
align-items: center;
.refund-amount,
.refund-time {
display: flex;
flex-direction: column;
align-items: flex-start;
.amount-label,
.time-label {
font-size: 24rpx;
color: #666;
margin-bottom: 8rpx;
}
.amount-value {
font-size: 32rpx;
color: #005bac;
font-weight: 600;
}
.time-value {
font-size: 28rpx;
color: #333;
}
}
.refund-time {
align-items: flex-end;
}
}
}
// 响应式适配
@media (max-width: 750rpx) {
.filter-section {
.date-filter {
flex-direction: column;
gap: 20rpx;
.date-separator {
display: none;
}
}
}
}
.date-filter {
display: flex;
align-items: center;
padding: 15px 10px;
background-color: #ffffff;
.date-picker-container {
flex: 1;
display: flex;
align-items: center;
background-color: #f2f2f2;
border-radius: 8px;
padding: 4px;
.date-input-wrapper {
flex: 1;
text-align: center;
padding: 6px 0;
font-size: 14px;
color: #333;
}
.separator {
color: #999;
margin: 0 5px;
}
}
.search-button {
width: 44px;
height: 36px;
margin-left: 10px;
background-color: #007aff;
color: white;
display: flex;
align-items: center;
justify-content: center;
border-radius: 4px;
padding: 0;
line-height: 1;
border: none;
&:after {
border: none;
}
}
}
</style>