detail.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701
  1. <template>
  2. <view class="app-container">
  3. <no-data v-if="isLoading" :showText="'加载中'"></no-data>
  4. <no-data v-if="!isLoading && noData" :showText="'加载失败'"></no-data>
  5. <block v-if="!isLoading && !noData">
  6. <view class="status-container" v-if="detail.orderStatus == 'NOPAY'">待付款,30分钟内未付款将会关闭订单</view>
  7. <view class="status-container" v-if="detail.orderStatus == 'DFH'">买家已付款,待卖家发货</view>
  8. <view class="status-container" v-if="detail.orderStatus == 'YFH'">卖家已发货,已签收状态将在签收7日后自动确认</view>
  9. <view class="status-container" v-if="detail.orderStatus == 'OVER'">已确定收货,7日后将关闭退款通道</view>
  10. <view class="status-container" v-if="detail.orderStatus == 'CLOSE'">订单已关闭</view>
  11. <view class="status-container" v-if="detail.orderStatus == 'REFUND'">订单售后中</view>
  12. <view class="status-container" v-if="detail.orderStatus == 'TIMEOUT'">订单超时未支付</view>
  13. <view class="main-container">
  14. <view class="address-container card">
  15. <view class="icon"><image src="@/static/icon/address.png"></image></view>
  16. <view class="right">
  17. <view class="name">{{detail.receUserName}}<text>{{detail.recePhone}}</text></view>
  18. <view class="address ellipsis-2">{{detail.province}}{{detail.city}}{{detail.area}}{{detail.street}}{{detail.receAddress}}{{detail.houseNo ? detail.houseNo : ''}}</view>
  19. </view>
  20. </view>
  21. <view class="goods-container card">
  22. <view class="title">商品信息</view>
  23. <block v-for="(item, index) in detail.orderDetails" :key='index'>
  24. <view class="item" @tap="toGoodsDetail(item.goodsId)">
  25. <image :src="item.imgUrl" mode="aspectFill"></image>
  26. <view class="right">
  27. <view class="top">
  28. <view class="name ellipsis-2">{{item.goodsName}}</view>
  29. <view class="des">{{item.goodsSpecValue}}</view>
  30. </view>
  31. <view class="bottom">
  32. <view class="price">¥{{item.price | numToFixed}}</view>
  33. <view class="num">x{{item.num}}</view>
  34. </view>
  35. </view>
  36. </view>
  37. </block>
  38. </view>
  39. <view class="peisong-container card">
  40. <view class="top">
  41. <view class="left">配送方式</view>
  42. <view class="right" v-if="detail.freight == 0">快递包邮</view>
  43. <view class="right" v-if="detail.freight != 0">快递自费</view>
  44. </view>
  45. <view class="bottom">
  46. <view class="left">买家留言</view>
  47. <view class="right">{{detail.buyerMsg}}</view>
  48. </view>
  49. </view>
  50. <view class="peisong-container card">
  51. <view class="top">
  52. <view class="left">商品金额</view>
  53. <view class="right">¥{{detail.totalProductAmount | numToFixed}}</view>
  54. </view>
  55. <view class="top">
  56. <view class="left">运费</view>
  57. <view class="right">¥{{detail.freight | numToFixed}}</view>
  58. </view>
  59. <view class="top">
  60. <view class="left">优惠券</view>
  61. <view class="right">-¥{{detail.couponValue | numToFixed}}</view>
  62. </view>
  63. <view class="top" v-if="detail.promotionDiscountRate">
  64. <view class="left">折扣优惠</view>
  65. <view class="right">-¥{{detail.promotionDiscountAmount | numToFixed}}({{detail.promotionDiscountRate*10}}折)</view>
  66. </view>
  67. </view>
  68. <view class="apply-container">
  69. <view class="user-info2">
  70. <view class="left">
  71. <view class="label">业务员:</view>
  72. <view class="value">{{detail.workerName}}/{{detail.workerPhone}}</view>
  73. </view>
  74. <view class="right">{{detail.promotionApplyStatus | statusFilter2}}</view>
  75. </view>
  76. <view class="data">
  77. <view class="row">
  78. <view class="label">订单号:</view>
  79. <view class="value">{{detail.orderId}}</view>
  80. </view>
  81. <view class="row">
  82. <view class="label">订单金额:</view>
  83. <view class="value">¥{{((detail.promotionApplyAmount*100 + detail.promotionApplyPayAmount*100))/100 | numToFixed}}</view>
  84. </view>
  85. <view class="row">
  86. <view class="label">减免金额:</view>
  87. <view class="value">¥{{detail.promotionApplyAmount | numToFixed}}</view>
  88. </view>
  89. <view class="row">
  90. <view class="label">应付金额:</view>
  91. <view class="value">¥{{detail.promotionApplyPayAmount | numToFixed}}</view>
  92. </view>
  93. </view>
  94. <template v-if="detail.promotionApplyStatus > 0">
  95. <view class="remark" v-if="detail.promotionApplyStatus === 4">
  96. <view class="label">备注:</view>
  97. <view class="value">在业务员可优惠折扣设置值内,自动审核通过。</view>
  98. </view>
  99. <view class="remark" v-if="detail.promotionApplyRemark">
  100. <view class="label">申请备注:</view>
  101. <view class="value">{{detail.promotionApplyRemark}}</view>
  102. </view>
  103. <view class="remark" v-if="detail.promotionApplyStatus === 3 && detail.promotionApplyRejectRemark">
  104. <view class="label">驳回备注:</view>
  105. <view class="value">{{detail.promotionApplyRejectRemark}}</view>
  106. </view>
  107. </template>
  108. </view>
  109. <!-- 业务员端:未申请且待支付 -->
  110. <view class="bottom-container" v-if="!isExamineUser && detail.promotionApplyStatus === 0 && detail.orderStatus === 'NOPAY'">
  111. <div class="button red" @tap.stop="toApply()">申请减免金额</div>
  112. </view>
  113. <!-- 业务员端:已驳回 -->
  114. <view class="bottom-container" v-if="!isExamineUser && detail.promotionApplyStatus === 3">
  115. <div class="button red" @tap.stop="toApply()">重新申请优惠</div>
  116. </view>
  117. <!-- 审核端:待审核-->
  118. <view class="bottom-container" v-if="isExamineUser && detail.promotionApplyStatus === 1">
  119. <div class="button white" @tap.stop="toExamine(false)">不同意</div>
  120. <div class="button red" @tap.stop="toExamine(true)">同意</div>
  121. </view>
  122. <!-- 审核端:已通过未付款 -->
  123. <view class="bottom-container" v-if="isExamineUser && detail.promotionApplyStatus === 2 && detail.orderStatus === 'NOPAY'">
  124. <view class="button red" @tap="toExamine(false)">驳回</view>
  125. </view>
  126. </view>
  127. </block>
  128. <view class="apply-dialog" v-show="isApplyDialog">
  129. <view class="dialog">
  130. <view class="title">减免金额申请</view>
  131. <view class="content">
  132. <view class="row">
  133. <text>订单金额:¥{{detail.payAmount | numToFixed}}</text>
  134. </view>
  135. <view class="row">
  136. <text>减免金额:</text><input type="digit" v-model="applyForm.amount" /><text></text>
  137. </view>
  138. <view class="row">
  139. <text>应付金额:¥{{((detail.payAmount*100 - applyForm.amount*100)/100) | numToFixed}}</text>
  140. </view>
  141. <view class="row">
  142. <text>备注:</text><input type="text" v-model="applyForm.remark" /><text></text>
  143. </view>
  144. </view>
  145. <view class="btn">
  146. <view class="left" @tap="cancelApplyForm">取消</view>
  147. <view class="right" @tap="confirmApplyForm">提交</view>
  148. </view>
  149. </view>
  150. </view>
  151. <view class="apply-dialog" v-show="isExamineDialog">
  152. <view class="dialog">
  153. <view class="title">减免金额审核</view>
  154. <view class="content">
  155. <view class="row">
  156. <text>订单金额:¥{{detail.payAmount | numToFixed}}</text>
  157. </view>
  158. <view class="row">
  159. <text>减免金额:¥{{detail.promotionApplyAmount | numToFixed}}</text>
  160. </view>
  161. <view class="row">
  162. <text>应付金额:¥{{detail.promotionApplyPayAmount | numToFixed}}</text>
  163. </view>
  164. <view class="row">
  165. <text>申请备注:{{detail.promotionApplyRemark}}</text>
  166. </view>
  167. <view class="row">
  168. <text>审核备注:</text><input type="text" v-model="examineForm.remark" />
  169. </view>
  170. </view>
  171. <view class="btn">
  172. <view class="left" @tap="cancelExamineForm">取消</view>
  173. <view class="right" @tap="confirmExamineForm">提交</view>
  174. </view>
  175. </view>
  176. </view>
  177. </view>
  178. </template>
  179. <script>
  180. import {mapState} from 'vuex';
  181. import modalDialog from '@/components/modalDialog.vue';
  182. export default {
  183. components:{
  184. modalDialog
  185. },
  186. filters: {
  187. statusFilter2(val) {
  188. const statusMap = {
  189. 0: '未申请',
  190. 1: '待审核',
  191. 2: '已通过',
  192. 3: '已驳回',
  193. 4: '自动审核通过',
  194. }
  195. return statusMap[val]
  196. },
  197. },
  198. data() {
  199. return {
  200. configInfo: uni.getStorageSync('configInfo'),
  201. isLoading: true,
  202. noData: true,
  203. orderId: null, // 订单id
  204. orderRefundId: null, // 售后订单id
  205. isReturnOrder: false, // 是否售后订单
  206. detail: {}, // 订单详情
  207. isApplyDialog: false,
  208. applyForm: {
  209. amount: '',
  210. remark: '',
  211. },
  212. isExamineDialog: false,
  213. examineForm: {
  214. status: true,
  215. remark: '',
  216. },
  217. }
  218. },
  219. computed:{
  220. ...mapState(['userInfo', 'isLogin', 'userId']),
  221. isExamineUser() { // 是否审核人员
  222. return this.userInfo.promotionApplyExamineby;
  223. },
  224. },
  225. onLoad({orderId, orderRefundId}) {
  226. this.orderId = orderId;
  227. if(orderRefundId != 'undefined') {
  228. this.isReturnOrder = true;
  229. this.orderRefundId = orderRefundId;
  230. }
  231. this.getOrderDetail();
  232. uni.$on('refreshOrderDetail',(data) => {
  233. if(data) {
  234. this.isReturnOrder = true;
  235. this.orderRefundId = data;
  236. }else {
  237. this.isReturnOrder = false;
  238. this.orderRefundId = null;
  239. }
  240. this.getOrderDetail();
  241. })
  242. },
  243. methods: {
  244. getOrderDetail() {
  245. let url = '', params = {};
  246. if(this.isReturnOrder) {
  247. url = '/order/refund/detail';
  248. params = {
  249. orderRefundId: this.orderRefundId
  250. }
  251. }else {
  252. url = '/order/detail';
  253. params = {
  254. orderId: this.orderId
  255. }
  256. }
  257. this.$axios({
  258. url,
  259. method: 'get',
  260. params
  261. }).then(res => {
  262. this.noData = false;
  263. this.isLoading = false;
  264. this.detail = res.data;
  265. }).catch(res => {
  266. this.noData = true;
  267. this.isLoading = false;
  268. })
  269. },
  270. // 去商品详情
  271. toGoodsDetail(id) {
  272. uni.navigateTo({
  273. url: '/packageGoods/pages/detail?id=' + id
  274. })
  275. },
  276. toApply() {
  277. this.isApplyDialog = true;
  278. },
  279. // 取消表单
  280. cancelApplyForm() {
  281. this.applyForm.amount = '';
  282. this.isApplyDialog = false;
  283. },
  284. // 提交表单
  285. confirmApplyForm() {
  286. if(!this.applyForm.amount) {
  287. return this.$toast('请填写优惠金额');
  288. }
  289. this.$axios({
  290. url: '/promotion/apply/apply',
  291. method: 'post',
  292. type: 'application/json',
  293. params: {
  294. orderId: this.detail.orderId,
  295. applyAmount: this.applyForm.amount,
  296. applyRemark: this.applyForm.remark,
  297. },
  298. isLoading: 1
  299. }).then(res => {
  300. this.$successToast('提交成功');
  301. this.cancelApplyForm();
  302. this.getOrderDetail();
  303. })
  304. },
  305. toExamine(status) {
  306. this.examineForm.status = status;
  307. this.isExamineDialog = true;
  308. },
  309. // 取消表单
  310. cancelExamineForm() {
  311. this.examineForm.remark = '';
  312. this.isExamineDialog = false;
  313. },
  314. // 提交表单
  315. confirmExamineForm() {
  316. this.$axios({
  317. url: '/promotion/apply/examine',
  318. method: 'post',
  319. params: {
  320. orderId: this.detail.orderId,
  321. examineReuslt: this.examineForm.status,
  322. promotionApplyRejectRemmark: this.examineForm.remark,
  323. },
  324. isLoading: 1
  325. }).then(res => {
  326. this.$successToast('提交成功');
  327. this.cancelExamineForm();
  328. this.getOrderDetail();
  329. })
  330. },
  331. }
  332. }
  333. </script>
  334. <style lang="scss">
  335. .app-container {
  336. background: #F4F2F2;
  337. box-sizing: border-box;
  338. }
  339. .status-container {
  340. background: #FE781F;
  341. line-height: 80rpx;
  342. text-align: center;
  343. font-size: 32rpx;
  344. color: #FFFFFF;
  345. }
  346. .card {
  347. background: #FFFFFF;
  348. border-radius: 20rpx;
  349. margin-bottom: 20rpx;
  350. }
  351. .main-container {
  352. padding: 20rpx 20rpx 120rpx;
  353. }
  354. .address-container {
  355. display: flex;
  356. justify-content: space-between;
  357. align-items: center;
  358. height: 150rpx;
  359. padding: 0 20rpx;
  360. .icon {
  361. width: 52rpx;
  362. height: 52rpx;
  363. border-radius: 50%;
  364. background: linear-gradient(-90deg,#ff4042 0%, #fe781f 100%);
  365. display: flex;
  366. flex-shrink: 0;
  367. justify-content: center;
  368. align-items: center;
  369. margin-right: 20rpx;
  370. image {
  371. width: 28rpx;
  372. height: 36rpx;
  373. }
  374. }
  375. .right {
  376. width: 600rpx;
  377. .name {
  378. font-size: 32rpx;
  379. color: #333333;
  380. text {
  381. font-size: 28rpx;
  382. color: #999999;
  383. margin-left: 16rpx;
  384. }
  385. }
  386. .address {
  387. font-size: 28rpx;
  388. color: #666666;
  389. line-height: 34rpx;
  390. margin-top: 10rpx;
  391. }
  392. }
  393. }
  394. .goods-container {
  395. padding: 0 20rpx;
  396. .title {
  397. font-size: 32rpx;
  398. color: #333333;
  399. line-height: 32rpx;
  400. padding-top: 20rpx;
  401. }
  402. .item {
  403. padding: 20rpx 0;
  404. border-bottom: 1px solid #eaeaea;
  405. display: flex;
  406. justify-content: space-between;
  407. align-items: center;
  408. &:last-child {
  409. border: none;
  410. }
  411. image {
  412. width: 180rpx;
  413. height: 180rpx;
  414. flex-shrink: 0;
  415. margin-right: 20rpx;
  416. }
  417. .right {
  418. width: 470rpx;
  419. height: 180rpx;
  420. display: flex;
  421. flex-direction: column;
  422. justify-content: space-between;
  423. .top {
  424. .name {
  425. font-size: 28rpx;
  426. color: #333333;
  427. line-height: 36rpx;
  428. }
  429. .des {
  430. font-size: 28rpx;
  431. color: #999999;
  432. margin-top: 10rpx;
  433. }
  434. }
  435. .bottom {
  436. display: flex;
  437. justify-content: space-between;
  438. align-items: center;
  439. .price {
  440. font-size: 28rpx;
  441. color: #FF3F42;
  442. line-height: 28rpx;
  443. }
  444. .num {
  445. font-size: 28rpx;
  446. color: #333333;
  447. line-height: 28rpx;
  448. }
  449. }
  450. }
  451. }
  452. }
  453. .peisong-container {
  454. padding: 0 20rpx;
  455. font-size: 28rpx;
  456. color: #333333;
  457. .top {
  458. display: flex;
  459. justify-content: space-between;
  460. align-items: center;
  461. height: 88rpx;
  462. border-bottom: 1px solid #eaeaea;
  463. }
  464. .bottom {
  465. display: flex;
  466. align-items: center;
  467. min-height: 88rpx;
  468. .left {
  469. margin-right: 40rpx;
  470. flex-shrink: 0;
  471. }
  472. .right {
  473. padding: 10rpx 0;
  474. line-height: 32rpx;
  475. }
  476. }
  477. }
  478. .apply-container {
  479. background: #FFFFFF;
  480. border-radius: 20rpx;
  481. padding: 20rpx;
  482. .user-info {
  483. height: 70rpx;
  484. display: flex;
  485. justify-content: space-between;
  486. align-items: center;
  487. padding: 10rpx 0;
  488. .left {
  489. display: flex;
  490. align-items: center;
  491. .head {
  492. display: block;
  493. width: 60rpx;
  494. height: 60rpx;
  495. border-radius: 50%;
  496. }
  497. .user {
  498. font-size: 24rpx;
  499. color: #333333;
  500. line-height: 30rpx;
  501. margin-left: 10rpx;
  502. }
  503. }
  504. .right {
  505. font-size: 28rpx;
  506. color: #FE781F;
  507. }
  508. }
  509. .user-info2 {
  510. height: 48rpx;
  511. display: flex;
  512. justify-content: space-between;
  513. align-items: center;
  514. .left {
  515. display: flex;
  516. align-items: center;
  517. line-height: 48rpx;
  518. .label {
  519. width: 128rpx;
  520. font-size: 26rpx;
  521. color: #999999;
  522. }
  523. .value {
  524. flex: 1;
  525. font-size: 28rpx;
  526. color: #666666;
  527. }
  528. }
  529. .right {
  530. font-size: 28rpx;
  531. color: #FE781F;
  532. }
  533. }
  534. .data {
  535. .row {
  536. display: flex;
  537. align-items: center;
  538. line-height: 48rpx;
  539. .label {
  540. width: 128rpx;
  541. font-size: 26rpx;
  542. color: #999999;
  543. }
  544. .value {
  545. flex: 1;
  546. font-size: 28rpx;
  547. color: #666666;
  548. }
  549. }
  550. }
  551. .remark {
  552. font-size: 26rpx;
  553. color: #999999;
  554. display: flex;
  555. padding: 6rpx 0;
  556. .label {
  557. flex-shrink: 0;
  558. line-height: 36rpx;
  559. }
  560. .value {
  561. flex: 1;
  562. line-height: 36rpx;
  563. }
  564. }
  565. }
  566. .bottom-container {
  567. position: fixed;
  568. bottom: 0;
  569. left: 0;
  570. width: 100%;
  571. padding: 0 20rpx;
  572. box-sizing: border-box;
  573. height: 100rpx;
  574. display: flex;
  575. justify-content: flex-end;
  576. align-items: center;
  577. background: #FFFFFF;
  578. border-top: 1px solid #F4F2F2;
  579. .button {
  580. width: 150rpx;
  581. height: 56rpx;
  582. border-radius: 56rpx;
  583. text-align: center;
  584. line-height: 56rpx;
  585. font-size: 28rpx;
  586. margin-left: 15rpx;
  587. &:first-child {
  588. margin-left: 0;
  589. }
  590. &.gray {
  591. color: #999999;
  592. border: 1px solid #999999;
  593. }
  594. &.white {
  595. color: #FF3F42;
  596. border: 1px solid #FF3F42;
  597. }
  598. &.red {
  599. color: #FFFFFF;
  600. border: 1px solid #FF3F42;
  601. background: #FF3F42;
  602. }
  603. }
  604. }
  605. .apply-dialog {
  606. position: fixed;
  607. top: 0;
  608. left: 0;
  609. z-index: 9999;
  610. width: 100%;
  611. height: 100%;
  612. background: rgba(0, 0, 0, 0.3);
  613. display: flex;
  614. align-items: center;
  615. justify-content: center;
  616. .dialog {
  617. width: 600rpx;
  618. background: #FFFFFF;
  619. border-radius: 20rpx;
  620. overflow: hidden;
  621. .title {
  622. font-size: 36rpx;
  623. font-weight: 500;
  624. text-align: center;
  625. line-height: 100rpx;
  626. padding-bottom: 10rpx;
  627. }
  628. .content {
  629. padding: 0 40rpx 30rpx;
  630. .row {
  631. height: 50rpx;
  632. display: flex;
  633. align-items: center;
  634. margin-bottom: 10rpx;
  635. font-size: 28rpx;
  636. color: #333333;
  637. input {
  638. flex: 1;
  639. height: 50rpx;
  640. border: 1px solid #E5E5E5;
  641. margin-right: 16rpx;
  642. padding: 0 12rpx;
  643. font-size: 28rpx;
  644. }
  645. }
  646. }
  647. .btn {
  648. border-top: 1px solid #eaeaea;
  649. display: flex;
  650. &> view {
  651. flex: 1;
  652. text-align: center;
  653. line-height: 100rpx;
  654. font-size: 32rpx;
  655. &.left {
  656. color: #666666;
  657. }
  658. &.right {
  659. color: #FFFFFF;
  660. background: #FF3F42;
  661. }
  662. }
  663. }
  664. }
  665. }
  666. </style>