index.vue 9.9 KB

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