shopping-cart1.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690
  1. <!-- 购物车 -->
  2. <template>
  3. <view class="container">
  4. <view class="discount-box" v-if="discountstate == true">
  5. <view class="discount">
  6. <view class="discount-off" @click="discountclose">×</view>
  7. <view class="discount-title">金额明细</view>
  8. <view class="discount-tips">实际优惠金额以下单页为准</view>
  9. <view class="discount-list">
  10. <text>商品总价</text>
  11. <text>¥1200</text>
  12. </view>
  13. <view class="discount-list2">
  14. <text>满减</text>
  15. <text>-¥200</text>
  16. </view>
  17. <view class="discount-list2">
  18. <text>优惠券</text>
  19. <text>-¥200</text>
  20. </view>
  21. <view class="discount-list2">
  22. <text>共减</text>
  23. <text>-¥200</text>
  24. </view>
  25. <view class="discount-list">
  26. <text>合计(合计金额不含运费、进口税)</text>
  27. <text>¥1200</text>
  28. </view>
  29. </view>
  30. </view>
  31. <view class="status_bar">
  32. <!-- 这里是状态栏 -->
  33. </view>
  34. <view class="page-header">
  35. <view class="text-center">购物车</view>
  36. <block v-if="cartList.length > 0">
  37. <view class="edit" v-if="btnType === 'edit'" @click="$u.throttle(btnTool('edit'), 2000)">管理</view>
  38. <view class="edit" v-if="btnType === 'done'" @click="$u.throttle(btnTool('done'), 2000)">完成</view>
  39. </block>
  40. </view>
  41. <view class="tips-box">
  42. <view>限时30元购物券免费领,一人一次限领一张</view>
  43. <view class="tips-del">×</view>
  44. </view>
  45. <!-- <u-checkbox-group> -->
  46. <view class="cart-box" v-if="cartList.length > 0">
  47. <u-swipe-action :show="item.show" :disabled="swipeAction" btn-width="160" :index="index"
  48. v-for="(item, index) in cartList" :key="item._id"
  49. @click="$u.throttle(actionClick($event, item._id), 2000)" @open="actionOpen" :options="options"
  50. class="card-warp">
  51. <view class="item u-border-bottom">
  52. <u-checkbox-group @change="checkboxChange" width="36rpx">
  53. <u-checkbox v-model="item.check" size="36" shape="circle" active-color="#F2501A"
  54. class="checkbox"></u-checkbox>
  55. <view class="shop-title">店铺名称</view>
  56. </u-checkbox-group>
  57. </view>
  58. <view class="item u-border-bottom">
  59. <u-checkbox-group @change="checkboxChange" width="36rpx">
  60. <u-checkbox v-model="item.check" size="36" shape="circle" active-color="#F2501A"
  61. class="checkbox"></u-checkbox>
  62. </u-checkbox-group>
  63. <view class="cart-img">
  64. <image mode="widthFix" :src="item.product.image"
  65. @click.stop="$u.throttle(gotoDetail(item.product._id), 2000)"></image>
  66. </view>
  67. <!-- 此层wrap在此为必写的,否则可能会出现标题定位错误 -->
  68. <view class="title-wrap" @click.stop="$u.throttle(gotoDetail(item.product._id), 2000)">
  69. <view class="title u-line-1 ">{{ item.product.title }}</view>
  70. <view class="sku">{{ item.product.cate_name }}</view>
  71. <view class="price">
  72. ¥{{ priceInt(item.product.price) }}.{{ priceDecimal(item.product.price) }}
  73. <del>10.0</del>
  74. </view>
  75. </view>
  76. <u-number-box v-model="item.product.number" :min="1" :max="item.product.stock" input-width="100"
  77. input-height="50" @change="valChange($event, item.product._id)"></u-number-box>
  78. </view>
  79. </u-swipe-action>
  80. </view>
  81. <view class="page-box" v-else>
  82. <view>
  83. <view class="emputy">
  84. <image src="../../static/normal-1.png"></image>
  85. <view class="explain">
  86. <view class="tips">暂无购物车</view>
  87. </view>
  88. <view class="btn" @click="$u.throttle(gotoHome, 2000)">去逛逛</view>
  89. </view>
  90. </view>
  91. </view>
  92. <view class="bottom" v-if="cartList.length > 0">
  93. <u-checkbox @change="checkedAll" v-model="checked" size="40" shape="circle" active-color="#F2501A"
  94. class="checkall"><span>全选</span></u-checkbox>
  95. <text class="price fill" v-if="btnType === 'edit'">
  96. <view>
  97. <text class="sml">总计(不含运费):</text>
  98. ¥{{ totalPrice }}
  99. </view>
  100. <view class="discount-but" v-if="totalPrice != 0" @click="discount">
  101. <text>共优惠20.00元 查看明细</text>
  102. <image src="../../static/normal-1.png"></image>
  103. </view>
  104. </text>
  105. <u-button throttle-time="2000" v-if="btnType === 'edit'" @click="$u.throttle(btnClick('order'), 2000)"
  106. size="medium" :ripple="true" ripple-bg-color="#F2501A" shape="circle" :customStyle="customStyle">
  107. <text>
  108. <text v-if="this.discountstate == false">结算</text>
  109. <text v-else @click="discountclose">领卷结算</text>
  110. </text>
  111. <text v-if="totalCount > 0">{{ '(' + totalCount + ')' }}</text>
  112. </u-button>
  113. <u-button throttle-time="2000" v-if="btnType === 'done'" size="medium" :ripple="true" ripple-bg-color="#ccc"
  114. shape="circle" :customStyle="customStyle3" @click="$u.throttle(btnClick('del'), 2000)">
  115. 移入收藏夹
  116. </u-button>
  117. <u-button throttle-time="2000" v-if="btnType === 'done'" size="medium" :ripple="true" ripple-bg-color="#ccc"
  118. shape="circle" :customStyle="customStyle2" @click="$u.throttle(btnClick('del'), 2000)">
  119. 删除
  120. </u-button>
  121. </view>
  122. <u-modal v-model="showModal" :title="title" :content="content" :title-style="titleStyle"
  123. :content-style="contentStyle" :confirm-style="confirmStyle" :show-cancel-button="true" @confirm="confirm">
  124. </u-modal>
  125. <u-toast ref="uToast" />
  126. </view>
  127. </template>
  128. <script>
  129. import sortdata from '../../static/data/cartData.js';
  130. import uswipeaction from "@/uview-ui/components/u-swipe-action/u-swipe-action.vue";
  131. export default {
  132. data() {
  133. return {
  134. loadStatus: 'loadmore',
  135. cartList: [], //购物车列表
  136. showModal: false, //modal弹窗
  137. title: '提示', //弹窗标题
  138. content: '确认将1个宝贝删除?', //弹窗内容
  139. delId: '', //删除id,
  140. options: [{
  141. text: '移入收藏夹',
  142. style: {
  143. backgroundColor: '#F28A1A'
  144. }
  145. }, {
  146. text: '删除',
  147. style: {
  148. backgroundColor: '#F2501A'
  149. }
  150. }], //u-swipe-action样式
  151. btnType: 'edit', //按钮类型
  152. checked: false, //是否全选
  153. swipeAction: false, //是否禁止滑动
  154. // u-button样式
  155. customStyle: {
  156. color: '#fff',
  157. backgroundColor: '#F2501A',
  158. margin: '0',
  159. padding: '0 20rpx',
  160. width: '200rpx',
  161. fontSize: '32rpx'
  162. },
  163. customStyle2: {
  164. color: '#F2501A',
  165. backgroundColor: '#fff',
  166. margin: '0',
  167. padding: '0 20rpx',
  168. border: 'none',
  169. width: '200rpx',
  170. fontSize: '32rpx'
  171. },
  172. customStyle3: {
  173. color: '#F28A1A',
  174. backgroundColor: '#fff',
  175. margin: '10rpx',
  176. padding: '0 10rpx',
  177. border: 'none',
  178. width: '200rpx',
  179. fontSize: '30rpx'
  180. },
  181. // 模态窗样式
  182. titleStyle: {
  183. fontSize: '36rpx'
  184. },
  185. contentStyle: {
  186. fontSize: '36rpx'
  187. },
  188. confirmStyle: {
  189. color: '#fff',
  190. backgroundColor: '#F2501A'
  191. },
  192. discountstate: false //优惠页面状态true打开 false关闭
  193. };
  194. },
  195. onShow() {
  196. this.checked = false;
  197. this.getCart();
  198. console.log(this.cartList);
  199. },
  200. computed: {
  201. // 价格合计
  202. totalPrice() {
  203. let sumPrice = 0,
  204. items = this.cartList;
  205. items.forEach(val => {
  206. let priceVal = parseFloat(val.product.price),
  207. salesVal = parseFloat(val.product.number);
  208. if (val.check) {
  209. sumPrice += priceVal * salesVal;
  210. }
  211. });
  212. // sumPrice = sumPrice == 0 ? 0 : sumPrice.toFixed(2);
  213. let tofixNum = sumPrice.toFixed(2);
  214. return parseFloat(tofixNum);
  215. },
  216. // 数量统计
  217. totalCount() {
  218. let totalNumber = 0,
  219. items = this.cartList;
  220. items.forEach(val => {
  221. if (val.check) {
  222. totalNumber += val.number;
  223. }
  224. });
  225. return totalNumber;
  226. },
  227. // 价格整数
  228. priceInt() {
  229. return val => {
  230. let priceStr = val.toString();
  231. if (priceStr != parseInt(priceStr)) {
  232. return priceStr.split('.')[0];
  233. } else {
  234. return priceStr;
  235. }
  236. };
  237. },
  238. // 价格小数
  239. priceDecimal() {
  240. return val => {
  241. let priceStr = val.toString();
  242. if (priceStr != parseInt(priceStr)) {
  243. return priceStr.split('.')[1];
  244. } else {
  245. return '00';
  246. }
  247. };
  248. }
  249. },
  250. methods: {
  251. //打开优惠页面
  252. discount() {
  253. this.discountstate = true
  254. },
  255. //关闭优惠页面
  256. discountclose() {
  257. this.discountstate = false
  258. },
  259. // 购物车列表
  260. getCart() {
  261. const resData = sortdata.cartData;
  262. resData.forEach(val => {
  263. val.show = false;
  264. val.check = false;
  265. });
  266. this.cartList = resData;
  267. },
  268. gotoHome() {
  269. uni.switchTab({
  270. url: '/pages/tabbar/home/home'
  271. });
  272. },
  273. // 管理按钮
  274. btnTool(type) {
  275. switch (type) {
  276. case 'edit':
  277. this.btnType = 'done';
  278. this.swipeAction = true;
  279. let items = this.cartList;
  280. items.forEach(val => {
  281. val.show = false;
  282. });
  283. break;
  284. case 'done':
  285. this.btnType = 'edit';
  286. this.swipeAction = false;
  287. break;
  288. }
  289. },
  290. // 删除
  291. actionClick(index, del) {
  292. this.showModal = true;
  293. this.delId = del;
  294. },
  295. // 删除确认
  296. confirm() {
  297. let _self = this;
  298. let delArr = [],
  299. type = _self.btnType,
  300. items = _self.cartList;
  301. if (type === 'edit') {
  302. delArr.push(_self.delId);
  303. } else {
  304. items.forEach(val => {
  305. if (val.check) {
  306. delArr.push(val._id);
  307. }
  308. });
  309. _self.btnType = 'edit';
  310. }
  311. // 为了模拟清空购物车
  312. if (delArr.length == items.length) {
  313. _self.cartList = [];
  314. }
  315. _self.$refs.uToast.show({
  316. title: '删除了id为' + delArr + '的商品',
  317. icon: false
  318. });
  319. },
  320. // 如果打开一个的时候,不需要关闭其他,则无需实现本方法
  321. actionOpen(index) {
  322. // 先将正在被操作的swipeAction标记为打开状态,否则由于props的特性限制,
  323. // 原本为'false',再次设置为'false'会无效
  324. let items = this.cartList;
  325. items[index].show = true;
  326. items.forEach((val, idx) => {
  327. if (index != idx) {
  328. items[idx].show = false;
  329. }
  330. });
  331. },
  332. // 跳转详情
  333. gotoDetail(id) {
  334. console.log(id);
  335. // uni.navigateTo({
  336. // url: '/pages/good/detail?good_id=' + id
  337. // })
  338. },
  339. // 选中商品
  340. checkboxChange(e) {
  341. let items = this.cartList;
  342. let checkedArr = [];
  343. items.forEach(val => {
  344. if (val.check) {
  345. checkedArr.push(val);
  346. }
  347. });
  348. let len = checkedArr.length;
  349. // 如果选择的数组中有值,并且长度等于列表长度,就是全选
  350. if (len > 0 && len == items.length) {
  351. this.checked = true;
  352. } else {
  353. this.checked = false;
  354. }
  355. },
  356. // 数量增减
  357. valChange(e, id) {
  358. // console.log(e, id);
  359. },
  360. // 全选
  361. checkedAll() {
  362. this.checked = !this.checked;
  363. let flag = this.checked,
  364. items = this.cartList;
  365. items.forEach(val => {
  366. if (flag) {
  367. val.check = true;
  368. } else {
  369. val.check = false;
  370. }
  371. });
  372. },
  373. // 结算
  374. btnClick(type) {
  375. let _self = this;
  376. switch (type) {
  377. case 'order':
  378. let items = _self.cartList;
  379. let checkedArr = [];
  380. items.forEach(val => {
  381. if (val.check) {
  382. checkedArr.push(val._id);
  383. }
  384. });
  385. let shopIds = checkedArr.toString();
  386. if (!shopIds) {
  387. _self.$refs.uToast.show({
  388. title: '请选择要结算的商品!',
  389. icon: false
  390. });
  391. return;
  392. }
  393. console.log(shopIds);
  394. // uni.navigateTo({
  395. // url: '/pages/order/createOrder?cartId=' + shopIds
  396. // });
  397. break;
  398. case 'del':
  399. if (!_self.totalCount) {
  400. _self.$refs.uToast.show({
  401. title: '请选择要删除的商品!',
  402. icon: false
  403. });
  404. return;
  405. }
  406. _self.showModal = true;
  407. break;
  408. }
  409. }
  410. }
  411. };
  412. </script>
  413. <style lang="scss" scoped>
  414. page {
  415. background-color: #F2F2F2;
  416. }
  417. // 状态栏占位
  418. .status_bar {
  419. background-color: #F2501A;
  420. }
  421. .page-header {
  422. padding: 26rpx 30rpx;
  423. display: flex;
  424. position: relative;
  425. color: #000;
  426. font-size: 36rpx;
  427. background-color: #fff;
  428. .text-center {
  429. width: 100%;
  430. text-align: center;
  431. }
  432. .edit {
  433. position: absolute;
  434. top: 50%;
  435. transform: translateY(-50%);
  436. right: 30rpx;
  437. font-size: 32rpx;
  438. }
  439. }
  440. .item {
  441. display: flex;
  442. justify-content: space-between;
  443. padding: 20rpx;
  444. align-items: center;
  445. .cart-img {
  446. width: 200rpx;
  447. border-radius: 12rpx;
  448. overflow: hidden;
  449. image {
  450. width: 100%;
  451. }
  452. }
  453. .title-wrap {
  454. width: 440rpx;
  455. .sku {
  456. margin: 20rpx 0;
  457. font-size: 28rpx;
  458. color: #666;
  459. }
  460. .price {
  461. font-size: 32rpx;
  462. color: #F2501A;
  463. .large {
  464. margin-left: 6rpx;
  465. font-size: 40rpx;
  466. font-weight: bold;
  467. }
  468. del {
  469. margin-left: 10rpx;
  470. color: #919191;
  471. font-size: 24rpx;
  472. }
  473. }
  474. .clamp {
  475. overflow: hidden;
  476. text-overflow: ellipsis;
  477. white-space: nowrap;
  478. display: block;
  479. }
  480. .title {
  481. text-align: left;
  482. font-size: 30rpx;
  483. color: #333;
  484. // margin-top: 20rpx;
  485. line-height: 46rpx;
  486. }
  487. }
  488. .u-numberbox {
  489. position: absolute;
  490. bottom: 20rpx;
  491. right: 20rpx;
  492. }
  493. }
  494. // emputy
  495. .emputy {
  496. text-align: center;
  497. margin: 200rpx auto;
  498. font-size: 32rpx;
  499. image {
  500. width: 582rpx;
  501. height: 370rpx;
  502. margin-bottom: 20rpx;
  503. }
  504. .tips {
  505. font-size: 24rpx;
  506. color: #999999;
  507. margin-top: 20rpx;
  508. }
  509. .btn {
  510. margin: 80rpx auto;
  511. width: 200rpx;
  512. border-radius: 32rpx;
  513. line-height: 64rpx;
  514. color: #F2501A;
  515. font-size: 26rpx;
  516. border: 1px solid #F2501A;
  517. }
  518. }
  519. .bottom {
  520. border-radius: 100rpx;
  521. background-color: #F8F9FC;
  522. padding: 0 20rpx;
  523. width: 100%;
  524. position: fixed;
  525. left: 0;
  526. bottom: var(--window-bottom);
  527. display: flex;
  528. align-items: center;
  529. justify-content: flex-end;
  530. z-index: 99;
  531. height: 110rpx;
  532. box-sizing: border-box;
  533. padding-bottom: constant(safe-area-inset-bottom);
  534. padding-bottom: env(safe-area-inset-bottom);
  535. .checkall {
  536. position: absolute;
  537. left: 20rpx;
  538. top: 50%;
  539. transform: translateY(-50%);
  540. span {
  541. color: #000;
  542. font-weight: bold;
  543. font-size: 32rpx;
  544. }
  545. }
  546. .price {
  547. margin-right: 20rpx;
  548. color: #F2501A;
  549. font-size: 26rpx;
  550. .sml {
  551. margin-right: 10rpx;
  552. color: #202020;
  553. font-size: 30rpx;
  554. }
  555. }
  556. }
  557. .tips-box {
  558. width: 100%;
  559. height: 80rpx;
  560. display: flex;
  561. align-items: center;
  562. justify-content: space-between;
  563. padding: 0px 20rpx;
  564. background-color: #FFF0C7;
  565. color: #F2501A;
  566. margin-bottom: 40rpx;
  567. }
  568. .tips-del {
  569. font-size: 50rpx;
  570. }
  571. .card-warp {
  572. margin-top: 20rpx;
  573. }
  574. .shop-title {
  575. margin: 5rpx 0rpx 0 10rpx;
  576. font-weight: bold;
  577. }
  578. .discount-but {
  579. display: flex;
  580. align-items: center;
  581. image {
  582. width: 40rpx;
  583. height: 40rpx;
  584. }
  585. }
  586. // 金额明细
  587. .discount-box {
  588. width: 100%;
  589. min-height: 87vh;
  590. background-color: rgba(0, 0, 0, 0.2);
  591. position: fixed;
  592. top: 0;
  593. z-index: 999;
  594. display: flex;
  595. align-items: flex-end;
  596. }
  597. .discount {
  598. width: 100%;
  599. height: 450rpx;
  600. background-color: #fff;
  601. border-top-left-radius: 20rpx;
  602. border-top-right-radius: 20rpx;
  603. background-color: #fff;
  604. display: flex;
  605. flex-direction: column;
  606. align-items: center;
  607. .discount-off {
  608. width: 100%;
  609. color: #777777;
  610. font-size: 60rpx;
  611. height: 30rpx;
  612. display: flex;
  613. justify-content: flex-end;
  614. margin-right: 10rpx;
  615. }
  616. .discount-title {
  617. color: #202020;
  618. font-size: 32rpx;
  619. }
  620. .discount-tips {
  621. color: #A7A7A7;
  622. margin-top: 10rpx;
  623. }
  624. .discount-list,
  625. .discount-list2 {
  626. width: 100%;
  627. display: flex;
  628. align-items: center;
  629. justify-content: space-between;
  630. padding: 15rpx;
  631. text:last-child {
  632. font-weight: bold;
  633. }
  634. }
  635. .discount-list2 {
  636. text:last-child {
  637. color: #F2501A;
  638. }
  639. }
  640. }
  641. </style>