zhugaoliang 7b7e00bd20 Add menu page with product listing and shopping cart functionality
diff --git a/yshop-drink-uniapp-vue3/pages/menu/menu.vue b/yshop-drink-uniapp-vue3/pages/menu/menu.vue
new file mode 100644
index 0000
2025-04-10 19:30:08 +08:00

1491 lines
34 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<template>
<layout>
<uv-navbar :fixed="false" :title="title" left-arrow @leftClick="$onClickLeft" />
<view class="container" v-if="!loading">
<!-- <view class="banner-container">
<image :src="shopAd" mode="aspectFill" class="shop-banner"></image>
</view> -->
<!-- <view class="notice-bar" v-if="store.notice">
<uv-notice-bar :text="store.notice"></uv-notice-bar>
</view> -->
<view class="main">
<view class="nav">
<view class="header flex-row justify-between">
<view class="store-info flex-row">
<view class="mr-1">
<image :src="store.image" class="store-logo"></image>
</view>
<view class="left" v-if="orderType == 'takein'">
<view class="store-name flex-row align-center" @click="selectShop()">
<text style="font-weight: 800">{{ store.name }}</text>
<view class="iconfont iconarrow-right"></view>
</view>
<view class="store-location flex-row align-center">
<img src="../../static/images/menu/map.png"
style="width: 36rpx; height: 36rpx; margin-right: 22rpx" alt="" />
<text>距离您 {{ kmUnit(store.dis) }}</text>
</view>
</view>
<view class="left overflow-hidden" v-else>
<view class="store-name flex-row align-center" @click="selectShop()">
<view>{{ store.name }}
<text class="small" v-if="store.distance > 0 && orderType == 'takeout'">(配送距离:
{{ store.distance }}km)</text>
<text class="small" v-else-if="orderType == 'takeout'">(本店不支持外卖)</text>
</view>
<view class="iconfont iconarrow-right"></view>
</view>
</view>
</view>
<view class="order-type flex-row">
<view class="dinein" :class="{ active: orderType == 'takein' }" @tap="takein">
<text>自取</text>
</view>
<view class="takeout" :class="{ active: orderType == 'takeout' }" @tap="takout">
<text>外卖</text>
</view>
</view>
</view>
</view>
<!-- 广告部分 -->
<view class="banner-container flex justify-center align-center" style="height: 250rpx; margin: 1ch;">
<!-- <view class="" style=" width: 95%; height: 85%; border-radius: 26px; background-color: lightgray;">
</view> -->
<!-- {{ ads.list[0].image }} -->
<image :src="ads.list[0].image" mode="aspectFill" class="shop-banner" style="background-color:lightgray; border-radius: 2ch;"></image>
</view>
<!-- #ifdef H5 -->
<view class="content" :style="{
height: 'calc(100vh - ' + (headerHeight + (store.notice ? 0 : 60) + footerHeight) + 'rpx)',
paddingBottom: '160rpx'
}">
<!-- #endif -->
<!-- #ifndef H5 -->
<view class="content" :style="{
height: 'calc(100vh - ' + (headerHeight + (store.notice ? 0 : 60) + footerHeight) + 'rpx)',
paddingBottom: '160rpx'
}">
<!-- #endif -->
<scroll-view class="menu-sidebar" :scroll-into-view="menuScrollIntoView" scroll-with-animation
scroll-y>
<view class="sidebar-wrapper">
<view class="menu-category" :id="`menu-${item.id}`"
:class="{'current': item.id === currentCateId}" v-for="(item, index) in goods"
:key="index" @tap="handleMenuTap(item.id)">
<text>{{ item.name }}</text>
<view class="category-badge" v-show="menuCartNum(item.id)">
{{ menuCartNum(item.id) }}
</view>
</view>
</view>
</scroll-view>
<!-- goods list begin -->
<scroll-view class="goods-container" scroll-with-animation scroll-y :scroll-top="cateScrollTop"
@scroll="handleGoodsScroll" id="goodsContainer">
<view class="goods-wrapper">
<view class="goods-list">
<!-- category begin -->
<view class="category-section" v-for="(item, index) in goods" :key="index"
:id="`cate-${item.id}`">
<view class="category-title">
<text>{{ item.name }}</text>
<image mode="aspectFill" :src="item.icon" class="category-icon"></image>
</view>
<view class="category-items">
<!-- 商品 begin -->
<view class="good-card" v-for="(good, key) in item.goodsList" :key="key"
:class="{'sold-out': good.stock <= 0}">
<image mode="aspectFill" :src="good.image" class="good-image"
@tap="showGoodDetailModal(item, good)"></image>
<!-- <image mode="aspectFill"
src="https://lanhu-oss-2537-2.lanhuapp.com/FigmaDDSSlicePNG62414071ec6e0e4e00603407efd42e53.png"
class="good-image" @tap="showGoodDetailModal(item, good)"></image> -->
<!-- https://lanhu-oss-2537-2.lanhuapp.com/FigmaDDSSlicePNG62414071ec6e0e4e00603407efd42e53.png -->
<view class="good-details">
<text class="good-name">{{ good.storeName }}</text>
<text class="good-description">{{ good.storeInfo }}</text>
<view class="price-action flex-row justify-between align-center">
<text class="good-price">¥{{ good.price }}</text>
<view class="action-buttons" v-if="good.stock > 0">
<button type="primary" class="spec-button"
hover-class="none" size="mini"
@tap="showGoodDetailModal(item, good)">
<view class="iconfont iconadd-select"></view>
</button>
<view class="item-badge" v-show="goodCartNum(good.id)">
{{ goodCartNum(good.id) }}
</view>
</view>
<view class="sold-out-label" v-if="good.stock == 0">已售罄</view>
</view>
</view>
</view>
<!-- 商品 end -->
</view>
</view>
<!-- category end -->
</view>
</view>
</scroll-view>
<!-- goods list end -->
</view>
<!-- content end -->
<!-- 购物车栏 begin -->
<view class="cart-bar" v-if="cart.length > 0 && isCartShow">
<view class="cart-icon-container">
<image src="/static/images/menu/cart.png" class="cart-icon" @tap="openCartPopup"></image>
<view class="cart-badge">{{ getCartGoodsNumber }}</view>
</view>
<view class="cart-total" @tap="openCartShow">¥{{ getCartGoodsPrice }}</view>
<button type="primary" class="checkout-button" @tap="toPay" :disabled="disabledPay">
{{ disabledPay ? `差${spread}元起送` : '去结算' }}
</button>
</view>
<!-- 购物车栏 end -->
</view>
<!-- 商品详情模态框 begin -->
<modal :show="goodDetailModalVisible" class="good-detail-modal" color="#5A5B5C" width="90%" custom
padding="0rpx" radius="12rpx">
<template #default>
<view class="modal-header">
<view class="close-button">
<image src="/static/images/menu/close.png" @tap="closeGoodDetailModal"></image>
</view>
</view>
<scroll-view class="modal-body" scroll-y>
<view v-if="good.image" class="modal-image">
<image mode="aspectFill" style="width: 100%; height: 100%; border-radius: 22rpx;" :src="good.image"></image>
</view>
<view class="modal-content">
<view class="good-basic-info">
<view class="good-title">{{ good.storeName }}</view>
<view class="good-subtitle flex justify-between">{{ good.storeInfo }} <text
class="points-info">可获积分:10</text></view>
</view>
<view class="good-properties">
<view class="property-item" v-for="(item, index) in good.productAttr" :key="index">
<view class="property-name">
<text class="name">{{ item.attrName }}</text>
</view>
<view class="property-values">
<view class="property-value" v-for="(value, key) in item.attrValueArr"
:key="key" :class="{ selected: value == newValue[index] }"
@tap="changePropertyDefault(index, key, false)">
{{ value }}
</view>
</view>
</view>
</view>
</view>
</scroll-view>
<view class="modal-footer">
<view class="price-info">
<view class="final-price">¥{{ good.price }}</view>
<view class="selected-props">
{{ good.valueStr }}
</view>
</view>
<view class="quantity-control">
<!-- <text class="stock-info">库存:{{ good.stock }} </text> -->
<button type="default" plain class="quantity-button" size="mini" hover-class="none"
@tap="handlePropertyReduce">
<view class="iconfont iconsami-select"></view>
</button>
<view class="quantity-number">{{ good.number }}</view>
<button type="primary" class="quantity-button" size="min" hover-class="none"
@tap="handlePropertyAdd">
<view class="iconfont iconadd-select"></view>
</button>
</view>
</view>
<view class="add-cart-button" @tap="handleAddToCartInModal">
<view>加入购物车</view>
</view>
</template>
</modal>
<!-- 商品详情模态框 end -->
<!-- 购物车详情popup -->
<uv-popup ref="popup" mode="bottom" class="cart-popup">
<template #default>
<view class="cart-popup-container">
<view class="popup-header">
<text @tap="handleCartClear">清空</text>
</view>
<scroll-view class="cart-items" scroll-y>
<view class="items-wrapper">
<view class="cart-item" v-for="(item, index) in cart" :key="index">
<view class="item-info">
<view class="item-name">{{ item.name }}</view>
<view class="item-properties">{{ item.valueStr }}</view>
</view>
<view class="item-price">
<text>{{ item.price }}</text>
</view>
<view class="item-controls">
<button type="default" plain size="mini" hover-class="none"
@tap="handleCartItemReduce(index)">
<view class="iconfont iconsami-select"></view>
</button>
<view class="item-quantity">{{ item.number }}</view>
<button type="primary" size="mini" hover-class="none"
@tap="handleCartItemAdd(index)">
<view class="iconfont iconadd-select"></view>
</button>
</view>
</view>
</view>
</scroll-view>
</view>
</template>
</uv-popup>
</view>
<!--轻提示-->
<view class="loading" v-if="loading">
<uv-loading-icon color="#DA5650" size=40 mode="circle"></uv-loading-icon>
<button type="primary" style="z-index: 3001;position: absolute;top: 650rpx;" @click="init"
class="cu-btn bg-red">刷新</button>
</view>
<uv-toast ref="uToast"></uv-toast>
</layout>
</template>
<script setup>
import {
ref,
computed,
nextTick
} from "vue";
import {
useMainStore
} from "@/store/store";
import {
storeToRefs
} from "pinia";
import {
onLoad,
onShow,
onPullDownRefresh,
onHide
} from "@dcloudio/uni-app";
import {
formatDateTime,
kmUnit
} from "@/utils/util";
import {
shopNearby,
menuGoods
} from "@/api/goods";
import {
menuAds
} from "@/api/market";
const main = useMainStore();
const {
orderType,
address,
store,
location,
isLogin
} = storeToRefs(main);
const title = ref("点餐");
const text = ref("滚动通知");
const goods = ref([]);
const ads = ref([]);
const loading = ref(true);
const currentCateId = ref(0);
const cateScrollTop = ref(0);
const menuScrollIntoView = ref("");
const cart = ref([]);
const goodDetailModalVisible = ref(false);
const good = ref({});
const category = ref({});
const cartPopupVisible = ref(false);
const sizeCalcState = ref(false);
const newValue = ref([]);
const shopAd = ref(
"https://lanhu-oss-2537-2.lanhuapp.com/FigmaDDSSlicePNGca115cd446d280796935fea74b5cd20f.png"
);
const isCartShow = ref(true);
const popup = ref();
const headerHeight = ref(300); // 头部区域高度rpx
const footerHeight = ref(100); // 底部区域高度rpx
const newkmUnit = computed(() => (param) => {
console.log("param:", param);
return "10km";
});
const goodCartNum = computed(() => {
//计算单个饮品添加到购物车的数量
return (id) =>
cart.value.reduce((acc, cur) => {
if (cur.id === id) {
return (acc += cur.number);
}
return acc;
}, 0);
});
const menuCartNum = computed(() => {
return (id) =>
cart.value.reduce((acc, cur) => {
if (cur.cate_id === id) {
return (acc += cur.number);
}
return acc;
}, 0);
});
const getCartGoodsNumber = computed(() => {
//计算购物车总数
return cart.value.reduce((acc, cur) => acc + cur.number, 0);
});
const getCartGoodsPrice = computed(() => {
//计算购物车总价
let price = cart.value.reduce((acc, cur) => acc + cur.number * cur.price, 0);
return parseFloat(price).toFixed(2);
});
const disabledPay = computed(() => {
//是否达到起送价
return orderType.value == "takeout" &&
getCartGoodsPrice < parseFloat(store.value.min_price) ?
true :
false;
});
const spread = computed(() => {
//差多少元起送
if (orderType.value != "takeout") return;
return parseFloat((store.value.min_price - getCartGoodsPrice).toFixed(2));
});
// 监听自定义事件
uni.$on("refreshMenu", () => {
// 在这里执行onLoad逻辑
console.log("refreshMenu1:", store.value.id);
init();
});
onPullDownRefresh(() => {
init();
});
onLoad(() => {
init();
refreshCart();
nextTick(() => {
if (goods.value.length > 0) {
currentCateId.value = goods.value[0].id;
}
});
});
onHide(() => {
// 重新进入要重新计算页面高度,否则有问题
sizeCalcState.value = false;
});
onShow(() => {
//init();
refreshCart();
shopAd.value = uni.getStorageSync("shopAd");
// 初始化页面高度计算
nextTick(() => {
calcLayoutHeights();
});
});
const openCartShow = () => {
isCartShow.value = false;
};
const in_array = (search, array) => {
for (var i in array) {
if (array[i] == search) {
return true;
}
}
return false;
};
const selectShop = () => {
uni.navigateTo({
url: "/pages/components/pages/shop/shop",
});
};
const uToast = ref();
const init = async () => {
//页面初始化
loading.value = true;
//return
let error = {},
result = location.value;
console.log("result:", result);
if (!location.value.hasOwnProperty("latitude")) {
console.log("result1:", location.value);
uni.getLocation({
type: "wgs84",
success: function(res) {
console.log("location1:", res);
result = {
latitude: res.latitude,
longitude: res.longitude,
};
getShopList(result);
},
fail: function(res) {
uni.showToast({
title: "获取位置失败,请检查是否开启相关权限",
duration: 2000,
icon: "error",
});
// 默认地为你为北京地址
result = {
latitude: 39.91999,
longitude: 116.45627,
};
getShopList(result);
},
complete: function(res) {},
});
return;
}
getShopList(result);
};
const getShopList = async (res) => {
console.log("location9:", res);
if (res) {
main.SET_LOCATION(res);
let shop_id = 0;
if (store.value.id) {
shop_id = store.value.id;
}
let shop = await shopNearby({
lat: res.latitude,
lng: res.longitude,
shop_id: shop_id,
kw: "",
});
if (shop) {
//广告图
getAds(shop.id);
shop.notice =
shop.status == 1 ?
shop.notice :
"店铺营业时间为:" +
formatDateTime(shop.startTime, "hh:mm") +
" - " +
formatDateTime(shop.endTime, "hh:mm") +
",不在营业时间内无法下单";
// 设置店铺信息
main.SET_STORE(shop);
let mygoods = await menuGoods({
shopId: shop.id,
});
if (mygoods) {
goods.value = mygoods;
refreshCart();
// 设置初始分类
if (mygoods.length > 0) {
currentCateId.value = mygoods[0].id;
}
// 数据加载完成后,重新计算布局
nextTick(() => {
calcLayoutHeights();
});
loading.value = false;
}
console.log("goods:", mygoods);
console.log("goods:", goods.value);
// loading.value = false;
// uni.stopPullDownRefresh();
}
}
uni.stopPullDownRefresh();
};
const refreshCart = () => {
if (goods.value && goods.value.length > 0) {
let newGoods = goods.value;
cart.value = [];
let newCart = uni.getStorageSync("cart") || [];
let tmpCart = [];
if (newCart) {
for (let i in newCart) {
for (let ii in newGoods) {
for (let iii in newGoods[ii].goodsList) {
if (newCart[i].id == newGoods[ii].goodsList[iii].id) {
tmpCart.push(newCart[i]);
}
}
}
}
cart.value = tmpCart;
cartPopupVisible.value = false;
}
}
};
const getAds = async (shop_id) => {
let data = await menuAds({
shop_id: shop_id ? shop_id : 0
});
if (data) {
ads.value = data;
console.log("ads1111:", ads.value);
}
}
const takout = (force = false) => {
if (orderType.value == "takeout" && force == false) return;
main.SET_ORDER_TYPE("takeout");
if (!isLogin.value) {
uni.navigateTo({
url: "/pages/components/pages/login/login",
});
return;
}
};
const takein = (force = false) => {
if (orderType.value == "takein" && force == false) return;
main.SET_ORDER_TYPE("takein");
if (!isLogin.value) {
uni.navigateTo({
url: "/pages/components/pages/login/login",
});
return;
}
};
const handleMenuTap = (id) => {
//点击菜单项事件
if (!sizeCalcState.value) {
calcSize();
}
currentCateId.value = id;
// nextTick(() => cateScrollTop.value = goods.value.find(item => item.id == id).top)
menuScrollIntoView.value = `menu-${id}`;
// 查找对应的分类区域并滚动到该位置
const category = goods.value.find((item) => item.id === id);
if (category && category.top !== undefined) {
// 使用nextTick确保DOM更新后再滚动
nextTick(() => {
cateScrollTop.value = category.top;
// 强制触发滚动以确保位置正确
const goodsContainer = uni
.createSelectorQuery()
.select("#goodsContainer");
if (goodsContainer) {
goodsContainer
.boundingClientRect((data) => {
if (data) {
console.log(
"Scrolling to category:",
id,
"at position:",
category.top
);
}
})
.exec();
}
});
}
};
const handleGoodsScroll = ({
detail
}) => {
//商品列表滚动事件
if (!sizeCalcState.value) {
calcSize();
}
console.log("scrollTop:", detail);
const {
scrollTop
} = detail;
let tabs = goods.value.filter((item) => item.top <= scrollTop).reverse();
if (tabs.length > 0) {
currentCateId.value = tabs[0].id;
}
};
const calcSize = () => {
let h = 0;
// 更精确计算每个分类的位置
goods.value.forEach((item, index) => {
let view = uni.createSelectorQuery().select(`#cate-${item.id}`);
view
.fields({
size: true,
rect: true,
},
(data) => {
if (data) {
// 对于第一个分类顶部位置为0
if (index === 0) {
item.top = 0;
} else {
item.top = h;
}
h += data.height;
item.bottom = h;
console.log(
`Category ${item.id} calculated: top=${item.top}, height=${data.height}, bottom=${item.bottom}`
);
}
}
)
.exec();
});
sizeCalcState.value = true;
};
// 增强页面布局高度计算的准确性
const calcLayoutHeights = () => {
// 获取系统信息
const systemInfo = uni.getSystemInfoSync();
const rpxRatio = 750 / systemInfo.windowWidth;
// 获取头部区域高度
const headerQuery = uni.createSelectorQuery().select(".nav");
headerQuery
.boundingClientRect((data) => {
if (data) {
// 将px转换为rpx
headerHeight.value = data.height * rpxRatio;
console.log("Header height:", headerHeight.value);
}
})
.exec();
// 获取底部购物车栏高度
const footerQuery = uni.createSelectorQuery().select(".cart-bar");
footerQuery
.boundingClientRect((data) => {
if (data) {
// 将px转换为rpx
footerHeight.value = data ? data.height * rpxRatio : 100;
console.log("Footer height:", footerHeight.value);
} else {
// 如果没有购物车栏,设置一个默认值
footerHeight.value = 100;
}
// 重新计算各个分类的位置
nextTick(() => {
sizeCalcState.value = false;
calcSize();
});
})
.exec();
};
// const calcSize = () => {
// let h = 10
// let view = uni.createSelectorQuery().select('#ads')
// if (view) {
// view.fields({
// size: true
// }, data => {
// if (data) {
// h += Math.floor(data.height)
// }
// }).exec()
// }
// goods.value.forEach(item => {
// let view = uni.createSelectorQuery().select(`#cate-${item.id}`)
// view.fields({
// size: true
// }, data => {
// console.log('h3:',h)
// item.top = h
// h += data.height
// item.bottom = h
// }).exec()
// })
// sizeCalcState.value = true
// }
const handleAddToCart = (cate, newGood, num) => {
//添加到购物车
const index = cart.value.findIndex((item) => {
if (newGood) {
return item.id === newGood.id && item.props_text === good.value.valueStr;
} else {
return item.id === newGood.id;
}
});
if (index > -1) {
cart.value[index].number += num;
} else {
cart.value.push({
id: newGood.id,
cate_id: cate.id,
name: newGood.storeName,
price: newGood.price,
number: num,
image: newGood.image,
valueStr: good.value.valueStr,
});
}
uni.setStorageSync("cart", JSON.parse(JSON.stringify(cart.value)));
};
const handleReduceFromCart = (item, good) => {
const index = cart.value.findIndex((item) => item.id === good.id);
cart.value[index].number -= 1;
if (cart.value[index].number <= 0) {
cart.value.splice(index, 1);
}
uni.setStorageSync("cart", JSON.parse(JSON.stringify(cart.value)));
};
const showGoodDetailModal = (item, newGood) => {
isCartShow.value = true;
good.value = JSON.parse(
JSON.stringify({
...newGood,
number: 1,
})
);
category.value = JSON.parse(JSON.stringify(item));
goodDetailModalVisible.value = true;
console.log("goodDetailModalVisible:", goodDetailModalVisible.value);
changePropertyDefault(0, 0, true);
};
const closeGoodDetailModal = () => {
//关闭饮品详情模态框
goodDetailModalVisible.value = false;
category.value = {};
good.value = {};
};
const changePropertyDefault = (index, key, isDefault) => {
//改变默认属性值
let valueStr = "";
console.log("good:", good.value);
if (isDefault) {
newValue.value = [];
for (let i = 0; i < good.value.productAttr.length; i++) {
newValue.value[i] = good.value.productAttr[i].attrValueArr[0];
}
valueStr = newValue.value.join(",");
} else {
newValue.value[index] = good.value.productAttr[index].attrValueArr[key];
// //valueStr = newValue.value.join(',')
// }
valueStr = newValue.value.join(",");
// let productValue = good.value.productValue[valueStr]
// if(!productValue) {
// let skukey = JSON.parse(JSON.stringify(newValue.value))
// skukey.sort((a, b) => a.localeCompare(b))
// //console.log('skukey:',skukey)
// valueStr = skukey.join(',')
// productValue = good.value.productValue[valueStr]
}
let productValue = good.value.productValue[valueStr];
good.value.number = 1;
good.value.price = parseFloat(productValue.price).toFixed(2);
good.value.stock = productValue.stock;
good.value.image = productValue.image ? productValue.image : good.value.image;
good.value.valueStr = valueStr;
};
const handlePropertyAdd = () => {
good.value.number += 1;
};
const handlePropertyReduce = () => {
if (good.value.number === 1) return;
good.value.number -= 1;
};
const handleAddToCartInModal = () => {
if (good.value.stock <= 0) {
uToast.value.show({
message: "商品库存不足",
type: "error"
});
return;
}
handleAddToCart(category.value, good.value, good.value.number);
closeGoodDetailModal();
};
const openCartPopup = () => {
//打开/关闭购物车列表popup
popup.value.open();
};
const handleCartClear = () => {
//清空购物车
uni.showModal({
title: "提示",
content: "确定清空购物车么",
success: ({
confirm
}) => {
if (confirm) {
popup.value.close();
cart.value = [];
uni.setStorageSync("cart", JSON.parse(JSON.stringify(cart.value)));
}
},
});
};
const handleCartItemAdd = (index) => {
cart.value[index].number += 1;
uni.setStorageSync("cart", JSON.parse(JSON.stringify(cart.value)));
};
const handleCartItemReduce = (index) => {
if (cart.value[index].number === 1) {
cart.value.splice(index, 1);
} else {
cart.value[index].number -= 1;
}
if (!cart.value.length) {
cartPopupVisible.value = false;
}
uni.setStorageSync("cart", JSON.parse(JSON.stringify(cart.value)));
};
const toPay = () => {
if (!isLogin.value) {
uni.navigateTo({
url: "/pages/components/pages/login/login",
});
return;
} else {
if (store.value.status == 0) {
uToast.value.show({
message: "不在店铺营业时间内",
type: "error"
});
return;
}
// 判断当前是否在配送范围内
if (
orderType.value == "takeout" &&
store.value.distance < store.value.far
) {
uToast.value.show({
message: "选中的地址不在配送范围",
type: "error"
});
return;
}
uni.showLoading({
title: "加载中",
});
uni.setStorageSync("cart", JSON.parse(JSON.stringify(cart.value)));
uni.navigateTo({
url: "/pages/components/pages/pay/pay",
});
}
uni.hideLoading();
};
</script>
<style lang="scss" scoped>
@import "./common.css";
/* #ifdef H5 */
page {
height: auto;
min-height: 100%;
}
/* #endif */
.container {
overflow: hidden;
position: relative;
background-color: #ffffff;
}
.loading {
width: 100%;
height: 100%;
display: flex;
align-items: center;
justify-content: center;
image {
width: 260rpx;
height: 260rpx;
position: relative;
margin-top: -200rpx;
/* #ifdef h5 */
margin-top: 0;
/* #endif */
}
}
.flex-row {
display: flex;
flex-direction: row;
}
.justify-between {
justify-content: space-between;
}
.align-center {
align-items: center;
}
.shop-banner {
width: 100%;
height: 250rpx;
object-fit: cover;
}
.notice-bar {
height: 60rpx;
background-color: #ffffff;
}
.main {
position: relative;
}
.nav {
background-color: #ffffff;
padding: 20rpx;
box-shadow: 0 2rpx 10rpx rgba(0, 0, 0, 0.1);
}
.header {
padding: 10rpx 0;
}
.store-info {
flex: 1;
}
.store-logo {
width: 80rpx;
height: 80rpx;
border-radius: 8rpx;
}
.store-name {
font-size: 32rpx;
font-weight: 500;
color: #000;
}
.store-location {
font-size: 24rpx;
color: #999;
margin-top: 4rpx;
}
.order-type {
background-color: #f7f3e9;
border-radius: 50rpx;
overflow: hidden;
border: 1px solid #fff;
height: 60rpx;
}
.dinein,
.takeout {
padding: 0 20rpx;
height: 100%;
display: flex;
align-items: center;
justify-content: center;
font-size: 24rpx;
}
.dinein.active {
background-color: #52ac41;
color: #fff;
}
.takeout.active {
background-color: #52ac41;
color: #fff;
}
.content {
width: 100%;
display: flex;
flex-direction: row;
padding-bottom: 160rpx;
}
.menu-sidebar {
width: 170rpx;
height: 100%;
background-color: #f9faf7;
}
.sidebar-wrapper {
display: flex;
flex-direction: column;
}
.menu-category {
width: 100%;
height: 148rpx;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
position: relative;
box-shadow: inset 0px -1px 0px 0px rgba(249, 246, 239, 1);
color: #717171;
font-size: 24rpx;
text-align: center;
}
.menu-category.current {
background-image: linear-gradient(270deg,
rgba(255, 255, 255, 1) 0,
rgba(243, 203, 90, 1) 100%);
color: #000;
font-weight: 600;
font-size: 24rpx;
}
.menu-category.current:before {
content: "";
position: absolute;
left: 0;
top: 0;
width: 10rpx;
height: 100%;
background-color: #52ac41;
}
.category-badge {
position: absolute;
top: 10rpx;
right: 10rpx;
background-color: #eab729;
color: #fff;
border-radius: 50%;
width: 32rpx;
height: 32rpx;
font-size: 22rpx;
display: flex;
align-items: center;
justify-content: center;
}
.goods-container {
flex: 1;
height: 100%;
}
.goods-wrapper {
padding: 20rpx;
}
.category-section {
margin-bottom: 40rpx;
}
.category-title {
font-size: 32rpx;
font-weight: 600;
margin-bottom: 20rpx;
display: flex;
align-items: center;
}
.category-icon {
width: 32rpx;
height: 32rpx;
margin-left: 10rpx;
}
.good-card {
display: flex;
margin-bottom: 30rpx;
border-radius: 12rpx;
padding: 20rpx;
background-color: #fff;
box-shadow: 0 2rpx 10rpx rgba(0, 0, 0, 0.05);
}
.good-image {
width: 200rpx;
height: 200rpx;
border-radius: 8rpx;
object-fit: cover;
}
.good-details {
flex: 1;
margin-left: 20rpx;
display: flex;
flex-direction: column;
}
.good-name {
font-size: 28rpx;
font-weight: 500;
color: #000;
margin-bottom: 8rpx;
}
.good-description {
font-size: 22rpx;
color: #999;
margin-bottom: 30rpx;
}
.price-action {
margin-top: auto;
}
.good-price {
font-size: 32rpx;
font-weight: 500;
color: #000;
}
.action-buttons {
display: flex;
align-items: center;
}
.spec-button {
height: 50rpx;
line-height: 50rpx;
padding: 0 20rpx;
font-size: 24rpx;
background-color: #52ac41;
border-radius: 25rpx;
}
.item-badge {
background-color: #eab729;
color: #fff;
border-radius: 50%;
width: 36rpx;
height: 36rpx;
font-size: 22rpx;
display: flex;
align-items: center;
justify-content: center;
margin-left: 10rpx;
}
.sold-out-label {
color: #999;
font-size: 24rpx;
}
.good-card.sold-out {
opacity: 0.7;
}
.cart-bar {
position: fixed;
bottom: 140rpx;
left: 0;
width: calc(100% - 80rpx);
height: 100rpx;
background-color: #f7f3e9;
display: flex;
align-items: center;
padding: 0 20rpx;
box-shadow: 0 -2rpx 10rpx rgba(0, 0, 0, 0.1);
border-radius: 50rpx;
margin: 0 40rpx;
z-index: 10;
border: 2px solid #fff;
}
.cart-icon-container {
position: relative;
}
.cart-icon {
width: 84rpx;
height: 84rpx;
}
.cart-badge {
position: absolute;
top: 0;
right: -10rpx;
background-color: #eab729;
color: #000;
border-radius: 50%;
min-width: 32rpx;
height: 32rpx;
font-size: 22rpx;
display: flex;
align-items: center;
justify-content: center;
padding: 0 6rpx;
}
.cart-total {
font-size: 40rpx;
font-weight: 500;
margin-left: 20rpx;
}
.checkout-button {
height: 100rpx;
line-height: 100rpx;
background-color: #52ac41;
color: #fff;
font-size: 32rpx;
border-radius: 50rpx;
margin-left: auto;
width: 192rpx;
}
/* 商品详情模态框 */
.good-detail-modal {
width: 100%;
height: 100%;
display: flex;
flex-direction: column;
}
.modal-header {
padding: 20rpx;
display: flex;
justify-content: flex-end;
}
.close-button image {
width: 40rpx;
height: 40rpx;
}
.modal-body {
flex: 1;
overflow-y: auto;
}
.modal-image {
width: 100%;
height: 400rpx;
padding: 10rpx;
}
.modal-image image {
width: 100%;
height: 100%;
object-fit: cover;
}
.modal-content {
padding: 20rpx;
}
.good-basic-info {
margin-bottom: 30rpx;
}
.good-title {
font-size: 32rpx;
font-weight: 600;
margin-bottom: 10rpx;
}
.good-subtitle {
font-size: 24rpx;
color: #999;
}
.points-info {
color: #f00;
}
.good-properties {
margin-bottom: 30rpx;
}
.property-item {
margin-bottom: 20rpx;
}
.property-name {
font-size: 28rpx;
font-weight: 500;
margin-bottom: 15rpx;
}
.property-values {
display: flex;
flex-wrap: wrap;
}
.property-value {
padding: 10rpx 20rpx;
background-color: #f5f5f5;
border-radius: 6rpx;
margin-right: 15rpx;
margin-bottom: 15rpx;
font-size: 24rpx;
}
.property-value.selected {
background-color: #52ac41;
color: #fff;
}
.modal-footer {
padding: 20rpx;
display: flex;
justify-content: space-between;
align-items: center;
border-top: 1px solid #eee;
}
.price-info {
display: flex;
flex-direction: column;
}
.final-price {
font-size: 32rpx;
font-weight: 600;
color: #000;
}
.selected-props {
font-size: 22rpx;
color: #999;
margin-top: 5rpx;
}
.quantity-control {
display: flex;
align-items: center;
}
.stock-info {
font-size: 24rpx;
color: #999;
margin-right: 15rpx;
}
.quantity-button {
width: 60rpx;
height: 60rpx;
padding: 0;
display: flex;
align-items: center;
justify-content: center;
border-radius: 50%;
}
.quantity-number {
width: 60rpx;
text-align: center;
font-size: 28rpx;
}
.add-cart-button {
height: 100rpx;
line-height: 100rpx;
background-color: #52ac41;
color: #fff;
font-size: 32rpx;
text-align: center;
}
/* 购物车popup */
.cart-popup-container {
max-height: 70vh;
display: flex;
flex-direction: column;
}
.popup-header {
height: 80rpx;
line-height: 80rpx;
padding: 0 30rpx;
background-color: #f7f7f7;
color: #52ac41;
font-size: 26rpx;
text-align: right;
}
.cart-items {
max-height: calc(70vh - 80rpx);
}
.items-wrapper {
padding: 20rpx 30rpx;
}
.cart-item {
display: flex;
justify-content: space-between;
align-items: center;
padding: 20rpx 0;
border-bottom: 1px solid #eee;
}
.item-info {
flex: 1;
}
.item-name {
font-size: 28rpx;
color: #000;
margin-bottom: 5rpx;
}
.item-properties {
font-size: 22rpx;
color: #999;
}
.item-price {
font-size: 28rpx;
font-weight: 500;
padding: 0 20rpx;
}
.item-controls {
display: flex;
align-items: center;
}
.item-quantity {
width: 60rpx;
text-align: center;
font-size: 28rpx;
}
button[size="mini"] {
min-width: 60rpx !important;
width: 60rpx !important;
height: 60rpx !important;
padding: 0 !important;
display: flex;
align-items: center;
justify-content: center;
}
button[type="primary"] {
background-color: #52ac41 !important;
}
.image_3 {
width: 48rpx;
height: 6rpx;
margin: 194rpx 0 0 319rpx;
}
</style>