index.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358
  1. <template>
  2. <ContentWrap>
  3. <!-- 订单信息 -->
  4. <el-descriptions title="订单信息">
  5. <el-descriptions-item label="订单号: ">{{ formData.no }}</el-descriptions-item>
  6. <el-descriptions-item label="配送方式: ">
  7. <dict-tag :type="DICT_TYPE.TRADE_DELIVERY_TYPE" :value="formData.deliveryType!" />
  8. </el-descriptions-item>
  9. <!-- TODO 营销活动待实现 -->
  10. <el-descriptions-item label="营销活动: ">秒杀活动</el-descriptions-item>
  11. <el-descriptions-item label="订单类型: ">
  12. <dict-tag :type="DICT_TYPE.TRADE_ORDER_TYPE" :value="formData.type!" />
  13. </el-descriptions-item>
  14. <el-descriptions-item label="收货人: ">{{ formData.receiverName }}</el-descriptions-item>
  15. <el-descriptions-item label="买家留言: ">{{ formData.userRemark }}</el-descriptions-item>
  16. <el-descriptions-item label="订单来源: ">
  17. <dict-tag :type="DICT_TYPE.TERMINAL" :value="formData.terminal!" />
  18. </el-descriptions-item>
  19. <el-descriptions-item label="联系电话: ">{{ formData.receiverMobile }}</el-descriptions-item>
  20. <el-descriptions-item label="商家备注: ">{{ formData.remark }}</el-descriptions-item>
  21. <el-descriptions-item label="支付单号: ">{{ formData.payOrderId }}</el-descriptions-item>
  22. <el-descriptions-item label="付款方式: ">
  23. <dict-tag :type="DICT_TYPE.PAY_CHANNEL_CODE" :value="formData.payChannelCode!" />
  24. </el-descriptions-item>
  25. <!-- <el-descriptions-item label="买家: ">{{ formData.user.nickname }}</el-descriptions-item> -->
  26. <!-- TODO @puhui999:待实现:跳转会员 -->
  27. <el-descriptions-item label="收货地址: ">
  28. {{ formData.receiverAreaName }} {{ formData.receiverDetailAddress }}
  29. <el-link
  30. v-clipboard:copy="formData.receiverAreaName + ' ' + formData.receiverDetailAddress"
  31. v-clipboard:success="clipboardSuccess"
  32. icon="ep:document-copy"
  33. type="primary"
  34. />
  35. </el-descriptions-item>
  36. </el-descriptions>
  37. <!-- 订单状态 -->
  38. <el-descriptions :column="1" title="订单状态">
  39. <el-descriptions-item label="订单状态: ">
  40. <dict-tag :type="DICT_TYPE.TRADE_ORDER_STATUS" :value="formData.status!" />
  41. </el-descriptions-item>
  42. <el-descriptions-item label-class-name="no-colon">
  43. <el-button v-if="formData.status! === 0" type="primary" @click="updatePrice">
  44. 调整价格
  45. </el-button>
  46. <el-button type="primary" @click="remark">备注</el-button>
  47. <el-button v-if="formData.status! === 10" type="primary" @click="delivery">
  48. 发货
  49. </el-button>
  50. <el-button v-if="formData.status! === 10" type="primary" @click="updateAddress">
  51. 修改地址
  52. </el-button>
  53. </el-descriptions-item>
  54. <el-descriptions-item>
  55. <template #label><span style="color: red">提醒: </span></template>
  56. 买家付款成功后,货款将直接进入您的商户号(微信、支付宝)<br />
  57. 请及时关注你发出的包裹状态,确保可以配送至买家手中 <br />
  58. 如果买家表示没收到货或货物有问题,请及时联系买家处理,友好协商
  59. </el-descriptions-item>
  60. </el-descriptions>
  61. <!-- 商品信息 -->
  62. <el-descriptions title="商品信息">
  63. <el-descriptions-item labelClassName="no-colon">
  64. <el-row :gutter="20">
  65. <el-col :span="15">
  66. <el-table :data="formData.items" border>
  67. <el-table-column label="商品" prop="spuName" width="auto">
  68. <template #default="{ row }">
  69. {{ row.spuName }}
  70. <el-tag v-for="property in row.properties" :key="property.propertyId">
  71. {{ property.propertyName }}: {{ property.valueName }}
  72. </el-tag>
  73. </template>
  74. </el-table-column>
  75. <el-table-column label="商品原价" prop="price" width="150">
  76. <template #default="{ row }">{{ floatToFixed2(row.price) }}元</template>
  77. </el-table-column>
  78. <el-table-column label="数量" prop="count" width="100" />
  79. <el-table-column label="合计" prop="payPrice" width="150">
  80. <template #default="{ row }">{{ floatToFixed2(row.payPrice) }}元</template>
  81. </el-table-column>
  82. <el-table-column label="售后状态" prop="afterSaleStatus" width="120">
  83. <template #default="{ row }">
  84. <dict-tag
  85. :type="DICT_TYPE.TRADE_ORDER_ITEM_AFTER_SALE_STATUS"
  86. :value="row.afterSaleStatus"
  87. />
  88. </template>
  89. </el-table-column>
  90. </el-table>
  91. </el-col>
  92. <el-col :span="10" />
  93. </el-row>
  94. </el-descriptions-item>
  95. </el-descriptions>
  96. <el-descriptions :column="6">
  97. <el-descriptions-item label="商品总额: ">
  98. {{ floatToFixed2(formData.totalPrice!) }}元
  99. </el-descriptions-item>
  100. <el-descriptions-item label="运费金额: ">
  101. {{ floatToFixed2(formData.deliveryPrice!) }}元
  102. </el-descriptions-item>
  103. <el-descriptions-item label="订单调价: ">
  104. {{ floatToFixed2(formData.adjustPrice!) }}元
  105. </el-descriptions-item>
  106. <el-descriptions-item>
  107. <template #label><span style="color: red">商品优惠: </span></template>
  108. {{ floatToFixed2(formData.couponPrice!) }}元
  109. </el-descriptions-item>
  110. <el-descriptions-item>
  111. <template #label><span style="color: red">订单优惠: </span></template>
  112. {{ floatToFixed2(formData.discountPrice!) }}元
  113. </el-descriptions-item>
  114. <el-descriptions-item>
  115. <template #label><span style="color: red">积分抵扣: </span></template>
  116. {{ floatToFixed2(formData.pointPrice!) }}元
  117. </el-descriptions-item>
  118. <el-descriptions-item v-for="item in 5" :key="item" label-class-name="no-colon" />
  119. <!-- 占位 -->
  120. <el-descriptions-item label="应付金额: ">
  121. {{ floatToFixed2(formData.payPrice!) }}元
  122. </el-descriptions-item>
  123. </el-descriptions>
  124. <!-- TODO 芋艿:需要改改 -->
  125. <el-descriptions :column="4" title="物流信息">
  126. <el-descriptions-item label="物流公司: ">
  127. {{ deliveryExpressList.find((item) => item.id === formData.logisticsId)?.name }}
  128. </el-descriptions-item>
  129. <el-descriptions-item label="运单号: ">{{ formData.logisticsNo }}</el-descriptions-item>
  130. <el-descriptions-item label="发货时间: ">
  131. {{ formatDate(formData.deliveryTime!) }}
  132. </el-descriptions-item>
  133. <el-descriptions-item label="物流状态: ">
  134. <!-- TODO 物流状态怎么获取? -->
  135. <dict-tag :type="DICT_TYPE.TRADE_ORDER_STATUS" :value="formData.deliveryStatus!" />
  136. </el-descriptions-item>
  137. <!-- 占位 4 -->
  138. <el-descriptions-item v-for="item in 4" :key="item" label-class-name="no-colon" />
  139. <el-descriptions-item label="物流详情: ">
  140. <el-timeline>
  141. <el-timeline-item
  142. v-for="(express, index) in expressTrackList"
  143. :key="index"
  144. :timestamp="formatDate(express.time)"
  145. >
  146. {{ express.content }}
  147. </el-timeline-item>
  148. </el-timeline>
  149. </el-descriptions-item>
  150. </el-descriptions>
  151. <el-descriptions title="订单操作日志">
  152. <el-descriptions-item labelClassName="no-colon">
  153. <el-timeline>
  154. <el-timeline-item
  155. v-for="(log, index) in formData.logs"
  156. :key="index"
  157. :timestamp="formatDate(log.createTime!)"
  158. placement="top"
  159. >
  160. <div class="el-timeline-right-content">
  161. {{ log.content }}
  162. </div>
  163. <template #dot>
  164. <span
  165. :style="{ backgroundColor: getUserTypeColor(log.userType!) }"
  166. class="dot-node-style"
  167. >
  168. {{ getDictLabel(DICT_TYPE.USER_TYPE, log.userType)[0] }}
  169. </span>
  170. </template>
  171. </el-timeline-item>
  172. </el-timeline>
  173. </el-descriptions-item>
  174. </el-descriptions>
  175. </ContentWrap>
  176. <!-- 各种操作的弹窗 -->
  177. <OrderDeliveryForm ref="deliveryFormRef" @success="getDetail" />
  178. <OrderUpdateRemarkForm ref="updateRemarkForm" @success="getDetail" />
  179. <OrderUpdateAddressForm ref="updateAddressFormRef" @success="getDetail" />
  180. <OrderUpdatePriceForm ref="updatePriceFormRef" @success="getDetail" />
  181. </template>
  182. <script lang="ts" setup>
  183. import * as TradeOrderApi from '@/api/mall/trade/order'
  184. import { floatToFixed2 } from '@/utils'
  185. import { formatDate } from '@/utils/formatTime'
  186. import { DICT_TYPE, getDictLabel, getDictObj } from '@/utils/dict'
  187. import OrderUpdateRemarkForm from '@/views/mall/trade/order/form/OrderUpdateRemarkForm.vue'
  188. import OrderDeliveryForm from '@/views/mall/trade/order/form/OrderDeliveryForm.vue'
  189. import OrderUpdateAddressForm from '@/views/mall/trade/order/form/OrderUpdateAddressForm.vue'
  190. import OrderUpdatePriceForm from '@/views/mall/trade/order/form/OrderUpdatePriceForm.vue'
  191. import * as DeliveryExpressApi from '@/api/mall/trade/delivery/express'
  192. import { useTagsViewStore } from '@/store/modules/tagsView'
  193. defineOptions({ name: 'TradeOrderDetail' })
  194. const message = useMessage() // 消息弹窗
  195. /** 获得 userType 颜色 */
  196. const getUserTypeColor = (type: number) => {
  197. const dict = getDictObj(DICT_TYPE.USER_TYPE, type)
  198. switch (dict?.colorType) {
  199. case 'success':
  200. return '#67C23A'
  201. case 'info':
  202. return '#909399'
  203. case 'warning':
  204. return '#E6A23C'
  205. case 'danger':
  206. return '#F56C6C'
  207. }
  208. return '#409EFF'
  209. }
  210. // 订单详情
  211. const formData = ref<TradeOrderApi.OrderVO>({
  212. logs: []
  213. })
  214. /** 各种操作 */
  215. const updateRemarkForm = ref() // 订单备注表单 Ref
  216. const remark = () => {
  217. updateRemarkForm.value?.open(formData.value)
  218. }
  219. const deliveryFormRef = ref() // 发货表单 Ref
  220. const delivery = () => {
  221. deliveryFormRef.value?.open(formData.value)
  222. }
  223. const updateAddressFormRef = ref() // 收货地址表单 Ref
  224. const updateAddress = () => {
  225. updateAddressFormRef.value?.open(formData.value)
  226. }
  227. const updatePriceFormRef = ref() // 订单调价表单 Ref
  228. const updatePrice = () => {
  229. updatePriceFormRef.value?.open(formData.value)
  230. }
  231. /** 获得详情 */
  232. const { params } = useRoute() // 查询参数
  233. const getDetail = async () => {
  234. const id = params.orderId as unknown as number
  235. if (id) {
  236. const res = (await TradeOrderApi.getOrder(id)) as TradeOrderApi.OrderVO
  237. // 没有表单信息则关闭页面返回
  238. if (res === null) {
  239. close()
  240. }
  241. formData.value = res
  242. }
  243. }
  244. /** 关闭 tag */
  245. const { delView } = useTagsViewStore() // 视图操作
  246. const { push, currentRoute } = useRouter() // 路由
  247. const close = () => {
  248. delView(unref(currentRoute))
  249. push({ name: 'TradeAfterSale' })
  250. }
  251. /** 复制 */
  252. const clipboardSuccess = () => {
  253. message.success('复制成功')
  254. }
  255. /** 初始化 **/
  256. const deliveryExpressList = ref([]) // 物流公司
  257. const expressTrackList = ref([]) // 物流详情
  258. onMounted(async () => {
  259. await getDetail()
  260. deliveryExpressList.value = await DeliveryExpressApi.getSimpleDeliveryExpressList()
  261. expressTrackList.value = await TradeOrderApi.getExpressTrackList(formData.value.id!)
  262. })
  263. </script>
  264. <style lang="scss" scoped>
  265. :deep(.el-descriptions) {
  266. &:not(:nth-child(1)) {
  267. margin-top: 20px;
  268. }
  269. .el-descriptions__title {
  270. display: flex;
  271. align-items: center;
  272. &::before {
  273. display: inline-block;
  274. width: 3px;
  275. height: 20px;
  276. margin-right: 10px;
  277. background-color: #409eff;
  278. content: '';
  279. }
  280. }
  281. .el-descriptions-item__container {
  282. margin: 0 10px;
  283. .no-colon {
  284. margin: 0;
  285. &::after {
  286. content: '';
  287. }
  288. }
  289. }
  290. }
  291. // 时间线样式调整
  292. :deep(.el-timeline) {
  293. margin: 10px 0px 0px 160px;
  294. .el-timeline-item__wrapper {
  295. position: relative;
  296. top: -20px;
  297. .el-timeline-item__timestamp {
  298. position: absolute !important;
  299. top: 10px;
  300. left: -150px;
  301. }
  302. }
  303. .el-timeline-right-content {
  304. display: flex;
  305. align-items: center;
  306. min-height: 30px;
  307. padding: 10px;
  308. background-color: #f7f8fa;
  309. &::before {
  310. content: ''; /* 必须设置 content 属性 */
  311. position: absolute;
  312. top: 10px;
  313. left: 13px; /* 将伪元素水平居中 */
  314. border-width: 8px; /* 调整尖角大小 */
  315. border-style: solid;
  316. border-color: transparent #f7f8fa transparent transparent; /* 尖角颜色,左侧朝向 */
  317. }
  318. }
  319. .dot-node-style {
  320. width: 20px;
  321. height: 20px;
  322. position: absolute;
  323. left: -5px;
  324. display: flex;
  325. justify-content: center;
  326. align-items: center;
  327. border-radius: 50%;
  328. color: #fff;
  329. font-size: 10px;
  330. }
  331. }
  332. </style>