github:https://github.com/tuanliang/eribbit_client-pc
06-首页-底部布局
目的: 完成首页底部布局。
首先,在 src/components/
下新建 app-footer.vue
组件,基础布局如下:
<template>
<footer class="app-footer">
<!-- 联系我们 -->
<div class="contact">
<div class="container">
<dl>
<dt>客户服务</dt>
<dd><i class="iconfont icon-kefu"></i> 在线客服</dd>
<dd><i class="iconfont icon-question"></i> 问题反馈</dd>
</dl>
<dl>
<dt>关注我们</dt>
<dd><i class="iconfont icon-weixin"></i> 公众号</dd>
<dd><i class="iconfont icon-weibo"></i> 微博</dd>
</dl>
<dl>
<dt>下载APP</dt>
<dd class="qrcode"><img src="../assets/images/qrcode.jpg" /></dd>
<dd class="download">
<span>扫描二维码</span>
<span>立马下载APP</span>
<a href="javascript:;">下载页面</a>
</dd>
</dl>
<dl>
<dt>服务热线</dt>
<dd class="hotline">400-0000-000 <small>周一至周日 8:00-18:00</small></dd>
</dl>
</div>
</div>
<!-- 其它 -->
<div class="extra">
<div class="container">
<div class="slogan">
<a href="javascript:;">
<i class="iconfont icon-footer01"></i>
<span>价格亲民</span>
</a>
<a href="javascript:;">
<i class="iconfont icon-footer02"></i>
<span>物流快捷</span>
</a>
<a href="javascript:;">
<i class="iconfont icon-footer03"></i>
<span>品质新鲜</span>
</a>
</div>
<!-- 版权信息 -->
<div class="copyright">
<p>
<a href="javascript:;">关于我们</a>
<a href="javascript:;">帮助中心</a>
<a href="javascript:;">售后服务</a>
<a href="javascript:;">配送与验收</a>
<a href="javascript:;">商务合作</a>
<a href="javascript:;">搜索推荐</a>
<a href="javascript:;">友情链接</a>
</p>
<p>CopyRight © 小兔鲜儿</p>
</div>
</div>
</div>
</footer>
</template>
<script>
export default {
name: 'AppFooter'
}
</script>
<style scoped lang='less'>
.app-footer {
overflow: hidden;
background-color: #f5f5f5;
padding-top: 20px;
.contact {
background: #fff;
.container {
padding: 60px 0 40px 25px;
display: flex;
}
dl {
height: 190px;
text-align: center;
padding: 0 72px;
border-right: 1px solid #f2f2f2;
color: #999;
&:first-child {
padding-left: 0;
}
&:last-child {
border-right: none;
padding-right: 0;
}
}
dt {
line-height: 1;
font-size: 18px;
}
dd {
margin: 36px 12px 0 0;
float: left;
width: 92px;
height: 92px;
padding-top: 10px;
border: 1px solid #ededed;
.iconfont {
font-size: 36px;
display: block;
color: #666;
}
&:hover {
.iconfont {
color: @xtxColor;
}
}
&:last-child {
margin-right: 0;
}
}
.qrcode {
width: 92px;
height: 92px;
padding: 7px;
border: 1px solid #ededed;
}
.download {
padding-top: 5px;
font-size: 14px;
width: auto;
height: auto;
border: none;
span {
display: block;
}
a {
display: block;
line-height: 1;
padding: 10px 25px;
margin-top: 5px;
color: #fff;
border-radius: 2px;
background-color: @xtxColor;
}
}
.hotline {
padding-top: 20px;
font-size: 22px;
color: #666;
width: auto;
height: auto;
border: none;
small {
display: block;
font-size: 15px;
color: #999;
}
}
}
.extra {
background-color: #333;
}
.slogan {
height: 178px;
line-height: 58px;
padding: 60px 100px;
border-bottom: 1px solid #434343;
display: flex;
justify-content: space-between;
a {
height: 58px;
line-height: 58px;
color: #fff;
font-size: 28px;
i {
font-size: 50px;
vertical-align: middle;
margin-right: 10px;
font-weight: 100;
}
span {
vertical-align: middle;
text-shadow: 0 0 1px #333;
}
}
}
.copyright {
height: 170px;
padding-top: 40px;
text-align: center;
color: #999;
font-size: 15px;
p {
line-height: 1;
margin-bottom: 20px;
}
a {
color: #999;
line-height: 1;
padding: 0 10px;
border-right: 1px solid #999;
&:last-child {
border-right: none;
}
}
}
}
</style>
最后,在 src/views/Layout.vue
中导入使用。
<template>
<AppTopnav/>
<AppHeader/>
<main class="app-body">
<!-- 二级路由 -->
<router-view></router-view>
</main>
+ <AppFooter/>
</template>
<script>
import AppTopnav from '@/components/app-topnav'
import AppHeader from '@/components/app-header'
+import AppFooter from '@/components/app-footer'
export default {
name: 'XtxLayout',
+ components: { AppTopnav, AppHeader, AppFooter }
}
</script>
<style scoped lang='less'>
+.app-body {
+ min-height: 600px;
+}
</style>
07-首页-头部分类导航组件
目的: 提取头部分类导航组件,提供给头部,和将来的吸顶头部使用。
第一步:提取头部导航为一个组件
- 新建
src/components/app-header-nav.vue
组件。
<template>
<ul class="app-header-nav">
<li class="home"><RouterLink to="/">首页</RouterLink></li>
<li><a href="#">美食</a></li>
<li><a href="#">餐厨</a></li>
<li><a href="#">艺术</a></li>
<li><a href="#">电器</a></li>
<li><a href="#">居家</a></li>
<li><a href="#">洗护</a></li>
<li><a href="#">孕婴</a></li>
<li><a href="#">服装</a></li>
<li><a href="#">杂货</a></li>
</ul>
</template>
<script>
export default {
name: 'AppHeaderNav'
}
</script>
<style scoped lang='less'>
.app-header-nav {
width: 820px;
display: flex;
padding-left: 40px;
position: relative;
z-index: 998;
li {
margin-right: 40px;
width: 38px;
text-align: center;
a {
font-size: 16px;
line-height: 32px;
height: 32px;
display: inline-block;
}
&:hover {
a {
color: @xtxColor;
border-bottom: 1px solid @xtxColor;
}
}
}
}
</style>
- 在 app-header.vue 中使用组件。注意,删除结构和样式。
<template>
<header class='app-header'>
<div class="container">
<h1 class="logo"><RouterLink to="/">小兔鲜</RouterLink></h1>
+ <AppHeaderNav />
<div class="search">
<i class="iconfont icon-search"></i>
<input type="text" placeholder="搜一搜">
</div>
<div class="cart">
<a class="curr" href="#">
<i class="iconfont icon-cart"></i><em>2</em>
</a>
</div>
</div>
</header>
</template>
<script>
+import AppHeaderNav from './app-header-nav'
export default {
name: 'AppHeader',
+ components: { AppHeaderNav }
}
</script>
第二步:完善子级分类布局 src/components/app-header-nav.vue
<template>
<ul class="app-header-nav">
<li class="home"><RouterLink to="/">首页</RouterLink></li>
<li>
<a href="#">美食</a>
<div class="layer">
<ul>
<li v-for="i in 10" :key="i">
<a href="#">
<img src="http://zhoushugang.gitee.io/erabbit-client-pc-static/uploads/img/category%20(4).png" alt="">
<p>果干</p>
</a>
</li>
</ul>
</div>
</li>
<li><a href="#">餐厨</a></li>
<li><a href="#">艺术</a></li>
<li><a href="#">电器</a></li>
<li><a href="#">居家</a></li>
<li><a href="#">洗护</a></li>
<li><a href="#">孕婴</a></li>
<li><a href="#">服装</a></li>
<li><a href="#">杂货</a></li>
</ul>
</template>
<script>
export default {
name: 'AppHeaderNav'
}
</script>
<style scoped lang='less'>
.app-header-nav {
width: 820px;
display: flex;
justify-content: space-around;
padding-left: 40px;
position: relative;
z-index: 998;
> li {
margin-right: 40px;
width: 38px;
text-align: center;
> a {
font-size: 16px;
line-height: 32px;
height: 32px;
display: inline-block;
}
&:hover {
> a {
color: @xtxColor;
border-bottom: 1px solid @xtxColor;
}
> .layer {
height: 132px;
opacity: 1;
}
}
}
}
.layer {
width: 1240px;
background-color: #fff;
position: absolute;
left: -200px;
top: 56px;
height: 0;
overflow: hidden;
opacity: 0;
box-shadow: 0 0 5px #ccc;
transition: all .2s .1s;
ul {
display: flex;
flex-wrap: wrap;
padding: 0 70px;
align-items: center;
height: 132px;
li {
width: 110px;
text-align: center;
img {
width: 60px;
height: 60px;
}
p {
padding-top: 10px;
}
&:hover {
p {
color: @xtxColor;
}
}
}
}
}
</style>
08-首页-头部分类导航渲染
目的: 实现头部一级分类和二级分类渲染。
基本步骤:
- 定义一个常量数据和后台保持一致(约定好9大分类),这样不请求后台就能展示一级分类,不至于白屏。
- 在API目录定义接口函数
- 在vuex中的category模块,基于常量数据定义state数据,定义修改分类列表函数,定义获取数据函数。
- 在Layout组件获取调用actions获取数据,在头部导航组件渲染即可。
落地代码:
- 定义九个分类常量数据
src/api/constants.js
// 顶级分类
export const topCategory = [
'居家',
'美食',
'服饰',
'母婴',
'个护',
'严选',
'数码',
'运动',
'杂货'
]
- 定义API函数
src/api/category.js
// 定义首页需要的接口函数
import request from '@/utils/request'
/**
* 获取首页头部分类数据
*/
export const findAllCategory = () => {
return request('/home/category/head', 'get')
}
- vuex在category模块,来存储分类数据,提供修改和获取的函数。
src/store/modules/category.js
// 存储的分类数据
import { topCategory } from '@/api/constants'
import { findAllCategory } from '@/api/category'
export default {
namespaced: true,
state: () => {
return {
// 如果默认是[]数组,看不见默认的9个分类,等你数据加载完毕才会看到。
// 所以:根据常量数据来生成一个默认的顶级分类数据,不会出现空白(没数据的情况)
list: topCategory.map(item => ({ name: item }))
}
},
// 加载数据成功后需要修改list所以需要mutations函数
mutations: {
setList (state, headCategory) {
state.list = headCategory
}
},
// 需要向后台加载数据,所以需要actions函数获取数据
actions: {
async getList ({ commit }) {
const { result } = await findAllCategory()
// 获取数据成功,提交mutations进行数据修改
commit('setCategory', result)
}
}
}
- 获取数据在
src/views/Layout.vue
初始化的时候
export default {
name: 'Layout',
components: {
AppTopnav,
AppHeader,
AppFooter
},
+ // 获取下分类数据
+ setup () {
+ const store = useStore()
+ store.dispatch('category/getList')
+ }
}
- 在头部导航组件渲染
src/compotents/app-header-nav.vue
import { useStore } from 'vuex'
import { computed } from 'vue'
export default {
name: 'AppHeaderNav',
setup () {
const store = useStore()
const list = computed(()=>{
return store.state.category.list
})
return { list }
}
}
<ul class="app-header-nav">
<li class="home"><RouterLink to="/">首页</RouterLink></li>
<li v-for="item in list" :key="item.id">
<RouterLink to="/">{{item.name}}</RouterLink>
<div class="layer">
<ul>
<li v-for="sub in item.children" :key="sub.id">
<RouterLink to="/">
<img :src="sub.picture" alt="">
<p>{{sub.name}}</p>
</RouterLink>
</li>
</ul>
</div>
</li>
</ul>
总结: 数据在vuex中管理,然后再组件使用数据进行渲染。
09-首页-头部分类导航交互
目的:实现点击的时候跳转,且能关闭二级分类弹窗。
描述:由于是单页面路由跳转不会刷新页面,css的hover一直触发无法关闭分类弹窗。
大致逻辑:
- 配置路由组件支持分类跳转
- 鼠标进入一级分类展示对应的二级分类弹窗
- 点击一级分类,二级分类,隐藏二级分类弹窗
- 离开一级分类,二级分类,隐藏二级分类弹窗
落地代码:
1) 配置路由和组件实现跳转
- 配置路由规则 src/router/index.js
+import TopCategory from '@/views/category'
+import SubCategory from '@/views/category/sub'
const routes = [
{
path: '/',
component: Layout,
children: [
{ path: '/', component: Home },
+ { path: '/category/:id', component: TopCategory },
+ { path: '/category/sub/:id', component: SubCategory }
]
}
]
- 创建分类组件 src/views/category/index.vue
<template>
<div>Top-Category</div>
</template>
<script>
export default {
name: 'TopCategory'
}
</script>
<style scoped lang="less"></style>
src/views/category/sub.vue
<template>
<div>Sub-Category</div>
</template>
<script>
export default {
name: 'SubCategory'
}
</script>
<style scoped lang="less"></style>
2)跳转后关闭二级分类弹窗
- 给每一个一级分类定义控制显示隐藏的数据,
open
布尔类型,通过open设置类名控制显示隐藏。 - 当进入一级分类的时候,将open改为true
- 当离开一级分类的时候,将open改为false
- 点击一级分类,二级分类,将open改为false
在vuex种给一级分类加open数据 src/store/modules/category.js
async getCategory ({ commit }) {
const { result } = await findHeadCategory()
// 给一级分类加上一个控制二级分类显示隐藏的数据open
+ result.forEach(item => {
+ item.open = false
+ })
// 获取数据成功,提交mutations进行数据修改
commit('setCategory', result)
}
添加了 show hide vuex的mutations函数修改 open src/store/modules/category.js
// 修改当前一级分类下的open数据为true
show (state, item) {
const category = state.list.find(category => category.id === item.id)
category.open = true
},
// 修改当前一级分类下的open数据为false
hide (state, item) {
const category = state.list.find(category => category.id === item.id)
category.open = false
}
再 头部导航组件 实现显示和隐藏 src/components/app-header-nav.vue
import { useStore } from 'vuex'
import { computed } from 'vue'
export default {
name: 'AppHeaderNav',
setup () {
const store = useStore()
const list = computed(()=>{
return store.state.category.list
})
+ const show = (item) => {
+ store.commit('category/show', item)
+ }
+ const hide = (item) => {
+ store.commit('category/hide', item)
+ }
+ return { list, show, hide}
}
}
+ <li v-for="item in list" :key="item.id" @mouseenter="show(item)" @mouseleave="hide(item)">
+ <RouterLink :to="`/category/${item.id}`" @click="hide(item)">{{item.name}}</RouterLink>
<div class="layer" :class="{open:item.open}">
<ul>
<li v-for="sub in item.children" :key="sub.id">
+ <RouterLink :to="`/category/sub/${sub.id}`" @click="hide(item)">
<img :src="sub.picture" alt="">
<p>{{sub.name}}</p>
</RouterLink>
</li>
</ul>
</div>
</li>
- // > .layer {
- // height: 132px;
- // opacity: 1;
- // }
}
}
}
.layer {
+ &.open {
+ height: 132px;
+ opacity: 1;
+ }
总结: 再组件中调用vuex的mutation函数控制每个一级分类下二级分类的显示隐藏。