文章目录
- 二级路由配置
- 模版代码
- 配置路由关系
- 跳转配置
- 面包屑导航实现
- 封装接口
- 渲染数据
- 分类基础列表实现
- 准备接口
- 渲染数据
- 列表筛选实现
- 无限加载实现
- 定制路由 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 = 1getGoodList() }
无限加载实现
核心实现逻辑:使用 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}}
})
本文链接:https://my.lmcjl.com/post/13244.html
4 评论