人力资源管理后台 === 左树右表

1.角色管理-编辑角色-进入行内编辑

image.png

image.png

  • 获取数据之后针对每个数据定义标识-使用$set-代码位置(src/views/role/index.vue)
 // 针对每一行数据添加一个编辑标记this.list.forEach(item => {// item.isEdit = false // 添加一个属性 初始值为false// 数据响应式的问题  数据变化 视图更新// 添加的动态属性 不具备响应式特点// this.$set(目标对象, 属性名称, 初始值) 可以针对目标对象 添加的属性 添加响应式this.$set(item, 'isEdit', false)})

为什么不使用item.isEdit = false , 因为动态添加的属性不具备响应式的特点,如果想要具备响应式,可以使用$set

  • 点击编辑时,将当前行的标记isEdit设置为true-代码位置(src/views/role/index.vue)
<el-table-column align="center" label="操作"><template><!-- 非编辑状态 --><el-button size="mini" type="text">分配权限</el-button><el-button size="mini" type="text" @click="btnEditRow(row)">编辑</el-button><el-button size="mini" type="text">删除</el-button></template>
</el-table-column>
  • 点击编辑的方法-代码位置(src/views/role/index.vue)
    // 点击编辑行btnEditRow(row) {row.isEdit = true // 改变行的编辑状态}
  • 表格列中根据当前的isEdit标记-渲染结构-代码位置(src/views/role/index.vue)
<el-table-column prop="name" align="center" width="200" label="角色"><template v-slot="{ row }"><!-- 条件判断 --><el-input v-if="row.isEdit" size="mini" /><span v-else>{{ row.name }}</span></template>
</el-table-column>
<el-table-column prop="state" align="center" width="200" label="启用"><!-- 自定义列结构 --><template v-slot="{ row }"><el-switch v-if="row.isEdit" /><span v-else>  {{ row.state === 1 ? "已启用" : row.state === 0 ? "未启用" : "无" }} </span></template>
</el-table-column>
<el-table-column prop="description" align="center" label="描述"><template v-slot="{ row }"><el-input v-if="row.isEdit" type="textarea" /><span v-else>{{ row.description }}</span></template>
</el-table-column><el-table-column align="center" label="操作"><template v-slot="{ row }"><template v-if="row.isEdit"><!-- 编辑状态 --><el-button type="primary" size="mini">确定</el-button><el-button size="mini">取消</el-button></template><template v-else><!-- 非编辑状态 --><el-button size="mini" type="text">分配权限</el-button><el-button size="mini" type="text" @click="btnEditRow(row)">编辑</el-button><el-button size="mini" type="text">删除</el-button></template></template>
</el-table-column>

$set的应用

  • this.$set(目标对象, 属性名称, 初始值 )
  • 等价于 Vue.set(目标对象, 属性名称, 初始值)
  • 向响应式对象中添加一个属性,并确保这个新属性同样是响应式的,且触发视图更新。

2.角色管理-行内编辑-数据缓存

为什么要做数据缓存?

答: 因为编辑时,可以取消会滚到之前的状态,所以编辑时的数据是临时的数据。

image.png


