Vue3:分类管理综合案例实现

综合案例

实现分类管理功能

请添加图片描述

路由

在main.js中引入router

访问根路径’/'后跳转到布局容器

加载布局容器后重定向到’/nav/manage’

加载我们需要的组件

这样可以在布局容器中切换功能模块时,只对需要修改的组件进行重新加载

const router = createRouter({history: createWebHistory(import.meta.env.BASE_URL),routes: [{path: '/',component: () => import('@/views/layout/LayoutContainer.vue'),redirect: '/nav/manage',children: [{path: '/nav/manage',component: () => import('@/views/nav/NavigationManage.vue')}]}]
})

封装组件

数据列表组件
<template><div> <el-table v-loading="loading" :data="dataList" style="width: 100%"><el-table-column prop="id" label="序号"></el-table-column><el-table-column prop="name" label="分类名称"></el-table-column><el-table-column prop="pictureImg" label="分类图片"><template #default="scope"><div><el-image style="width: auto; height: 40px; border: none; cursor: pointer":src="scope.row.pictureImg" :preview-src-list="[scope.row.pictureImg]":preview-teleported=true :hide-on-click-modal=true></el-image></div></template></el-table-column><el-table-column prop="sort" label="分类排序"></el-table-column><el-table-column label="状态"><template #default="scope"><div style="display: flex; align-items: center"><span style="margin-left: 10px">{{ scope.row.state == '0' ? '启用' : '禁用' }}</span></div></template></el-table-column><el-table-column prop="creator" label="创建人"></el-table-column><el-table-column label="操作"><!-- row 就是 channelList 的一项, $index 下标 --><template #default="{ row, $index }"><el-button :icon="Edit" circle plain type="primary" @click="onEditChannel(row, $index)"></el-button><el-button :icon="Delete" circle plain type="danger" @click="onDelChannel(row, $index)"></el-button></template></el-table-column><template #empty><el-empty description="没有数据"></el-empty></template></el-table></div>
</template><script setup>
import { Edit, Delete } from '@element-plus/icons-vue';defineProps({dataList:Object,loading:Boolean})const Emits = defineEmits(['onEditChannel','onDelChannel']);const onEditChannel = (row) => {Emits("onEditChannel",row)}const onDelChannel = (row) => {Emits("onDelChannel",row)}
</script><style lang='stylus' scoped></style>
查询组件
<template><div><!-- 表单区域 --><el-form inline><el-form-item label="分类:"><el-input v-model="searchParams.name" placeholder="请输入分类名字称" clearable /></el-form-item><el-form-item label="状态:"><el-select v-model="searchParams.state" placeholder="请选择状态" style="width: 120px" clearable><el-option label="启用" value="0"></el-option><el-option label="禁用" value="1"></el-option></el-select></el-form-item><el-form-item><el-button type="primary">搜索</el-button><el-button>重置</el-button></el-form-item></el-form></div>
</template><script setup>defineProps({searchParams: Object})
</script><style lang='stylus' scoped></style>
外层父组件(分页组件)

在引入时给子组件传递数据和方法,实现组件间通信

<script setup>
import { ref , onMounted} from 'vue'
import NavigationEdit from './components/NavigationEdit.vue'
import NavigationSearch from './components/NavigationSearch.vue'
import NavigationDataList from './components/NavigationDataList.vue'
import {getCategoryList} from '@/api/navigation'//测试数据
const testData = []
const loading = ref(false); //加载中,默认false
const dataList = ref(testData); //数据列表
const searchParams = ref({ //搜索参数name: '',state: ''
})
const pageData = ref({ //分页参数currentPage: 1,pageSize: 5,total: 0
});const param = {name: '',state: '',page: 2,pageSize: 5
}const getDataList = async() => {loading.value = true;let result = await getCategoryList(param);    dataList.value = result.data.result.records;pageData.value.total = result.data.result.total;console.log(result);loading.value = false;
}onMounted(()=>{getDataList();
})//对话框对象
const dialog = ref({});const onAddChannel = () => {dialog.value.open({})
}const onSuccess = () => {console.log("success...");
}const onEditChannel = (row) => { // 点击编辑执行的方法dialog.value.open(row)
}
const onDelChannel = async (row) => { // 点击删除执行的方法await ElMessageBox.confirm('你确认要删除该分类么', '温馨提示', {type: 'warning',confirmButtonText: '确认',cancelButtonText: '取消'})//TODO 待实现
}
</script> <template><div><!-- 引入自定义组件 --><NavigationEdit ref="dialog" @success="onSuccess"></NavigationEdit><el-card><template #header><span>分类管理</span><el-button style="float: right" @click="onAddChannel">添加分类</el-button></template><NavigationSearch:searchParams = "searchParams"/><NavigationDataList:dataList = "dataList":loading = "loading"@onEditChannel = "onEditChannel"@onDelChannel = "onDelChannel"/><!-- 分页插件 --><div class="pagination-block"><el-paginationbackgroundv-model:current-page="pageData.currentPage"v-model:page-size="pageData.pageSize":page-sizes="[3, 5, 10, 15, 20]"layout="->,total, sizes, prev, pager, next"v-model:total="pageData.total"/></div></el-card></div>
</template><style scoped>
.pagination-block {margin-top: 10px;
}
</style>

