480 lines
20 KiB
Vue
480 lines
20 KiB
Vue
<template>
|
|
<div class="app-container" v-loading="loading">
|
|
<el-row :gutter="10">
|
|
<el-col :span="18">
|
|
<el-card class="mb10">
|
|
<el-row :gutter="20" type="flex">
|
|
<el-col>
|
|
<el-statistic title="骑行费" :value="detail.ridingFee" :precision="2" suffix="元">
|
|
<template slot="prefix">
|
|
<i class="el-icon-bicycle" style="color: #409EFF"></i>
|
|
</template>
|
|
</el-statistic>
|
|
</el-col>
|
|
<el-col>
|
|
<el-statistic title="押金" :value="detail.depositFee" :precision="2" suffix="元">
|
|
<template slot="prefix">
|
|
<i class="el-icon-money" style="color: #67C23A"></i>
|
|
</template>
|
|
</el-statistic>
|
|
</el-col>
|
|
<el-col v-if="detail.ridePayAmount">
|
|
<el-statistic title="结算支付" :value="detail.ridePayAmount" :precision="2" suffix="元">
|
|
<template slot="prefix">
|
|
<i class="el-icon-money" style="color: #67C23A"></i>
|
|
</template>
|
|
</el-statistic>
|
|
</el-col>
|
|
<el-col>
|
|
<el-popover placement="bottom" width="200" trigger="hover">
|
|
<div>
|
|
<div>骑行费:{{detail.ridingFee | fix2 | dv}} 元</div>
|
|
<div>调度费:{{detail.dispatchFee | fix2 | dv}} 元</div>
|
|
<div>管理费:{{detail.manageFee | fix2 | dv}} 元</div>
|
|
<div>车损费:{{detail.deductionFee | fix2 | dv}} 元</div>
|
|
</div>
|
|
<el-statistic slot="reference" :value="detail.totalFee" :precision="2" suffix="元">
|
|
<template slot="prefix">
|
|
<i class="el-icon-wallet" style="color: #E6A23C"></i>
|
|
</template>
|
|
<template slot="title">
|
|
应收
|
|
<el-tag size="mini" type="danger" v-if="detail.priceChanged">改</el-tag>
|
|
<i class="el-icon-info" v-else style="color: #909399; margin-left: 4px; cursor: pointer"></i>
|
|
</template>
|
|
</el-statistic>
|
|
</el-popover>
|
|
</el-col>
|
|
<el-col v-if="OrderStatus.finishedList().includes(detail.status)">
|
|
<el-popover placement="bottom" width="200" trigger="hover">
|
|
<div>
|
|
<div>骑行费:{{detail.actualRidingFee | fix2 | dv}} 元</div>
|
|
<div>调度费:{{detail.actualDispatchFee | fix2 | dv}} 元</div>
|
|
<div>管理费:{{detail.actualManageFee | fix2 | dv}} 元</div>
|
|
<div>车损费:{{detail.actualDeductionFee | fix2 | dv}} 元</div>
|
|
<br/>
|
|
<div>合计:{{detail.actualAmount | fix2 | dv}} 元</div>
|
|
<div v-if="detail.depositDeductionAmount">押金抵扣:{{detail.depositDeductionAmount | fix2 | dv}} 元</div>
|
|
</div>
|
|
<el-statistic slot="reference"
|
|
title="实收"
|
|
:value="detail.actualReceivedAmount"
|
|
:precision="2"
|
|
suffix="元"
|
|
value-style="color: #67C23A">
|
|
<template slot="prefix">
|
|
<i class="el-icon-success" style="color: #67C23A"></i>
|
|
</template>
|
|
<template slot="title">
|
|
实收 <i class="el-icon-info" style="color: #909399; margin-left: 4px; cursor: pointer"></i>
|
|
</template>
|
|
</el-statistic>
|
|
</el-popover>
|
|
</el-col>
|
|
<el-col>
|
|
<el-statistic
|
|
title="总退款"
|
|
:value="detail.totalRefundAmount"
|
|
:precision="2"
|
|
suffix="元"
|
|
value-style="color: #F56C6C">
|
|
<template slot="prefix">
|
|
<i class="el-icon-refresh-left" style="color: #F56C6C"></i>
|
|
</template>
|
|
</el-statistic>
|
|
</el-col>
|
|
<el-col>
|
|
<el-statistic
|
|
title="自动退款"
|
|
:value="detail.payAutoRefund"
|
|
:precision="2"
|
|
suffix="元"
|
|
value-style="color: #F56C6C">
|
|
<template slot="prefix">
|
|
<i class="el-icon-timer" style="color: #F56C6C"></i>
|
|
</template>
|
|
</el-statistic>
|
|
</el-col>
|
|
<el-col>
|
|
<el-popover placement="bottom" width="200" trigger="hover">
|
|
<div>
|
|
<div>骑行费:{{detail.ridingRefund | fix2 | dv}} 元</div>
|
|
<div>调度费:{{detail.dispatchRefund | fix2 | dv}} 元</div>
|
|
<div>管理费:{{detail.manageRefund | fix2 | dv}} 元</div>
|
|
<div>车损费:{{detail.deductionRefund | fix2 | dv}} 元</div>
|
|
</div>
|
|
<el-statistic
|
|
slot="reference"
|
|
:value="detail.adminRefundAmount"
|
|
:precision="2"
|
|
suffix="元"
|
|
value-style="color: #F56C6C">
|
|
<template slot="prefix">
|
|
<i class="el-icon-user" style="color: #F56C6C"></i>
|
|
</template>
|
|
<template slot="title">
|
|
人工退款 <i class="el-icon-info" style="color: #909399; margin-left: 4px; cursor: pointer"></i>
|
|
</template>
|
|
</el-statistic>
|
|
</el-popover>
|
|
</el-col>
|
|
</el-row>
|
|
</el-card>
|
|
|
|
<el-card>
|
|
<el-row class="mb10" type="flex" justify="end">
|
|
<el-button
|
|
size="small"
|
|
plain
|
|
type="danger"
|
|
icon="el-icon-close"
|
|
@click="handleEnd(detail)"
|
|
v-has-permi="['bst:order:end']"
|
|
v-show="OrderStatus.canEnd().includes(detail.status)"
|
|
>结束订单</el-button>
|
|
<el-button
|
|
size="small"
|
|
plain
|
|
type="warning"
|
|
icon="el-icon-wallet"
|
|
@click="handleRefund(detail)"
|
|
v-has-permi="['bst:order:refund']"
|
|
v-show="OrderStatus.canRefund().includes(detail.status)"
|
|
>退款</el-button>
|
|
<el-button
|
|
size="small"
|
|
plain
|
|
type="warning"
|
|
icon="el-icon-s-check"
|
|
@click="handleVerify(detail)"
|
|
v-has-permi="['bst:order:verify']"
|
|
v-show="OrderStatus.canVerify().includes(detail.status)"
|
|
>审核</el-button>
|
|
<el-button
|
|
size="small"
|
|
plain
|
|
type="warning"
|
|
icon="el-icon-wallet"
|
|
@click="handleDeduct(detail)"
|
|
v-has-permi="['bst:order:deduct']"
|
|
v-show="OrderStatus.canDeduct().includes(detail.status)"
|
|
>押金抵扣</el-button>
|
|
<el-button
|
|
size="small"
|
|
plain
|
|
type="warning"
|
|
icon="el-icon-edit-outline"
|
|
@click="handleUpdatePrice(detail)"
|
|
v-has-permi="['bst:order:updatePrice']"
|
|
v-show="OrderStatus.canDeduct().includes(detail.status)"
|
|
>改价</el-button>
|
|
</el-row>
|
|
<collapse-panel :value="true" title="基础信息">
|
|
<el-descriptions :column="4" >
|
|
<el-descriptions-item label="订单编号">
|
|
{{ detail.no | dv}}
|
|
<el-link v-clipboard:copy="detail.no" v-clipboard:success="handleCopySuccess" type="primary" icon="el-icon-document"/>
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="订单状态">
|
|
<dict-tag :options="dict.type.order_status" :value="detail.status" size="small"/>
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="运营区">
|
|
<area-link :id="detail.areaId" :text="detail.areaName" />
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="创建时间">{{ detail.createTime | dv}}</el-descriptions-item>
|
|
<el-descriptions-item label="开始时间">{{ detail.startTime | dv}}</el-descriptions-item>
|
|
<el-descriptions-item label="结束时间">{{ detail.endTime | dv}}</el-descriptions-item>
|
|
<el-descriptions-item label="骑行时长">{{ orderDuration | dv}}</el-descriptions-item>
|
|
<el-descriptions-item label="骑行距离">{{ detail.distance / 1000 | fix2 | dv}} 公里</el-descriptions-item>
|
|
<el-descriptions-item label="结束原因" v-if="detail.endReason">{{ detail.endReason | dv }}</el-descriptions-item>
|
|
<el-descriptions-item label="取消原因" v-if="detail.cancelRemark">{{ detail.cancelRemark | dv }}</el-descriptions-item>
|
|
</el-descriptions>
|
|
</collapse-panel>
|
|
|
|
|
|
<collapse-panel :value="true" title="套餐信息">
|
|
<el-descriptions :column="4" >
|
|
<el-descriptions-item label="套餐名称" :span="2">
|
|
{{ detail.suitName }}
|
|
<dict-tag :options="dict.type.suit_type" :value="detail.suitType" size="mini" style="margin-left: 4px;"/>
|
|
<dict-tag :options="dict.type.suit_riding_rule" :value="detail.suitRidingRule" size="mini" style="margin-left: 4px;"/>
|
|
<el-tag v-if="detail.suitDepositDeduction" type="success" size="mini" style="margin-left: 4px;">自动抵扣</el-tag>
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="免费骑行时间" :span="3">
|
|
{{ detail.suitFreeRideTime | dv }} 分钟
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="起步规则" :span="3" v-if="SuitRidingRule.START === detail.suitRidingRule">
|
|
在{{detail.suitStartRule.startingTime}}{{unitLabel(detail.suitRentalUnit)}}以内,起步价{{detail.suitStartRule.startingPrice}}元;
|
|
超出起步时间后,超出的时间每{{detail.suitStartRule.timeoutTime}}{{unitLabel(detail.suitRentalUnit)}}收费{{detail.suitStartRule.timeoutPrice}}元,
|
|
不满{{detail.suitStartRule.timeoutTime}}{{unitLabel(detail.suitRentalUnit)}},按{{detail.suitStartRule.timeoutTime}}{{unitLabel(detail.suitRentalUnit)}}计算。
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="区间规则" :span="3" v-if="SuitRidingRule.INTERVAL === detail.suitRidingRule">
|
|
<template v-if="detail.suitIntervalRule && detail.suitIntervalRule.length > 0">
|
|
<div v-for="(rule, index) in detail.suitIntervalRule" :key="index">
|
|
<template v-if="index === detail.suitIntervalRule.length - 1">
|
|
在{{rule.start}}{{unitLabel(detail.suitRentalUnit)}}之后,
|
|
每{{rule.eachUnit}}{{unitLabel(detail.suitRentalUnit)}}收费{{rule.fee}}元;
|
|
</template>
|
|
<template v-else>
|
|
在{{rule.start}}~{{rule.end}}{{unitLabel(detail.suitRentalUnit)}}之间,
|
|
每{{rule.eachUnit}}{{unitLabel(detail.suitRentalUnit)}}收费{{rule.fee}}元;
|
|
</template>
|
|
</div>
|
|
</template>
|
|
<template v-else>
|
|
暂无区间规则
|
|
</template>
|
|
</el-descriptions-item>
|
|
</el-descriptions>
|
|
</collapse-panel>
|
|
</el-card>
|
|
</el-col>
|
|
<el-col :span="6">
|
|
<el-card>
|
|
<collapse-panel :value="true" title="设备信息">
|
|
<el-descriptions :column="1" >
|
|
<el-descriptions-item label="SN">
|
|
<device-link :id="detail.deviceId" :text="detail.deviceSn"/>
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="MAC">
|
|
<device-link :id="detail.deviceId" :text="detail.deviceMac"/>
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="车牌号">
|
|
<device-link :id="detail.deviceId" :text="detail.deviceVehicleNum"/>
|
|
</el-descriptions-item>
|
|
</el-descriptions>
|
|
</collapse-panel>
|
|
|
|
<collapse-panel :value="true" title="用户信息">
|
|
<el-descriptions :column="1" >
|
|
<el-descriptions-item label="用户">
|
|
<user-link :id="detail.userId" :text="detail.userName"/>
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="手机">
|
|
<user-link :id="detail.userId" :text="detail.userPhone"/>
|
|
</el-descriptions-item>
|
|
</el-descriptions>
|
|
</collapse-panel>
|
|
|
|
<collapse-panel :value="true" title="支付信息">
|
|
<el-descriptions :column="1" >
|
|
<el-descriptions-item label="支付方式">
|
|
<dict-tag :options="dict.type.order_pay_type" :value="detail.payType" size="mini"/>
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="支付时间">
|
|
{{ detail.ridePayTime | dv }}
|
|
</el-descriptions-item>
|
|
<el-descriptions-item label="押金抵扣" v-if="detail.depositDeductionAmount">
|
|
{{ detail.depositDeductionAmount | fix2 | dv }} 元
|
|
</el-descriptions-item>
|
|
</el-descriptions>
|
|
</collapse-panel>
|
|
</el-card>
|
|
</el-col>
|
|
</el-row>
|
|
|
|
<el-card class="box-card" v-if="detail.id" style="margin-top: 10px;">
|
|
<el-tabs >
|
|
<el-tab-pane lazy label="车辆轨迹" v-if="checkPermi(['bst:locationLog:list'])">
|
|
<device-location
|
|
:query="{orderId: detail.id, timeRange: orderTimeRange}"
|
|
:operQuery="{bizId: detail.id, bizType: LogBizType.ORDER, operTimeRange: []}"
|
|
:area-id="detail.areaId"
|
|
/>
|
|
</el-tab-pane>
|
|
<el-tab-pane lazy label="收益信息" v-if="checkPermi(['bst:bonus:list'])">
|
|
<bonus :query="{bstId: detail.id, bstType: BonusBstType.ORDER}" />
|
|
</el-tab-pane>
|
|
<el-tab-pane lazy label="订单车辆" v-if="checkPermi(['bst:orderDevice:list'])">
|
|
<order-device :query="{orderId: detail.id}" />
|
|
</el-tab-pane>
|
|
<el-tab-pane lazy label="支付信息" v-if="checkPermi(['bst:pay:list'])">
|
|
<pay :query="{bstId: detail.id, bstTypes: payBstTypes}"/>
|
|
</el-tab-pane>
|
|
<el-tab-pane lazy label="退款信息" v-if="checkPermi(['bst:refund:list'])">
|
|
<refund :query="{payBstId: detail.id, payBstTypes: payBstTypes}"/>
|
|
</el-tab-pane>
|
|
<el-tab-pane lazy label="命令日志" v-if="checkPermi(['bst:commandLog:list'])">
|
|
<command-log :query="{orderId: detail.id}"/>
|
|
</el-tab-pane>
|
|
<el-tab-pane lazy label="改价记录" v-if="checkPermi(['bst:feeLog:list'])">
|
|
<fee-log :query="{orderId: detail.id}"/>
|
|
</el-tab-pane>
|
|
<el-tab-pane lazy label="操作日志" v-if="checkPermi(['monitor:operlog:list'])">
|
|
<operlog :query="{bizId: detail.id, bizType: LogBizType.ORDER}"/>
|
|
</el-tab-pane>
|
|
</el-tabs>
|
|
</el-card>
|
|
|
|
<order-refund-dialog :id="detail.id" :visible.sync="showRefundDialog" @success="getDetail" />
|
|
|
|
<order-verify-dialog :id="detail.id" :visible.sync="showVerifyDialog" @success="getDetail" />
|
|
|
|
<order-deduct-dialog :id="detail.id" :visible.sync="showDeductDialog" @success="getDetail" />
|
|
|
|
<order-change-price-dialog :id="detail.id" :visible.sync="showUpdatePriceDialog" @success="getDetail" />
|
|
</div>
|
|
|
|
</template>
|
|
|
|
<script>
|
|
import { appCalcOrderFee } from '@/api/app/order'
|
|
import { endOrder, getOrder } from '@/api/bst/order'
|
|
import AreaLink from '@/components/Business/Area/AreaLink.vue'
|
|
import DeviceLink from '@/components/Business/Device/DeviceLink.vue'
|
|
import UserLink from '@/components/Business/User/UserLink.vue'
|
|
import CollapsePanel from '@/components/CollapsePanel/index.vue'
|
|
import { getLastDateTimeEndStr, toDescriptionFromSecond } from '@/utils/date'
|
|
import { BonusBstType, LogBizType, OrderStatus, PayBstType, SuitRidingRule } from '@/utils/enums'
|
|
import Bonus from '@/views/bst/bonus/index.vue'
|
|
import CommandLog from '@/views/bst/commandLog/index.vue'
|
|
import DeviceLocation from '@/views/bst/device/view/components/DeviceLocation.vue'
|
|
import FeeLog from '@/views/bst/feeLog/index.vue'
|
|
import OrderChangePriceDialog from '@/views/bst/order/components/OrderChangePriceDialog.vue'
|
|
import OrderDeductDialog from '@/views/bst/order/components/OrderDeductDialog.vue'
|
|
import OrderRefundDialog from '@/views/bst/order/components/OrderRefundDialog.vue'
|
|
import OrderVerifyDialog from '@/views/bst/order/components/OrderVerifyDialog.vue'
|
|
import { getOrderDuration } from '@/views/bst/order/util'
|
|
import OrderDevice from '@/views/bst/orderDevice/index.vue'
|
|
import Pay from '@/views/bst/pay/index.vue'
|
|
import Refund from '@/views/bst/refund/index.vue'
|
|
import Operlog from '@/views/monitor/operlog/index.vue'
|
|
|
|
export default {
|
|
name: 'OrderView',
|
|
dicts: [
|
|
'order_status',
|
|
'device_lock_status',
|
|
'suit_type',
|
|
'suit_rental_unit',
|
|
'suit_riding_rule',
|
|
'order_return_mode',
|
|
'order_return_type',
|
|
'order_pay_type'
|
|
],
|
|
components: {
|
|
CollapsePanel,
|
|
OrderDevice,
|
|
Pay,
|
|
Bonus,
|
|
DeviceLocation,
|
|
OrderRefundDialog,
|
|
OrderVerifyDialog,
|
|
Refund,
|
|
Operlog,
|
|
CommandLog,
|
|
DeviceLink,
|
|
UserLink,
|
|
AreaLink,
|
|
OrderDeductDialog,
|
|
FeeLog,
|
|
OrderChangePriceDialog
|
|
},
|
|
data() {
|
|
return {
|
|
payBstTypes: [PayBstType.ORDER, PayBstType.ORDER_RIDE],
|
|
OrderStatus,
|
|
id: null,
|
|
detail: {},
|
|
loading: false,
|
|
SuitRidingRule,
|
|
PayBstType,
|
|
BonusBstType,
|
|
LogBizType,
|
|
showRefundDialog: false,
|
|
showVerifyDialog: false,
|
|
showDeductDialog: false,
|
|
showUpdatePriceDialog: false,
|
|
}
|
|
},
|
|
computed: {
|
|
orderDuration() {
|
|
return toDescriptionFromSecond(getOrderDuration(this.detail)).text;
|
|
},
|
|
orderTimeRange() {
|
|
if (this.detail.startTime && this.detail.endTime) {
|
|
return [this.detail.startTime, this.detail.endTime];
|
|
}
|
|
return [this.detail.startTime, getLastDateTimeEndStr(0)];
|
|
}
|
|
},
|
|
created() {
|
|
this.id = this.$route.params.id
|
|
this.getDetail()
|
|
},
|
|
methods: {
|
|
handleCopySuccess() {
|
|
this.$message.success("复制成功");
|
|
},
|
|
toDescriptionFromSecond,
|
|
getDetail() {
|
|
this.loading = true
|
|
getOrder(this.id).then(res => {
|
|
this.detail = res.data;
|
|
}).finally(() => {
|
|
this.loading = false
|
|
})
|
|
},
|
|
handleRefund(row) {
|
|
this.showRefundDialog = true;
|
|
},
|
|
handleVerify(row) {
|
|
this.showVerifyDialog = true;
|
|
},
|
|
handleDeduct(row) {
|
|
this.showDeductDialog = true;
|
|
},
|
|
handleUpdatePrice(row) {
|
|
this.showUpdatePriceDialog = true;
|
|
},
|
|
async handleEnd(row) {
|
|
const calcRes = await appCalcOrderFee({orderId: row.id, checkLocation: false})
|
|
let fee = {}
|
|
if (calcRes.code === 200 && calcRes.data) {
|
|
fee = calcRes.data
|
|
}
|
|
|
|
if (fee.manageFee || fee.dispatchFee) {
|
|
this.$confirm(`订单${row.no}存在${fee.dispatchFee ? '调度费' + fee.dispatchFee + '元' : ''}${fee.manageFee ? '管理费' + fee.manageFee + '元' : ''},是否收取?`, '提示', {
|
|
confirmButtonText: '收取',
|
|
cancelButtonText: '不收取',
|
|
type: 'warning'
|
|
}).then(() => {
|
|
this.doEndOrder(row.id, true);
|
|
}).catch(() => {
|
|
this.doEndOrder(row.id, false);
|
|
});
|
|
} else {
|
|
this.$confirm(`确定结束订单${row.no}吗?`, '提示', {
|
|
confirmButtonText: '确定',
|
|
cancelButtonText: '取消',
|
|
type: 'warning'
|
|
}).then(() => {
|
|
this.doEndOrder(row.id, false);
|
|
});
|
|
}
|
|
},
|
|
doEndOrder(id, needDispatchFee) {
|
|
endOrder(id, needDispatchFee).then(res => {
|
|
this.$message.success("结束成功");
|
|
this.getList();
|
|
});
|
|
},
|
|
unitLabel(value) {
|
|
return this.dict.type.suit_rental_unit.find(item => item.value === value)?.label || value;
|
|
},
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<style scoped>
|
|
.red-text {
|
|
color: red;
|
|
}
|
|
|
|
.green-text {
|
|
color: green;
|
|
}
|
|
</style>
|
|
|