如图,editRow的数据是针对当前行的数据做了一份拷贝,针对这个拷贝,我们可以随意修改。

  • 初始化时缓存数据-代码位置(src/views/role/index.vue)
    this.list.forEach(item => {// item.isEdit = false // 添加一个属性 初始值为false// 数据响应式的问题  数据变化 视图更新// 添加的动态属性 不具备响应式特点// this.$set(目标对象, 属性名称, 初始值) 可以针对目标对象 添加的属性 添加响应式this.$set(item, 'isEdit', false)this.$set(item, 'editRow', {name: item.name,state: item.state,description: item.description})})
  • 点击编辑时更新缓存数据-代码位置(src/views/role/index.vue)
btnEditRow(row) {row.isEdit = true // 改变行的编辑状态// 更新缓存数据row.editRow.name = row.namerow.editRow.state = row.staterow.editRow.description = row.description
}
  • 将编辑时的表单双向绑定缓存数据-代码位置(src/views/role/index.vue)
        <el-table-column prop="name" align="center" width="200" label="角色"><template v-slot="{ row }"><!-- 条件判断 --><el-input v-if="row.isEdit" v-model="row.editRow.name" size="mini" /><span v-else>{{ row.name }}</span></template></el-table-column><el-table-column prop="state" align="center" width="200" label="启用"><!-- 自定义列结构 --><template v-slot="{ row }"><!-- 开 1 关 0 --><el-switch v-if="row.isEdit" v-model="row.editRow.state" :active-value="1" :inactive-value="0" /><span v-else>  {{ row.state === 1 ? "已启用" : row.state === 0 ? "未启用" : "无" }} </span></template></el-table-column><el-table-column prop="description" align="center" label="描述"><template v-slot="{ row }"><el-input v-if="row.isEdit" v-model="row.editRow.description" size="mini" type="textarea" /><span v-else>{{ row.description }}</span></template></el-table-column>

3.角色管理-编辑角色-确定取消

image.png

  • 封装更新角色信息的API-代码位置(src/api/role.js)
/*** 更新角色* ***/export function updateRole(data) {return request({url: `/sys/role/${data.id}`,method: 'put',data})
}
  • 确定方法-代码位置(src/views/role/index.vue)
<el-button type="primary" size="mini" @click="btnEditOK(row)">确定</el-button>// 点击确定时触发async  btnEditOK(row) {if (row.editRow.name && row.editRow.description) {// 下一步操作await updateRole({ ...row.editRow, id: row.id })// 更新成功this.$message.success('更新角色成功')// 更新显示数据  退出编辑状态// row.name = row.editRow.name // eslint的一校验 误判// Object.assign(target, source)Object.assign(row, {...row.editRow,isEdit: false // 退出编辑模式}) // 规避eslint的误判} else {this.$message.warning('角色和描述不能为空')}}

注意: 这里既然更新成功了,要把缓存数据回显到页面上,并且关闭编辑模式

  • 取消方法-代码位置(src/views/role/index.vue)
<el-button size="mini" @click="row.isEdit = false">取消</el-button>

4.角色管理-删除角色

image.png

image.png

  • 封装删除角色的API-代码位置(src/api/role.js)
/** ** 删除角色* **/export function delRole(id) {return request({url: `/sys/role/${id}`,method: 'delete'})
}
  • 在删除位置加入气泡框-代码位置(src/views/role/index.vue)
<el-popconfirm title="这是一段内容确定删除吗?" @onConfirm="confirmDel(row.id)" ><el-button slot="reference" style="margin-left:10px" size="mini" type="text">删除</el-button></el-popconfirm>

注意:这里element-ui文档有点问题,el-popconfirm的确认事件是onConfirm,但是文档上写的是confirm事件,这里需要注意

  • 删除方法-代码位置(src/views/role/index.vue)
 async  confirmDel(id) {await delRole(id) // 后端删除this.$message.success('删除角色成功')// 删除的如果是最后一个if (this.list.length === 1) this.pageParams.page--this.getRoleList()
}

5.员工管理-页面结构

image.png

image.png

  • 基本的页面结构-代码位置(src/views/employee/index.vue)
<template><div class="container"><div class="app-container"><div class="left"><el-input style="margin-bottom:10px" type="text" prefix-icon="el-icon-search" size="small" placeholder="输入员工姓名全员搜索" /><!-- 树形组件 --></div><div class="right"><el-row class="opeate-tools" type="flex" justify="end"><el-button size="mini" type="primary">添加员工</el-button><el-button size="mini">excel导入</el-button><el-button size="mini">excel导出</el-button></el-row><!-- 表格组件 --><!-- 分页 --></div></div></div>
</template><script>
export default {name: 'Employee'
}
</script><style lang="scss" scoped>
.app-container {background: #fff;display: flex;.left {width: 280px;padding: 20px;border-right: 1px solid #eaeef4;}.right {flex: 1;padding: 20px;.opeate-tools {margin:10px ;}.username {height: 30px;width: 30px;line-height: 30px;text-align: center;border-radius: 50%;color: #fff;background: #04C9BE;font-size: 12px;display:inline-block;}}
}</style>

6.员工管理-左侧树的加载

image.png

  • 定义树组件需要的数据-代码位置(src/views/employee/index.vue)
data() {return {depts: [], // 组织数据defaultProps: {label: 'name',children: 'children'}}
}
  • 放置树形组件-代码位置(src/views/employee/index.vue)
<el-tree:data="depts":props="defaultProps"default-expand-all:expand-on-click-node="false"highlight-current/>
  • 初始化时加载数据转化树形-代码位置(src/views/employee/index.vue)
  created() {this.getDepartment()},methods: {async getDepartment() {// 递归方法 将列表转化成树形// let result = await getDepartment()this.depts = transListToTreeData(await getDepartment(), 0)}}

7.员工管理-选中首个节点

image.png

image.png

  • 在data中声明一个记录id的参数-代码位置(src/views/employee/index.vue)
data () {return  {// 存储查询参数queryParams: {departmentId: null}}
}

为什么要放在queryParams中,因为后面的查询会有很多查询条件,到时候查询条件都会聚合到一起,所以使用一个公共的对象来管理更方便和合适

  • 初始化时首个id节点,并且选中-代码位置(src/views/employee/index.vue)
methods: {async getDepartment() {// 递归方法 将列表转化成树形// let result = await getDepartment()this.depts = transListToTreeData(await getDepartment(), 0)this.queryParams.departmentId = this.depts[0].id// 设置选中节点// 树组件渲染是异步的 等到更新完毕this.$nextTick(() => {// 此时意味着树渲染完毕this.$refs.deptTree.setCurrentKey(this.queryParams.departmentId)})}}

为什么使用��������,因为我们设置完树形之后立刻选中首个节点,此时更新还没有完成,必须等待更新完成后,再去选中首个节点,所以需要使用nextTick,因为我们设置完树形之后立刻选中首个节点,此时更新还没有完成,必须等待更新完成后,再去选中首个节点,所以需要使用nextTick

  • 监听树组件的节点切换事件-代码位置(src/views/employee/index.vue)
 <el-treeref="deptTree"node-key="id":data="depts":props="defaultProps"default-expand-all:expand-on-click-node="false"highlight-current@current-change="selectNode"/>

这里需要明白,需要给定node-key属性,否则setCurrentKey方法不知道设置的是哪个字段的值

  • 切换节点时再次记录id-代码位置(src/views/employee/index.vue)
selectNode(node) {this.queryParams.departmentId = node.id
}

8.员工管理-员工列表结构

image.png

image.png

  • 右侧列表页面结构-代码位置(src/views/employee/index.vue)
<el-table><el-table-column align="center" label="头像" /><el-table-column label="姓名" /><el-table-column label="手机号" sortable /><el-table-column label="工号" sortable /><el-table-column label="聘用形式" /><el-table-column label="部门" /><el-table-column label="入职时间" sortable /><el-table-column label="操作" width="280px"><template><el-button size="mini" type="text">查看</el-button><el-button size="mini" type="text">角色</el-button><el-button size="mini" type="text">删除</el-button></template></el-table-column>
</el-table>
<!-- 分页 -->
<el-row style="height: 60px" align="middle" type="flex" justify="end"><el-paginationlayout="total,prev, pager, next":total="1000"/>
</el-row>

9.员工管理-获取员工数据

image.png

image.png

第一次加载之后 或者是切换节点之后,都要去根据点击的节点去查询员工的数据

  • 封装获取员工数据的API-代码位置(src/api/employee.js)
export function getEmployeeList(params) {return request({url: '/sys/user',params // 地址参数 查询参数})
}
  • 声明一个list数据进行接受list-代码位置(src/views/employee/index.vue)
data () {return {list: []}
}
  • 初始化后获取右侧表格数据-代码位置(src/views/employee/index.vue)
methods: {async getDepartment() {// 递归方法 将列表转化成树形// let result = await getDepartment()this.depts = transListToTreeData(await getDepartment(), 0)this.queryParams.departmentId = this.depts[0].id// 设置选中节点// 树组件渲染是异步的 等到更新完毕this.$nextTick(() => {// 此时意味着树渲染完毕this.$refs.deptTree.setCurrentKey(this.queryParams.departmentId)})// 这个时候参数 记录了idthis.getEmployeeList()},// 获取员工列表的方法async getEmployeeList() {const { rows } = await getEmployeeList(this.queryParams)this.list = rows}}
  • 切换节点时,获取右侧表格数据-代码位置(src/views/employee/index.vue)
selectNode(node) {this.queryParams.departmentId = node.id // 重新记录了参数this.getEmployeeList()
},
  • 绑定表格-代码位置(src/views/employee/index.vue)
<el-table :data="list"><el-table-column prop="staffPhoto" align="center" label="头像" /><el-table-column prop="username" label="姓名" /><el-table-column prop="mobile" label="手机号" sortable /><el-table-column prop="workNumber" label="工号" sortable /><el-table-column prop="formOfEmployment" label="聘用形式" /><el-table-column prop="departmentName" label="部门" /><el-table-column prop="timeOfEntry" label="入职时间" sortable />...
</el-table>

10.员工管理-头像和聘用形式的处理

image.png

image.png

  • 头像的设置-代码位置(src/views/employee/index.vue)
 <el-table-column prop="staffPhoto" align="center" label="头像"><template v-slot="{ row }"><el-avatar v-if="row.staffPhoto" :src="row.staffPhoto" :size="30" /><span v-else class="username">{{ row.username?.charAt(0) }}</span></template>
</el-table-column>
  • 聘用形式的处理-代码位置(src/views/employee/index.vue)
<el-table-column prop="formOfEmployment" label="聘用形式"><template v-slot="{ row }"><span v-if="row.formOfEmployment === 1">正式</span><span v-else-if="row.formOfEmployment === 2">非正式</span><span v-else>无</span></template>
</el-table-column>

11.员工管理-员工分页处理

image.png

image.png

  • 定义分页参数-代码位置(src/views/employee/index.vue)
data () {return {queryParams: {departmentId: null,page: 1, // 当前页码pagesize: 10},total: 0, // 记录员工的总数}
}
  • 绑定分页参数-代码位置(src/views/employee/index.vue)
<el-paginationlayout="total,prev, pager, next":total="total":current-page="queryParams.page":page-size="queryParams.pagesize"@current-change="changePage"/>
  • 切换页码事件-代码位置(src/views/employee/index.vue)
// 切换页码changePage(newPage) {this.queryParams.page = newPage // 赋值新页码this.getEmployeeList() // 查询数据}
  • 切换部门时,查询第一页数据-代码位置(src/views/employee/index.vue)
selectNode(node) {this.queryParams.departmentId = node.id // 重新记录了参数this.queryParams.page = 1 // 设置第一页this.getEmployeeList()
},
  • 设置ElementUI的语言为中文-代码位置(src/main.js)
Vue.use(ElementUI)

12.员工管理-员工模糊搜索

image.png

image.png

  • 设置模糊搜索的参数字段-代码位置(src/views/employee/index.vue)
data () {return {queryParams: {departmentId: null,page: 1, // 当前页码pagesize: 10,keyword: '' // 模糊搜索字段},}
}
  • 双向绑定input输入框,监听值改变事件-代码位置(src/views/employee/index.vue)
<el-inputv-model="queryParams.keyword"style="margin-bottom:10px"type="text"prefix-icon="el-icon-search"size="small"placeholder="输入员工姓名全员搜索"@input="changeValue"
/>

这里监听的事件是input,有同学可能会问,为什么不用change事件,注意change事件是离开焦点触发,input是只要内容发生变化就会触发,所以这里使用input更符合使用场景

  • 值改变查询数据-支持防抖查询-代码位置(src/views/employee/index.vue)
 changeValue() {// 单位时间内只执行最后一次// this的实例上赋值了一个timer的属性clearTimeout(this.timer) // 清理上一次的定时器this.timer = setTimeout(() => {this.queryParams.page = 1this.getEmployeeList()}, 300)
}

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

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

相关文章

〔005〕虚幻 UE5 像素流多用户部署

✨ 目录 ▷ 为什么要部署多用户▷ 开启分发服务器▷ 配置启动多个信令服务器▷配置启动客户端▷多用户启动整体流程和预览▷注意事项▷ 为什么要部署多用户 之前的像素流部署,属于单用户,是有很大的弊端的打开多个窗口访问,可以看到当一个用户操作界面的时候,另一个界面也会…

为社会做贡献的EasyDarwin 4.0.1发布了,支持视频点播、文件直播、摄像机直播、直播录像、直播回放、录像MP4合成下载

经过几个月的不懈努力和测试&#xff0c;最新的EasyDarwin 4.0版本总算是发布出来了&#xff0c;功能还是老几样&#xff1a;文件点播、视频直播&#xff08;支持各种视频源&#xff09;、直播录像与回放、录像合成MP4下载&#xff0c;稍稍看一下细节&#xff1a; 文件上传与点…

cmdline

cmdline是一个kv结构,就是uboot参数传给kernel使用的 举例: Kernel command line: user_debug=31 storagemedia=mtd androidboot.storagemedia=mtd androidboot.mode=normal mac=00FA89112233 serial=LONBON12345 earlycon=uart8250,mmio32,0xff570000 console=ttyFIQ0…

PostgreSQL | EXTRACT | 获取时间的年月日字串

EXTRACT EXTRACT 函数是 PostgreSQL 中用于从日期和时间类型中提取特定部分&#xff08;如年、月、日、小时等&#xff09;的函数。 格式 EXTRACT(field FROM source) -- field 参数是要提取的部分&#xff0c;例如 YEAR、MONTH、DAY、HOUR 等。 -- source 参数是包含日期或…

Redis:持久化RDB和AOF

目录 概述RDB持久化流程指定备份文件的名称指定备份文件存放的目录触发RDB备份redis.conf 其他一些配置rdb的备份和恢复优缺点停止RDB AOF持久化流程AOF启动/修复/恢复AOF同步频率设置rewrite压缩原理触发机制重写流程no-appendfsync-on-rewrite 优缺点 如何选择 概述 Redis是…

带submodule的git仓库自动化一键git push、git pull脚本

前言 很久没写博客了&#xff0c;今天难得闲下来写一次。 不知道大家在使用git的时候有没有遇到过这样的问题&#xff1a;发现git submodule特别好用&#xff0c;适合用于满足同时开发和部署的需求&#xff0c;并且结构清晰&#xff0c;方便我们对整个代码层次有一个大概的了…

数据库第十第十一章 恢复和并发简答题

数据库第一章 概论简答题 数据库第二章 关系数据库简答题 数据库第三章 SQL简答题 数据库第四第五章 安全性和完整性简答题 数据库第七章 数据库设计简答题 数据库第九章 查询处理和优化简答题 1.什么是数据库中的事务&#xff1f;它有哪些特性&#xff1f;这些特性的含义是什么…

函数指针数组指针数组传参的本质字符指针

&#x1f680; 作者&#xff1a;阿辉不一般 &#x1f680; 你说呢&#xff1a;不服输的你&#xff0c;他们拿什么赢 &#x1f680; 专栏&#xff1a;爱上C语言 &#x1f680;作图工具&#xff1a;draw.io(免费开源的作图网站) 如果觉得文章对你有帮助的话&#xff0c;还请点赞…

XSTRING与STRING之间的互转,base64,长文本,科学计数法

XSTRING的介绍 SAP ABAP 理解RAWSTRING(XSTRING) 类型-腾讯云开发者社区-腾讯云 XString,String以及SString 类型区别 | 摆渡SAP SAP ABAP 理解RAWSTRING(XSTRING) 类型 RAWSTRING 和 STRING 类型具有可变长度。可以指定这些类型的最大长度&#xff0c;但没有上限。 SSTRI…

代码浅析DLIO(二)---预积分与单点去畸变

0. 简介 我们刚刚了解过DLIO的整个流程&#xff0c;我们发现相比于Point-LIO而言&#xff0c;这个方法更适合我们去学习理解&#xff0c;同时官方给出的结果来看DLIO的结果明显好于现在的主流方法&#xff0c;当然指的一提的是&#xff0c;这个DLIO是必须需要六轴IMU的&#x…

【ZYNQ】SD 卡读写及文件扫描实验

SD 卡控制器&#xff08;SD/SDIO Controller&#xff09; ZYNQ 中的 SD 卡控制器符合 SD2.0 协议规范&#xff0c;接口兼容 eMMC、MMC3.31、SDIO2.0、SD2.0、SPI&#xff0c;支持 SDHC、SDHS 器件。SD 卡控制器支持 SDMA&#xff08;单操作 DMA&#xff09;、ADMA1&#xff08…

数据结构-顺序表

文章目录 线性表概念顺序表静态顺序表动态顺序表 总结 线性表概念 线性表是最基本、最简单、也是最常用的一种数据结构&#xff0c;常见的线性表:顺序表、链表、栈、队列、字符串…线性表&#xff08;linear> list&#xff09;是数据结构的一种&#xff0c;一个线性表是n个具…

蚂蚁庄园小课堂答题今日答案最新

蚂蚁庄园小课堂答题今日答案最新 温馨提醒&#xff1a;由于本文章会停留在一个固定的更新时间上&#xff0c;包含当前日期最新的支付宝蚂蚁庄园小课堂答题今日答案。如果您看到这篇文章已成为过去时&#xff0c;请按下面的方法进入查看天天保持更新的最新今日答案&#xff1b; …

Linux 网络通信

(一)套接字Socket概念 Socket 中文意思是“插座”&#xff0c;在 Linux 环境下&#xff0c;用于表示进程 x 间网络通信的特殊文件 类型。本质为内核借助缓冲区形成的伪文件。 既然是文件&#xff0c;那么理所当然的&#xff0c;我们可以使用文件描述符引用套接字。Linux 系统…

Windows11安装后跳过联网登录

Windows11安装后跳过联网登录 实验设备&#xff1a; VMware17Pro虚拟机中使用Windows11镜像安装Windows11操作系统&#xff0c;并且在虚拟机中测试跳过联网登录。 步骤 说明&#xff1a;物理卸载网卡&#xff08;在虚拟机上禁用网卡&#xff09;没用 思路&#xff1a; sh…

8.统一异常处理 + 统一记录日志

目录 1.统一异常处理 2.统一记录日志 1.统一异常处理 在 HomeController 类中添加请求方法&#xff08;服务器发生异常之后需要统一处理异常&#xff0c;记录日志&#xff0c;然后转到 500 页面&#xff0c;需要人工处理重定向到 500 页面&#xff0c;提前把 500 页面请求访问…

经典神经网络——AlexNet模型论文详解及代码复现

一、背景 AlexNet是在2012年由Alex Krizhevsky等人提出的&#xff0c;该网络在2012年的ImageNet大赛上夺得了冠军&#xff0c;并且错误率比第二名高了很多。Alexnet共有8层结构&#xff0c;前5层为卷积层&#xff0c;后三层为全连接层。 论文地址&#xff1a;ImageNet Classif…

ModuleNotFoundError: No module named ‘mdtex2html‘ module已经安装还是报错,怎么办?

用streamlit运行ChatGLM/basic_model/web_demo.py的时候&#xff0c;出现了module not found&#xff1a; ModuleNotFoundError: No module named mdtex2html Traceback: File "/home/haiyue/.local/lib/python3.10/site-packages/streamlit/runtime/scriptrunner/script…

【阿里云】图像识别 智能分类识别 增加网络控制功能点(三)

一、增加网络控制功能 实现需求TCP 心跳机制解决Soket异常断开问题 二、Linux内核提供了通过sysctl命令查看和配置TCP KeepAlive参数的方法。 查看当前系统的TCP KeepAlive参数修改TCP KeepAlive参数 三、C语言实现TCP KeepAlive功能 四、setsockopt用于设置套接字选项的系…

Qt4利用MVC开发曲线数据编辑器

目录 1 需求 2 开发流程 1 搭建框架 2 构造函数 3 打开工程 4 实现应用程序参数加载 5 QCustomPlot和TableView的联动 6 数据的可视化修改 7 列表点击事件事先键盘控制 8 表格实现复制&#xff0c;粘贴&#xff0c;删除等一系列功能 9 曲线实现自适应范围和统一范围…