index.vue 26 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010
  1. <template>
  2. <view class="content">
  3. <!-- 骨架屏 -->
  4. <u-skeleton el-color="#efefef" bg-color="#fff" :loading="loading && isFirstComeIn" :animation="true"></u-skeleton>
  5. <global-loading />
  6. <!-- 购物车 -->
  7. <view class="u-skeleton">
  8. <view v-if="!isEmpty">
  9. <u-sticky bg-color="#fff">
  10. <view class="cart-bg u-skeleton-fillet">
  11. <view class="cart-num-box">
  12. <image class=" u-skeleton-fillet" src="../../../static/images/logo/jf-name.png"></image>
  13. <text v-if="!showManage" class="btn-box " @click="showManage = !showManage">
  14. 管理
  15. </text>
  16. <text v-if="showManage" class="btn-box" @click="showManage = !showManage">
  17. 完成
  18. </text>
  19. </view>
  20. <view>
  21. <text class="num-box u-skeleton-fillet">
  22. <text class="num">{{ settleAccountsObj.allNum }}</text>
  23. 件宝贝
  24. </text>
  25. </view>
  26. </view>
  27. </u-sticky>
  28. <view class="cart-list-box">
  29. <view v-for="(item, index) in dataList" :key="item.shopId" class="itemBox">
  30. <view v-if="item.skus.length > 0" class="item">
  31. <view class="shop-box">
  32. <image
  33. v-if="item.selected === 1" mode="aspectFill u-skeleton-fillet"
  34. src="../../../static/images/origin/selectActive.png" class="cart-select-img"
  35. @click.stop="handleSelectShop(index, 0)"
  36. ></image>
  37. <image
  38. v-else mode="aspectFill u-skeleton-fillet" src="../../../static/images/origin/selectEmpty.png"
  39. class="cart-select-img" @click.stop="handleSelectShop(index, 1)"
  40. ></image>
  41. <view class="shop-name-box u-skeleton-fillet" @click="$jump(`${jumpObj.store}?storeId=${item.shopId}`)">
  42. <image src="../../../static/images/origin/orderStoreIcon.png" class="shop-img"></image>
  43. <text class="shop-name">{{ item.shopName }}</text>
  44. <image src="../../../static/images/origin/arrowRight.png" class="arrow-right-img"></image>
  45. </view>
  46. </view>
  47. <view v-if="item.currentRules && item.currentRules.number" class="rulesBox flex-items">
  48. <image class="mar-right-20" src="../../../static/images/origin/zuheIcon.png"></image>
  49. <view class="fs24 font-color-C83732">
  50. 已满足【{{ item.currentRules.price }}元任选{{ item.currentRules.number }}件】!
  51. </view>
  52. </view>
  53. <view v-for="(skuItem, cIndex) in dataList[index].skus" class="product-list-box ">
  54. <view
  55. class="pro-item"
  56. @click="$jump(`${jumpObj.detail}?shopId=${item.shopId}&productId=${skuItem.productId}&skuId=${skuItem.skuId}`)"
  57. >
  58. <image
  59. v-if="skuItem.selected == 1" mode="aspectFill u-skeleton-fillet"
  60. src="../../../static/images/origin/selectActive.png" class="cart-select-img"
  61. @click.stop="handleSelectSku(index, cIndex, 0)"
  62. ></image>
  63. <image
  64. v-else mode="aspectFill u-skeleton-fillet" src="../../../static/images/origin/selectEmpty.png"
  65. class="cart-select-img" @click.stop="handleSelectSku(index, cIndex, 1)"
  66. ></image>
  67. <view class="pro-r">
  68. <image :src="skuItem.image" class="pro-img default-img u-skeleton-fillet"></image>
  69. <view class="pro-r-r u-skeleton-fillet">
  70. <view class="pro-name">{{ skuItem.productName }}</view>
  71. <view class="sku-box">
  72. <text v-if="skuItem.value">{{ skuItem.value }}</text>
  73. <text v-else>默认规格</text>
  74. <!-- <text></text> -->
  75. </view>
  76. <view class="pro-price-num-box">
  77. <view class="pro-price-box">
  78. <text class="fuhao">¥</text>
  79. <text>{{ skuItem.price }}</text>
  80. </view>
  81. <view class="pro-num-box">
  82. <text class="num-btn r" @click.stop="handleSubSkuNumber(index, cIndex)">
  83. -
  84. </text>
  85. <text class="num">{{ skuItem.number }}</text>
  86. <text class="num-btn l" @click.stop="handleAddSkuNumber(index, cIndex)">
  87. +
  88. </text>
  89. </view>
  90. </view>
  91. </view>
  92. </view>
  93. </view>
  94. </view>
  95. </view>
  96. </view>
  97. </view>
  98. <!-- #ifdef H5 -->
  99. <view class="cart-bottom-box-h5">
  100. <!-- #endif -->
  101. <!-- #ifndef H5 -->
  102. <view class="cart-bottom-box-app">
  103. <!-- #endif -->
  104. <view class="cart-bottom">
  105. <view class="left">
  106. <image
  107. v-if="settleAccountsObj.isAllCheck" mode="aspectFill"
  108. src="../../../static/images/origin/selectActive.png" class="cart-select-img"
  109. @click="handleSelectAll(0)"
  110. ></image>
  111. <image
  112. v-else mode="aspectFill" src="../../../static/images/origin/selectEmpty.png"
  113. class="cart-select-img" @click="handleSelectAll(1)"
  114. ></image>
  115. <text>全选</text>
  116. </view>
  117. <view v-if="!showManage" class="right">
  118. <view class="price-box">
  119. <text>合计:</text>
  120. <text class="price">¥{{ settleAccountsObj.checkMoney }}</text>
  121. </view>
  122. <view class="btn-confirm" @click="settlementTap">
  123. 结算({{ settleAccountsObj.checkNum }})
  124. </view>
  125. </view>
  126. <view v-if="showManage" class="right">
  127. <view class="btn-delete" @click="handleOpenDelete">
  128. 删除
  129. </view>
  130. </view>
  131. </view>
  132. </view>
  133. </view>
  134. </view>
  135. <!-- 购物车为空 -->
  136. <view v-if="isEmpty" class="emptyCart-box flex-items-plus flex-column">
  137. <image class="emptyCart-img" src="../../../static/images/origin/cartEmpty.png"></image>
  138. <label class="font-color-999 fs26 mar-top-30">你的购物车还没有宝贝哦</label>
  139. <label class="font-color-999 fs26 mar-top-10">快去首页选一个吧~</label>
  140. <view class="goToShopping" @click="$jumpToTabbar(jumpObj.shopping)">
  141. 去购物
  142. </view>
  143. </view>
  144. <!-- 热门推荐 -->
  145. <HotTemplate class="u-skeleton-fillet" />
  146. <view style="width: 100%;height: 120rpx;background-color:#fff;"></view>
  147. <!-- 删除确认弹窗 -->
  148. <DeleteModal :show-tip.sync="showDeleteModal" @confirm="handleDoDelete"></DeleteModal>
  149. </view>
  150. </view>
  151. </template>
  152. <script>
  153. import HotTemplate from '../../../components/hoteRecommed/index.vue'
  154. import DeleteModal from './components/DeleteModal'
  155. import api from '../../../components/canvasShow/config/api'
  156. import { J_STORAGE_KEY } from '../../../config/constant'
  157. import { defaultCartList, getCartNumberBySelect, getPriceBySelect } from './cartUtils'
  158. import lodash from 'lodash'
  159. let cacheKey = ''
  160. const NET = require('../../../utils/request')
  161. const API = require('../../../config/api')
  162. export default {
  163. name: 'Cart',
  164. components: {
  165. HotTemplate,
  166. DeleteModal
  167. },
  168. data() {
  169. return {
  170. isFirstComeIn: true, // 是否是首次进入
  171. loading: true, // 是否在加载
  172. showManage: false, // 是否开启管理
  173. dataList: [
  174. { skus: [] }
  175. ], // 购物车数据
  176. showDeleteModal: false, // 是否展示删除
  177. isEmpty: false, // 购物车是否为空
  178. userInfo: {}, // 用户信息
  179. // 跳转对象
  180. jumpObj: {
  181. store: '/pages_category_page1/store/index',
  182. detail: '/pages_category_page1/goodsModule/goodsDetails',
  183. shopping: '/pages/tabbar/index/index'
  184. },
  185. // 底部结算条对象
  186. settleAccountsObj: {
  187. allNum: 0, // 所有sku数量(头部)
  188. checkNum: 0, // 选中sku的数量
  189. checkMoney: 0, // 选中sku的总价
  190. isAllCheck: false // 是否宣布选中
  191. }
  192. }
  193. },
  194. onShow() {
  195. this.isFirstComeIn = true
  196. this.loading = true
  197. this.userInfo = uni.getStorageSync(J_STORAGE_KEY)
  198. cacheKey = this.userInfo.buyerUserId + 'cart_info'
  199. this.dataList = defaultCartList
  200. this.isEmpty = false
  201. this.getDataList()
  202. },
  203. methods: {
  204. /**
  205. * 获取购物车列表
  206. */
  207. getDataList: lodash.debounce(async function () {
  208. this.isEmpty = false
  209. this.loading = true
  210. try {
  211. const res = await NET.request(API.ShoppingCart, {}, 'GET')
  212. this.dataList = res.data
  213. this.settleAccountsObj.allNum = this.dataList.length
  214. if (this.dataList.length === 0) {
  215. this.isEmpty = true
  216. uni.setStorageSync('allCartNum', 0)
  217. uni.removeTabBarBadge({
  218. index: 3
  219. })
  220. }
  221. // sku为空的山沟
  222. const emptySkuShopArray = []
  223. this.dataList.forEach((shopObj, shopIndex) => {
  224. shopObj.currentIds = []
  225. shopObj.priceNumber = 0
  226. shopObj.rules = []
  227. shopObj.currentRules = {}
  228. shopObj.ids = 0
  229. // 处理下架商品
  230. for (let i = shopObj.skus.length - 1; i >= 0; i--) {
  231. // shelveState是否上架
  232. if (shopObj.skus[i].shelveState === 0) {
  233. // 删掉下架商品
  234. // todo 失效商品
  235. shopObj.skus.splice(i, 1)
  236. continue
  237. }
  238. if (shopObj.skus[i].activityType === 6 && shopObj.skus[i].selected === 1) {
  239. shopObj.currentIds.push(shopObj.skus[i].priceId)
  240. shopObj.priceNumber += shopObj.skus[i].number
  241. }
  242. }
  243. for (let i = 0; i < shopObj.skus.length; i++) {
  244. if (shopObj.skus[i].activityType === 6) {
  245. shopObj.ids = shopObj.skus[i].priceId
  246. break
  247. }
  248. }
  249. // 根据店铺索引获取规则
  250. this.getData(shopObj).then((res) => {
  251. shopObj.rules = res.data ? res.data[0].rules : {}
  252. this.handleSetGroupGood(shopIndex)
  253. })
  254. shopObj.skus.length === 0 ? emptySkuShopArray.push(shopObj) : undefined
  255. })
  256. this.isEmpty = emptySkuShopArray.length >= this.dataList.length
  257. this.handleRenderCart()
  258. // 数据回来就直接关闭骨架屏
  259. this.loading = false
  260. this.isFirstComeIn = false
  261. await this.handleUpdateMoneyAndNum()
  262. } finally {
  263. uni.hideLoading()
  264. }
  265. }, 500),
  266. /**
  267. * 获取组合定价
  268. * @param item
  269. * @return {Promise<unknown>}
  270. */
  271. getData(item) {
  272. return new Promise((resolve, reject) => {
  273. if (item.ids) {
  274. NET.request(api.getPrices, {
  275. shopId: item.shopId,
  276. ids: item.ids,
  277. page: 1,
  278. pageSize: 10
  279. }, 'GET').then((res) => {
  280. resolve(res)
  281. })
  282. .catch((e) => {
  283. reject(e)
  284. })
  285. } else {
  286. resolve([])
  287. }
  288. })
  289. },
  290. /**
  291. * 单个SKU数量减
  292. * @param shopIndex 店铺索引
  293. * @param skuIndex index店铺下sku商品索引
  294. */
  295. async handleSubSkuNumber(shopIndex, skuIndex) {
  296. const selectSku = this.dataList[shopIndex].skus[skuIndex]
  297. if (selectSku.number <= 1) {
  298. return uni.showToast({
  299. title: '亲!至少一件哦!',
  300. icon: 'none'
  301. })
  302. }
  303. --selectSku.number
  304. await this.handleUpdateCart(selectSku.skuId, selectSku.number)
  305. setTimeout(async () => {
  306. await this.getDataList()
  307. }, 500)
  308. },
  309. /**
  310. * 单个SKU数量加
  311. * @param shopIndex 店铺索引
  312. * @param skuIndex index店铺下sku商品索引
  313. */
  314. async handleAddSkuNumber(shopIndex, skuIndex) {
  315. const selectSku = this.dataList[shopIndex].skus[skuIndex]
  316. if (selectSku.number >= selectSku.stockNumber) {
  317. selectSku.number = selectSku.stockNumber
  318. return uni.showToast({
  319. title: '库存不足!',
  320. icon: 'none'
  321. })
  322. }
  323. if (selectSku.number < selectSku.stockNumber) {
  324. ++selectSku.number
  325. await this.handleUpdateCart(selectSku.skuId, selectSku.number)
  326. setTimeout(async () => {
  327. await this.getDataList()
  328. }, 500)
  329. }
  330. },
  331. /**
  332. * 更新总价和总数(底部结算栏,头部总数)
  333. * @return {Promise<void>}
  334. */
  335. async handleUpdateMoneyAndNum() {
  336. const { allNumber, checkNumber, isAllCheck } = await getCartNumberBySelect(this.dataList)
  337. const { money } = await getPriceBySelect(this.dataList)
  338. this.settleAccountsObj.checkMoney = money
  339. this.settleAccountsObj.isAllCheck = isAllCheck
  340. this.settleAccountsObj.allNum = allNumber
  341. this.settleAccountsObj.checkNum = checkNumber
  342. },
  343. /**
  344. * 请求服务端更新购物车数量
  345. * @param skuId :需要更新的skuId
  346. * @param number: 数量
  347. */
  348. handleUpdateCart: lodash.debounce(async function (skuId, number) {
  349. // 重新算钱和数量
  350. await NET.request(API.UpdateNumberCart, {
  351. skuId,
  352. number
  353. }, 'POST')
  354. }, 500),
  355. /**
  356. * 选中店铺
  357. * @param shopIndex 店铺索引
  358. * @param type 0否1是
  359. */
  360. handleSelectShop(shopIndex, type) {
  361. const shopObj = this.dataList[shopIndex]
  362. const shopCarts = [ {
  363. shopId: shopObj.shopId,
  364. skus: []
  365. } ]
  366. shopObj.selected = type
  367. // 设置当前店铺下的所有sku
  368. shopObj.skus.forEach((skuObj) => {
  369. skuObj.selected = type
  370. shopCarts[0].skus.push({
  371. skuId: skuObj.skuId,
  372. selected: skuObj.selected
  373. })
  374. })
  375. this.handleSetGroupGood(shopIndex)
  376. this.handleUpdateSelected(shopCarts)
  377. },
  378. /**
  379. * 商品单选
  380. * @param shopIndex 店铺索引dataList
  381. * @param skuIndex sku索引dataList[index].skus
  382. * @param type 是否选中 0否1是
  383. */
  384. handleSelectSku(shopIndex, skuIndex, type) {
  385. const shopObj = this.dataList[shopIndex]
  386. const skuObj = this.dataList[shopIndex].skus[skuIndex]
  387. skuObj.selected = type
  388. const shopCarts = [ {
  389. shopId: shopObj.shopId,
  390. skus: [ {
  391. skuId: skuObj.skuId,
  392. selected: skuObj.selected
  393. } ]
  394. } ]
  395. if (type === 1) {
  396. // 过滤店铺内未选择的sku
  397. const noSelectSkuList = shopObj.skus.filter((sku) => sku.selected === 0)
  398. if (noSelectSkuList.length >= 0) {
  399. shopObj.selected = 0
  400. } else {
  401. shopObj.selected = 1
  402. }
  403. } else {
  404. shopObj.selected = type
  405. }
  406. // 渲染组合商品
  407. this.handleSetGroupGood(shopIndex)
  408. this.handleUpdateSelected(shopCarts)
  409. },
  410. /**
  411. * 全选
  412. * @param type 是否选中 0否1是
  413. */
  414. handleSelectAll(type) {
  415. this.dataList.forEach((shopObj, shopIndex) => {
  416. // 组合支付商品数量
  417. const goodsOfJointNumber = shopObj.skus.reduce((prev, skuObj) => {
  418. skuObj.selected = type
  419. // 如果是组合支付
  420. if (skuObj.selected === 1 && skuObj.activityType === 6) {
  421. return prev + skuObj.number
  422. }
  423. }, 0)
  424. shopObj.selected = type
  425. shopObj.priceNumber = goodsOfJointNumber
  426. shopObj.currentRules = {}
  427. // 处理选中的组合商品
  428. if (type === 1) {
  429. this.handleSetGroupGood(shopIndex)
  430. }
  431. })
  432. this.handleUpdateSelected([])
  433. },
  434. /**
  435. * 处理组合商品(设置currentRules渲染横幅)
  436. * @param shopIndex
  437. */
  438. handleSetGroupGood(shopIndex) {
  439. const shopObj = this.dataList[shopIndex]
  440. shopObj.currentRules = {}
  441. shopObj.priceNumber = 0
  442. shopObj.skus.forEach((skuObj) => {
  443. if (skuObj.activityType === 6 && skuObj.selected === 1) {
  444. shopObj.priceNumber += skuObj.number
  445. }
  446. })
  447. const shopRules = this.dataList[shopIndex].rules
  448. for (let i = 0; i < shopRules.length; i++) {
  449. if (shopRules[i].number === shopObj.priceNumber) {
  450. shopObj.currentRules = shopRules[i]
  451. break
  452. } else if (shopRules[shopRules.length - 1].number < shopObj.priceNumber) {
  453. shopObj.currentRules = shopRules[shopRules.length - 1]
  454. break
  455. }
  456. }
  457. },
  458. /**
  459. * 更新缓存sku勾选和价格、数量显示
  460. * @param shopCarts:{shopId:number,skus:{skuId:number,select:number}}[] 只有一个对象(店铺)全选传空数组
  461. */
  462. handleUpdateSelected(shopCarts) {
  463. this.handleSetCache(shopCarts)
  464. this.handleUpdateMoneyAndNum()
  465. },
  466. /**
  467. * 设置购物车本地缓存(先存入本地缓存,再调用handleRenderCart根据本地缓存渲染)
  468. * @param shopCarts:{shopId:number,skus:{skuId:number,select:number}}[] 只有一个对象(店铺)全选传空数组
  469. */
  470. handleSetCache(shopCarts) {
  471. let cartInfo = uni.getStorageSync(cacheKey)
  472. if (cartInfo === '') {
  473. // 全选
  474. if (shopCarts.length <= 0) {
  475. // 全选直接缓存整个列表
  476. uni.setStorageSync(cacheKey, JSON.stringify(this.dataList))
  477. // 渲染视图
  478. this.handleRenderCart()
  479. return
  480. }
  481. // 无购物车信息
  482. cartInfo = shopCarts
  483. uni.setStorageSync(cacheKey, JSON.stringify(cartInfo))
  484. } else {
  485. cartInfo = JSON.parse(cartInfo)
  486. // 全选
  487. if (shopCarts.length <= 0) {
  488. // 全选直接缓存整个列表
  489. uni.setStorageSync(cacheKey, JSON.stringify(this.dataList))
  490. // 渲染视图
  491. this.handleRenderCart()
  492. return
  493. }
  494. // 看了代码逻辑结构,一次只会传一个商铺过来,大胆取0
  495. const shopItem = shopCarts[0]
  496. const cacheHaveInfo = cartInfo.findIndex((item) => item.shopId === shopItem.shopId)
  497. if (cacheHaveInfo < 0) {
  498. // 如果缓存中不存在当前商店信息,写入缓存
  499. cartInfo.push(shopItem)
  500. } else {
  501. // 获取到缓存项
  502. const cacheShopItem = cartInfo[cacheHaveInfo]
  503. // 判断传入的sku大小,sku length为1就是点单项,sku length>1就是点击了整个店铺
  504. if (shopItem.skus.length > 1) {
  505. // 点击整个店铺,直接赋值
  506. cartInfo[cacheHaveInfo] = shopItem
  507. } else {
  508. // 点击单项sku,获取到sku // 数据结构只会传入一项
  509. const shopItemSkuItem = shopItem.skus[0]
  510. // 在缓存中寻找
  511. const cacheShopItemSkuItemIndex = cacheShopItem.skus.findIndex((item) => item.skuId === shopItemSkuItem.skuId)
  512. cacheShopItemSkuItemIndex >= 0 ? cacheShopItem.skus[cacheShopItemSkuItemIndex] = shopItemSkuItem : cacheShopItem.skus.push(shopItemSkuItem)
  513. }
  514. }
  515. // 逻辑处理完毕更新缓存
  516. uni.setStorageSync(cacheKey, JSON.stringify(cartInfo))
  517. // 渲染视图
  518. this.handleRenderCart()
  519. }
  520. },
  521. /**
  522. * 根据本地缓存渲染购物车勾选
  523. * @constructor
  524. */
  525. handleRenderCart() {
  526. // 取消所有勾选
  527. this.dataList.forEach((shop) => {
  528. shop.selected = 0
  529. shop.skus.forEach((sku) => {
  530. sku.selected = 0
  531. })
  532. })
  533. // 校验缓存中的数据是否存在于购物车中
  534. this.handleCheckCacheAndUpdate()
  535. // 缓存内购物车信息
  536. let cartInfo = uni.getStorageSync(cacheKey)
  537. if (cartInfo === '') return
  538. cartInfo = JSON.parse(cartInfo)
  539. // 遍历购物车信息,寻找缓存比对
  540. this.dataList.forEach((nowCartShopItem) => {
  541. let shopSelect = 1
  542. const cacheCartShopItem = cartInfo.find((item) => item.shopId === nowCartShopItem.shopId)
  543. if (cacheCartShopItem) {
  544. // 如果缓存中有当前店铺,遍历当前购物车sku
  545. nowCartShopItem.skus.forEach((nowCartSkuItem) => {
  546. const cacheCartSkuItem = cacheCartShopItem.skus.find((item) => item.skuId === nowCartSkuItem.skuId)
  547. if (cacheCartSkuItem) {
  548. // 如果有一个未选中当前店铺就不能全选
  549. !cacheCartSkuItem.selected ? shopSelect = 0 : ''
  550. nowCartSkuItem.selected = cacheCartSkuItem.selected
  551. } else {
  552. shopSelect = 0
  553. }
  554. })
  555. } else {
  556. shopSelect = 0
  557. }
  558. nowCartShopItem.selected = shopSelect
  559. })
  560. },
  561. /**
  562. * 比较缓存内数据和后端数据是否一致,并且更新缓存
  563. * @constructor
  564. */
  565. handleCheckCacheAndUpdate() {
  566. // 缓存内购物车信息
  567. let cartInfo = uni.getStorageSync(cacheKey)
  568. if (cartInfo === '') return
  569. cartInfo = JSON.parse(cartInfo)
  570. // 校验缓存中的数据是否存在于购物车中
  571. cartInfo.forEach((cacheCartShopItem, cacheCartShopIndex) => {
  572. const nowCartShopItem = this.dataList.find((item) => item.shopId === cacheCartShopItem.shopId)
  573. if (!nowCartShopItem) {
  574. cartInfo.splice(cacheCartShopIndex, 1)
  575. } else {
  576. // 存在就校验缓存中的sku在不在后端返回的列表内
  577. cacheCartShopItem.skus.forEach((cacheCartSkuItem, cacheCartSkuIndex) => {
  578. const nowCartSkuItem = nowCartShopItem.skus.find((item) => item.skuId === cacheCartSkuItem.skuId)
  579. if (!nowCartSkuItem) {
  580. cacheCartShopItem.skus.splice(cacheCartSkuIndex, 1)
  581. }
  582. })
  583. }
  584. })
  585. uni.setStorageSync(cacheKey, JSON.stringify(cartInfo))
  586. },
  587. /**
  588. * 打开删除弹窗
  589. */
  590. handleOpenDelete() {
  591. if (!this.settleAccountsObj.checkNum) {
  592. return uni.showToast({
  593. title: '请先选择对应商品',
  594. icon: 'none'
  595. })
  596. }
  597. this.showDeleteModal = true
  598. },
  599. /**
  600. * 执行删除
  601. * @return {Promise<void>}
  602. */
  603. async handleDoDelete() {
  604. let ids = []
  605. for (const shopObj of this.dataList) {
  606. ids = [...ids, ...shopObj.skus.filter((sku) => sku.selected === 1 || sku.selected === true).map((sku) => sku.skuId)]
  607. }
  608. await NET.request(API.DeleteCart, { ids }, 'POST')
  609. this.showDeleteModal = false
  610. await this.getDataList()
  611. },
  612. /**
  613. * 结算购物车
  614. * @return {Promise<void>}
  615. */
  616. async settlementTap() {
  617. const { shopList } = await getPriceBySelect(this.dataList)
  618. uni.setStorageSync('skuItemDTOList', shopList)
  619. this.$jump('/pages_category_page1/orderModule/orderConfirm?type=2')
  620. }
  621. }
  622. }
  623. </script>
  624. <style
  625. lang="scss"
  626. scoped
  627. >
  628. .content {
  629. //overflow: hidden;
  630. //opacity: 0;
  631. .cart-bg {
  632. width: 100%;
  633. height: 180rpx;
  634. background-color: #fff;
  635. .cart-num-box {
  636. display: flex;
  637. flex-direction: row;
  638. align-items: center;
  639. justify-content: space-between;
  640. image {
  641. width: 286rpx;
  642. height: 72rpx;
  643. }
  644. .btn-box {
  645. font-size: 30rpx;
  646. color: #333333;
  647. padding: 30rpx;
  648. box-sizing: border-box;
  649. display: inline-block;
  650. }
  651. }
  652. .num-box {
  653. padding: 30rpx 0 30rpx 30rpx;
  654. box-sizing: border-box;
  655. font-size: 30rpx;
  656. color: #C5CACF;
  657. }
  658. }
  659. .cart-list-box {
  660. box-sizing: border-box;
  661. .itemBox {
  662. .item {
  663. background: #fff;
  664. border-bottom: 16rpx solid #F8F9FA;
  665. .shop-box {
  666. margin-top: 5rpx;
  667. display: flex;
  668. flex-direction: row;
  669. align-items: center;
  670. border-bottom: 1px solid #eee;
  671. position: relative;
  672. .cart-select-img {
  673. width: 40rpx;
  674. height: 40rpx;
  675. margin: 30rpx;
  676. box-sizing: border-box;
  677. }
  678. .shop-name-box {
  679. display: flex;
  680. flex-direction: row;
  681. align-items: center;
  682. .shop-img {
  683. width: 36rpx;
  684. height: 36rpx;
  685. margin-right: 10rpx;
  686. }
  687. .shop-name {
  688. font-size: 30rpx;
  689. color: #333;
  690. font-weight: bold;
  691. display: inline-block;
  692. margin-left: 10rpx;
  693. }
  694. .arrow-right-img {
  695. width: 30rpx;
  696. height: 30rpx;
  697. box-sizing: border-box;
  698. margin-left: 30rpx;
  699. position: absolute;
  700. right: 30rpx;
  701. }
  702. }
  703. }
  704. .rulesBox {
  705. height: 86rpx;
  706. background: #F9F6F1;
  707. padding: 0 20rpx;
  708. image {
  709. width: 126rpx;
  710. height: 46rpx;
  711. }
  712. }
  713. .product-list-box {
  714. margin: 8rpx 0;
  715. .pro-item {
  716. display: flex;
  717. flex-direction: row;
  718. align-items: center;
  719. .cart-select-img {
  720. width: 40rpx;
  721. height: 40rpx;
  722. margin: 30rpx;
  723. box-sizing: border-box;
  724. }
  725. .pro-r {
  726. flex: 1;
  727. border-bottom: 1px solid #eee;
  728. display: flex;
  729. flex-direction: row;
  730. padding: 30rpx 30rpx 30rpx 0;
  731. box-sizing: border-box;
  732. overflow: hidden;
  733. .pro-img {
  734. width: 180rpx;
  735. height: 180rpx;
  736. border-radius: 10rpx;
  737. margin-right: 30rpx;
  738. }
  739. .pro-r-r {
  740. flex: 1;
  741. font-size: 26rpx;
  742. color: #333;
  743. overflow: hidden;
  744. display: flex;
  745. flex-direction: column;
  746. justify-content: space-between;
  747. .pro-name {
  748. height: 66rpx;
  749. line-height: 33rpx;
  750. display: -webkit-box;
  751. overflow: hidden;
  752. text-overflow: ellipsis;
  753. word-break: break-all;
  754. -webkit-box-orient: vertical;
  755. -webkit-line-clamp: 2;
  756. }
  757. .sku-box {
  758. width: auto;
  759. display: inline;
  760. height: 40rpx;
  761. border-radius: 4rpx;
  762. padding: 0 0 0 10rpx;
  763. box-sizing: border-box;
  764. font-size: 24rpx;
  765. color: #999;
  766. text {
  767. border: 2rpx solid #E4E5E6;
  768. padding: 2rpx 10rpx;
  769. }
  770. }
  771. .pro-price-num-box {
  772. display: flex;
  773. flex-direction: row;
  774. align-items: center;
  775. justify-content: space-between;
  776. .pro-price-box {
  777. font-size: 36rpx;
  778. color: #333333;
  779. font-weight: 400;
  780. .fuhao {
  781. font-size: 24rpx;
  782. }
  783. }
  784. .pro-num-box {
  785. width: 140rpx;
  786. height: 40rpx;
  787. border: 1px solid #ddd;
  788. border-radius: 4rpx;
  789. display: flex;
  790. flex-direction: row;
  791. justify-content: space-between;
  792. overflow: hidden;
  793. .num-btn {
  794. font-size: 34rpx;
  795. color: #999999;
  796. display: inline-block;
  797. width: 40rpx;
  798. text-align: center;
  799. line-height: 32rpx;
  800. height: 40rpx;
  801. }
  802. .num-btn.r {
  803. border-right: 1px solid #ddd;
  804. }
  805. .num-btn.l {
  806. border-left: 1px solid #ddd;
  807. }
  808. .num {
  809. font-size: 26rpx;
  810. color: #333;
  811. }
  812. }
  813. }
  814. }
  815. }
  816. }
  817. .pro-item:last-of-type .pro-r {
  818. border-bottom: none;
  819. }
  820. }
  821. }
  822. }
  823. .itemBox:first-child {
  824. .shop-box {
  825. border-top: 2rpx solid #eee;
  826. }
  827. }
  828. .itemBox:last-child {
  829. .item {
  830. border-bottom: none;
  831. }
  832. }
  833. }
  834. .emptyCart-box {
  835. margin: 100rpx 0;
  836. .emptyCart-img {
  837. width: 216rpx;
  838. height: 156rpx;
  839. }
  840. .goToShopping {
  841. width: 282rpx;
  842. height: 84rpx;
  843. line-height: 84rpx;
  844. text-align: center;
  845. background: #333333;
  846. margin-top: 40rpx;
  847. color: #FFEBC4;
  848. font-size: 28rpx;
  849. }
  850. }
  851. .cart-bottom-box-h5 {
  852. position: fixed;
  853. bottom: 80rpx;
  854. width: 100%;
  855. z-index: 99;
  856. }
  857. .cart-bottom-box-app {
  858. position: fixed;
  859. bottom: 0rpx;
  860. width: 100%;
  861. z-index: 99;
  862. }
  863. .cart-bottom {
  864. height: 120rpx;
  865. background: #fff;
  866. display: flex;
  867. flex-direction: row;
  868. align-items: center;
  869. justify-content: space-between;
  870. border-top: 1rpx solid #eee;
  871. }
  872. .left {
  873. display: flex;
  874. flex-direction: row;
  875. align-items: center;
  876. font-size: 28rpx;
  877. color: #666;
  878. .cart-select-img {
  879. width: 40rpx;
  880. height: 40rpx;
  881. margin: 30rpx;
  882. box-sizing: border-box;
  883. }
  884. }
  885. .right {
  886. display: flex;
  887. flex-direction: row;
  888. align-items: center;
  889. box-sizing: border-box;
  890. .price-box {
  891. font-size: 30rpx;
  892. color: #333;
  893. .price {
  894. font-size: 40rpx;
  895. color: #C83732;
  896. font-weight: bold;
  897. }
  898. }
  899. .btn-confirm {
  900. width: 232rpx;
  901. height: 120rpx;
  902. background: #333333;
  903. margin-left: 18rpx;
  904. text-align: center;
  905. line-height: 120rpx;
  906. font-size: 28rpx;
  907. color: #FFEBC4;
  908. }
  909. .btn-delete {
  910. width: 232rpx;
  911. height: 120rpx;
  912. line-height: 120rpx;
  913. text-align: center;
  914. font-size: 28rpx;
  915. color: #FFFFFF;
  916. background: #C83732;
  917. }
  918. }
  919. }
  920. </style>