list.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660
  1. <template>
  2. <view class="app-container">
  3. <view class="tab-container">
  4. <view class="item" :class="tabCurrent == '' ? 'current':''" @tap="changeTab('')"><text>全部</text></view>
  5. <view class="item" :class="tabCurrent == 'NOPAY' ? 'current':''" @tap="changeTab('NOPAY')"><text>待付款</text></view>
  6. <view class="item" :class="tabCurrent == 'DFH' ? 'current':''" @tap="changeTab('DFH')"><text>待发货</text></view>
  7. <view class="item" :class="tabCurrent == 'YFH' ? 'current':''" @tap="changeTab('YFH')"><text>待收货</text></view>
  8. <view class="item" :class="tabCurrent == 'OVER' ? 'current':''" @tap="changeTab('OVER')"><text>已完成</text></view>
  9. <view class="item" :class="tabCurrent == 'REFUND' ? 'current':''" @tap="changeTab('REFUND')"><text>售后中</text></view>
  10. </view>
  11. <view class="list-container">
  12. <block v-for="(item, index) in orderList" :key='index'>
  13. <view class="item" @tap="toOrderDetail(item.orderId, item.orderRefundId)">
  14. <!-- 普通订单 -->
  15. <block v-if="tabCurrent != 'REFUND'">
  16. <view class="top">
  17. <view class="left">{{item.orderId}}</view>
  18. <view class="right">{{item.orderStatus | statusFilter}}</view>
  19. </view>
  20. <block v-for="(goodsItem, goodsIndex) in item.orderDetails" :key='goodsIndex'>
  21. <view class="goods" :class="'goods'+goodsIndex">
  22. <image :src="goodsItem.imgUrl" mode="aspectFill"></image>
  23. <view class="main">
  24. <view class="row1">
  25. <view class="name ellipsis-2">{{goodsItem.goodsName}}</view>
  26. <view class="price">¥{{goodsItem.price}}</view>
  27. </view>
  28. <view class="row2">
  29. <view>{{goodsItem.goodsSpecValue}}</view>
  30. <view>x{{goodsItem.num}}</view>
  31. </view>
  32. </view>
  33. </view>
  34. </block>
  35. <view class="bottom">
  36. <view class="left">{{item.createTime}}</view>
  37. <view class="total">共{{item.totalNum}}件 订单总额:<text>¥{{item.payAmount | numToFixed}}</text></view>
  38. </view>
  39. </block>
  40. <!-- 售后订单 -->
  41. <block v-if="tabCurrent == 'REFUND'">
  42. <view class="top">
  43. <view class="left">{{item.orderId}}</view>
  44. <view class="right">{{item.orderStatus | statusFilter2}}</view>
  45. </view>
  46. <!-- 拒绝申请 -->
  47. <block v-if="item.examineStatus === 'FAIL'">
  48. <block v-for="(goodsItem, goodsIndex) in item.orderDetails" :key='goodsIndex'>
  49. <view class="goods" :class="'goods'+goodsIndex">
  50. <image :src="goodsItem.imgUrl" mode="aspectFill"></image>
  51. <view class="main">
  52. <view class="row1">
  53. <view class="name ellipsis-2">{{goodsItem.goodsName}}</view>
  54. <view class="price">¥{{goodsItem.price}}</view>
  55. </view>
  56. <view class="row2">
  57. <view>{{goodsItem.goodsSpecValue}}</view>
  58. <view>x{{goodsItem.refundNum}}</view>
  59. </view>
  60. </view>
  61. </view>
  62. </block>
  63. <view class="bottom">
  64. <view class="left">{{item.createTime}}</view>
  65. <view class="total">共{{item.totalNum}}件 {{item.orderStatus == 'NOPAY' ? '应付':'实付'}}总额:<text>¥{{item.payAmount | numToFixed}}</text></view>
  66. </view>
  67. </block>
  68. <!-- 其他 -->
  69. <block v-else>
  70. <block v-for="(goodsItem, goodsIndex) in item.orderDetails" :key='goodsIndex'>
  71. <view class="goods" :class="'goods'+goodsIndex" v-if="goodsItem.refund">
  72. <image :src="goodsItem.imgUrl" mode="aspectFill"></image>
  73. <view class="main">
  74. <view class="row1">
  75. <view class="name ellipsis-2">{{goodsItem.goodsName}}</view>
  76. <view class="price">¥{{goodsItem.price}}</view>
  77. </view>
  78. <view class="row2">
  79. <view>{{goodsItem.goodsSpecValue}}</view>
  80. <view>x{{goodsItem.refundNum}}</view>
  81. </view>
  82. </view>
  83. </view>
  84. </block>
  85. <view class="bottom">
  86. <view class="left">{{item.createTime}}</view>
  87. <view class="total">共{{getRefundNum(item.orderDetails)}}件 {{item.orderStatus == 'OVER' ? '实退':'应退'}}总额:<text>¥{{item.refundAmount | numToFixed}}</text></view>
  88. </view>
  89. </block>
  90. </block>
  91. <!-- 按钮:待付款 -->
  92. <view class="btn-group btn-group2" v-if="item.orderStatus == 'NOPAY'">
  93. <div class="tips">请在30分钟内付款</div>
  94. <div class="btns">
  95. <div class="button gray" @tap.stop="cancelOrder(item.orderId)">取消订单</div>
  96. <div class="button red" @tap.stop="payOrder(item.orderId)">立即付款</div>
  97. </div>
  98. </view>
  99. <!-- 按钮:待发货 -->
  100. <view class="btn-group" v-if="item.orderStatus == 'DFH'">
  101. <div class="button white" @tap.stop="remindShipment(item.orderId)">提醒发货</div>
  102. <div class="button white" v-if="checkCanReturn(item.orderShareStatus)" @tap.stop="toApplyReturn(item.orderId)">申请售后</div>
  103. </view>
  104. <!-- 按钮:待收货 -->
  105. <view class="btn-group" v-if="item.orderStatus == 'YFH'">
  106. <div class="button gray" @tap.stop="toLogistics(item.companyCode, item.logisticsNo)">查看物流</div>
  107. <div class="button red" @tap.stop="clickConfirmReceipt(item.orderId)">确认收货</div>
  108. <div class="button white" v-if="checkCanReturn(item.orderShareStatus)" @tap.stop="toApplyReturn(item.orderId)">申请售后</div>
  109. </view>
  110. <!-- 按钮:已完成 -->
  111. <view class="btn-group" v-if="item.orderStatus == 'OVER' && tabCurrent != 'REFUND'">
  112. <div class="button gray" @tap.stop="toLogistics(item.companyCode, item.logisticsNo)">查看物流</div>
  113. <div class="button white" v-if="checkCanReturn(item.orderShareStatus)" @tap.stop="toApplyReturn(item.orderId)">申请售后</div>
  114. </view>
  115. <!-- 按钮:已关闭 -->
  116. <view class="btn-group" v-if="item.orderStatus == 'CLOSE'">
  117. <div class="button gray">查看详情</div>
  118. </view>
  119. <!-- 按钮:超时未支付 -->
  120. <view class="btn-group" v-if="item.orderStatus == 'TIMEOUT'">
  121. <div class="button gray">查看详情</div>
  122. </view>
  123. <!-- 按钮:售后中 -->
  124. <view class="btn-group" v-if="item.orderStatus == 'REFUND'">
  125. <div class="button gray">查看详情</div>
  126. </view>
  127. <!-- 按钮:售后中 待商家处理 -->
  128. <view class="btn-group" v-if="item.orderStatus == 'DSJCL'">
  129. <div class="button gray" @tap.stop="toReturnDetail(item.orderRefundId)">售后详情</div>
  130. </view>
  131. <!-- 按钮:售后中 待买家处理 -->
  132. <view class="btn-group" v-if="item.orderStatus == 'DMJCL'">
  133. <div class="button red" @tap.stop="toReturnDetail(item.orderRefundId)" v-if="item.examineStatus == 'OK'">提交资料</div>
  134. <div class="button white" @tap.stop="toApplyReturn(item.orderId)" v-if="item.examineStatus == 'FAIL'">重新申请</div>
  135. </view>
  136. <!-- 按钮:售后中 待商家收货 -->
  137. <view class="btn-group" v-if="item.orderStatus == 'DSJSH'">
  138. <div class="button gray" @tap.stop="toReturnDetail(item.orderRefundId)">售后详情</div>
  139. </view>
  140. <!-- 按钮:售后中 退款成功 -->
  141. <view class="btn-group" v-if="item.orderStatus == 'OVER' && tabCurrent == 'REFUND'">
  142. <div class="button gray" @tap.stop="toReturnDetail(item.orderRefundId)">售后详情</div>
  143. </view>
  144. </view>
  145. </block>
  146. </view>
  147. <no-data v-if="!orderList.length" :showText="'暂无订单'"></no-data>
  148. <loading-text v-if="orderList.length" :loading="loading" :noMore="noMore" ></loading-text>
  149. <!-- 取消订单 -->
  150. <modal-dialog showText="确定要取消订单吗?" :isShowDialog="isCancelDialog" @cancel="isCancelDialog = false" @confirm="confirmCancelOrder"></modal-dialog>
  151. <!-- 确认收货 -->
  152. <modal-dialog showText="确定要确认收货吗?" :isShowDialog="isConfirmDialog" @cancel="isConfirmDialog = false" @confirm="confirmReceipt"></modal-dialog>
  153. </view>
  154. </template>
  155. <script>
  156. import {mapState} from 'vuex';
  157. import modalDialog from '@/components/modalDialog.vue';
  158. export default {
  159. components:{
  160. modalDialog
  161. },
  162. filters: {
  163. statusFilter(val) {
  164. const statusMap = {
  165. NOPAY: '待付款',
  166. DFH: '待发货',
  167. YFH: '待收货',
  168. OVER: '已完成',
  169. CLOSE: '已关闭',
  170. REFUND: '售后中',
  171. TIMEOUT: '超时未支付'
  172. }
  173. return statusMap[val]
  174. },
  175. statusFilter2(val) {
  176. const statusMap = {
  177. DSJCL: '待商家处理',
  178. DSJSH: '待商家收货',
  179. DMJCL: '待买家处理',
  180. OVER: '退款成功',
  181. CANCEL: '已取消',
  182. }
  183. return statusMap[val]
  184. },
  185. },
  186. data() {
  187. return {
  188. configInfo: uni.getStorageSync('configInfo'),
  189. tabCurrent: '', // 当前tab状态
  190. orderList: [], // 订单列表
  191. pageNum: 1,
  192. pageSize: 8,
  193. noMore: false,
  194. loading: false,
  195. isCancelDialog: false, // 是否显示取消订单弹窗
  196. cancelOrderId: null, // 取消订单id
  197. isConfirmDialog: false, // 是否显示确认收货弹窗
  198. confirmReceiptId: null, // 确认收货id
  199. autoDetailId: null, // 自动跳转到详情页id
  200. canRemindShipment: true, // 能否点击提醒发货
  201. }
  202. },
  203. computed:{
  204. ...mapState(['userInfo', 'isLogin', 'userId'])
  205. },
  206. onLoad({tab, orderId}) {
  207. this.tabCurrent = tab;
  208. console.log(this.tabCurrent);
  209. if(orderId) {
  210. this.autoDetailId = orderId;
  211. }
  212. this.getOrderList();
  213. uni.$on('refreshOrderList',(data) => {
  214. if(data) {
  215. this.tabCurrent = 'REFUND';
  216. }
  217. this.getOrderList();
  218. })
  219. },
  220. // 下拉刷新
  221. onPullDownRefresh() {
  222. this.pageNum = 1;
  223. this.getOrderList();
  224. },
  225. // 上拉加载
  226. onReachBottom() {
  227. this.getOrderList(1);
  228. },
  229. methods: {
  230. getOrderList(loadMore) {
  231. if(this.noMore && loadMore)return;
  232. this.noMore = false
  233. if(!loadMore){
  234. this.pageNum = 1;
  235. }else{
  236. this.loading = true;
  237. }
  238. let url = '';
  239. let orderStatus = '';
  240. if(this.tabCurrent == 'REFUND') {
  241. url = '/order/refund/list';
  242. orderStatus = '';
  243. }else {
  244. url = '/order/my/order';
  245. orderStatus = this.tabCurrent;
  246. }
  247. this.$axios({
  248. url: url,
  249. method: 'get',
  250. params: {
  251. pageNo: this.pageNum,
  252. pageSize: this.pageSize,
  253. orderStatus: orderStatus,
  254. userId: this.userId
  255. },
  256. isLoading: !loadMore
  257. }).then(res => {
  258. let _list = res.data.records;
  259. let pageTotal = res.data.pages;
  260. if(this.pageNum >= pageTotal){
  261. this.noMore = true;
  262. }
  263. if (_list.length) {
  264. this.pageNum += 1
  265. }
  266. if (loadMore) {
  267. this.orderList = this.orderList.concat(_list);
  268. this.loading = false;
  269. } else {
  270. this.orderList = _list;
  271. }
  272. uni.stopPullDownRefresh();
  273. if(this.autoDetailId) {
  274. this.toOrderDetail(this.autoDetailId);
  275. }
  276. })
  277. },
  278. getRefundNum(orderDetails) {
  279. let refundNum = 0;
  280. orderDetails.forEach(item => {
  281. if(item.refund) {
  282. refundNum = refundNum + item.refundNum;
  283. }
  284. })
  285. return refundNum;
  286. },
  287. // 检查是否可以申请售后(根据orderShareStatus)
  288. checkCanReturn(status) {
  289. if(!status || status === 'OVER' || status === 'CANCEL') {
  290. return false;
  291. }else {
  292. return true;
  293. }
  294. },
  295. // 切换类型
  296. changeTab(current) {
  297. this.orderList = [];
  298. this.tabCurrent = current;
  299. this.pageNum = 1;
  300. this.getOrderList();
  301. },
  302. // 申请退款
  303. toApplyReturn(orderId) {
  304. uni.navigateTo({
  305. url:'/pages/mine/order/return/apply?orderId=' + orderId
  306. })
  307. },
  308. // 去订单详情
  309. toOrderDetail(orderId, orderRefundId) {
  310. this.autoDetailId = null;
  311. if(orderRefundId) {
  312. return uni.navigateTo({
  313. url:'/pages/mine/order/return/detail?orderRefundId=' + orderRefundId
  314. })
  315. }
  316. uni.navigateTo({
  317. url:'/pages/mine/order/detail?orderId=' + orderId + '&orderRefundId=' + orderRefundId
  318. })
  319. },
  320. // 取消订单
  321. cancelOrder(orderId) {
  322. this.cancelOrderId = orderId;
  323. this.isCancelDialog = true;
  324. },
  325. // 确认取消订单
  326. confirmCancelOrder() {
  327. this.$axios({
  328. url: '/order/cancel',
  329. method: 'get',
  330. params: {
  331. orderId: this.cancelOrderId
  332. }
  333. }).then(res => {
  334. this.isCancelDialog = false;
  335. this.pageNum = 1;
  336. this.getOrderList();
  337. this.$successToast('取消订单成功');
  338. })
  339. },
  340. // 确认收货
  341. clickConfirmReceipt(orderId) {
  342. this.confirmReceiptId = orderId;
  343. this.isConfirmDialog = true;
  344. },
  345. // 确定 确认收货
  346. confirmReceipt() {
  347. this.$axios({
  348. url: '/order/ack',
  349. params: {
  350. orderId: this.confirmReceiptId
  351. }
  352. }).then(res => {
  353. this.isConfirmDialog = false;
  354. this.pageNum = 1;
  355. this.getOrderList();
  356. this.$successToast('确认收货成功');
  357. })
  358. },
  359. // 立即付款
  360. payOrder(orderId) {
  361. let that = this;
  362. this.$axios({
  363. url: '/order/wait/pay',
  364. params: {
  365. userId: this.userId,
  366. orderId: orderId,
  367. },
  368. isLoading: 1,
  369. }).then(res => {
  370. uni.getProvider({
  371. service: 'payment',
  372. success: (e) => {
  373. uni.requestPayment({
  374. provider: e.provider[0],
  375. orderInfo: res.data,
  376. timeStamp: res.data.timeStamp,
  377. nonceStr: res.data.nonceStr,
  378. package: res.data.payPackage,
  379. signType: 'MD5',
  380. paySign: res.data.paySign,
  381. success: (res) => {
  382. that.$successToast('支付成功');
  383. that.pageNum = 1;
  384. that.getOrderList();
  385. that.requestMessage();
  386. },
  387. fail: (err) => {
  388. that.$toast('支付失败');
  389. }
  390. })
  391. }
  392. })
  393. })
  394. },
  395. // 消息推送
  396. requestMessage() {
  397. let that = this;
  398. uni.showModal({
  399. title: '温馨提示',
  400. content: '为更好的促进您与买家的交流,需要在您的订单成交时向您发送消息',
  401. confirmText: "同意",
  402. cancelText: "拒绝",
  403. success: function (res) {
  404. if (res.confirm) {
  405. let tmplIds = [that.configInfo.template];
  406. uni.requestSubscribeMessage({
  407. tmplIds: tmplIds,
  408. success (res) {
  409. let status = null;
  410. tmplIds.map((item, index) => {
  411. if(res[item] == 'accept') {
  412. status = 'accept';
  413. }
  414. })
  415. if(status == 'accept') {
  416. that.$successToast('订阅成功');
  417. }else {
  418. that.$toast('订阅取消');
  419. }
  420. },
  421. fail (res) {
  422. console.log(res);
  423. that.$toast('订阅失败');
  424. }
  425. })
  426. } else if (res.cancel) {
  427. uni.showModal({
  428. title: '温馨提示',
  429. content: '拒绝后您将无法获取实时的与卖家(买家)的交易消息',
  430. confirmText: "知道了",
  431. showCancel: false,
  432. success: function (res) {
  433. }
  434. });
  435. }
  436. }
  437. });
  438. },
  439. // 去售后详情
  440. toReturnDetail(orderRefundId) {
  441. uni.navigateTo({
  442. url:'/pages/mine/order/return/detail?orderRefundId=' + orderRefundId
  443. })
  444. },
  445. // 去查看物流
  446. toLogistics(companyCode, logisticsNo) {
  447. uni.navigateTo({
  448. url:'/pages/mine/order/logistics?companyCode=' + companyCode + '&logisticsNo=' + logisticsNo
  449. })
  450. },
  451. // 提醒发货
  452. remindShipment(orderId) {
  453. if (!this.canRemindShipment) {
  454. return this.$toast('请勿频繁操作');
  455. };
  456. this.canRemindShipment = false;
  457. setTimeout(() => { this.canRemindShipment = true }, 3000)
  458. this.$axios({
  459. url: '/order/notice',
  460. params: {
  461. orderId: orderId,
  462. userId: this.userId
  463. }
  464. }).then(res => {
  465. this.$successToast('提醒发货成功');
  466. })
  467. },
  468. }
  469. }
  470. </script>
  471. <style lang="scss">
  472. .app-container {
  473. background: #F4F2F2;
  474. box-sizing: border-box;
  475. }
  476. .tab-container {
  477. position: fixed;
  478. top: 0;
  479. left: 0;
  480. width: 100%;
  481. background: #FFFFFF;
  482. display: flex;
  483. .item {
  484. width: 20%;
  485. text-align: center;
  486. text {
  487. display: inline-block;
  488. padding: 0 10rpx;
  489. line-height: 80rpx;
  490. padding-top: 8rpx;
  491. }
  492. &.current {
  493. text {
  494. color: #FF3F42;
  495. border-bottom: 6rpx solid #FF3F42;
  496. }
  497. }
  498. }
  499. }
  500. .list-container {
  501. padding: 110rpx 20rpx 0;
  502. .item {
  503. margin-bottom: 20rpx;
  504. background: #FFFFFF;
  505. border-radius: 20rpx;
  506. padding: 0 20rpx;
  507. .top {
  508. height: 70rpx;
  509. display: flex;
  510. justify-content: space-between;
  511. align-items: center;
  512. .left {
  513. font-size: 28rpx;
  514. color: #666666;
  515. }
  516. .right {
  517. font-size: 28rpx;
  518. color: #FE781F;
  519. }
  520. }
  521. .goods {
  522. display: flex;
  523. justify-content: space-between;
  524. margin-top: 20rpx;
  525. &.goods0 {
  526. margin-top: 0;
  527. }
  528. image {
  529. width: 140rpx;
  530. height: 140rpx;
  531. display: block;
  532. flex-shrink: 0;
  533. margin-right: 20rpx;
  534. }
  535. .main {
  536. width: 510rpx;
  537. .row1 {
  538. display: flex;
  539. justify-content: space-between;
  540. .name {
  541. font-size: 28rpx;
  542. line-height: 36rpx;
  543. }
  544. .price {
  545. font-size: 28rpx;
  546. margin-left: 20rpx;
  547. color: #666666;
  548. }
  549. }
  550. .row2 {
  551. display: flex;
  552. justify-content: space-between;
  553. font-size: 28rpx;
  554. color: #999999;
  555. margin-top: 10rpx;
  556. }
  557. }
  558. }
  559. .bottom {
  560. display: flex;
  561. align-items: center;
  562. justify-content: space-between;
  563. height: 50rpx;
  564. .left {
  565. font-size: 24rpx;
  566. color: #999999;
  567. }
  568. .total {
  569. font-size: 24rpx;
  570. color: #666666;
  571. text {
  572. color: #FF3F42;
  573. font-size: 28rpx;
  574. }
  575. }
  576. }
  577. .btn-group {
  578. border-top: 1px solid #eaeaea;
  579. height: 100rpx;
  580. display: flex;
  581. justify-content: flex-end;
  582. align-items: center;
  583. &.btn-group2 {
  584. justify-content: space-between;
  585. .tips {
  586. font-size: 28rpx;
  587. color: #FF3F42;
  588. }
  589. .btns {
  590. display: flex;
  591. }
  592. }
  593. .button {
  594. width: 140rpx;
  595. height: 48rpx;
  596. border-radius: 48rpx;
  597. text-align: center;
  598. line-height: 48rpx;
  599. font-size: 24rpx;
  600. margin-left: 20rpx;
  601. flex-shrink: 0;
  602. &:first-child {
  603. margin-left: 0;
  604. }
  605. &.gray {
  606. color: #999999;
  607. border: 1px solid #999999;
  608. }
  609. &.white {
  610. color: #FF3F42;
  611. border: 1px solid #FF3F42;
  612. }
  613. &.red {
  614. color: #FFFFFF;
  615. border: 1px solid #FF3F42;
  616. background: #FF3F42;
  617. }
  618. }
  619. }
  620. }
  621. }
  622. </style>