index.vue 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302
  1. <template>
  2. <div>
  3. <aflogo/>
  4. <div class="goods">
  5. <div class="menu-wrapper" ref="menuWrapper">
  6. <ul>
  7. <li v-for="(item,index) in goods" class="menu-item" :class="{'current':currentIndex===index}"
  8. @click="selectMenu(index,$event)" ref="menuList">
  9. <span class="text border-1px">
  10. <span v-show="false" class="icon"></span>{{item.name}}
  11. </span>
  12. </li>
  13. </ul>
  14. </div>
  15. <div class="foods-wrapper" ref="foodsWrapper">
  16. <ul>
  17. <li v-for="item in goods" class="food-list" ref="foodList">
  18. <h1 class="title">{{item.name}}</h1>
  19. <ul>
  20. <li @click="selectFood(food,$event)" v-for="food in item.foods" class="food-item border-1px">
  21. <div class="icon">
  22. <img width="88" height="88" :src="food.icon" alt="">
  23. </div>
  24. <div class="content">
  25. <div>
  26. <h2 class="name">{{food.name}}</h2>
  27. <p class="desc">{{food.description}}</p>
  28. </div>
  29. <div class="price">
  30. <span class="now">¥{{food.price}}</span>
  31. </div>
  32. <div class="cartcontrol-wrapper">
  33. <cartcontrol @add="addFood" :food="food"></cartcontrol>
  34. </div>
  35. </div>
  36. </li>
  37. </ul>
  38. </li>
  39. </ul>
  40. </div>
  41. <shopcart ref="shopcart" :selectFoods="selectFoods" :deliveryPrice="seller.deliveryPrice"
  42. :minPrice="seller.minPrice"></shopcart>
  43. </div>
  44. <food @add="addFood" :food="selectedFood" ref="food"></food>
  45. </div>
  46. </template>
  47. <script type="text/ecmascript-6">
  48. import BScroll from 'better-scroll'
  49. import aflogo from '../aflogo/index'
  50. import shopcart from '../shopcart/shopcart'
  51. import cartcontrol from '../cartcontrol/cartcontrol'
  52. import food from '../food/food'
  53. const mockDataJson = require('../../../mock/data.json')
  54. export default {
  55. data () {
  56. return {
  57. goods: [],
  58. listHeight: [],
  59. scrollY: 0,
  60. selectedFood: {},
  61. seller: {}
  62. }
  63. },
  64. computed: {
  65. currentIndex () {
  66. for (let i = 0; i < this.listHeight.length; i++) {
  67. let height1 = this.listHeight[i]
  68. let height2 = this.listHeight[i + 1]
  69. if (!height2 || (this.scrollY >= height1 && this.scrollY < height2)) {
  70. this._followScroll(i)
  71. return i
  72. }
  73. }
  74. return 0
  75. },
  76. selectFoods () {
  77. let foods = []
  78. this.goods.forEach((good) => {
  79. good.foods.forEach((food) => {
  80. if (food.count) {
  81. foods.push(food)
  82. }
  83. })
  84. })
  85. return foods
  86. }
  87. },
  88. async activated () {
  89. this.classMap = ['decrease', 'discount', 'special', 'invoice', 'guarantee']
  90. this.seller = mockDataJson.seller
  91. this.goods = mockDataJson.goods
  92. setTimeout(() => {
  93. this._initScroll()
  94. this._calculateHeight()
  95. }, 1000)
  96. },
  97. methods: {
  98. selectMenu (index, event) {
  99. if (!event._constructed) {
  100. return
  101. }
  102. let foodList = this.$refs.foodList
  103. let el = foodList[index]
  104. this.foodsScroll.scrollToElement(el, 300)
  105. },
  106. selectFood (food, event) {
  107. if (!event._constructed) {
  108. return
  109. }
  110. this.selectedFood = food
  111. this.$refs.food.show()
  112. },
  113. addFood (target) {
  114. this._drop(target)
  115. },
  116. _drop (target) {
  117. // 体验优化,异步执行下落动画
  118. this.$nextTick(() => {
  119. this.$refs.shopcart.drop(target)
  120. })
  121. },
  122. _initScroll () {
  123. this.meunScroll = new BScroll(this.$refs.menuWrapper, {
  124. click: true
  125. })
  126. this.foodsScroll = new BScroll(this.$refs.foodsWrapper, {
  127. click: true,
  128. probeType: 3
  129. })
  130. this.foodsScroll.on('scroll', (pos) => {
  131. // 判断滑动方向,避免下拉时分类高亮错误(如第一分类商品数量为1时,下拉使得第二分类高亮)
  132. if (pos.y <= 0) {
  133. this.scrollY = Math.abs(Math.round(pos.y))
  134. }
  135. })
  136. },
  137. _calculateHeight () {
  138. let foodList = this.$refs.foodList
  139. let height = 0
  140. this.listHeight.push(height)
  141. for (let i = 0; i < foodList.length; i++) {
  142. let item = foodList[i]
  143. height += item.clientHeight
  144. this.listHeight.push(height)
  145. }
  146. },
  147. _followScroll (index) {
  148. let menuList = this.$refs.menuList
  149. let el = menuList[index]
  150. this.meunScroll.scrollToElement(el, 300, 0, -100)
  151. }
  152. },
  153. components: {
  154. aflogo,
  155. shopcart,
  156. cartcontrol,
  157. food
  158. }
  159. }
  160. </script>
  161. <style lang="scss" scoped>
  162. .goods {
  163. display: flex;
  164. position: absolute;
  165. top: 107px;
  166. bottom: 68px;
  167. width: 100%;
  168. overflow: hidden;
  169. background: #fff;
  170. .menu-wrapper {
  171. flex: 0 0 100px;
  172. width: 100px;
  173. background: #FAF4F4;
  174. .menu-item {
  175. display: table;
  176. height: 60px;
  177. width: 100%;
  178. padding: 0 12px;
  179. &.current {
  180. position: relative;
  181. z-index: 10;
  182. margin-top: -1px;
  183. background: #fff;
  184. font-weight: 700;
  185. &:before {
  186. content: '';
  187. position: absolute;
  188. left: 0;
  189. top: 0;
  190. display: block;
  191. height: 100%;
  192. border-left: 3px solid #D32323;
  193. }
  194. }
  195. .text {
  196. display: table-cell;
  197. width: 100%;
  198. vertical-align: middle;
  199. font-size: 14px;
  200. color: #736F6F;
  201. line-height: 20px;
  202. }
  203. }
  204. }
  205. }
  206. .foods-wrapper {
  207. flex: 1;
  208. .title {
  209. padding: 10px 0 20px 19px;
  210. height: 42px;
  211. font-size: 14px;
  212. font-family: PingFangSC-Medium, PingFang SC;
  213. font-weight: 500;
  214. color: #736F6F;
  215. line-height: 20px;
  216. background: #F2F2F2;
  217. }
  218. .food-item {
  219. display: flex;
  220. margin: 8px 18px 0 16px;
  221. padding-bottom: 8px;
  222. @include border-1px(#F2F2F2);
  223. }
  224. &:last-child {
  225. @include border-none();
  226. margin-bottom: 0;
  227. }
  228. .icon {
  229. flex: 0 0 88px;
  230. margin-right: 8px;
  231. border-radius: 8px;
  232. overflow: hidden;
  233. }
  234. .content {
  235. flex: 1;
  236. display: flex;
  237. flex-direction: column;
  238. justify-content: space-between;
  239. .name {
  240. display: -webkit-box;
  241. -webkit-box-orient: vertical;
  242. -webkit-line-clamp: 1;
  243. overflow: hidden;
  244. margin: 4px 0;
  245. font-size: 16px;
  246. font-family: PingFangSC-Medium, PingFang SC;
  247. font-weight: 500;
  248. color: #1F1E1E;
  249. line-height: 22px;
  250. word-break: break-all;
  251. }
  252. .desc {
  253. display: -webkit-box;
  254. -webkit-box-orient: vertical;
  255. -webkit-line-clamp: 2;
  256. overflow: hidden;
  257. font-size: 12px;
  258. color: #736F6F;
  259. line-height: 17px;
  260. word-break: break-all;
  261. }
  262. .price {
  263. display: flex;
  264. flex-direction: column;
  265. }
  266. .now {
  267. font-size: 16px;
  268. font-family: PingFangSC-Medium, PingFang SC;
  269. font-weight: 600;
  270. color: #D32323;
  271. line-height: 22px;
  272. }
  273. .cartcontrol-wrapper {
  274. position: absolute;
  275. right: 0;
  276. bottom: 1px;
  277. }
  278. }
  279. }
  280. </style>