功能实现

向后端发送请求

工具类

返回一个对应链接和超时参数的axios对象

import axios from 'axios'
// import { useUserStore } from '@/stores'
// import { ElMessage } from 'element-plus'
// import router from '@/router'
const baseURL = '/api'const instance = axios.create({// TODO 1. 基础地址,超时时间baseURL: baseURL,timeout: 10000
})export default instance
export { baseURL }

数据列表请求

通过工具类的axios对象向后端对应资源链接发送get请求和传递参数,get方法的第二个参数传递一个config{}对象,params是其中的一个属性

import request from '@/utils/request'export const getCategoryList = (params) =>request.get('/home/category/head',{params:params});
在组件中定义方法对这个请求方法进行调用
const getDataList = async() => {loading.value = true;let result = await getCategoryList(param);    dataList.value = result.data.result.records;pageData.value.total = result.data.result.total;console.log(result);loading.value = false;
}
通过生命周期函数在组件对应生命周期进行挂载
onMounted(()=>{getDataList();
})

分页查询

在el-pagination组件中实现了属性和pageData的双向绑定

<!-- 分页插件 --><div class="pagination-block"><el-pagination background v-model:current-page="pageData.currentPage" v-model:page-size="pageData.pageSize":page-sizes="[3, 5, 10, 15, 20]" layout="->,total, sizes, prev, pager, next" v-model:total="pageData.total"@size-change="getDataList" @current-change="getDataList" /></div>

所以我们要从pageData中取数据

普通对象无法被二次修改,所以要使用一个响应式对象进行数据同步

const param = computed(() => {return {name: '',state: '',page: pageData.value.currentPage,pageSize: pageData.value.pageSize}}
);

最后因为在el-pagination组件中实现了

@size-change=“getDataList” @current-change=“getDataList”

事件和方法的绑定

所以调用

const getDataList = async () => {loading.value = true;let result = await getCategoryList(param.value);dataList.value = result.data.result.records;pageData.value.total = result.data.result.total;console.log(result);loading.value = false;
}

方法进行数据更新

搜索功能

分为搜索事件,重置事件,在输入框中删除事件,和在多选框中删除事件

