487 lines
11 KiB
Vue
487 lines
11 KiB
Vue
<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>
|
||
<!-- 退款列表 -->
|
||
<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>
|