refund.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469
  1. <template>
  2. <div class="page" :class="{ 'page--iphoneX': iphoneX }">
  3. <div class="tips">温馨提示:请优先联系用户协商处理</div>
  4. <div class="black-tit">请选择退款商品</div>
  5. <div v-if="orderData">
  6. <checkbox-group class="item-list" @change="itemChange">
  7. <label
  8. class="item"
  9. v-for="(item, index) of orderData.orderItems"
  10. :key="index"
  11. :class="{ invalid: item.invalid }"
  12. >
  13. <div class="check">
  14. <checkbox
  15. v-if="!item.invalid"
  16. class="checkbox"
  17. :value="item.id"
  18. />
  19. </div>
  20. <div class="main">
  21. <div class="info">
  22. <my-image
  23. :src="item.productImage"
  24. class="img"
  25. ></my-image>
  26. <div class="data">
  27. <div class="tit">
  28. {{ item.productName }}
  29. </div>
  30. <div>
  31. <span class="num"
  32. >数量:x {{ item.count }}</span
  33. >
  34. <span class="num" v-if="item.skuName"
  35. >规格: {{ item.skuName }}</span
  36. >
  37. </div>
  38. <div class="des">
  39. 支付金额:
  40. <span class="price"
  41. >¥{{
  42. item.payAmount | minuteToRmb
  43. }}</span
  44. >
  45. </div>
  46. </div>
  47. </div>
  48. <div
  49. class="difference"
  50. v-if="item.refundAmount && item.actualPickingWeight"
  51. >
  52. <div class="name">
  53. 已退差价:¥{{
  54. item.refundAmount | minuteToRmb
  55. }}
  56. </div>
  57. <div class="">
  58. 已退重量:{{ item.actualPickingWeight || "0" }}g
  59. </div>
  60. </div>
  61. </div>
  62. </label>
  63. </checkbox-group>
  64. </div>
  65. <div class="refund-info">
  66. <div class="head">
  67. <div class="tit">退款金额</div>
  68. <div class="money">¥{{ refundNum | minuteToRmb }}</div>
  69. </div>
  70. <!-- <div class="des">
  71. 备注:
  72. </div> -->
  73. </div>
  74. <div class="black-tit">请选择退款原因</div>
  75. <radio-group @change="radioChange" class="info-list">
  76. <label class="item" v-for="item of radioList" :key="item.key">
  77. <div class="radio">
  78. <radio class="radio-inp" :value="item.value" />
  79. </div>
  80. <div class="info">{{ item.key }}</div>
  81. </label>
  82. </radio-group>
  83. <div class="btn-box">
  84. <span class="btn all-btn" @click="cancelOrder">取消订单</span>
  85. <!-- <span class="btn all-btn" @click="disparity">退差价</span> -->
  86. <span class="btn" @click="submit">退款已选中</span>
  87. </div>
  88. </div>
  89. </template>
  90. <script>
  91. import MyImage from "../../components/image/index";
  92. export default {
  93. name: "",
  94. components: { MyImage },
  95. // 数据
  96. data() {
  97. return {
  98. orderData: {},
  99. radioList: [
  100. { key: "商品已售完", value: "商品已售完" },
  101. { key: "店铺太忙", value: "店铺太忙" },
  102. { key: "其他", value: "其他" },
  103. ],
  104. curReason: "",
  105. curGoodItemIds: [],
  106. submitLoading: false,
  107. };
  108. },
  109. onLoad() {
  110. let curOrder = this.$store.state.common.curOrder;
  111. this.orderData = this.initData(curOrder);
  112. console.log(curOrder);
  113. this.$store.commit("common/update", {
  114. curOrder: null,
  115. });
  116. if (!curOrder) {
  117. this.router.back();
  118. }
  119. },
  120. async onShow() {},
  121. // 函数
  122. methods: {
  123. // 初始化订单数据
  124. initData(data) {
  125. let d = JSON.parse(JSON.stringify(data));
  126. console.log(d);
  127. //没有现有售后则不处理
  128. if (!d.afterSaleList) {
  129. return d;
  130. }
  131. // 判断是否有退款
  132. let num = 0;
  133. for (let orderItem of d.orderItems) {
  134. for (let afterSale of d.afterSaleList) {
  135. if (afterSale.orderInfo.orderItemId === orderItem.id) {
  136. orderItem.invalid = true;
  137. num += 1;
  138. }
  139. }
  140. }
  141. d.validItemNum = d.orderItems.length - num;
  142. console.log(d.validItemNum);
  143. d.orderItems.sort((a, b) => (a.invalid ? 1 : -1));
  144. return d;
  145. },
  146. radioChange(e) {
  147. this.curReason = e.detail.value;
  148. },
  149. itemChange(e) {
  150. this.curGoodItemIds = e.detail.value;
  151. },
  152. submit() {
  153. if (this.submitLoading) {
  154. return;
  155. }
  156. if (this.curGoodItemIds.length === 0) {
  157. return this.fn.showToast("请选择退货项");
  158. }
  159. if (this.curGoodItemIds.length === this.orderData.validItemNum) {
  160. // return this.fn.showToast("退款所有请选择全部退款");
  161. return this.cancelOrder(true);
  162. }
  163. if (!this.curReason) {
  164. return this.fn.showToast("请选择退货原因");
  165. }
  166. let sendData = {
  167. orderItemIds: this.curGoodItemIds,
  168. reason: this.curReason,
  169. remark: this.curReason,
  170. };
  171. this.fn
  172. .showModal({
  173. content: "确认退款选中商品项",
  174. })
  175. .then((res) => {
  176. if (res.confirm) {
  177. this.submitLoading = true;
  178. uni.showLoading({
  179. title: "提交中...",
  180. mask: true,
  181. });
  182. this.api
  183. .post("/order/partialRefundApply", sendData, {
  184. pass: true,
  185. })
  186. .then((res) => {
  187. uni.hideLoading();
  188. this.submitLoading = false;
  189. if (res.success) {
  190. this.fn.showToast("提交成功");
  191. this.router.back();
  192. } else {
  193. this.fn.showToast(
  194. "提交失败:" + res.message
  195. );
  196. }
  197. });
  198. }
  199. });
  200. },
  201. // 退差价
  202. disparity() {
  203. if (this.submitLoading) {
  204. return;
  205. }
  206. if (this.curGoodItemIds.length === 0) {
  207. return this.fn.showToast("请选择退货项");
  208. }
  209. let arr = [];
  210. for (id of this.curGoodItemIds) {
  211. for (let item of this.orderData.orderItems) {
  212. if (item.id === id) {
  213. arr.push(item);
  214. }
  215. }
  216. }
  217. this.$store.commit("common/update", {
  218. disparityData: arr,
  219. });
  220. let id = this.orderData.orderInfo.id;
  221. console.log(this.orderData, id);
  222. this.router.replace({
  223. path: "/pages/index/disparity",
  224. query: {
  225. orderId: id,
  226. },
  227. });
  228. },
  229. // 取消订单
  230. cancelOrder(flag) {
  231. let str = "确认取消该订单,并全部退款";
  232. if (flag === true) {
  233. str = "你已选中所有商品,确认后将取消订单全部退款";
  234. }
  235. let val = this.orderData;
  236. this.fn
  237. .showModal({
  238. content: str,
  239. })
  240. .then((res) => {
  241. if (res.confirm) {
  242. if (val.loading) {
  243. return;
  244. }
  245. val.loading = true;
  246. uni.showLoading({
  247. title: "提交中...",
  248. mask: true,
  249. });
  250. this.api
  251. .post(
  252. "/Order/Cancel",
  253. {
  254. query: true,
  255. id: val.orderInfo.id,
  256. },
  257. { pass: true }
  258. )
  259. .then((res) => {
  260. uni.hideLoading();
  261. val.loading = false;
  262. if (res.success) {
  263. this.fn.showToast("取消成功");
  264. this.router.back();
  265. } else {
  266. this.fn.showToast(
  267. "取消失败:" + res.message
  268. );
  269. }
  270. });
  271. }
  272. });
  273. },
  274. },
  275. // 数据计算
  276. computed: {
  277. user() {
  278. return this.$store.state.user.user;
  279. },
  280. refundNum() {
  281. let num = 0;
  282. for (let id of this.curGoodItemIds) {
  283. for (let item of this.orderData.orderItems) {
  284. if (item.id === id) {
  285. num += item.payAmount;
  286. if(item.refundAmount){
  287. num-=item.refundAmount;
  288. }
  289. }
  290. }
  291. }
  292. num = num.toFixed(2);
  293. return num;
  294. },
  295. },
  296. // 数据监听
  297. watch: {},
  298. };
  299. </script>
  300. <style lang="scss" scoped>
  301. .page {
  302. padding-bottom: px(180);
  303. }
  304. .tips {
  305. background-color: #fef9f3;
  306. font-size: px(40);
  307. padding: 0 px(40);
  308. height: px(100);
  309. line-height: px(100);
  310. color: #ec883d;
  311. }
  312. .black-tit {
  313. padding: px(50) px(40) px(30);
  314. background-color: #f1f0f5;
  315. font-size: px(42);
  316. }
  317. .btn-box {
  318. position: fixed;
  319. bottom: 0;
  320. left: 0;
  321. right: 0;
  322. z-index: 100;
  323. background-color: #fff;
  324. padding: px(40);
  325. display: flex;
  326. justify-content: space-between;
  327. align-items: center;
  328. .btn {
  329. display: block;
  330. width: 47%;
  331. height: px(110);
  332. border-radius: px(10);
  333. color: #fff;
  334. text-align: center;
  335. line-height: px(110);
  336. background-color: rgb(0, 188, 38);
  337. }
  338. .all-btn {
  339. background-color: #f3f3f3;
  340. color: #666;
  341. }
  342. }
  343. .item-list {
  344. padding: px(20) px(40);
  345. .item {
  346. display: flex;
  347. justify-content: space-between;
  348. align-items: center;
  349. padding: px(40) 0;
  350. border-bottom: 1px solid #f1f1f1;
  351. &.invalid {
  352. background-color: #efefef;
  353. }
  354. }
  355. .check {
  356. width: px(100);
  357. flex-shrink: 0;
  358. margin-right: px(20);
  359. }
  360. .checkbox {
  361. transform: scale(0.8);
  362. }
  363. .main {
  364. width: 100%;
  365. }
  366. .info {
  367. width: 100%;
  368. display: flex;
  369. justify-content: space-between;
  370. align-items: stretch;
  371. }
  372. .img {
  373. width: px(180);
  374. height: px(180);
  375. flex-shrink: 0;
  376. margin-right: px(20);
  377. /deep/ img {
  378. width: px(180);
  379. height: px(180);
  380. background: #f1f1f1;
  381. }
  382. }
  383. .data {
  384. width: 100%;
  385. }
  386. .tit {
  387. font-size: px(44);
  388. @include omits(1);
  389. }
  390. .des {
  391. margin-top: px(10);
  392. font-size: px(40);
  393. }
  394. .price {
  395. font-size: px(40);
  396. color: #ff475f;
  397. }
  398. .num {
  399. font-size: px(36);
  400. color: #666;
  401. display: inline-block;
  402. width: px(300);
  403. }
  404. }
  405. .refund-info {
  406. padding: px(50) px(40) px(30);
  407. .head {
  408. display: flex;
  409. justify-content: space-between;
  410. align-items: center;
  411. }
  412. .tit {
  413. font-size: px(44);
  414. font-weight: 700;
  415. color: #666;
  416. }
  417. .money {
  418. font-size: px(48);
  419. color: #ff475f;
  420. font-weight: 700;
  421. }
  422. }
  423. .info-list {
  424. background-color: #fff;
  425. .item {
  426. padding: px(30) px(40);
  427. border-bottom: 1px solid #f1f1f1;
  428. display: flex;
  429. justify-content: space-between;
  430. align-items: center;
  431. }
  432. .radio {
  433. width: px(100);
  434. flex-shrink: 0;
  435. .radio-inp {
  436. transform: scale(0.8);
  437. }
  438. }
  439. .info {
  440. width: 100%;
  441. font-size: px(44);
  442. }
  443. }
  444. .difference {
  445. display: flex;
  446. justify-content: space-between;
  447. align-items: center;
  448. background-color: #e9e9e9;
  449. color: #666;
  450. padding: px(20) px(10);
  451. font-size: px(38);
  452. .link {
  453. text-decoration: underline;
  454. }
  455. }
  456. </style>