This commit is contained in:
aaron 2025-04-05 21:17:36 +08:00
parent e2720eecbb
commit 79ec214050

View File

@ -65,6 +65,7 @@
:loading="loading" :loading="loading"
@change="handleTableChange" @change="handleTableChange"
row-key="id" row-key="id"
:scroll="{ x: 1500 }"
> >
<template #bodyCell="{ column, record }"> <template #bodyCell="{ column, record }">
<!-- 订单状态列 --> <!-- 订单状态列 -->
@ -74,17 +75,87 @@
</a-tag> </a-tag>
</template> </template>
<!-- 金额列 --> <!-- 金额列 -->
<template v-if="['original_amount', 'coupon_discount_amount', 'point_discount_amount', 'final_amount'].includes(column.key)"> <template v-if="['original_amount', 'coupon_discount_amount', 'point_discount_amount', 'final_amount', 'refund_amount'].includes(column.key)">
¥{{ record[column.dataIndex] }} ¥{{ (record[column.dataIndex] || 0).toFixed(2) }}
</template>
<!-- 退款原因如果字段为空则显示占位符 -->
<template v-if="column.key === 'refund_reason'">
{{ record.refund_reason || '-' }}
</template>
<!-- 操作列 -->
<template v-if="column.key === 'action'">
<a-button
type="primary"
size="small"
@click="showRefundModal(record)"
:disabled="record.status === 'CANCELLED' || record.refund_amount > 0"
>
退款
</a-button>
</template> </template>
</template> </template>
</a-table> </a-table>
<!-- 退款模态框 -->
<a-modal
v-model:visible="refundModalVisible"
title="订单退款"
@ok="handleRefundSubmit"
@cancel="handleRefundCancel"
:confirmLoading="refundLoading"
width="500px"
okText="确定"
cancelText="取消"
>
<div v-if="currentOrder" class="refund-modal-content">
<div class="order-info">
<div class="order-info-item">
<span class="label">订单号</span>
<span class="value">{{ currentOrder.orderid }}</span>
</div>
<a-divider style="margin: 12px 0" />
<div class="order-amount-info">
<div class="amount-item">
<span class="label">订单金额</span>
<span class="amount">¥{{ currentOrder.final_amount.toFixed(2) }}</span>
</div>
<div class="amount-item">
<span class="label">可退款金额</span>
<span class="amount available">¥{{ calculateAvailableRefund(currentOrder).toFixed(2) }}</span>
</div>
</div>
</div>
<div class="refund-form">
<div class="form-item">
<div class="form-label"><span class="required">*</span> 退款金额</div>
<a-input-number
v-model:value="refundForm.amount"
:min="0.01"
:max="calculateAvailableRefund(currentOrder)"
:precision="2"
class="refund-input"
placeholder="请输入退款金额"
/>
</div>
<div class="form-item">
<div class="form-label"><span class="required">*</span> 退款原因</div>
<a-input
v-model:value="refundForm.reason"
placeholder="请输入退款原因"
class="refund-input"
allowClear
/>
</div>
</div>
</div>
</a-modal>
</div> </div>
</page-container> </page-container>
</template> </template>
<script> <script>
import { defineComponent, ref, onMounted } from 'vue' import { defineComponent, ref, onMounted, onBeforeUnmount } from 'vue'
import { message } from 'ant-design-vue' import { message } from 'ant-design-vue'
import PageContainer from '@/components/PageContainer.vue' import PageContainer from '@/components/PageContainer.vue'
import request from '@/utils/request' import request from '@/utils/request'
@ -116,61 +187,75 @@ export default defineComponent({
title: '订单号', title: '订单号',
dataIndex: 'orderid', dataIndex: 'orderid',
key: 'orderid', key: 'orderid',
width: 180 width: 160
}, },
{ {
title: '用户ID', title: '用户ID',
dataIndex: 'userid', dataIndex: 'userid',
key: 'userid', key: 'userid',
width: 100, width: 80,
align: 'center' align: 'center'
}, },
{ {
title: '包裹数量', title: '包裹数量',
dataIndex: 'package_count', dataIndex: 'package_count',
key: 'package_count', key: 'package_count',
width: 100, width: 80,
align: 'center' align: 'center'
}, },
{ {
title: '订单金额', title: '订单金额',
dataIndex: 'original_amount', dataIndex: 'original_amount',
key: 'original_amount', key: 'original_amount',
width: 120, width: 100,
align: 'right' align: 'right'
}, },
{ {
title: '服务券抵扣', title: '服务券抵扣',
dataIndex: 'coupon_discount_amount', dataIndex: 'coupon_discount_amount',
key: 'coupon_discount_amount', key: 'coupon_discount_amount',
width: 120, width: 100,
align: 'right' align: 'right'
}, },
{ {
title: '积分抵扣', title: '积分抵扣',
dataIndex: 'point_discount_amount', dataIndex: 'point_discount_amount',
key: 'point_discount_amount', key: 'point_discount_amount',
width: 120, width: 100,
align: 'right' align: 'right'
}, },
{ {
title: '最终金额', title: '最终金额',
dataIndex: 'final_amount', dataIndex: 'final_amount',
key: 'final_amount', key: 'final_amount',
width: 120, width: 100,
align: 'right' align: 'right'
}, },
{
title: '退款金额',
dataIndex: 'refund_amount',
key: 'refund_amount',
width: 100,
align: 'right'
},
{
title: '退款原因',
dataIndex: 'refund_reason',
key: 'refund_reason',
width: 120,
ellipsis: true
},
{ {
title: '订单状态', title: '订单状态',
dataIndex: 'status', dataIndex: 'status',
key: 'status', key: 'status',
width: 100, width: 90,
align: 'center' align: 'center'
}, },
{ {
title: '配送地址', title: '配送地址',
key: 'address', key: 'address',
width: 300, width: 250,
ellipsis: true, ellipsis: true,
customRender: ({ record }) => { customRender: ({ record }) => {
const { address } = record const { address } = record
@ -181,7 +266,14 @@ export default defineComponent({
title: '下单时间', title: '下单时间',
dataIndex: 'create_time', dataIndex: 'create_time',
key: 'create_time', key: 'create_time',
width: 180 width: 160
},
{
title: '操作',
key: 'action',
width: 100,
align: 'center',
fixed: 'right'
} }
] ]
@ -276,10 +368,106 @@ export default defineComponent({
fetchData() fetchData()
} }
// ResizeObserver
const handleResizeObserverError = () => {
const errorHandler = (error) => {
if (error.message && error.message.includes('ResizeObserver')) {
error.stopImmediatePropagation();
console.warn('ResizeObserver error intercepted');
return true;
}
return false;
};
window.addEventListener('error', errorHandler, true);
return () => {
window.removeEventListener('error', errorHandler, true);
};
};
const clearErrorHandler = ref(null);
onMounted(() => { onMounted(() => {
fetchData() fetchData();
clearErrorHandler.value = handleResizeObserverError();
});
onBeforeUnmount(() => {
if (clearErrorHandler.value) {
clearErrorHandler.value();
}
});
// 退
const refundModalVisible = ref(false)
const refundLoading = ref(false)
const currentOrder = ref(null)
const refundForm = ref({
amount: undefined,
reason: ''
}) })
// 退
const showRefundModal = (record) => {
currentOrder.value = record
refundForm.value = {
amount: undefined,
reason: ''
}
refundModalVisible.value = true
}
// 退
const calculateAvailableRefund = (order) => {
if (!order) return 0
const alreadyRefunded = order.refund_amount || 0
return Number((order.final_amount - alreadyRefunded).toFixed(2))
}
// 退
const handleRefundSubmit = async () => {
//
if (!refundForm.value.amount) {
message.error('请输入退款金额')
return
}
if (!refundForm.value.reason) {
message.error('请输入退款原因')
return
}
try {
refundLoading.value = true
const params = {
order_id: currentOrder.value.orderid,
amount: refundForm.value.amount,
reason: refundForm.value.reason
}
const res = await request.put('/api/order/admin/refund_order_amount', params)
if (res.code === 200) {
message.success('退款操作成功')
refundModalVisible.value = false
//
fetchData()
} else {
message.error(res.message || '退款操作失败')
}
} catch (error) {
console.error('退款操作失败:', error)
message.error('退款操作失败')
} finally {
refundLoading.value = false
}
}
// 退
const handleRefundCancel = () => {
refundModalVisible.value = false
}
return { return {
loading, loading,
columns, columns,
@ -291,7 +479,15 @@ export default defineComponent({
handleSearch, handleSearch,
handleReset, handleReset,
getStatusText, getStatusText,
getStatusColor getStatusColor,
refundModalVisible,
refundLoading,
currentOrder,
refundForm,
showRefundModal,
calculateAvailableRefund,
handleRefundSubmit,
handleRefundCancel
} }
} }
}) })
@ -329,4 +525,127 @@ export default defineComponent({
min-width: 60px; min-width: 60px;
text-align: center; text-align: center;
} }
/* 优化表格组件性能 */
:deep(.ant-table-wrapper) {
contain: content;
will-change: transform;
}
:deep(.ant-table) {
table-layout: fixed;
}
:deep(.ant-table-container) {
contain: layout;
}
:deep(.ant-table-tbody > tr > td) {
padding: 12px 8px;
contain: content;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
:deep(.ant-table-thead > tr > th) {
padding: 12px 8px;
contain: content;
}
/* 退款模态框样式 */
.refund-modal-content {
padding: 0;
}
.order-info {
background-color: #f9f9f9;
border-radius: 4px;
padding: 16px;
margin-bottom: 20px;
}
.order-info-item {
display: flex;
align-items: center;
}
.order-info-item .label {
font-weight: 500;
color: #666;
width: 80px;
}
.order-info-item .value {
color: #333;
font-family: monospace;
font-size: 15px;
}
.order-amount-info {
display: flex;
justify-content: space-between;
}
.amount-item {
display: flex;
flex-direction: column;
padding: 8px 12px;
width: 48%;
border-radius: 4px;
background-color: #fff;
margin: 4px 0;
box-shadow: 0 1px 2px rgba(0, 0, 0, 0.06);
}
.amount-item .label {
font-size: 12px;
color: #666;
margin-bottom: 4px;
width: auto;
}
.amount-item .amount {
font-size: 16px;
font-weight: bold;
color: #333;
}
.amount-item .available {
color: #1890ff;
}
.refund-form {
padding: 0 5px;
}
.form-item {
margin-bottom: 16px;
}
.form-label {
margin-bottom: 8px;
font-size: 14px;
color: rgba(0, 0, 0, 0.85);
}
.required {
color: #ff4d4f;
margin-right: 4px;
}
.refund-input {
width: 100%;
height: 32px;
border-radius: 2px;
}
:deep(.ant-input-number-input) {
height: 30px;
}
:deep(.ant-input-number),
:deep(.ant-input) {
width: 100%;
}
</style> </style>