ElementUI table+dialog实现一个简单的可编辑的表格
table组件如何实现可编辑呢?
我的需求是把table组件那样的表格,实现它点击可以弹出一个框,然后在这个框里面输入你的东西,然后将他回显回去,当然,输入的有可能是时间啥的。
为什么要弹出弹层不在框上直接编辑呢?因为框太小了,用户可能看不见输入的是啥。
我们可以利用element组件的插槽,获取数值,然后回显到表格里。
HTML部分
我用了一个弹层来写这个功能,如果测试效果,可以直接在主页面引入 ,将弹层的控制开关改为true。
<template><el-drawer v-model="drawerVisible" size="calc(100% - 210px)" @close="resetForm"><el-container><el-form ref="ruleFormRef2" label-position="right" :model="form" label-width="138px"><div class="form-left"><div class="sepcial_about"><div class="tableForm"><el-table :data="testDatas" border style="width: 700px"><el-table-column v-for="(col, idx) in columnList" :key="col.prop" :index="idx" :width="idx === 1 ? '180px' : ''"><template #header><p>{{ col.label }}</p></template><template #default="{ row }"><p :class="col.prop == 'remark' ? 'p_remark' : 'p1111'" @click="e => handleEdit(0, row, col)">{{ col.prop == "time" ? changeTime(row[col.prop]) : row[col.prop] }}</p></template></el-table-column><el-table-column width="120"><template #header><p class="p_specail">操作</p></template><template #default="scope"><p class="p1111"><el-button link type="primary" size="large" @click.prevent="delRow(scope, 0)">删除</el-button></p></template></el-table-column></el-table><el-button type="primary" class="mt-4" style="width: 100%" @click="addRow(0)">新增</el-button></div></div></div></el-form><el-dialog v-model="dialogVisible" title="添加" width="30%" center align-center><span>{{ sureMsg }}</span><template #footer><span class="dialog-footer"><el-button type="primary" @click="toSure">确 认</el-button></span></template></el-dialog><el-dialog v-model="addDialog" :title="addTitle" width="30%" center align-center><el-date-picker v-if="isTime" v-model="addinput" type="daterange" format="YYYY/MM/DD" value-format="YYYY-MM-DD" /><el-inputv-if="isNum"resize="none"maxlength="15"show-word-limitv-model="addinput"type="textarea"size="large"placeholder="请输入"/><el-inputv-if="!isNum && !isTime"resize="none"maxlength="500"show-word-limitv-model="addinput"type="textarea"size="large"placeholder="请输入"/><template #footer><span class="dialog-footer"><el-button type="primary" @click="sava">保存</el-button></span></template></el-dialog></el-container></el-drawer>
</template>
JS
1.渲染数据集合
自定义的表头,HTML结构中通过v-for 循环表头列表,进行label的取值来构建最基础的表格
// 表格之中的数据
const testDatas: any = ref([]);// 表头列表
const columnList: any = ref([{ prop: "projectName", label: "测试1" },{ prop: "time", label: "时间段" },{ prop: "projectContentSummary", label: "测试2" }
]);
2.handleEdit函数
这个函数主要是用来控制弹层的弹出,通过行的标签名字判断需要什么输入的值。根据whatSava的值来判断你点击的是哪个表格,这个参数在HTML结构写死,cellRow和cellCol就是在渲染的时候带入的行的值和列的值。可以获得标签名字或者prop字段用来做判断。当然,这些名字也可以用来设置弹出弹框的弹框名字。
然后判断prop的值来看是不是需要输入 或者是 时间段格式的,让各自的开关变为true或者false
并且根据cellRow[cellCol.prop]这个值来对弹出来的框进行赋值,也就是回显。
const addinput = ref("");
const addTitle = ref("");
const addCode = ref();
const addProp = ref();
const isTime = ref(false);
const savaWhat = ref();
const isNum = ref(false);
const handleEdit = (whatSava: any, cellRow: any, cellCol: any) => {if (whatSava == "0") {savaWhat.value = 0;}if (cellCol.prop == "ceshi1" || cellCol.prop == "ceshi2") {isTime.value = false;if (cellCol.prop == "ceshi2") {isNum.value = true;} else {isNum.value = false;}addTitle.value = cellCol.label;addProp.value = cellCol.prop;addCode.value = cellRow.index;addinput.value = cellRow[cellCol.prop];addDialog.value = true;} else {isTime.value = true;addTitle.value = cellCol.label;addProp.value = cellCol.prop;addCode.value = cellRow.index;addinput.value = cellRow[cellCol.prop];addDialog.value = true;}
};
3.保存弹框函数
这个函数就是弹出层的保存函数,先找到你点击的是哪个框,然后把弹框里面输入的东西赋值给他,如果是数字的话,就用正则限制一下,不让他输入数字以外的东西。最后要清空一下,防止下次回显。
const sava = () => {let idx = testDatas.value.findIndex((p: any) => p.index == addCode.value);if (addProp.value == "ceshi2") {if (!/^[0-9]*$/.test(addinput.value)) {ElMessage.warning("请输入数字");return;}testDatas.value[idx][addProp.value] = addinput.value;} else {testDatas.value[idx][addProp.value] = addinput.value;}addinput.value = "";addDialog.value = false;
};
4.清洗时间格式函数
因为我们日期时间选择器,得到的是一个数组,所以我们回显的时候,应该拿数组的第一项和第二项,然后拼接起来,得到我们回显的正确格式。
//时间格式清洗
const changeTime = (data: any) => {if (!data) {return "";} else {let data1 = data[0] + "~" + data[1];return data1;}
};
5.添加函数
只需要我们知道添加的是哪个表格的,单表格就可以忽略whatAdd这个参数。
逻辑就是先判断他是不是为空,为空的话给他单纯的加一行,不为空的话,得到数组的长度,给他加一个index,我在写的过程中,发现我们没有办法获得一个标识,所以我写了一个index来当作唯一标识,当然,这个在获取后端接口的时候,需要你获取后端给你数组的长度,然后再依次相加就不会导致错误的出现。
// 添加一行
const addRow = (whatAdd: number) => {if (whatAdd == 0) {let obj: any = {};if (testDatas.value.length == 0) {obj.index = "0";columnList.value.forEach((p: any) => {obj[p.prop] = "";});testDatas.value[0] = obj;} else {const weiyi = testDatas.value[testDatas.value.length - 1].index;columnList.value.forEach((p: any) => {obj[p.prop] = "";});obj.index = String(Number(weiyi) + 1);testDatas.value[testDatas.value.length] = obj;}}
};
6.删除函数
因为删除标签在table表格之中,所以我们可以用scope准确的获得是哪一行,然后用whatDel判断你删除的是哪个表格。
//删除
const delRow = (scope: any, whatDel: number) => {const arrIndex = scope.$index; //唯一标识//将testDatas中的第arrIndex项数据删除if (whatDel == 0) {testDatas.value.splice(arrIndex, 1);}
};
CSS
<style scoped lang="scss">
:deep(.el-textarea__inner) {height: 200px;
}
:deep(.cell) {text-align: center !important;height: 45px;padding: 0;
}
.p1111 {height: 50px;line-height: 50px !important;overflow: hidden;white-space: nowrap;text-overflow: ellipsis;padding: 0;margin: 0;text-align: center !important;
}.tableForm {width: 700px;margin-left: 35px;margin-top: 20px;
}
.sepcial_about {width: 500px;padding-left: 40px;box-sizing: border-box;margin-bottom: 60px;position: relative;> :first-child {width: 1000px;margin-bottom: 10px;}
}
.sepcial_about .mt-4 {position: absolute;width: 40px !important;height: 30px !important;bottom: 10px;left: 800px;z-index: 999;
}
.sepcial_about :deep(.el-table) {display: block;height: 100%;
}
:deep(.el-date-editor) {width: 100% !important;height: 50px;text-align: center;
}.form-left {padding-top: 20px;
}:deep(.el-form-item__content) {/* margin-top: 20px; */margin-left: 40px !important;
}:deep(.el-form-item__label) {font-size: 16px;color: black;
}
</style>
全部代码
<template><el-drawer v-model="drawerVisible" size="calc(100% - 210px)" @close="resetForm"><el-container><el-form ref="ruleFormRef2" label-position="right" :model="form" label-width="138px"><div class="form-left"><div class="sepcial_about"><div class="tableForm"><el-table :data="testDatas" border style="width: 700px"><el-table-column v-for="(col, idx) in columnList" :key="col.prop" :index="idx" :width="idx === 1 ? '180px' : ''"><template #header><p>{{ col.label }}</p></template><template #default="{ row }"><p :class="col.prop == 'remark' ? 'p_remark' : 'p1111'" @click="e => handleEdit(0, row, col)">{{ col.prop == "time" ? changeTime(row[col.prop]) : row[col.prop] }}</p></template></el-table-column><el-table-column width="120"><template #header><p class="p_specail">操作</p></template><template #default="scope"><p class="p1111"><el-button link type="primary" size="large" @click.prevent="delRow(scope, 0)">删除</el-button></p></template></el-table-column></el-table><el-button type="primary" class="mt-4" style="width: 100%" @click="addRow(0)">新增</el-button></div></div></div></el-form><el-dialog v-model="addDialog" :title="addTitle" width="30%" center align-center><el-date-picker v-if="isTime" v-model="addinput" type="daterange" format="YYYY/MM/DD" value-format="YYYY-MM-DD" /><el-inputv-if="isNum"resize="none"maxlength="15"show-word-limitv-model="addinput"type="textarea"size="large"placeholder="请输入"/><el-inputv-if="!isNum && !isTime"resize="none"maxlength="500"show-word-limitv-model="addinput"type="textarea"size="large"placeholder="请输入"/><template #footer><span class="dialog-footer"><el-button type="primary" @click="sava">保存</el-button></span></template></el-dialog></el-container></el-drawer>
</template><script setup lang="tsx" name="addConfig">
import { ref, reactive } from "vue";
import { ElMessage } from "element-plus";
const drawerVisible = ref(false);
const addDialog = ref(false);
const addinput = ref("");
const testDatas: any = ref([]);
const columnList: any = ref([{ prop: "ceshi1", label: "测试1" },{ prop: "time", label: "时间段" },{ prop: "ceshi2", label: "测试2" }
]);
const addTitle = ref("");
const addCode = ref();
const addProp = ref();
const isTime = ref(false);
const savaWhat = ref();
const isNum = ref(false);
const handleEdit = (whatSava: any, cellRow: any, cellCol: any) => {if (whatSava == "0") {savaWhat.value = 0;}if (cellCol.prop == "ceshi1" || cellCol.prop == "ceshi2") {isTime.value = false;if (cellCol.prop == "ceshi2") {isNum.value = true;} else {isNum.value = false;}addTitle.value = cellCol.label;addProp.value = cellCol.prop;addCode.value = cellRow.index;addinput.value = cellRow[cellCol.prop];addDialog.value = true;} else {isTime.value = true;addTitle.value = cellCol.label;addProp.value = cellCol.prop;addCode.value = cellRow.index;addinput.value = cellRow[cellCol.prop];addDialog.value = true;}
};
const sava = () => {let idx = testDatas.value.findIndex((p: any) => p.index == addCode.value);if (addProp.value == "ceshi2") {if (!/^[0-9]*$/.test(addinput.value)) {ElMessage.warning("请输入数字");return;}testDatas.value[idx][addProp.value] = addinput.value;} else {testDatas.value[idx][addProp.value] = addinput.value;}addinput.value = "";addDialog.value = false;
};
let form: any = reactive({});
//时间格式清洗
const changeTime = (data: any) => {if (!data) {return "";} else {let data1 = data[0] + "~" + data[1];return data1;}
};
//清空数据
const resetForm = () => {form.companyName = "";form.linkman = "";form.companyType = "";form.contactTitle = "";form.contactPhone = "";form.id1 = [];form.id2 = [];form.idCardPhoto = [];form.publicCreditInformationReportOfEnterpriseInstitutions = [];form.businessLicenseOrLegalRepresentativeCertificate = [];form.lastYearBankCertificationDocuments = [];form.taxPaymentMaterialsForThePastYear = [];form.employeePaymentCertificateForThePastYear = [];form.declarationOfMajorIllegalRecords = [];form.representativeProjects = [];form.actualImplementationSituation = [];form.other = [];testDatas.value = [];
};
// 添加一行
const addRow = (whatAdd: number) => {if (whatAdd == 0) {let obj: any = {};if (testDatas.value.length == 0) {obj.index = "0";columnList.value.forEach((p: any) => {obj[p.prop] = "";});testDatas.value[0] = obj;} else {const weiyi = testDatas.value[testDatas.value.length - 1].index;columnList.value.forEach((p: any) => {obj[p.prop] = "";});obj.index = String(Number(weiyi) + 1);testDatas.value[testDatas.value.length] = obj;}}
};
//删除
const delRow = (scope: any, whatDel: number) => {const arrIndex = scope.$index; //唯一标识//将testDatas中的第arrIndex项数据删除if (whatDel == 0) {testDatas.value.splice(arrIndex, 1);}
};
const acceptParams = () => {drawerVisible.value = true;
};
defineExpose({acceptParams
});
</script><style scoped lang="scss">
:deep(.el-textarea__inner) {height: 200px;
}
:deep(.cell) {text-align: center !important;height: 45px;padding: 0;
}
.p1111 {height: 50px;line-height: 50px !important;overflow: hidden;white-space: nowrap;text-overflow: ellipsis;padding: 0;margin: 0;text-align: center !important;
}
.tableForm {width: 700px;margin-left: 35px;margin-top: 20px;
}
.sepcial_about {width: 500px;padding-left: 40px;box-sizing: border-box;margin-bottom: 60px;position: relative;> :first-child {width: 1000px;margin-bottom: 10px;}
}
.sepcial_about .mt-4 {position: absolute;width: 40px !important;height: 30px !important;bottom: 10px;left: 800px;z-index: 999;
}
.sepcial_about :deep(.el-table) {display: block;height: 100%;
}
:deep(.el-date-editor) {width: 100% !important;height: 50px;text-align: center;
}
.form-left {padding-top: 20px;
}
:deep(.el-form-item__content) {/* margin-top: 20px; */margin-left: 40px !important;
}
:deep(.el-form-item__label) {font-size: 16px;color: black;
}
</style>
相关文章:

ElementUI table+dialog实现一个简单的可编辑的表格
table组件如何实现可编辑呢? 我的需求是把table组件那样的表格,实现它点击可以弹出一个框,然后在这个框里面输入你的东西,然后将他回显回去,当然,输入的有可能是时间啥的。 为什么要弹出弹层不在框上直接…...

Rust语言精讲:数据类型全解析
大家好!我是lincyang。 今天,我们将深入探讨Rust语言中的数据类型,这是理解和掌握Rust的基础。 Rust语言数据类型概览 Rust是静态类型语言,所有变量类型在编译时确定。Rust的数据类型分为两类:标量类型和复合类型。…...

《数据结构、算法与应用C++语言描述》-代码实现散列表(线性探查与链式散列)
散列表 完整可编译运行代码:Github:Data-Structures-Algorithms-and-Applications/_22hash/ 定义 字典的另一种表示方法是散列(hashing)。它用一个散列函数(也称哈希函数)把字典的数对映射到一个散列表(…...

Hadoop学习笔记:运行wordcount对文件字符串进行统计案例
文/朱季谦 我最近使用四台Centos虚拟机搭建了一套分布式hadoop环境,简单模拟了线上上的hadoop真实分布式集群,主要用于业余学习大数据相关体系。 其中,一台服务器作为NameNode,一台作为Secondary NameNode,剩下两台当…...
python编写简单登录系统(密码混淆加密)
输入非123的数字会显示输入123选项,输入空格或者回车会报错,因为choice设置成int型先输入2个正常账户进去预防了用户名为空,密码为空或者小于3个,用户名已存在3种情况只有用户名和对应的密码都输入正确才能登录成功输入选项3退出代…...
UVA1025 城市里的间谍 A Spy in the Metro
UVA1025 城市里的间谍 A Spy in the Metro 题面翻译 题目大意 某城市地铁是一条直线,有 n n n( 2 ≤ n ≤ 50 2\leq n\leq 50 2≤n≤50)个车站,从左到右编号 1 … n 1\ldots n 1…n。有 M 1 M_1 M1 辆列车从第 1 1 1 站开…...

【科普知识】什么是步进电机?
德国百格拉公司于1973年发明了五相混合式步进电机及其驱动器,1993年又推出了性能更加优越的三相混合式步进电机。我国在80年代以前,一直是反应式步进电机占统治地位,混合式步进电机是80年代后期才开始发展。 步进电机是一种用电脉冲信号进行…...

AWS云服务器EC2实例实现ByConity快速部署
1. 前言 亚马逊是全球最大的在线零售商和云计算服务提供商。AWS云服务器在全球范围内都备受推崇,被众多业内人士誉为“云计算服务的行业标准”。在国内,亚马逊AWS也以其卓越的性能和服务满足了众多用户的需求,拥有着较高的市场份额和竞争力。…...

Docker的项目资源参考
Docker的项目资源包括以下内容: Docker官方网站:https://www.docker.com/ Docker Hub:https://hub.docker.com/ Docker文档:https://docs.docker.com/ Docker GitHub仓库:https://github.com/docker Docker官方博客…...
wsl-ubuntu 系统端口总被主机端口占用问题解决
wsl-ubuntu 系统端口总被主机端口占用问题解决 0. 问题描述1. 解决方法 0. 问题描述 wsl-ubuntu 子系统中的服务,总是启动失败,错误信息是端口被占用。 用一些命令查看,被占用的端口也没有用服务启动。 1. 解决方法 运行, ne…...

详解自动化之单元测试工具Junit
目录 1.注解 1.1 Test 1.2 BeforeEach 1.3 BeforeAll 1.4 AfterEach 1.5 AfterAll 2. 用例的执行顺序 通过 order() 注解来排序 3. 参数化 3.1 单参数 3.2 多参数 3.3 多参数(从第三方csv文件读取数据源) 3.4 动态参数ParameterizedTest MethodSource() 4. 测试…...

超声波雪深传感器冬季里的科技魔法
在冬季的某个清晨,当你打开大门,被厚厚的积雪覆盖的大地映入眼帘,你是否曾想过,这片雪地的深度是多少?它又如何影响着我们的生活和环境?今天,我们将为你揭开这个谜团,介绍一款神秘的…...

2023年【熔化焊接与热切割】免费试题及熔化焊接与热切割模拟考试
题库来源:安全生产模拟考试一点通公众号小程序 熔化焊接与热切割免费试题是安全生产模拟考试一点通生成的,熔化焊接与热切割证模拟考试题库是根据熔化焊接与热切割最新版教材汇编出熔化焊接与热切割仿真模拟考试。2023年【熔化焊接与热切割】免费试题及…...

【数据结构】—搜索二叉树(C++实现,超详细!)
🎬慕斯主页:修仙—别有洞天 ♈️今日夜电波:消えてしまいそうです—真夜中 1:15━━━━━━️💟──────── 4:18 🔄 ◀️ ⏸ ▶️…...

机器人算法—ROS TF坐标变换
1.TF基本概念 (1)什么是TF? TF是Transformations Frames的缩写。在ROS中,是一个工具包,提供了坐标转换等方面的功能。 tf工具包,底层实现采用的是一种树状数据结构,根据时间缓冲并维护多个参考…...

路由VRRP配置例子
拓朴如下: 主要配置如下: [R1] interface GigabitEthernet0/0/0ip address 10.1.1.1 255.255.255.0 vrrp vrid 1 virtual-ip 10.1.1.254vrrp vrid 1 priority 200vrrp vrid 1 preempt-mode timer delay 20 # interface GigabitEthernet0/0/1ip address …...
OpenGL 绘制点与三角形(Qt)
文章目录 一、简介二、实现代码三、实现效果一、简介 这里对OpenGL中点与三角形相关绘制操作进行封装,方便后续点云数据与模型数据的渲染。 二、实现代码 这里我们先创建一个基类Drawable,后续的点、线、面等,均会继承该类: Drawable.h #ifndef DRAWABLE_H #define DRAWABL…...
究竟什么是阻塞与非阻塞、同步与异步
文章目录 前言阻塞与非阻塞同步与异步复杂的网络IO真正的异步IOIO分类与示例总结 前言 这几个名词在程序开发时经常听到,但是突然问起来各个词的含义一时间还真是说不清楚,貌似这几个词都是翻译过来的,每个人的解释都不太一样,我…...

Openlayer【三】—— 绘制多边形GeoJson边界绘制
1.1、绘制多边形 在绘制多边形和前面绘制线有异曲同工之妙,多边形本质上就是由多个点组成的线然后连接组成的面,这个面就是最终的结果,那么这里使用到的是Polygon对象,而传给这个对象的值也是多个坐标,坐标会一个个的…...

用SOLIDWORKS画个高尔夫球,看似简单的建模却大有学问
SOLIDWORKS软件提供了大量的建模功能,如果工程师能灵活使用这些功能,就可以绘制得到各式各样的模型,我们尝试使用SOLIDWORKS绘制高尔夫球模型,如下图所示。 为什么选用solid works进行建模? solid works是一款功能强大…...

VB.net复制Ntag213卡写入UID
本示例使用的发卡器:https://item.taobao.com/item.htm?ftt&id615391857885 一、读取旧Ntag卡的UID和数据 Private Sub Button15_Click(sender As Object, e As EventArgs) Handles Button15.Click轻松读卡技术支持:网站:Dim i, j As IntegerDim cardidhex, …...
Java 8 Stream API 入门到实践详解
一、告别 for 循环! 传统痛点: Java 8 之前,集合操作离不开冗长的 for 循环和匿名类。例如,过滤列表中的偶数: List<Integer> list Arrays.asList(1, 2, 3, 4, 5); List<Integer> evens new ArrayList…...

如何在看板中体现优先级变化
在看板中有效体现优先级变化的关键措施包括:采用颜色或标签标识优先级、设置任务排序规则、使用独立的优先级列或泳道、结合自动化规则同步优先级变化、建立定期的优先级审查流程。其中,设置任务排序规则尤其重要,因为它让看板视觉上直观地体…...

Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility
Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility 1. 实验室环境1.1 实验室环境1.2 小测试 2. The Endor System2.1 部署应用2.2 检查现有策略 3. Cilium 策略实体3.1 创建 allow-all 网络策略3.2 在 Hubble CLI 中验证网络策略源3.3 …...

蓝牙 BLE 扫描面试题大全(2):进阶面试题与实战演练
前文覆盖了 BLE 扫描的基础概念与经典问题蓝牙 BLE 扫描面试题大全(1):从基础到实战的深度解析-CSDN博客,但实际面试中,企业更关注候选人对复杂场景的应对能力(如多设备并发扫描、低功耗与高发现率的平衡)和前沿技术的…...
linux 下常用变更-8
1、删除普通用户 查询用户初始UID和GIDls -l /home/ ###家目录中查看UID cat /etc/group ###此文件查看GID删除用户1.编辑文件 /etc/passwd 找到对应的行,YW343:x:0:0::/home/YW343:/bin/bash 2.将标红的位置修改为用户对应初始UID和GID: YW3…...
[Java恶补day16] 238.除自身以外数组的乘积
给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法,且在 O(n) 时间复杂度…...

在WSL2的Ubuntu镜像中安装Docker
Docker官网链接: https://docs.docker.com/engine/install/ubuntu/ 1、运行以下命令卸载所有冲突的软件包: for pkg in docker.io docker-doc docker-compose docker-compose-v2 podman-docker containerd runc; do sudo apt-get remove $pkg; done2、设置Docker…...
全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比
目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec? IPsec VPN 5.1 IPsec传输模式(Transport Mode) 5.2 IPsec隧道模式(Tunne…...
MySQL用户和授权
开放MySQL白名单 可以通过iptables-save命令确认对应客户端ip是否可以访问MySQL服务: test: # iptables-save | grep 3306 -A mp_srv_whitelist -s 172.16.14.102/32 -p tcp -m tcp --dport 3306 -j ACCEPT -A mp_srv_whitelist -s 172.16.4.16/32 -p tcp -m tcp -…...