index.vue 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  1. <template>
  2. <div>
  3. <!--todo 临时隐藏-->
  4. <aflogo
  5. :objCurrentBarInfo="objCurrentBarInfo" style="visibility: hidden;"/>
  6. <div class="goods">
  7. <div
  8. ref="menuWrapper"
  9. class="menu-wrapper">
  10. <ul>
  11. <li
  12. v-for="(item, index) in goods"
  13. :key="index"
  14. ref="menuList"
  15. :class="{'current':currentIndex === index}"
  16. class="menu-item"
  17. @click="selectMenu(index, $event)">
  18. <span class="text border-1px">
  19. <span
  20. v-show="false"
  21. class="icon"></span>
  22. {{ item.category_name }}
  23. </span>
  24. </li>
  25. </ul>
  26. </div>
  27. <div
  28. ref="foodsWrapper"
  29. class="foods-wrapper">
  30. <ul>
  31. <li
  32. v-for="(item, index) in goods"
  33. :key="index"
  34. ref="foodList"
  35. class="food-list">
  36. <h1 class="title">{{ item.category_name }}</h1>
  37. <ul>
  38. <li
  39. v-for="(food, idx) in item.products"
  40. :key="idx"
  41. class="food-item border-1px"
  42. @click="selectFood(food, $event)">
  43. <div class="icon">
  44. <img
  45. :src="food.product_img_url"
  46. alt=""
  47. height="88"
  48. width="88">
  49. </div>
  50. <div class="content">
  51. <div>
  52. <h2 class="name">{{ food.product_name }}</h2>
  53. <p class="desc">{{ food.product_sale_num }}人推荐</p>
  54. </div>
  55. <div class="price">
  56. <span class="now">¥{{ food.product_price | fen2Yuan }}</span>
  57. </div>
  58. <!--第一个版本暂时不用-->
  59. <div
  60. v-if="false"
  61. class="cartcontrol-wrapper">
  62. <cartcontrol
  63. :food="food"
  64. @add="addFood"/>
  65. </div>
  66. <van-button
  67. v-if="food.is_sell_out === 0"
  68. class="btn-show-sku"
  69. type="danger">选规格
  70. </van-button>
  71. <!--售罄图标-->
  72. <img
  73. v-if="food.is_sell_out === 1"
  74. alt=""
  75. class="sell-out"
  76. src="./image/ic_sold_out@2x.png">
  77. </div>
  78. </li>
  79. </ul>
  80. </li>
  81. </ul>
  82. </div>
  83. <shopcart
  84. ref="shopcart"
  85. :deliveryPrice="seller.deliveryPrice"
  86. :minPrice="seller.minPrice"
  87. :selectFoods="selectFoods"
  88. @handleJumpPay="handleJumpPay"/>
  89. </div>
  90. <food
  91. ref="food"
  92. :food="selectedFood"
  93. @add="addFood"></food>
  94. <editTableNum ref="editTableNum"/>
  95. </div>
  96. </template>
  97. <script type="text/ecmascript-6">
  98. import BScroll from 'better-scroll'
  99. import aflogo from '../aflogo/index'
  100. import shopcart from '../shopcart/shopcart'
  101. import cartcontrol from '../cartcontrol/cartcontrol'
  102. import food from '../food/food'
  103. import editTableNum from '../editTableNum/index'
  104. import { apiProductList, apiProductDetail } from './api'
  105. import { mapGetters } from 'vuex'
  106. import { Toast, Button } from 'vant'
  107. const mockDataJson = require('../../../mock/data.json')
  108. // todo 登录成功不会掉接口
  109. export default {
  110. data () {
  111. return {
  112. goods: [],
  113. listHeight: [],
  114. scrollY: 0,
  115. selectedFood: {},
  116. seller: {}
  117. }
  118. },
  119. computed: {
  120. ...mapGetters({
  121. objCurrentBarInfo: 'common/objCurrentBarInfo'
  122. }),
  123. currentIndex () {
  124. for (let i = 0; i < this.listHeight.length; i++) {
  125. const height1 = this.listHeight[i]
  126. const height2 = this.listHeight[i + 1]
  127. if (!height2 || (this.scrollY >= height1 && this.scrollY < height2)) {
  128. this._followScroll(i)
  129. return i
  130. }
  131. }
  132. return 0
  133. },
  134. selectFoods () {
  135. const foods = []
  136. this.goods.forEach((good) => {
  137. good.products.forEach((food) => {
  138. if (food.count) {
  139. foods.push(food)
  140. }
  141. })
  142. })
  143. return foods
  144. }
  145. },
  146. watch: {
  147. 'objCurrentBarInfo.id': {
  148. immediate: true,
  149. handler: function () {
  150. this.fetchProductList()
  151. }
  152. }
  153. },
  154. async activated () {
  155. this.seller = mockDataJson.seller
  156. },
  157. methods: {
  158. selectMenu (index, event) {
  159. if (!event._constructed) {
  160. return
  161. }
  162. const foodList = this.$refs.foodList
  163. const el = foodList[index]
  164. // offsetY的值改为3,是因为px转rem有误差,iOS12.4.1点击左侧导航时,当前点击的菜单无法被选中,类似一个修正值
  165. this.foodsScroll.scrollToElement(el, 300, true, 3)
  166. },
  167. async selectFood (food, event) {
  168. // better-scroll 默认会阻止浏览器的原生 click 事件。当设置为 true,better-scroll 会派发一个 click 事件,我们会给派发的 event 参数加一个私有属性 _constructed,值为 true
  169. // todo 这里暂时用不到
  170. // if (!event._constructed) {
  171. // return
  172. // }
  173. try {
  174. const { data, status, msg } = await apiProductDetail(food.id)
  175. if (status) {
  176. this.selectedFood = data
  177. } else {
  178. Toast(msg)
  179. }
  180. this.$refs.food.show()
  181. } catch (err) {}
  182. },
  183. addFood (target) {
  184. this._drop(target)
  185. },
  186. _drop (target) {
  187. // 体验优化,异步执行下落动画
  188. this.$nextTick(() => {
  189. this.$refs.shopcart.drop(target)
  190. })
  191. },
  192. _initScroll () {
  193. this.meunScroll = new BScroll(this.$refs.menuWrapper, {
  194. click: true
  195. })
  196. this.foodsScroll = new BScroll(this.$refs.foodsWrapper, {
  197. click: true,
  198. probeType: 3
  199. })
  200. this.foodsScroll.on('scroll', (pos) => {
  201. // 判断滑动方向,避免下拉时分类高亮错误(如第一分类商品数量为1时,下拉使得第二分类高亮)
  202. if (pos.y <= 0) {
  203. this.scrollY = Math.abs(Math.round(pos.y))
  204. }
  205. })
  206. },
  207. _calculateHeight () {
  208. const foodList = this.$refs.foodList
  209. let height = 0
  210. this.listHeight.push(height)
  211. for (let i = 0; i < foodList.length; i++) {
  212. const item = foodList[i]
  213. height += item.clientHeight
  214. this.listHeight.push(height)
  215. }
  216. },
  217. _followScroll (index) {
  218. const menuList = this.$refs.menuList
  219. const el = menuList[index]
  220. this.meunScroll.scrollToElement(el, 300, 0, -100)
  221. },
  222. handleJumpPay () {
  223. this.$refs.editTableNum.init()
  224. },
  225. // 获取商品列表
  226. async fetchProductList () {
  227. const { id } = this.objCurrentBarInfo
  228. if (!id) {
  229. return
  230. }
  231. try {
  232. const { data, status, msg } = await apiProductList(id)
  233. if (status) {
  234. if (Array.isArray(data) && data.length) {
  235. this.goods = data
  236. setTimeout(() => {
  237. this._initScroll()
  238. this._calculateHeight()
  239. }, 1000)
  240. }
  241. } else {
  242. Toast(msg)
  243. }
  244. } catch (err) {}
  245. }
  246. },
  247. components: {
  248. aflogo,
  249. shopcart,
  250. cartcontrol,
  251. food,
  252. editTableNum,
  253. 'van-button': Button
  254. }
  255. }
  256. </script>
  257. <style lang="scss" scoped>
  258. .goods {
  259. display: flex;
  260. position: absolute;
  261. top: 107px;
  262. bottom: 68px;
  263. width: 100%;
  264. overflow: hidden;
  265. background: #fff;
  266. .menu-wrapper {
  267. flex: 0 0 100px;
  268. width: 100px;
  269. background: #FAF4F4;
  270. .menu-item {
  271. display: table;
  272. height: 60px;
  273. width: 100%;
  274. padding: 0 12px;
  275. &.current {
  276. position: relative;
  277. z-index: 10;
  278. margin-top: -1px;
  279. background: #fff;
  280. font-weight: 700;
  281. &:before {
  282. content: '';
  283. position: absolute;
  284. left: 0;
  285. top: 0;
  286. display: block;
  287. height: 100%;
  288. border-left: 3px solid #D32323;
  289. }
  290. }
  291. .text {
  292. display: table-cell;
  293. width: 100%;
  294. vertical-align: middle;
  295. font-size: 14px;
  296. color: #736F6F;
  297. line-height: 20px;
  298. }
  299. }
  300. }
  301. }
  302. .foods-wrapper {
  303. flex: 1;
  304. .title {
  305. padding: 10px 0 20px 19px;
  306. height: 42px;
  307. font-size: 14px;
  308. font-family: PingFangSC-Medium, PingFang SC;
  309. font-weight: 500;
  310. color: #736F6F;
  311. line-height: 20px;
  312. background: #F2F2F2;
  313. }
  314. .food-item {
  315. display: flex;
  316. margin: 8px 18px 0 16px;
  317. padding-bottom: 8px;
  318. @include border-1px(#F2F2F2);
  319. }
  320. &:last-child {
  321. @include border-none();
  322. margin-bottom: 0;
  323. }
  324. .icon {
  325. flex: 0 0 88px;
  326. margin-right: 8px;
  327. border-radius: 8px;
  328. overflow: hidden;
  329. }
  330. .content {
  331. flex: 1;
  332. display: flex;
  333. flex-direction: column;
  334. justify-content: space-between;
  335. .name {
  336. display: -webkit-box;
  337. -webkit-box-orient: vertical;
  338. -webkit-line-clamp: 1;
  339. overflow: hidden;
  340. margin: 4px 0;
  341. font-size: 16px;
  342. font-family: PingFangSC-Medium, PingFang SC;
  343. font-weight: 500;
  344. color: #1F1E1E;
  345. line-height: 22px;
  346. word-break: break-all;
  347. }
  348. .desc {
  349. display: -webkit-box;
  350. -webkit-box-orient: vertical;
  351. -webkit-line-clamp: 2;
  352. overflow: hidden;
  353. font-size: 12px;
  354. color: #736F6F;
  355. line-height: 17px;
  356. word-break: break-all;
  357. }
  358. .price {
  359. display: flex;
  360. flex-direction: column;
  361. }
  362. .now {
  363. font-size: 16px;
  364. font-family: PingFangSC-Medium, PingFang SC;
  365. font-weight: 600;
  366. color: #D32323;
  367. line-height: 22px;
  368. }
  369. .cartcontrol-wrapper {
  370. position: absolute;
  371. right: 0;
  372. bottom: 1px;
  373. }
  374. .btn-show-sku {
  375. position: absolute;
  376. right: 0;
  377. bottom: 9px;
  378. width: 70px;
  379. height: 24px;
  380. border: none;
  381. background: #D32323;
  382. border-radius: 12px;
  383. ::v-deep .van-button__text {
  384. font-size: 12px;
  385. color: #FFFFFF;
  386. line-height: 24px;
  387. }
  388. }
  389. .sell-out {
  390. position: absolute;
  391. right: 0;
  392. bottom: 9px;
  393. display: block;
  394. width: 46px;
  395. height: 32px;
  396. }
  397. }
  398. }
  399. </style>