mirror of
https://gitee.com/niucloud-team/niucloud-admin.git
synced 2026-02-22 16:40:25 +00:00
417 lines
21 KiB
Vue
417 lines
21 KiB
Vue
<template>
|
|
<view :style="themeColor()">
|
|
<view class="bg-[var(--page-bg-color)] min-h-screen overflow-hidden" v-if="!loading">
|
|
<view class="pb-[200rpx]" v-if="type != 'logistics'">
|
|
<view class="bg-linear">
|
|
<!-- #ifdef MP-WEIXIN || APP-PLUS -->
|
|
<top-tabbar :data="topTabbarData" :scrollBool="topTabarObj.getScrollBool()" />
|
|
<!-- #endif -->
|
|
<view v-if="detail.status_name"
|
|
class="flex justify-between items-center pl-[40rpx] pr-[50rpx] h-[280rpx] box-border pb-[90rpx]">
|
|
<view class="text-[36rpx] font-500 leading-[42rpx] text-[#fff]">{{ detail.status_name }}</view>
|
|
<view class="flex items-center">
|
|
<image v-if="['1','2','4','6','7'].indexOf(detail.status) != -1" class="w-[180rpx] h-[140rpx]" :src="img('addon/shop/detail/payment.png')" mode="aspectFit" />
|
|
<image v-if="['8'].indexOf(detail.status) != -1" class="w-[180rpx] h-[140rpx]" :src="img('addon/shop/detail/complete.png')" mode="aspectFit" />
|
|
<image v-if="['3','5','-1'].indexOf(detail.status) != -1" class="w-[180rpx] h-[140rpx]" :src="img('addon/shop/detail/close.png')" mode="aspectFit" />
|
|
</view>
|
|
</view>
|
|
</view>
|
|
<view class="sidebar-margin card-template flex justify-between flex-wrap mt-[-76rpx]">
|
|
<view class="w-[150rpx] h-[150rpx] rounded-[var(--goods-rounded-big)] overflow-hidden" @click="goodsEvent(detail.order_goods.goods_id)">
|
|
<up-image radius="var(--goods-rounded-big)" width="150rpx" height="150rpx" :src="img(detail.order_goods.goods_image_thumb_small ? detail.order_goods.goods_image_thumb_small : '')" model="aspectFill">
|
|
<template #error>
|
|
<image class="w-[150rpx] h-[150rpx] rounded-[var(--goods-rounded-big)] overflow-hidden" :src="img('static/resource/images/diy/shop_default.jpg')" mode="aspectFill"/>
|
|
</template>
|
|
</up-image>
|
|
</view>
|
|
<view class="ml-[20rpx] flex flex-1 flex-col justify-between">
|
|
<view>
|
|
<view class="text-[28rpx] max-w-[490rpx] truncate leading-[40rpx]">{{ detail.order_goods.goods_name }}</view>
|
|
<view class="text-[22rpx] mt-[10rpx] text-[var(--text-color-light9)] truncate max-w-[490rpx] leading-[28rpx]" v-if="detail.order_goods.sku_name">{{ detail.order_goods.sku_name }}</view>
|
|
</view>
|
|
<view class="flex justify-between items-center leading-[28rpx] ">
|
|
<view class="price-font">
|
|
<text class="text-[24rpx]">¥</text>
|
|
<text class="text-[40rpx] font-500">{{ parseFloat(detail.order_goods.price).toFixed(2).split('.')[0] }}</text>
|
|
<text class="text-[24rpx] font-500">.{{ parseFloat(detail.order_goods.price).toFixed(2).split('.')[1] }}</text>
|
|
</view>
|
|
<text class="text-right text-[26rpx]">x{{ detail.order_goods.num }}</text>
|
|
</view>
|
|
</view>
|
|
</view>
|
|
<view class="sidebar-margin mt-[var(--top-m)] card-template">
|
|
<view class="justify-between text-[28rpx] card-template-item">
|
|
<view>{{ t('refundMoney') }}</view>
|
|
<view class="price-font text-[var(--price-text-color)]">
|
|
<text class="text-[24rpx] mr-[4rpx]">¥</text>
|
|
<text class="text-[28rpx]" v-if="detail.status == 8">{{ parseFloat(detail.money).toFixed(2) }}</text>
|
|
<text class="text-[28rpx]" v-else>{{ parseFloat(detail.apply_money).toFixed(2) }}</text>
|
|
</view>
|
|
</view>
|
|
<view class="justify-between text-[28rpx] card-template-item">
|
|
<view>{{ t('refundType') }}</view>
|
|
<view>{{ detail.refund_type_name }}</view>
|
|
</view>
|
|
<view class="justify-between text-[28rpx] card-template-item">
|
|
<view>{{ t('refundCause') }}</view>
|
|
<view class="w-[400rpx] multi-hidden text-right">{{ detail.reason || '--' }}</view>
|
|
</view>
|
|
<view class="justify-between text-[28rpx] card-template-item">
|
|
<view>{{ t('refundNo') }}</view>
|
|
<view>{{ detail.order_refund_no }}</view>
|
|
</view>
|
|
<view class="justify-between text-[28rpx] card-template-item">
|
|
<view>{{ t('createTime') }}</view>
|
|
<view>{{ detail.create_time }}</view>
|
|
</view>
|
|
<view class="justify-between text-[28rpx] card-template-item !items-baseline">
|
|
<view>{{ t('createExplain') }}</view>
|
|
<view class="flex-1 ml-[60rpx] text-right leading-[1.5] flex justify-end break-all">
|
|
{{ detail.remark }}
|
|
</view>
|
|
</view>
|
|
<view class="justify-between text-[28rpx] card-template-item !items-baseline break-all"
|
|
v-if="detail.shop_reason">
|
|
<view>{{ t('reasonRefusal') }}</view>
|
|
<view class="flex-1 ml-[60rpx] leading-[1.5] text-right text-[#333]">{{ detail.shop_reason }}</view>
|
|
</view>
|
|
</view>
|
|
|
|
<view class="sidebar-margin mt-[var(--top-m)] card-template">
|
|
<view class="justify-between text-[28rpx] card-template-item">
|
|
<view>{{ t('record') }}</view>
|
|
<view class="flex items-center"
|
|
@click="redirect({url: '/addon/shop/pages/refund/log', param: { order_refund_no: orderRefundNo }})">
|
|
<text class="text-[26rpx] text-[var(--text-color-light9)]">{{ t('check') }}</text>
|
|
<text class="nc-iconfont nc-icon-youV6xx text-[24rpx] text-[var(--text-color-light9)] pt-[2rpx]"></text>
|
|
</view>
|
|
</view>
|
|
</view>
|
|
|
|
<view
|
|
class="flex tab-bar justify-between items-center bg-[#fff] fixed left-0 right-0 bottom-0 min-h-[100rpx] pl-[30rpx] pr-[20rpx] flex-wrap">
|
|
<view class="flex">
|
|
<!-- <view class="flex mr-[20rpx] flex-col justify-center items-center"
|
|
@click="redirect({ url: '/addon/shop/pages/index', mode: 'reLaunch' })">
|
|
<view class="nc-iconfont nc-icon-shouyeV6xx11 text-[36rpx]"></view>
|
|
<text class="text-[20rpx] mt-[10rpx]">{{ t('index') }}</text>
|
|
</view> -->
|
|
<!-- #ifdef MP-WEIXIN -->
|
|
<!-- <view>
|
|
<nc-contact
|
|
:send-message-title="sendMessageTitle"
|
|
:send-message-path="sendMessagePath"
|
|
:send-message-img="sendMessageImg">
|
|
<view class="flex flex-col justify-center items-center">
|
|
<text class="nc-iconfont nc-icon-kefuV6xx-1 text-[36rpx]"></text>
|
|
<text class="text-[20rpx] mt-[10rpx]">客服</text>
|
|
</view>
|
|
</nc-contact>
|
|
</view> -->
|
|
<!-- #endif -->
|
|
</view>
|
|
|
|
<view class="flex justify-end">
|
|
<view @click="redirect({ url: '/addon/shop/pages/index', mode: 'reLaunch' })" class="order-grey-hollow-btn ml-[20rpx]">
|
|
{{ t('index') }}
|
|
</view>
|
|
<!-- #ifdef MP-WEIXIN -->
|
|
<view>
|
|
<nc-contact :send-message-title="sendMessageTitle" :send-message-path="sendMessagePath" :send-message-img="sendMessageImg">
|
|
<view class="order-grey-hollow-btn ml-[20rpx]">
|
|
客服
|
|
</view>
|
|
</nc-contact>
|
|
</view>
|
|
<!-- #endif -->
|
|
<view class="order-grey-hollow-btn ml-[20rpx]"
|
|
@click="refundBtnFn('cancel')" v-if="['6','7','8','-1','-3'].indexOf(detail.status) == -1">{{ t('refundApply') }}</view>
|
|
<view v-if="['3'].indexOf(detail.status) != -1" class="order-grey-hollow-btn ml-[20rpx]"
|
|
@click.stop="refundBtnFn('edit')">编辑退款信息</view>
|
|
<view v-if="['2'].indexOf(detail.status) != -1" class="order-grey-hollow-btn ml-[20rpx]"
|
|
@click.stop="refundBtnFn('logistics')">填写发货物流</view>
|
|
<view v-if="['5'].indexOf(detail.status) != -1" class="order-grey-hollow-btn ml-[20rpx]"
|
|
@click.stop="refundBtnFn('editLogistics')">编辑发货物流</view>
|
|
</view>
|
|
</view>
|
|
</view>
|
|
<view v-else>
|
|
<view class="bg-linear">
|
|
<!-- #ifdef MP-WEIXIN || APP-PLUS -->
|
|
<top-tabbar :data="topTabbarData" :scrollBool="topTabarObj.getScrollBool()" />
|
|
<!-- #endif -->
|
|
<view
|
|
class="flex justify-between items-center pl-[40rpx] pr-[50rpx] h-[280rpx] box-border pb-[90rpx]">
|
|
<view class="text-[36rpx] font-500 leading-[42rpx] text-[#fff]">{{ detail.status_name }}</view>
|
|
<view class="flex items-center">
|
|
<image class="w-[180rpx] h-[140rpx]" :src="img('addon/shop/detail/payment.png')" mode="aspectFit" />
|
|
</view>
|
|
</view>
|
|
</view>
|
|
<view class="sidebar-margin card-template mt-[-79rpx] flex justify-between flex-wrap">
|
|
<view class="w-[150rpx] h-[150rpx] flex-2" @click="goodsEvent(detail.order_goods.goods_id)">
|
|
<up-image radius="var(--goods-rounded-big)" width="150rpx" height="150rpx" :src="img(detail.order_goods.sku_image ? detail.order_goods.sku_image.split(',')[0] : '')" model="aspectFill">
|
|
<template #error>
|
|
<image class="w-[150rpx] h-[150rpx] rounded-[var(--goods-rounded-big)] overflow-hidden" :src="img('static/resource/images/diy/shop_default.jpg')" mode="aspectFill"/>
|
|
</template>
|
|
</up-image>
|
|
</view>
|
|
<view class="ml-[20rpx] flex flex-1 flex-col justify-between">
|
|
<view>
|
|
<view class="text-[28rpx] max-w-[490rpx] truncate leading-[40rpx]">{{ detail.order_goods.goods_name }}</view>
|
|
<view class="text-[24rpx] mt-[14rpx] text-[var(--text-color-light9)] truncate max-w-[490rpx] leading-[28rpx]" v-if="detail.order_goods.sku_name">{{ detail.order_goods.sku_name }}</view>
|
|
</view>
|
|
<view class="flex justify-between items-center leading-[28rpx]">
|
|
<view class="price-font">
|
|
<text class="text-[24rpx] font-500">¥</text>
|
|
<text class="text-[40rpx] font-500">{{ parseFloat(detail.order_goods.price).toFixed(2).split('.')[0] }}</text>
|
|
<text class="text-[24rpx] font-500">.{{ parseFloat(detail.order_goods.price).toFixed(2).split('.')[1] }}</text>
|
|
</view>
|
|
<text class="text-right text-[26rpx]">x{{ detail.order_goods.num }}</text>
|
|
</view>
|
|
</view>
|
|
</view>
|
|
<view class="sidebar-margin card-template top-mar">
|
|
<view class="card-template-item justify-between text-[28rpx]">
|
|
<view>联系人</view>
|
|
<view>{{ detail.refund_address.contact_name }}</view>
|
|
</view>
|
|
<view class="card-template-item justify-between text-[28rpx]">
|
|
<view>手机号</view>
|
|
<view>{{ detail.refund_address.mobile }}</view>
|
|
</view>
|
|
<view class="card-template-item justify-between text-[28rpx]">
|
|
<view>退货地址</view>
|
|
<view class="w-[460rpx] text-sm text-right" v-if="detail.refund_address">
|
|
{{ detail.refund_address.full_address || '--' }}
|
|
</view>
|
|
</view>
|
|
</view>
|
|
<view class="sidebar-margin card-template top-mar py-[var(--top-m)] mb-[var(--top-m)]">
|
|
<view class="title">物流信息</view>
|
|
<up-form labelPosition="left" :model="formData" :rules="rules" errorType='toast' ref="deliveryForm" labelWidth="140rpx" :labelStyle="{'fontSize': '28rpx'}">
|
|
<u-form-item label="物流公司" prop="express_company" :borderBottom="false">
|
|
<up-input border="none" v-model="formData.express_company" placeholder="请输入物流公司"
|
|
placeholderClass="text-sm !text-[var(--text-color-light9)]" fontSize="28rpx"
|
|
maxlength="50"></up-input>
|
|
</u-form-item>
|
|
<view class="mt-[16rpx]">
|
|
<u-form-item label="物流单号" prop="express_number" :borderBottom="false">
|
|
<up-input border="none" placeholder="请输入物流单号" v-model="formData.express_number"
|
|
placeholderClass="text-sm !text-[var(--text-color-light9)]" fontSize="28rpx"
|
|
maxlength="100"></up-input>
|
|
</u-form-item>
|
|
</view>
|
|
<view class="mt-[16rpx]">
|
|
<u-form-item label="物流说明" :borderBottom="false">
|
|
<up-input border="none" placeholder="选填" v-model="formData.remark"
|
|
placeholderClass="text-sm !text-[var(--text-color-light9)]" fontSize="28rpx"
|
|
maxlength="100"></up-input>
|
|
</u-form-item>
|
|
</view>
|
|
</up-form>
|
|
</view>
|
|
<u-tabbar :fixed="true" :placeholder="true" :safeAreaInsetBottom="true" zIndex="10">
|
|
<view class="flex-1 pl-[var(--sidebar-m)] pr-[var(--sidebar-m)] bg-[var(--page-bg-color)]">
|
|
<button class=" primary-btn-bg text-[#fff] h-[80rpx] leading-[80rpx] rounded-[100rpx] text-[26rpx] font-500"
|
|
hover-class="none" @click="deliverySave">提交</button>
|
|
</view>
|
|
</u-tabbar>
|
|
|
|
</view>
|
|
<logistics-tracking ref="materialRef"></logistics-tracking>
|
|
<u-modal :show="cancelRefundShow" confirmColor="var(--primary-color)" :content="t('cancelRefundContent')"
|
|
:showCancelButton="true" :closeOnClickOverlay="true" @cancel="refundCancel"
|
|
@confirm="refundConfirm"></u-modal>
|
|
</view>
|
|
|
|
<loading-page :loading="loading"></loading-page>
|
|
|
|
<!-- #ifdef MP-WEIXIN -->
|
|
<!-- 小程序隐私协议 -->
|
|
<wx-privacy-popup ref="wxPrivacyPopupRef"></wx-privacy-popup>
|
|
<!-- #endif -->
|
|
</view>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ref, reactive, computed, nextTick } from 'vue';
|
|
import { onLoad } from '@dcloudio/uni-app'
|
|
import { t } from '@/locale'
|
|
import { img, redirect, copy, goback } from '@/utils/common';
|
|
import { topTabar } from '@/utils/topTabbar'
|
|
import { getRefundDetail, refundDelivery, editRefundDelivery, closeRefund } from '@/addon/shop/api/refund';
|
|
import logisticsTracking from '@/addon/shop/pages/order/components/logistics-tracking/logistics-tracking.vue'
|
|
|
|
/********* 自定义头部 - start ***********/
|
|
const topTabarObj = topTabar()
|
|
let topTabbarData = topTabarObj.setTopTabbarParam({ title: '退款详情' })
|
|
/********* 自定义头部 - end ***********/
|
|
|
|
const detail = ref<Object>({});
|
|
const loading = ref<boolean>(true);
|
|
const orderRefundNo = ref('');
|
|
const type = ref('');
|
|
const isEditDelivery = ref(false);
|
|
|
|
const sendMessageTitle = ref('')
|
|
const sendMessagePath = ref('')
|
|
const sendMessageImg = ref('')
|
|
// 物流信息
|
|
const formData = ref({
|
|
express_number: '',
|
|
express_company: '',
|
|
remark: ''
|
|
})
|
|
|
|
// 物流验证
|
|
const rules = computed(() => {
|
|
return {
|
|
'express_number': {
|
|
type: 'string',
|
|
required: true,
|
|
message: '请输入物流单号',
|
|
trigger: ['blur', 'change']
|
|
},
|
|
'express_company': {
|
|
type: 'string',
|
|
required: true,
|
|
message: '请输入物流公司',
|
|
trigger: ['blur', 'change']
|
|
},
|
|
}
|
|
})
|
|
|
|
const wxPrivacyPopupRef: any = ref(null)
|
|
|
|
onLoad((option: any) => {
|
|
orderRefundNo.value = option.order_refund_no;
|
|
if (orderRefundNo.value) {
|
|
type.value = option.type;
|
|
isEditDelivery.value = option.is_edit_delivery;
|
|
|
|
refundDetailFn(orderRefundNo.value);
|
|
// #ifdef MP
|
|
nextTick(() => {
|
|
if (wxPrivacyPopupRef.value) wxPrivacyPopupRef.value.proactive();
|
|
})
|
|
// #endif
|
|
} else {
|
|
let parameter = {
|
|
url: '/addon/shop/pages/refund/list',
|
|
title: '缺少订单号'
|
|
};
|
|
goback(parameter);
|
|
}
|
|
});
|
|
|
|
const refundDetailFn = (refundNo: any) => {
|
|
loading.value = true;
|
|
getRefundDetail(refundNo).then((res: any) => {
|
|
detail.value = res.data;
|
|
// 赋值物流信息
|
|
if (isEditDelivery.value && detail.value.delivery) {
|
|
formData.value.express_number = detail.value.delivery.express_number
|
|
formData.value.express_company = detail.value.delivery.express_company
|
|
formData.value.remark = detail.value.delivery.remark
|
|
}
|
|
|
|
sendMessageTitle.value = detail.value.order_goods.goods_name
|
|
sendMessageImg.value = img(detail.value.order_goods.goods_image_thumb_small || '')
|
|
loading.value = false;
|
|
}).catch(() => {
|
|
loading.value = false;
|
|
})
|
|
}
|
|
|
|
const goodsEvent = (id: number) => {
|
|
redirect({
|
|
url: '/addon/shop/pages/goods/detail',
|
|
param: {
|
|
goods_id: id
|
|
}
|
|
})
|
|
}
|
|
|
|
// 提交物流信息
|
|
const deliveryForm = ref()
|
|
const deliverySave = () => {
|
|
deliveryForm.value.validate().then(res => {
|
|
let obj = { delivery: formData.value, order_refund_no: detail.value.order_refund_no }
|
|
let api = isEditDelivery.value ? editRefundDelivery(obj) : refundDelivery(obj);
|
|
api.then((res) => {
|
|
setTimeout(() => {
|
|
redirect({ url: '/addon/shop/pages/refund/list' })
|
|
}, 500)
|
|
}).catch(() => {
|
|
})
|
|
})
|
|
}
|
|
|
|
const refundBtnFn = (type: any) => {
|
|
if (type == 'cancel') {
|
|
currRefundOn = detail.value.order_refund_no;
|
|
cancelRefundShow.value = true;
|
|
} else if (type == 'edit') {
|
|
redirect({
|
|
url: '/addon/shop/pages/refund/edit_apply',
|
|
param: { order_refund_no: detail.value.order_refund_no }
|
|
})
|
|
} else if (type == 'logistics') {
|
|
redirect({
|
|
url: '/addon/shop/pages/refund/detail',
|
|
param: { order_refund_no: detail.value.order_refund_no, type: 'logistics' }
|
|
})
|
|
} else if (type == 'editLogistics') {
|
|
redirect({
|
|
url: '/addon/shop/pages/refund/detail',
|
|
param: { order_refund_no: detail.value.order_refund_no, type: 'logistics', is_edit_delivery: true }
|
|
})
|
|
}
|
|
}
|
|
|
|
// 撤销售后
|
|
const cancelRefundShow = ref(false);
|
|
let currRefundOn = "";
|
|
|
|
const refundConfirm = () => {
|
|
closeRefund(currRefundOn).then((res) => {
|
|
cancelRefundShow.value = false;
|
|
refundDetailFn(orderRefundNo.value);
|
|
}).catch(() => {
|
|
cancelRefundShow.value = false;
|
|
})
|
|
}
|
|
|
|
const refundCancel = () => {
|
|
cancelRefundShow.value = false;
|
|
}
|
|
|
|
</script>
|
|
<style lang="scss" scoped>
|
|
.text-item {
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
display: -webkit-box;
|
|
-webkit-line-clamp: 2;
|
|
-webkit-box-orient: vertical;
|
|
}
|
|
|
|
.bg-linear {
|
|
background: linear-gradient(94deg, #F84949 8%, #FF9A68 99%);
|
|
}
|
|
|
|
:deep(.u-form-item__body__left__content__label) {
|
|
height: 40rpx !important;
|
|
}
|
|
|
|
:deep(.u-form-item__body) {
|
|
padding: 0 !important;
|
|
height: 40rpx !important;
|
|
}
|
|
|
|
:deep(.u-form-item) {
|
|
margin-bottom: 34rpx;
|
|
}
|
|
</style>
|