黑马程序员前端 Vue3 小兔鲜电商项目——(六)二级分类页

文章目录

    • 二级路由配置
      • 模版代码
      • 配置路由关系
      • 跳转配置
    • 面包屑导航实现
      • 封装接口
      • 渲染数据
    • 分类基础列表实现
      • 准备接口
      • 渲染数据
    • 列表筛选实现
    • 无限加载实现
    • 定制路由 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,重新发送列表请求。

  1. 修改模板 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>
    
  2. 定义回调函数,重新发送请求获取数据:

    // tab切换回调
    const tabChange = () => {console.log('tab切换了', reqData.value.sortField)reqData.value.page = 1getGoodList()
    }
    

无限加载实现

核心实现逻辑:使用 elementPlus 提供的 v-infinite-scroll 指令监听是否满足触底条件,满足加载条件时让页数参数加一获取下一页数据,做新老数据拼接渲染。

  1. 修改 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>
    
  2. 添加 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 评论

留下您的评论.