123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423 |
- <template>
- <div>
- <!--todo 临时隐藏-->
- <aflogo
- :objCurrentBarInfo="objCurrentBarInfo" style="visibility: hidden;"/>
- <div class="goods">
- <div
- ref="menuWrapper"
- class="menu-wrapper">
- <ul>
- <li
- v-for="(item, index) in goods"
- :key="index"
- ref="menuList"
- :class="{'current':currentIndex === index}"
- class="menu-item"
- @click="selectMenu(index, $event)">
- <span class="text border-1px">
- <span
- v-show="false"
- class="icon"></span>
- {{ item.category_name }}
- </span>
- </li>
- </ul>
- </div>
- <div
- ref="foodsWrapper"
- class="foods-wrapper">
- <ul>
- <li
- v-for="(item, index) in goods"
- :key="index"
- ref="foodList"
- class="food-list">
- <h1 class="title">{{ item.category_name }}</h1>
- <ul>
- <li
- v-for="(food, idx) in item.products"
- :key="idx"
- class="food-item border-1px"
- @click="selectFood(food, $event)">
- <div class="icon">
- <img
- :src="food.product_img_url"
- alt=""
- height="88"
- width="88">
- </div>
- <div class="content">
- <div>
- <h2 class="name">{{ food.product_name }}</h2>
- <p class="desc">{{ food.product_sale_num }}人推荐</p>
- </div>
- <div class="price">
- <span class="now">¥{{ food.product_price | fen2Yuan }}</span>
- </div>
- <!--第一个版本暂时不用-->
- <div
- v-if="false"
- class="cartcontrol-wrapper">
- <cartcontrol
- :food="food"
- @add="addFood"/>
- </div>
- <van-button
- v-if="food.is_sell_out === 0"
- class="btn-show-sku"
- type="danger">选规格
- </van-button>
- <!--售罄图标-->
- <img
- v-if="food.is_sell_out === 1"
- alt=""
- class="sell-out"
- src="./image/ic_sold_out@2x.png">
- </div>
- </li>
- </ul>
- </li>
- </ul>
- </div>
- <shopcart
- ref="shopcart"
- :deliveryPrice="seller.deliveryPrice"
- :minPrice="seller.minPrice"
- :selectFoods="selectFoods"
- @handleJumpPay="handleJumpPay"/>
- </div>
- <food
- ref="food"
- :food="selectedFood"
- @add="addFood"></food>
- <editTableNum ref="editTableNum"/>
- </div>
- </template>
- <script type="text/ecmascript-6">
- import BScroll from 'better-scroll'
- import aflogo from '../aflogo/index'
- import shopcart from '../shopcart/shopcart'
- import cartcontrol from '../cartcontrol/cartcontrol'
- import food from '../food/food'
- import editTableNum from '../editTableNum/index'
- import { apiProductList, apiProductDetail } from './api'
- import { mapGetters } from 'vuex'
- import { Toast, Button } from 'vant'
- const mockDataJson = require('../../../mock/data.json')
- // todo 登录成功不会掉接口
- export default {
- data () {
- return {
- goods: [],
- listHeight: [],
- scrollY: 0,
- selectedFood: {},
- seller: {}
- }
- },
- computed: {
- ...mapGetters({
- objCurrentBarInfo: 'common/objCurrentBarInfo'
- }),
- currentIndex () {
- for (let i = 0; i < this.listHeight.length; i++) {
- const height1 = this.listHeight[i]
- const height2 = this.listHeight[i + 1]
- if (!height2 || (this.scrollY >= height1 && this.scrollY < height2)) {
- this._followScroll(i)
- return i
- }
- }
- return 0
- },
- selectFoods () {
- const foods = []
- this.goods.forEach((good) => {
- good.products.forEach((food) => {
- if (food.count) {
- foods.push(food)
- }
- })
- })
- return foods
- }
- },
- watch: {
- 'objCurrentBarInfo.id': {
- immediate: true,
- handler: function () {
- this.fetchProductList()
- }
- }
- },
- async activated () {
- this.seller = mockDataJson.seller
- },
- methods: {
- selectMenu (index, event) {
- if (!event._constructed) {
- return
- }
- const foodList = this.$refs.foodList
- const el = foodList[index]
- // offsetY的值改为3,是因为px转rem有误差,iOS12.4.1点击左侧导航时,当前点击的菜单无法被选中,类似一个修正值
- this.foodsScroll.scrollToElement(el, 300, true, 3)
- },
- async selectFood (food, event) {
- // better-scroll 默认会阻止浏览器的原生 click 事件。当设置为 true,better-scroll 会派发一个 click 事件,我们会给派发的 event 参数加一个私有属性 _constructed,值为 true
- // todo 这里暂时用不到
- // if (!event._constructed) {
- // return
- // }
- try {
- const { data, status, msg } = await apiProductDetail(food.id)
- if (status) {
- this.selectedFood = data
- } else {
- Toast(msg)
- }
- this.$refs.food.show()
- } catch (err) {}
- },
- addFood (target) {
- this._drop(target)
- },
- _drop (target) {
- // 体验优化,异步执行下落动画
- this.$nextTick(() => {
- this.$refs.shopcart.drop(target)
- })
- },
- _initScroll () {
- this.meunScroll = new BScroll(this.$refs.menuWrapper, {
- click: true
- })
- this.foodsScroll = new BScroll(this.$refs.foodsWrapper, {
- click: true,
- probeType: 3
- })
- this.foodsScroll.on('scroll', (pos) => {
- // 判断滑动方向,避免下拉时分类高亮错误(如第一分类商品数量为1时,下拉使得第二分类高亮)
- if (pos.y <= 0) {
- this.scrollY = Math.abs(Math.round(pos.y))
- }
- })
- },
- _calculateHeight () {
- const foodList = this.$refs.foodList
- let height = 0
- this.listHeight.push(height)
- for (let i = 0; i < foodList.length; i++) {
- const item = foodList[i]
- height += item.clientHeight
- this.listHeight.push(height)
- }
- },
- _followScroll (index) {
- const menuList = this.$refs.menuList
- const el = menuList[index]
- this.meunScroll.scrollToElement(el, 300, 0, -100)
- },
- handleJumpPay () {
- this.$refs.editTableNum.init()
- },
- // 获取商品列表
- async fetchProductList () {
- const { id } = this.objCurrentBarInfo
- if (!id) {
- return
- }
- try {
- const { data, status, msg } = await apiProductList(id)
- if (status) {
- if (Array.isArray(data) && data.length) {
- this.goods = data
- setTimeout(() => {
- this._initScroll()
- this._calculateHeight()
- }, 1000)
- }
- } else {
- Toast(msg)
- }
- } catch (err) {}
- }
- },
- components: {
- aflogo,
- shopcart,
- cartcontrol,
- food,
- editTableNum,
- 'van-button': Button
- }
- }
- </script>
- <style lang="scss" scoped>
- .goods {
- display: flex;
- position: absolute;
- top: 107px;
- bottom: 68px;
- width: 100%;
- overflow: hidden;
- background: #fff;
- .menu-wrapper {
- flex: 0 0 100px;
- width: 100px;
- background: #FAF4F4;
- .menu-item {
- display: table;
- height: 60px;
- width: 100%;
- padding: 0 12px;
- &.current {
- position: relative;
- z-index: 10;
- margin-top: -1px;
- background: #fff;
- font-weight: 700;
- &:before {
- content: '';
- position: absolute;
- left: 0;
- top: 0;
- display: block;
- height: 100%;
- border-left: 3px solid #D32323;
- }
- }
- .text {
- display: table-cell;
- width: 100%;
- vertical-align: middle;
- font-size: 14px;
- color: #736F6F;
- line-height: 20px;
- }
- }
- }
- }
- .foods-wrapper {
- flex: 1;
- .title {
- padding: 10px 0 20px 19px;
- height: 42px;
- font-size: 14px;
- font-family: PingFangSC-Medium, PingFang SC;
- font-weight: 500;
- color: #736F6F;
- line-height: 20px;
- background: #F2F2F2;
- }
- .food-item {
- display: flex;
- margin: 8px 18px 0 16px;
- padding-bottom: 8px;
- @include border-1px(#F2F2F2);
- }
- &:last-child {
- @include border-none();
- margin-bottom: 0;
- }
- .icon {
- flex: 0 0 88px;
- margin-right: 8px;
- border-radius: 8px;
- overflow: hidden;
- }
- .content {
- flex: 1;
- display: flex;
- flex-direction: column;
- justify-content: space-between;
- .name {
- display: -webkit-box;
- -webkit-box-orient: vertical;
- -webkit-line-clamp: 1;
- overflow: hidden;
- margin: 4px 0;
- font-size: 16px;
- font-family: PingFangSC-Medium, PingFang SC;
- font-weight: 500;
- color: #1F1E1E;
- line-height: 22px;
- word-break: break-all;
- }
- .desc {
- display: -webkit-box;
- -webkit-box-orient: vertical;
- -webkit-line-clamp: 2;
- overflow: hidden;
- font-size: 12px;
- color: #736F6F;
- line-height: 17px;
- word-break: break-all;
- }
- .price {
- display: flex;
- flex-direction: column;
- }
- .now {
- font-size: 16px;
- font-family: PingFangSC-Medium, PingFang SC;
- font-weight: 600;
- color: #D32323;
- line-height: 22px;
- }
- .cartcontrol-wrapper {
- position: absolute;
- right: 0;
- bottom: 1px;
- }
- .btn-show-sku {
- position: absolute;
- right: 0;
- bottom: 9px;
- width: 70px;
- height: 24px;
- border: none;
- background: #D32323;
- border-radius: 12px;
- ::v-deep .van-button__text {
- font-size: 12px;
- color: #FFFFFF;
- line-height: 24px;
- }
- }
- .sell-out {
- position: absolute;
- right: 0;
- bottom: 9px;
- display: block;
- width: 46px;
- height: 32px;
- }
- }
- }
- </style>
|