You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

488 lines
15 KiB

10 months ago
<!--页面头部-->
<template>
<div class="relative w-100%">
<!--黑条-->
<div class="header bg-#F5F5F5" :class="isShowTop && useApp.isHomePage ? 'headerfixed' : ''">
<div class="headerCon acea-row row-between-wrapper">
<div class="flex-y-center text-12px">
<div class="mr-30px flex-y-center">
<nuxt-link :to="{ path: '/' }" class="flex-y-center"
><span class="iconfont icon-shangchengshouye inline-block"></span>商城首页</nuxt-link
>
</div>
<el-popover
v-if="pcHomeCon && pcHomeCon.goPhoneQrCodeType"
:width="auto"
:minWidth="auto"
popper-style="box-shadow: rgb(14 18 22 / 35%) 0px 10px 38px -10px, rgb(14 18 22 / 20%) 0px 10px 20px -15px; padding: 20px;"
>
<template #reference>
<div class="flex-y-center mobileMall">
<span class="iconfont icon-shoujishangcheng inline-block"></span>手机商城
</div>
</template>
<template #default>
<div class="acea-row">
<div v-if="pcHomeCon.goPhoneQrCodeType.includes('1')">
<div :class="pcHomeCon.goPhoneQrCodeType.includes('2')?'mr-26px':''">
<div class="borderSol-eee w-80px h-80px flex-center b-rd-4px mb-10px">
<el-image :src="wechatQrcode" class="w-72px h-72px"></el-image>
</div>
<div class="font-400 text-12px text-#333 w-80px text-center">小程序商城</div>
</div>
</div>
<div v-if="pcHomeCon.goPhoneQrCodeType.includes('2')">
<div class="borderSol-eee w-80px h-80px flex-center b-rd-4px mb-10px">
<qrcode-vue :value="indexDomain" :size="72" level="H" />
</div>
<div class="font-400 text-12px text-#333 w-80px text-center">H5商城</div>
</div>
</div>
</template>
</el-popover>
</div>
<div class="user acea-row row-middle">
<div v-if="!userStore.isLogin" class="item" @click="handlerLogin">登录/注册</div>
<div v-else class="acea-row row-middle" @click.stop="handlerNuxtLink('/users/user_info', 11)">
<span class="line1 font-color" style="max-width: 135px">Hi{{ userStore.userInfo.nikeName }}</span>
<p class="ml-10px item" @click.stop="handleHeaderLogout">退出</p>
</div>
<div class="item" @click="handlerNuxtLink('/users/order_list', 1)">我的订单</div>
<div class="item" @click="handlerNuxtLink('/users/collect_products', 3)">我的收藏</div>
<el-dropdown class="user">
<span class="el-dropdown-link text-12px item">
商户入驻
<el-icon class="el-icon--right">
<arrow-down />
</el-icon>
</span>
<template #dropdown>
<el-dropdown-menu>
<el-dropdown-item class="text-12px" @click="handlerNuxtLink('/merchant/merchant_settled', 0)"
>商户入驻</el-dropdown-item
>
<el-dropdown-item class="text-12px" @click="handlerNuxtLink('/merchant/application_record', 0)"
>申请记录</el-dropdown-item
>
</el-dropdown-menu>
</template>
</el-dropdown>
</div>
</div>
</div>
<!--菜单/搜索框/菜单-->
<div class="w-100% bg-white">
<!--搜索框-->
<div :class="isShowTop && useApp.isHomePage ? 'menufixed' : ''" class="w-100% bg-white">
<div class="min_wrapper_1200">
<div class="flex flex-justify-between pt-30px">
<!--logo-->
<nuxt-link :to="{ path: '/' }"
><el-image lazy :src="pcHomeCon?.leftTopLogo" class="w100px h36px"></el-image
></nuxt-link>
<!--搜索框商品详情页商户主页-->
<div v-if="useApp.routePath === '/merchant/merchant_home' || useApp.routePath.includes('product/detail')" class="acea-row">
<div class="w-560px ml-5px">
<el-input
v-model="searchVal"
placeholder="搜索全站/本店"
class="input-with-select"
@keyup.enter="handleEnterKey"
>
<template #append>
<div @click="handleEnterKey" class="cursors w-80px h-40px bg-color b-rd-20px text-center lh-40px">
<span class="text-#fff text-14px font-400">搜全站</span>
</div>
</template>
</el-input>
</div>
<div @click="handleEnterKeyHome" class="cursors w-80px h-40px bg-#333333 b-rd-20px text-center lh-40px ml-10px">
<span class="text-#fff text-14px font-400">搜本店</span>
</div>
</div>
<!--搜索框-->
<div v-else class="w-560px ml-5px">
<el-input
v-model="searchVal"
placeholder="搜索商品/店铺"
class="input-with-select"
@keyup.enter="handleEnterKey"
>
<template #prepend>
<el-select v-model="selectVal" placeholder="Select" style="width: 115px">
<el-option label="商品" value="1" />
<el-option label="店铺" value="2" />
</el-select>
</template>
<template #append>
<div @click="handleEnterKey" class="cursors w-80px h-40px bg-color b-rd-20px text-center lh-40px">
<span class="iconfont icon-sousuo text-#fff font-400"></span>
</div>
</template>
</el-input>
</div>
<!--购物车-->
<div
@click="handlerNuxtLink('/order/shopping_cart', 0)"
class="h-40px borderSol b-rd-20px font-color text-14px flex-center mb-26px cursors w-140px mr-5px"
>
<span class="iconfont icon-gouwuche mr-10px"></span>购物车({{ useApp.carNumber }})
</div>
</div>
</div>
</div>
<!--菜单分类-->
<div v-if="useApp.isShowHeaderMenu" class="w-100% bg-white header-menu">
<div class="min_wrapper_1200">
<div @mouseenter="handlerMenuEnter" @mouseleave="handleSubmitLeave">
<!--菜单分类-->
<div class="menu flex-y-center">
<div
v-for="(item, index) in menuList"
:key="item.id"
@mouseenter="handlerMenuEnter(index)"
@click="handleGoPage(item, index)"
:class="index === 0 ? 'classification' : current===item.id ? 'font-color':''"
class="cursor-pointer menu-item fontColor333 text-16px mr-40px font-500 oppoSans-M"
>
<span v-show="index === 0" class="iconfont icon-shangpinfenlei mr-8px"></span>{{ item.name }}
</div>
</div>
<!--分类显示 -->
<div v-if="showMenu" class="absolute w-1200px yop-179px z-11" :class="showSeach ? 'on' : ''">
<div class="classify z-8">
<!--分类组件-->
<classify-card @handleSubmitLeave="handleSubmitLeave"></classify-card>
</div>
</div>
</div>
</div>
</div>
</div>
<!--登录弹窗-->
<user-login ref="userLoginRef" @on-login-succeeded="onLoginSucceeded"></user-login>
</div>
</template>
<script setup lang="ts">
import { headerMenuListDefault } from '~/components/defaultComponents'
import { ArrowDown } from '@element-plus/icons-vue'
import {b64toBlob, Debounce, linkNavigateTo} from '~/utils/util'
import { loginLogout } from '~/server/userApi'
import feedback from '~/utils/feedback'
defineOptions({ name: 'layoutHeader' })
import { useUserStore } from '@/stores/user'
import {onMounted, ref, watch} from 'vue'
import { searcKeywordApi } from '~/server/goodsApi'
import { useAppStore } from '~/stores/app'
import QrcodeVue from 'qrcode.vue'
import { indexDomainApi, wechatQrcodeApi } from '~/server/homeApi'
import {auto} from "@popperjs/core";
const userStore = useUserStore()
const useApp = useAppStore()
const route = useRoute()
/**
* 监听页面滚动
*/
onMounted(() => {
window.addEventListener('scroll', handleScroll)
})
const isShowTop = ref<boolean>(false)
const handleScroll = () => {
const t = document.documentElement.scrollTop || document.body.scrollTop
if (t > 100) {
isShowTop.value = true
} else {
isShowTop.value = false
}
}
//头部信息,首页接口调用
const props = defineProps({
pcHomeCon: {
type: Object,
default: null,
},
})
const { pcHomeCon } = toRefs(props)
//移动端域名
const { data: indexDomain } = useAsyncData(() => indexDomainApi())
//移动端域名
const wechatQrcode = ref<string>('')
const getWechatQrcode = async () => {
let data = await wechatQrcodeApi({
scene: 'id=0',
path: 'pages/index/index',
env_version: 'trial',
})
let blob = b64toBlob(data.code)
wechatQrcode.value = URL.createObjectURL(blob)
}
getWechatQrcode();
//搜索
const searchVal = ref<string>('') //搜索框输入值
const selectVal = ref<string>('1') //搜索框选择值
searchVal.value = route.query.searchValue ? route.query.searchValue.toString() : searchVal.value.toString()
onMounted(() => {
selectVal.value = '1'
});
//搜本店
const handleEnterKeyHome = async()=>{
linkNavigateTo(`/merchant/merchant_home`, { merId: useApp.pcMerId, searchValue: searchVal.value, currentVal: 1 })
}
//搜索跳入页面
const handleEnterKey = async () => {
//useApp.routePath === '/merchant/merchant_home' || useApp.routePath.includes('product/detail')
if(useApp.routePath === '/merchant/merchant_home'|| useApp.routePath.includes('product/detail')){
await linkNavigateTo(`/product/product_list`, { searchValue: searchVal.value })
}else{
if (selectVal.value === '1') {
await linkNavigateTo(`/product/product_list`, { searchValue: searchVal.value })
} else {
await linkNavigateTo(`/merchant/merchant_list`, { searchValue: searchVal.value })
}
}
showSeach.value = !showSeach.value
showMenu.value = false
}
//头部菜单
const menuList = ref<ItemObject[]>(headerMenuListDefault())
//动画
const { $aos } = useNuxtApp()
onMounted(() => {
$aos().init({
easing: 'ease-out-back',
duration: 1000,
})
})
//登录成功后回调
const onLoginSucceeded = () => {
// getCartCount()
}
/**
* 热门搜索词
*/
const showSeach = ref<boolean>(false)
const { data: searcKeywordList } = await useAsyncData(async () => searcKeywordApi())
//清空搜索词
const nuxtApp = useNuxtApp()
nuxtApp.provide('onClearSearchVal', () => {
searchVal.value = ''
})
/**
* 菜单移入
*/
const checkedIndex = ref<number>(10) //分类鼠标悬浮索引
const showMenu = ref<boolean>(false) //是否展示分类菜单
const handlerMenuEnter = (idx: number | undefined) => {
checkedIndex.value = idx ? idx : 10
if (idx === 0 && !useApp.isHomePage) {
showMenu.value = true
} else {
showMenu.value = false
}
}
//鼠标移出回调
const handleSubmitLeave = () => {
showMenu.value = false
}
// 跳入页面
const current = ref<string>(route.query.type?<string>route.query.type:'')
watch(
() => <string>route.query.type,
(newValue) => {
current.value = newValue
},
)
const handleGoPage = async (obj: any, idx: number) => {
current.value = obj.id
if (idx === 0) return
await linkNavigateTo(obj.pc_url, { type: obj.id })
}
/**
* 登录
*/
const userLoginRef = shallowRef()
const handlerLogin = async () => {
if (userStore.isLogin) {
// await linkNavigateTo('/forum/create')
} else {
//data.backUrl = '/forum/create'
userLoginRef.value.open()
}
}
/**
* 地址跳转
*/
const handlerNuxtLink = async (url: string, type: number) => {
if (userStore.token) {
await linkNavigateTo(url, { type: type })
} else {
userLoginRef.value.open()
}
}
//退出登录
//购物车没登录状态下显示0
const { getCarNumber } = useAppStore()
const handleHeaderLogout = Debounce(async () => {
await feedback.confirm('确定退出吗?')
await loginLogout()
await getCarNumber(0)
await userStore.logout()
},500)
</script>
<style scoped lang="scss">
:deep(.el-dropdown-menu__item){
font-size: 12px !important;
}
:deep(.el-dropdown__popper.el-popper), :deep(.el-popper){
font-size: 12px !important;
}
.header-menu {
border-bottom: 2px solid #e93323;
}
.mobileMall{
&:hover {
color: #e93323;
}
}
.classification {
width: 200px;
height: 40px;
background: #e93323;
border-radius: 12px 12px 0px 0px;
display: flex;
align-items: center;
justify-content: center;
color: #fff;
}
:deep(.el-input-group) {
height: 40px;
background: #ffffff;
border-radius: 20px 20px 20px 20px;
border: 1px solid #e93323;
overflow: hidden;
}
:deep(.el-input__wrapper) {
padding-left: 20px !important;
}
:deep(.el-input__wrapper),
:deep(.el-input-group__prepend),
:deep(.el-input-group__prepend:hover),
:deep(.el-input-group__append) {
padding: 0;
box-shadow: none !important;
--el-fill-color-light: rgba(255, 255, 255, 0) !important;
}
:deep(.el-input-group__append) {
padding: 0;
}
:deep(.el-input) {
--el-input-hover-border-color: none !important;
--el-input-focus-border-color: none !important;
}
:deep(.el-select) {
margin: 0 !important;
width: 74px !important;
--el-select-input-focus-border-color: none !important;
}
:deep(.el-input-group__prepend:hover) {
border: none;
}
.headerfixed {
top: 0;
z-index: 99;
position: fixed;
}
.menufixed {
top: 40px;
z-index: 99;
position: fixed;
}
.seachfixed {
position: fixed;
top: 124px;
z-index: 99;
}
.classify {
border-radius: 0px 0px 26px 26px;
overflow: hidden;
box-shadow: 0px 1px 6px 0px rgba(0, 0, 0, 0.1);
}
.search {
&-ipt {
outline: none;
border: 0;
background: none;
}
&-history {
border-radius: 0px 0px 16px 16px;
}
}
.header {
flex: 1;
width: 100%;
height: 40px;
font-size: 12px;
color: #666666;
cursor: pointer;
.headerCon {
height: 100%;
position: relative;
max-width: 1200px;
margin: 0 auto;
a {
color: #666666;
&:hover {
color: #e93323;
}
}
.iconfont {
margin-right: 5px;
}
.user {
.item {
margin-right: 8px;
position: relative;
padding-left: 8px;
color: #666666;
&:hover {
color: #e93323;
}
}
}
}
}
</style>