選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。

cart.vue 11 KiB

4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
4年前
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496
  1. <template>
  2. <view class="container">
  3. <!-- 空白页 -->
  4. <view v-if="(!hasLogin || empty===true) && state != 'load'" class="empty">
  5. <image src="/static/emptyCart.jpg" mode="aspectFit"></image>
  6. <view v-if="hasLogin" class="empty-tips">
  7. 空空如也
  8. <navigator class="navigator" v-if="hasLogin" url="../index/index" open-type="switchTab">随便逛逛></navigator>
  9. </view>
  10. <view v-else class="empty-tips">
  11. 空空如也
  12. <view class="navigator" @click="navToLogin">去登陆></view>
  13. </view>
  14. </view>
  15. <view v-else>
  16. <!-- 列表 -->
  17. <view class="cart-list">
  18. <block v-for="(item, index) in cartList" :key="item.id">
  19. <view class="cart-item" :class="{'b-b': index!==cartList.length-1}" :style="{'background':item.isset?'':'#f5f5f5'}"
  20. @click="navTo(`/pages/product/product?id=${item.product_id}&flash=0`)">
  21. <view class="image-wrapper">
  22. <image :src="item.image" class="loaded" mode="aspectFill"></image>
  23. <view v-if="item.isset == true" class="yticon icon-xuanzhong checkbox" :class="{checked: item.choose}"
  24. @click.stop="check('item', index)"></view>
  25. </view>
  26. <view class="item-right">
  27. <text class="clamp title">{{item.title}}</text>
  28. <text class="attr" v-if="item.spec">{{item.spec}}</text>
  29. <text class="price">¥{{item.nowPrice}} <text style="color:red"> {{cartPrice(item.oldPrice, item.nowPrice)}}</text></text>
  30. <uni-number-box class="step" :min="1" :max="item.stock" :disabled="item.number>=item.stock" :value="cartList[index].number"
  31. :isMax="item.number>=item.stock?true:false" :isMin="item.number===1" :index="index" @eventChange="numberChange"></uni-number-box>
  32. </view>
  33. <text class="del-btn yticon icon-lajitong" @click.stop="deleteCartItem(index)"></text>
  34. <text class="invalid" v-if="item.isset == false">失效</text>
  35. <text class="invalid" v-if="item.stock == 0 && item.isset == true">库存不足</text>
  36. </view>
  37. </block>
  38. </view>
  39. <!-- 底部菜单栏 -->
  40. <view class="action-section" v-if="state != 'load'">
  41. <view class="checkbox">
  42. <image style="position: absolute;" :src="allChoose?'/static/selected.png':'/static/select.png'" mode="aspectFit"
  43. @click="check('all')"></image>
  44. <view class="clear-btn" :class="{show: allChoose}" @click="clearCart">
  45. Clear
  46. </view>
  47. </view>
  48. <view class="total-box">
  49. <text class="price">¥{{total}}</text>
  50. </view>
  51. <button type="primary" class="no-border confirm-btn" @click="createOrder">Buy Now</button>
  52. </view>
  53. </view>
  54. </view>
  55. </template>
  56. <script>
  57. import {
  58. mapState
  59. } from 'vuex';
  60. import uniNumberBox from '@/components/uni-number-box.vue'
  61. export default {
  62. components: {
  63. uniNumberBox
  64. },
  65. data() {
  66. return {
  67. total: 0, //总价格
  68. allChoose: false, //全选状态 true|false
  69. empty: false, //空白页现实 true|false
  70. cartList: [],
  71. state: 'load'
  72. };
  73. },
  74. onLoad() {
  75. },
  76. onPullDownRefresh() {
  77. this.state = 'load';
  78. this.cartList = [];
  79. this.getCart();
  80. },
  81. onShow() {
  82. this.state = 'load';
  83. this.cartList = [];
  84. this.getCart();
  85. },
  86. watch: {
  87. //显示空白页
  88. cartList(e) {
  89. let empty = e.length === 0 ? true : false;
  90. if (this.empty !== empty) {
  91. this.empty = empty;
  92. }
  93. }
  94. },
  95. computed: {
  96. ...mapState(['hasLogin'])
  97. },
  98. methods: {
  99. async getCart() {
  100. let login = await this.$api.checkLogin();
  101. if (login) {
  102. let data = await this.$api.request('/cart');
  103. uni.stopPullDownRefresh();
  104. this.state = 'loaded';
  105. if (data) {
  106. this.cartList = data;
  107. this.calcTotal();
  108. }
  109. }
  110. },
  111. cartPrice(oldPrice, nowPrice) {
  112. let string = '';
  113. if (oldPrice < nowPrice) {
  114. let number = (nowPrice - oldPrice).toFixed(2);
  115. string = ' ↑涨价 ' + number + '元';
  116. } else if (oldPrice > nowPrice) {
  117. let number = (oldPrice - nowPrice).toFixed(2);
  118. string = ' ↓降价 ' + number + '元';
  119. }
  120. return string;
  121. },
  122. navToLogin() {
  123. uni.navigateTo({
  124. url: '/pages/public/login'
  125. })
  126. },
  127. //选中状态处理
  128. async check(type, index) {
  129. let trueArr = [];
  130. let falseArr = [];
  131. let oldChoose = [];
  132. const list = this.cartList;
  133. //保存旧的数据
  134. list.forEach(item => {
  135. if (item.choose) {
  136. oldChoose.push(item.cart_id);
  137. }
  138. })
  139. //本地处理
  140. if (type === 'item') {
  141. this.cartList[index].choose = !this.cartList[index].choose;
  142. if (this.cartList[index].choose) {
  143. trueArr.push(this.cartList[index].cart_id);
  144. } else {
  145. falseArr.push(this.cartList[index].cart_id);
  146. }
  147. } else {
  148. const choose = !this.allChoose
  149. list.forEach(item => {
  150. item.choose = choose;
  151. if (item.isset) {
  152. if (choose) {
  153. trueArr.push(item.cart_id);
  154. } else {
  155. falseArr.push(item.cart_id);
  156. }
  157. }
  158. })
  159. this.allChoose = choose;
  160. }
  161. this.calcTotal(type);
  162. //远程处理
  163. let result = await this.$api.request('/cart/choose_change', 'POST', {
  164. trueArr,
  165. falseArr
  166. });
  167. if (!result) {
  168. //恢复原来勾选的状态
  169. list.forEach(item => {
  170. if (oldChoose.indexOf(item.cart_id) >= 0) {
  171. item.choose = 1;
  172. } else {
  173. item.choose = 0;
  174. }
  175. })
  176. this.calcTotal(type);
  177. }
  178. },
  179. //数量
  180. async numberChange(data) {
  181. let oldNumber = this.cartList[data.index].number;
  182. let newNumber = data.number;
  183. this.cartList[data.index].number = newNumber;
  184. this.calcTotal();
  185. let cart_id = this.cartList[data.index].cart_id;
  186. let result = await this.$api.request('/cart/number_change?id=' + cart_id, 'GET', {
  187. number: newNumber
  188. }, false);
  189. if (!result) {
  190. this.cartList[data.index].number = oldNumber;
  191. this.calcTotal();
  192. }
  193. },
  194. //删除
  195. async deleteCartItem(index) {
  196. let list = this.cartList;
  197. let row = list[index];
  198. let id = row.cart_id;
  199. uni.showModal({
  200. cancelText: 'Cancel',
  201. confirmText: 'OK',
  202. content: 'Confirm delete ' + list[index].title + '?',
  203. success: async (e) => {
  204. if (e.confirm) {
  205. let result = await this.$api.request('/cart/delete?', 'POST', {
  206. id: id
  207. });
  208. if (result) {
  209. let tempCart = this.cartList.splice(index, 1);
  210. this.calcTotal();
  211. }
  212. }
  213. }
  214. })
  215. },
  216. //清空
  217. async clearCart() {
  218. let [error, res] = await uni.showModal({
  219. title: 'Confirm empty?'
  220. });
  221. if (res.confirm) {
  222. let id = [];
  223. this.cartList.forEach(item => {
  224. id.push(item.cart_id);
  225. });
  226. let data = this.$api.request('/cart/delete', 'POST', {
  227. id: id
  228. });
  229. let that = this;
  230. if (data) {
  231. setTimeout(function() {
  232. that.state = 'load';
  233. that.cartList = [];
  234. that.getCart();
  235. }, 300);
  236. }
  237. }
  238. },
  239. //计算总价
  240. calcTotal() {
  241. let list = this.cartList;
  242. if (list.length === 0) {
  243. this.empty = true;
  244. return;
  245. }
  246. let total = 0;
  247. let choose = true;
  248. list.forEach(item => {
  249. if (item.isset) {
  250. if (item.choose == 1) {
  251. total += item.nowPrice * item.number;
  252. } else if (choose === true) {
  253. choose = false;
  254. }
  255. }
  256. })
  257. this.allChoose = choose;
  258. this.total = total.toFixed(2);
  259. },
  260. //创建订单
  261. createOrder() {
  262. let list = this.cartList;
  263. let cartId = [];
  264. list.forEach(item => {
  265. if (item.choose) {
  266. cartId.push(item.cart_id);
  267. }
  268. })
  269. if (cartId.length == 0) {
  270. this.$api.msg('没有选中商品');
  271. return;
  272. }
  273. this.$api.navTo(`/pages/order/createOrder?cart=${cartId.join(',')}`);
  274. },
  275. navTo(url) {
  276. this.$api.navTo(url);
  277. }
  278. }
  279. }
  280. </script>
  281. <style lang='scss'>
  282. .container {
  283. margin-bottom: 134upx;
  284. /* 空白页 */
  285. .empty {
  286. position: fixed;
  287. left: 0;
  288. top: 0;
  289. width: 100%;
  290. height: 100vh;
  291. padding-bottom: 100upx;
  292. display: flex;
  293. justify-content: center;
  294. flex-direction: column;
  295. align-items: center;
  296. background: #fff;
  297. image {
  298. width: 240upx;
  299. height: 160upx;
  300. margin-bottom: 30upx;
  301. }
  302. .empty-tips {
  303. display: flex;
  304. font-size: $font-sm+2upx;
  305. color: $font-color-disabled;
  306. .navigator {
  307. color: $uni-color-primary;
  308. margin-left: 16upx;
  309. }
  310. }
  311. }
  312. }
  313. /* 购物车列表项 */
  314. .cart-item {
  315. display: flex;
  316. position: relative;
  317. padding: 30upx 40upx;
  318. .image-wrapper {
  319. width: 230upx;
  320. height: 230upx;
  321. flex-shrink: 0;
  322. position: relative;
  323. image {
  324. border-radius: 8upx;
  325. }
  326. }
  327. .checkbox {
  328. position: absolute;
  329. left: -16upx;
  330. top: -16upx;
  331. z-index: 8;
  332. font-size: 44upx;
  333. line-height: 1;
  334. padding: 4upx;
  335. color: $font-color-disabled;
  336. background: #fff;
  337. border-radius: 50px;
  338. }
  339. .item-right {
  340. display: flex;
  341. flex-direction: column;
  342. flex: 1;
  343. overflow: hidden;
  344. position: relative;
  345. padding-left: 30upx;
  346. .title,
  347. .price {
  348. font-size: $font-base + 2upx;
  349. color: $font-color-dark;
  350. height: 40upx;
  351. line-height: 40upx;
  352. }
  353. .attr {
  354. font-size: $font-sm + 2upx;
  355. color: $font-color-light;
  356. height: 50upx;
  357. line-height: 50upx;
  358. }
  359. .price {
  360. height: 50upx;
  361. line-height: 50upx;
  362. }
  363. }
  364. .del-btn {
  365. padding: 4upx 10upx;
  366. font-size: 34upx;
  367. height: 50upx;
  368. color: $font-color-light;
  369. }
  370. .invalid {
  371. position: absolute;
  372. right: 0;
  373. bottom: 0;
  374. background: #999999;
  375. color: #ffffff;
  376. padding: 6upx 12upx;
  377. border-radius: 10upx;
  378. font-size: 26upx;
  379. margin-right: 50upx;
  380. margin-bottom: 32upx;
  381. }
  382. }
  383. /* 底部栏 */
  384. .action-section {
  385. /* #ifdef H5 */
  386. margin-bottom: 100upx;
  387. /* #endif */
  388. position: fixed;
  389. max-width: 1024px;
  390. margin: auto;
  391. left: 30upx;
  392. right: 30upx;
  393. bottom: 60px;
  394. z-index: 95;
  395. display: flex;
  396. align-items: center;
  397. height: 100upx;
  398. padding: 0 30upx;
  399. background: rgba(255, 255, 255, .9);
  400. box-shadow: 0 0 20upx 0 rgba(0, 0, 0, .5);
  401. border-radius: 16upx;
  402. .checkbox {
  403. height: 52upx;
  404. position: relative;
  405. image {
  406. width: 52upx;
  407. height: 100%;
  408. position: relative;
  409. z-index: 5;
  410. }
  411. }
  412. .clear-btn {
  413. position: absolute;
  414. left: 26upx;
  415. top: 0;
  416. z-index: 4;
  417. width: 0;
  418. height: 52upx;
  419. line-height: 52upx;
  420. padding-left: 38upx;
  421. font-size: $font-base;
  422. color: #fff;
  423. background: $font-color-disabled;
  424. border-radius: 0 50px 50px 0;
  425. opacity: 0;
  426. transition: .2s;
  427. &.show {
  428. opacity: 1;
  429. width: 120upx;
  430. }
  431. }
  432. .total-box {
  433. flex: 1;
  434. display: flex;
  435. flex-direction: column;
  436. text-align: right;
  437. padding-right: 40upx;
  438. .price {
  439. font-size: $font-lg;
  440. color: $font-color-dark;
  441. }
  442. }
  443. .confirm-btn {
  444. padding: 0 38upx;
  445. margin: 0;
  446. border-radius: 100px;
  447. height: 76upx;
  448. line-height: 76upx;
  449. font-size: $font-base + 2upx;
  450. background: $uni-color-primary;
  451. box-shadow: 1px 2px 5px rgba(217, 60, 93, 0.72)
  452. }
  453. }
  454. /* 复选框选中状态 */
  455. .action-section .checkbox.checked,
  456. .cart-item .checkbox.checked {
  457. color: $uni-color-primary;
  458. }
  459. </style>