文章目录
- 二级路由配置
- 模版代码
- 配置路由关系
- 跳转配置
- 面包屑导航实现
- 封装接口
- 渲染数据
- 分类基础列表实现
- 准备接口
- 渲染数据
- 列表筛选实现
- 无限加载实现
- 定制路由 scrollBehavior
二级路由配置
模版代码
创建 src\views\SubCategory\index.vue 文件,添加以下代码:
<script setup>
</script>
<template>
<div class="container ">
<!-- 面包屑 -->
<div class="bread-container">
<el-breadcrumb separator=">">
<el-breadcrumb-item :to="{ path: '/' }">首页</el-breadcrumb-item>
<el-breadcrumb-item :to="{ path: '/' }">居家
</el-breadcrumb-item>
<el-breadcrumb-item>居家生活用品</el-breadcrumb-item>
</el-breadcrumb>
</div>
<div class="sub-container">
<el-tabs>
<el-tab-pane label="最新商品" name="publishTime"></el-tab-pane>
<el-tab-pane label="最高人气" name="orderNum"></el-tab-pane>
<el-tab-pane label="评论最多" name="evaluateNum"></el-tab-pane>
</el-tabs>
<div class="body">
<!-- 商品列表-->
</div>
</div>
</div>
</template>
<style lang="scss" scoped>
.bread-container {
padding: 25px 0;
color: #666;
}
.sub-container {
padding: 20px 10px;
background-color: #fff;
.body {
display: flex;
flex-wrap: wrap;
padding: 0 10px;
}
.goods-item {
display: block;
width: 220px;
margin-right: 20px;
padding: 20px 30px;
text-align: center;
img {
width: 160px;
height: 160px;
}
p {
padding-top: 10px;
}
.name {
font-size: 16px;
}
.desc {
color: #999;
height: 29px;
}
.price {
color: $priceColor;
font-size: 20px;
}
}
.pagination-container {
margin-top: 20px;
display: flex;
justify-content: center;
}
}
</style>
配置路由关系
在 src\router\index.js 中配置二级分类路由,URL 样式【`/category/sub/categoryId】:
import SubCategory from '@/views/SubCategory/index.vue'
routes: [
{
path: '/',
component: Layout,
children: [
{
path: '',
component: Home,
},
{
path: 'category/:id',
component: Category
},
{
path: 'category/sub/:id',
component: SubCategory
}
]
}
]
跳转配置
修改 src\views\Category\index.vue 文件中的 RouterLink 的 to 属性,修改为二级分类的链接【/category/sub/categoryId
】:
<!-- 分类数据 -->
<div class="sub-list">
<h3>全部分类</h3>
<ul>
<li v-for="i in categoryData.children" :key="i.id">
<RouterLink :to="`/category/sub/${i.id}`">
<img :src="i.picture" />
<p>{{ i.name }}</p>
</RouterLink>
</li>
</ul>
</div>
面包屑导航实现
封装接口
在 src\apis\category.js 文件中封装接口,用于获取二级分类列表数据:
/**
* @description: 获取二级分类列表数据
* @param {*} id 分类id
* @return {*}
*/
export const getCategoryFilterAPI = (id) => {
return http({
url: '/category/sub/filter',
params: {
id
}
})
}
渲染数据
在 src\views\SubCategory\index.vue 中调用 getCategoryFilterAPI()
接口获取分类数据,并在模板中进行渲染:
<script setup>
import { getCategoryFilterAPI } from '@/apis/category';
import { ref, onMounted } from 'vue'
import { useRoute } from 'vue-router'
const route = useRoute()
const categoryData = ref({})
const getCategoryData = async () => {
const res = await getCategoryFilterAPI(route.params.id)
categoryData.value = res.result
}
onMounted(() => {
getCategoryData()
})
</script>
<template>
<div class="container ">
<!-- 面包屑 -->
<div class="bread-container">
<el-breadcrumb separator=">">
<el-breadcrumb-item :to="{ path: '/' }">首页</el-breadcrumb-item>
<el-breadcrumb-item :to="{ path: `/category/${categoryData.parentId}` }">{{ categoryData.parentName }}
</el-breadcrumb-item>
<el-breadcrumb-item>{{ categoryData.name }}</el-breadcrumb-item>
</el-breadcrumb>
</div>
</div>
</template>
分类基础列表实现
准备接口
在 src\apis\category.js 文件中封装接口,用于获取二级分类商品列表数据:
/**
* @description: 获取导航数据
* @data {
categoryId: 1005000 ,
page: 1,
pageSize: 20,
sortField: 'publishTime' | 'orderNum' | 'evaluateNum'
}
* @return {*}
*/
export const getSubCategoryAPI = (data) => {
return request({
url: '/category/goods/temporary',
method: 'POST',
data
})
}
渲染数据
在 src\views\SubCategory\index.vue 中调用 getSubCategoryAPI()
接口获取分类数据,并在模板中进行渲染:
<script setup>
import { getCategoryFilterAPI, getSubCategoryAPI } from '@/apis/category'
import { ref, onMounted } from 'vue'
import { useRoute } from 'vue-router'
import GoodsItem from '@/views/Home/components/GoodsItem.vue'
const route = useRoute()
// 获取基础列表数据
const goodList = ref([])
const reqData = ref({
categoryId: route.params.id,
page: 1,
pageSize: 20,
sortField: 'publishTime'
})
const getGoodList = async() => {
const res = await getSubCategoryAPI(reqData.value)
goodList.value = res.result.items
}
onMounted(()=>{
getGoodList()
})
</script>
在模板中渲染商品数据:
<div class="body">
<!-- 商品列表-->
<GoodsItem v-for="good in goodList" :good="good" :key="good.id" />
</div>
列表筛选实现
核心逻辑:使用 v-model 指令双向绑定,点击 tab,切换筛选条件参数 sortField,重新发送列表请求。
-
修改模板 HTML 代码,v-model 指令双向绑定 reqData.sortField 参数,@tab-change 执行回调函数:
<el-tabs v-model="reqData.sortField" @tab-change="tabChange"> <el-tab-pane label="最新商品" name="publishTime"></el-tab-pane> <el-tab-pane label="最高人气" name="orderNum"></el-tab-pane> <el-tab-pane label="评论最多" name="evaluateNum"></el-tab-pane> </el-tabs>
-
定义回调函数,重新发送请求获取数据:
// tab切换回调 const tabChange = () => { console.log('tab切换了', reqData.value.sortField) reqData.value.page = 1 getGoodList() }
无限加载实现
核心实现逻辑:使用 elementPlus 提供的 v-infinite-scroll 指令监听是否满足触底条件,满足加载条件时让页数参数加一获取下一页数据,做新老数据拼接渲染。
-
修改 src\views\SubCategory\index.vue 中的代码,在要实现滚动加载的列表上上添加
v-infinite-scroll
,并赋值相应的加载方法,可实现滚动到底部时自动执行加载方法。绑定infinite-scroll-disabled
全部加载完毕后禁用加载。<div class="body" v-infinite-scroll="load" :infinite-scroll-disabled="disabled"> <!-- 商品列表--> <GoodsItem v-for="good in goodList" :good="good" :key="good.id" /> </div>
-
添加 load 自动加载方法:
//加载更多 const disabled = ref(false) const load = async () => { console.log("加载更多数据") //获取下一页数据 reqData.value.page++ const res = await getSubCategoryAPI(reqData.value) goodList.value = [...goodList.value, ...res.result.items] if (res.result.items === 0) { disabled = true } }
使用
...
扩展运算符进行拼接数据。
定制路由 scrollBehavior
优化:在不同路由之间切换的时候,窗口可以自动滚动到页面的顶部,而不是停留在原先的位置。
在 src\router\index.js 中配置 vue-router 的 scrollBehavior 属性,可以指定路由切换时的滚动位置:
const router = createRouter({
history: createWebHistory(import.meta.env.BASE_URL),
// path 和 component 对应关系的位置
routes: [
...
],
//路由滚动行为定制
scrollBehavior() {
return {
top: 0
}
}
})