搜索事件
const search = () => {//拿到父组件的搜索方法emits('search');}
<el-button type="primary" @click="search">搜索</el-button>
//绑定搜索方法
const param = computed(() => {return {name: searchParams.value.name,state: searchParams.value.state,page: pageData.value.currentPage,pageSize: pageData.value.pageSize}
}
);
//在父组件中绑定搜索的参数
const search = () => {pageData.value.currentPage = 1;getDataList();
}
//父组件中定义搜索方法
重置事件
 const reset = ()  => {//拿到父组件的重置方法emits('reset')}
<el-button @click="reset">重置</el-button>
//绑定重置按钮
const reset = () => {searchParams.value.name = '';searchParams.value.state = '';pageData.value.currentPage = 1;pageData.value.pageSize = 5;search();
}
//父组件中定义重置方法
搜索/下拉框重置
<el-input v-model="searchParams.name" placeholder="请输入分类名字称" clearable  @clear="search"/>
//input组件自带clear事件绑定search方法

图片上传

<el-form-item label="分类图片" prop="pictureImg"><el-uploadclass="avatar-uploader"action="/api/upload"//设置上传的资源链接name="pictureImg"//设置参数名字:show-file-list="false":on-success="handleAvatarSuccess":before-upload="beforeAvatarUpload">

新增分类/修改分类

export const  navEditCategoryService = (params) => request.put('/home/category/edit',{params});
//发起修改请求export const  navAddCategoryService = (params) => request.post('/home/category/add',params);
//发起新增请求

对请求方法进行调用

const onSubmit = async () => {await formRef.value.validate()const isEdit = formModel.value.idif (isEdit) {await navEditCategoryService(formModel.value)//修改数据ElMessage.success('编辑成功')} else {await navAddCategoryService(formModel.value)//新增数据ElMessage.success('添加成功')}dialogVisible.value = falseemit('success')
}

查询一级分类(在分类dialog中显示)

export const getCategoryParenList = () => request.get('/home/category/parent')
//发起查询请求
onMounted(async() =>{let result = await getCategoryParenList();parentCategory.value = result.data.result;
})
//通过钩子函数在组件加载完成后自动执行

删除分类(前端加后端)

export const deleteCategoryService = (id) => request.delete(`/home/category/del/${id}`)
//发起删除请求const onDelChannel = async (row) => { // 点击删除执行的方法await ElMessageBox.confirm('你确认要删除该分类么', '温馨提示', {type: 'warning',confirmButtonText: '确认',cancelButtonText: '取消'})//TODO 待实现console.log(row);deleteCategoryService(row.id);search();
}
//父组件中的删除方法<el-button :icon="Delete" circle plain type="danger" @click="onDelChannel(row, $index)"></el-button>
//绑定删除方法,将对应行的id传入
@DeleteMapping("/category/del/{id}")public R deleteCategory(@PathVariable String id){frontService.deleteCategory(id);return R.ok();}//控制层接收@Overridepublic void deleteCategory(String id) {frontMapper.delectCategory(id);}  //业务层处理@Delete("delete from classification_front where id = ${id}")void delectCategory(String id);//持久层和数据库交互

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/13140.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

APP封装后防止破解的全方位策略

移动应用开发完成后&#xff0c;封装&#xff08;编译打包&#xff09;是发布前的重要步骤。然而&#xff0c;一旦APP发布&#xff0c;就可能面临被逆向工程破解的风险&#xff0c;从而导致源代码泄露、数据被盗取等严重后果。 本文将介绍一系列实用的策略和技术&#xff0c;帮…

邦注科技 工业冷水机的风冷和水冷的区别介绍

工业冷水机在工业生产中扮演着重要角色&#xff0c;特别是在需要精确控制温度的应用中。风冷式冷水机和水冷式冷水机是两种常见的类型&#xff0c;它们之间存在一些显著的区别。 热交换的来源不同&#xff1a; 风冷式冷水机&#xff1a;热交换的来源是气体。它采用空气冷却方…

STL——vector容器【动态数组】

vector基本概念 功能&#xff1a;vector数据结构和数组非常相似&#xff0c;也成为单端数组 头文件&#xff1a;<vector> vector与普通数组的区别&#xff1a;不同之处在于数组是静态空间&#xff0c;而vector可以动态扩展 动态扩展&#xff1a; 并不是在原空间之后续…

代理IP与网络隐私

随着科技的发展&#xff0c;越来越多的人开始享受网络的便利&#xff0c;人们逐步在社交媒体上分享自己的生活。这些公开信息是可以被所有人浏览观看的&#xff0c;但是也会存在部分隐私信息我们并不想要被人知晓&#xff0c;这就牵扯到网络隐私保护问题。 代理IP对于网络隐私保…

链接表存储图(C++注释详解): 构建表 深度优先遍历 (DFS)

链接表的结构体单元: #define size 100 typedef struct node {int idx;//下一个节点的索引int wt;//权重, 也可根据实际情景存储边的信息struct node* next; }Node; Node* hd[size]; // 存储图的邻接表 链接表的的构建: int main() {int n, m;cin >> n >> m; //…

【2024系统架构设计】回顾历史,查缺补漏篇 ③

前言 hello,大家好: 💡💡💡 我们一起来备考软考高级系统架构设计师吧,本专栏提供综合知识、案例科目、论文(论点和部分示例范文)等内容,包括知识点总结和记忆小妙招哦。 🚀🚀🚀 可以减少资料查找和收集的时间,提高效率,我们一起集中精力学习干货吧! 💡…

iOS 创建pch文件

1.参考链接&#xff08;xcode8添加方法&#xff0c;之前的跟这个差不多&#xff09;&#xff1a; 参考链接 2.自我总结&#xff1a; &#xff08;1&#xff09;创建pch文件: 注意点&#xff1a;1&#xff09;注意选中所有的targets&#xff08;看图明义&#xff09; 2&…

Android系统不同版本存储权限

一、Android存储简介 Android系统分为内部存储和外部存储 从Android6.0开始不断在更新存储&#xff08;读写&#xff09;权限&#xff0c;除了在AndroidManifest.xml文件里声明&#xff0c;app运行时也要动态申请使用对应的权限 提醒&#xff1a;应用私有存储不需要动态申请权…

基于Java的飞机大战游戏的设计与实现(论文 + 源码)

关于基于Java的飞机大战游戏.zip资源-CSDN文库https://download.csdn.net/download/JW_559/89313362 基于Java的飞机大战游戏的设计与实现 摘 要 现如今&#xff0c;随着智能手机的兴起与普及&#xff0c;加上4G&#xff08;the 4th Generation mobile communication &#x…

欧拉公式的讲解

啊&#xff0c;哈喽&#xff0c;小伙伴们大家好。我是#张亿&#xff0c;今天呐&#xff0c;学的是欧拉公式 在不同的学科中有着不同的含义和应用。在复变函数中&#xff0c;欧拉公式表述为e^(ix)(cos xisin x)&#xff0c;其中e是自然对数的底&#xff0c;i是虚数单位&#x…

Java 自然排序和比较器排序区别?Comparable接口和Comparator比较器区别?

注&#xff1a;如果你对排序不理解&#xff0c;请您耐心看完&#xff0c;你一定会明白的。文章通俗易懂。建议用idea运行一下案例。 1&#xff09;自然排序和比较器排序的区别&#xff1f; 自然排序是对象本身定义的排序规则&#xff0c;由对象实现 Comparable 接口&#xff…

Tomcat无法连通的调试方法1-service方式无法连通

作者&#xff1a;私语茶馆 1.局域网Tomcat服务不通 组网如下&#xff1a; 问题&#xff1a; Tomcat Server 服务方式启动后&#xff0c;无法访问&#xff0c;但命令行方式启动可以。IP地址都在同网段或不同网段现象都一样。 2.Tomcat 服务安装与调试 在Windows下&#xff0c;…

计算机系列之排序算法

20、排序算法 1、直接插入排序&#xff08;这里以从小到大排序为例&#xff09; ◆要注意的是&#xff0c;前提条件是前i-1个元素是有序的&#xff0c;第i个元素依次从第i-1个元素往前比较&#xff0c;直到找到一个比第i个元素值小的元素&#xff0c;而后插入&#xff0c;插入…

表现层设计模式_1.MVC模式

1.MVC模式三个核心模块 MVC是一种目前广泛流行的软件设计模式。近年来&#xff0c;随着Java EE的成熟&#xff0c;MVC成为了Java EE平台上推荐的一种设计模式。MVC强制性地把一个应用的输入、处理、输出流程按照视图、控制、模型的方式进行分离&#xff0c;形成了控制器…

有关登录安全,测试人该知道些什么?

作为测试&#xff0c;给我们一个关键词“登录”&#xff0c;我们可能想到的用例设计更多的是什么用户名、密码校验是否合法、是否为空、是否正确等等之类的场景。 但在如今信息化的时代&#xff0c;“登录安全”已经是一个很热门且普遍的的话题了&#xff0c;今天给大家简单分…

基于国产LoRa的智慧农业解决方案--ASR6601、SX1278

我国《数字乡村发展战略纲要》明确指出“要推进农业数字化转型”&#xff0c;加快推广云计算、大数据、物联网、人工智能在农业生产经营管理中的运用。 然而&#xff0c;目前我国的农业数字化转型还面临着诸多挑战。我国整体农业机械化程度和自动化控制水平仍然较低。由于农田面…

Golang RPC实现-day02

导航 Golang RPC实现一、客户端异步并发多个请求1、 客户端结构体2、 一个客户端&#xff0c;异步发送多个请求&#xff0c;使用call结构体代表客户端的每次请求3、客户端并发多个请求4、客户端接收请求 Golang RPC实现 day01 我们实现了简单的服务端和客户端。我们简单总结一…

蓝桥杯-外卖店优先级(简单写法)

“饱了么”外卖系统中维护着 N 家外卖店&#xff0c;编号 1∼N。 每家外卖店都有一个优先级&#xff0c;初始时 (0 时刻) 优先级都为 0。 每经过 1 个时间单位&#xff0c;如果外卖店没有订单&#xff0c;则优先级会减少 1&#xff0c;最低减到 0&#xff1b;而如果外卖店有订…

【数据结构】双向循环链表专题解析

实现自己既定的目标&#xff0c;必须能耐得住寂寞单干。&#x1f493;&#x1f493;&#x1f493; 目录 •✨说在前面 &#x1f34b;知识点一&#xff1a;双向链表的结构 • &#x1f330;1."哨兵位"节点 • &#x1f330;2.双向带头循环链表的结构 &#x1f34b;…

基于java的超级玛丽游戏的设计与实现(论文 + 源码)

Java的超级玛丽游戏.zip资源-CSDN文库https://download.csdn.net/download/JW_559/89313347 基于java的超级玛丽游戏的设计与实现 摘要 近年来&#xff0c;Java作为一种新的编程语言&#xff0c;以其简单性、可移植性和平台无关性等优点&#xff0c;得到了广泛地应用。J2SE称…