Vue的高级表格组件库【vxe-table】
文章目录
- 前言
- vxe-table
- 官网
- 实现表头拖拽
- 树形表格
- 全键盘操作
- 后言
前言
hello world欢迎来到前端的新世界
😜当前文章系列专栏:前端系列文章
🐱👓博主在前端领域还有很多知识和技术需要掌握,正在不断努力填补技术短板。(如果出现错误,感谢大家指出)🌹
💖感谢大家支持!您的观看就是作者创作的动力
vxe-table
一款专门用来对Vue3表格进行复杂操作的ui组件库,虚拟滚动,普通表格的行拖拽,表格的全屏展示,表格的全键盘操作,禁用编辑等。
官网
官网地址
下载
npm install xe-utils vxe-table
配置在main.js或者main.ts里面
import { App, createApp } from 'vue'
import VXETable from 'vxe-table'
import 'vxe-table/lib/style.css'function useTable (app: App) {app.use(VXETable)
}createApp(App).use(useTable).mount('#app')
实现表头拖拽
直接复制
<template><div><vxe-grid v-bind="gridOptions"><template #toolbar_buttons><vxe-button @click="gridOptions.align = 'left'">居左</vxe-button><vxe-button @click="gridOptions.align = 'center'">居中</vxe-button><vxe-button @click="gridOptions.align = 'right'">居右</vxe-button></template></vxe-grid></div>
</template><script lang="ts" setup>
import { reactive } from 'vue'
import { VxeGridProps } from 'vxe-table'interface RowVO {id: numbername: stringnickname: stringrole: stringsex: stringage: numberaddress: string
}const gridOptions = reactive<VxeGridProps<RowVO>>({border: true,height: 300,align: null,columnConfig: {resizable: true},columns: [{ type: 'seq', width: 50 },{ field: 'name', title: 'name' },{ field: 'sex', title: 'sex' },{ field: 'address', title: 'Address' }],toolbarConfig: {slots: {buttons: 'toolbar_buttons'}},data: [{ id: 10001, name: 'Test1', nickname: 'T1', role: 'Develop', sex: 'Man', age: 28, address: 'Shenzhen' },{ id: 10002, name: 'Test2', nickname: 'T2', role: 'Test', sex: 'Women', age: 22, address: 'Guangzhou' },{ id: 10003, name: 'Test3', nickname: 'T3', role: 'PM', sex: 'Man', age: 32, address: 'Shanghai' },{ id: 10004, name: 'Test4', nickname: 'T4', role: 'Designer', sex: 'Women', age: 23, address: 'Shenzhen' },{ id: 10005, name: 'Test5', nickname: 'T5', role: 'Develop', sex: 'Women', age: 30, address: 'Shanghai' },{ id: 10006, name: 'Test6', nickname: 'T6', role: 'Designer', sex: 'Women', age: 21, address: 'Shenzhen' },{ id: 10007, name: 'Test7', nickname: 'T7', role: 'Test', sex: 'Man', age: 29, address: 'Shenzhen' },{ id: 10008, name: 'Test8', nickname: 'T8', role: 'Develop', sex: 'Man', age: 35, address: 'Shenzhen' }]
})
</script>
树形表格
<template><div><vxe-toolbar><template #buttons><vxe-button @click="expandAllEvent">展开所有</vxe-button><vxe-button @click="claseExpandEvent">收起所有</vxe-button></template></vxe-toolbar><vxe-tableshow-overflowheight="400"ref="tableRef":loading="loading":tree-config="{transform: true}":scroll-y="{enabled: true, gt: 20}":data="tableData"><vxe-column type="seq" width="200" tree-node></vxe-column><vxe-column field="id" title="Id"></vxe-column><vxe-column field="name" title="Name"></vxe-column></vxe-table></div>
</template><script lang="ts" setup>
import { ref } from 'vue'
import { VxeTableInstance } from 'vxe-table'interface RowVO {id: numberparentId: number | nullname: string
}const tableRef = ref<VxeTableInstance<RowVO>>()const loading = ref(false)
const tableData = ref<RowVO[]>([])const loadList = () => {loading.value = truefetch('/resource/json/provinces_list.json').then(res => res.json()).then((data: RowVO[]) => {tableData.value = dataloading.value = false})
}const expandAllEvent = () => {const $table = tableRef.valueif ($table) {$table.setAllTreeExpand(true)}
}const claseExpandEvent = () => {const $table = tableRef.valueif ($table) {$table.clearTreeExpand()}
}loadList()
</script>
全键盘操作
template
<vxe-toolbar :refresh="{query: findList}"><template #buttons><vxe-button><template #default>新增操作</template><template #dropdowns><vxe-button type="text" @click="insertEvent(null)">从第一行插入</vxe-button><vxe-button type="text" @click="insertEvent(-1)">从最后插入</vxe-button><vxe-button type="text" @click="insertEvent($refs.xTable.getData(100))">插入到 100 行</vxe-button><vxe-button type="text" @click="insertEvent($refs.xTable.getData(400))">插入到 400 行</vxe-button></template></vxe-button><vxe-button><template #default>删除操作</template><template #dropdowns><vxe-button type="text" @click="$refs.xTable.removeCheckboxRow()">删除选中</vxe-button><vxe-button type="text" @click="$refs.xTable.remove($refs.xTable.getData(0))">删除第一行</vxe-button><vxe-button type="text" @click="$refs.xTable.remove($refs.xTable.getData($refs.xTable.getData().length - 1))">删除最后一行</vxe-button><vxe-button type="text" @click="$refs.xTable.remove($refs.xTable.getData(100))">删除第 100 行</vxe-button></template></vxe-button><vxe-button><template #default>校验操作</template><template #dropdowns><vxe-button type="text" @click="validEvent">快速校验</vxe-button><vxe-button type="text" @click="fullValidEvent">完整快速校验</vxe-button><vxe-button type="text" @click="selectValidEvent">选中行校验</vxe-button></template></vxe-button><vxe-button @click="getInsertEvent">获取新增</vxe-button><vxe-button @click="getRemoveEvent">获取删除</vxe-button><vxe-button @click="getUpdateEvent">获取修改</vxe-button><vxe-button><template #default>滚动操作</template><template #dropdowns><vxe-button type="text" @click="$refs.xTable.scrollToRow($refs.xTable.getData(10))">滚动到第 10 行</vxe-button><vxe-button type="text" @click="$refs.xTable.scrollToRow($refs.xTable.getData(400))">滚动第 400 行</vxe-button><vxe-button type="text" @click="$refs.xTable.scrollToColumn($refs.xTable.getColumns(1))">滚动第 1 列</vxe-button><vxe-button type="text" @click="$refs.xTable.scrollToColumn($refs.xTable.getColumns(10))">滚动第 10 列</vxe-button></template></vxe-button></template></vxe-toolbar><vxe-tablebordershow-overflowkeep-sourceref="xTable"height="300":column-config="{resizable: true}":loading="demo1.loading":edit-rules="demo1.validRules":mouse-config="{selected: true}":edit-config="{trigger: 'dblclick', mode: 'cell', showStatus: true}":keyboard-config="{isArrow: true, isDel: true, isEnter: true, isTab: true, isEdit: true}"><vxe-column type="checkbox" width="60"></vxe-column><vxe-column type="seq" width="100"></vxe-column><vxe-column field="name" title="Name" sortable width="200" :edit-render="{autofocus: '.vxe-input--inner'}"><template #edit="scope"><vxe-input v-model="scope.row.name" type="text" @change="$refs.xTable.updateStatus(scope)"></vxe-input></template></vxe-column><vxe-column field="age" title="Age" width="200" :edit-render="{autofocus: '.vxe-input--inner'}"><template #edit="scope"><vxe-input v-model="scope.row.age" type="text" @change="$refs.xTable.updateStatus(scope)"></vxe-input></template></vxe-column><vxe-column field="sex" title="Sex" width="200" :edit-render="{autofocus: '.vxe-input--inner'}"><template #edit="scope"><vxe-input v-model="scope.row.sex" type="text" @change="$refs.xTable.updateStatus(scope)"></vxe-input></template></vxe-column><vxe-column field="rate" title="Rate" width="200"></vxe-column><vxe-column field="region" title="Region" width="200"></vxe-column><vxe-column field="time" title="Time" width="200"></vxe-column><vxe-column field="address" title="Address" width="300" show-overflow></vxe-column><vxe-column field="updateTime" title="UpdateTime" width="200"></vxe-column><vxe-column field="createTime" title="CreateTime" width="200"></vxe-column></vxe-table>
script
import { defineComponent, reactive, ref } from 'vue'import { VXETable, VxeTableInstance, VxeTablePropTypes } from 'vxe-table'export default defineComponent({setup () {const xTable = ref<VxeTableInstance>()const demo1 = reactive({loading: false,validRules: {name: [{ required: true, message: 'app.body.valid.rName' },{ min: 3, max: 50, message: '名称长度在 3 到 50 个字符' }],sex: [{ required: true, message: '性别必须填写' }]} as VxeTablePropTypes.ValidConfig})const mockList = (size: number) => {const list: any[] = []for (let index = 0; index < size; index++) {list.push({checked: false,name: `名称${index}`,sex: '0',num: 123,age: 18,num2: 234,rate: 3,address: 'shenzhen'})}return list}const findList = () => {demo1.loading = truereturn new Promise(resolve => {setTimeout(() => {const tableData = mockList(600)// 阻断 vue 对大数组的监听,避免 vue 绑定大数据造成短暂的卡顿const $table = xTable.valueif ($table) {$table.loadData(tableData)}resolve(null)demo1.loading = false}, 300)})}const validEvent = async () => {const $table = xTable.valueconst errMap = await $table.validate()if (errMap) {VXETable.modal.message({ status: 'error', content: '校验不通过!' })} else {VXETable.modal.message({ status: 'success', content: '校验成功!' })}}const fullValidEvent = async () => {const $table = xTable.valueconst errMap = await $table.fullValidate()if (errMap) {const msgList: string[] = []Object.values(errMap).forEach((errList) => {errList.forEach(params => {const { rowIndex, column, rules } = paramsrules.forEach(rule => {msgList.push(`第 ${rowIndex + 1} 行 ${column.title} 校验错误:${rule.message}`)})})})VXETable.modal.message({status: 'error',slots: {default () {return [<div class="red" style="max-height: 400px;overflow: auto;">{msgList.map(msg => {return <div>{ msg }</div>})}</div>]}}})} else {VXETable.modal.message({ status: 'success', content: '校验成功!' })}}const selectValidEvent = async () => {const $table = xTable.valueconst selectRecords = $table.getCheckboxRecords()if (selectRecords.length > 0) {const errMap = await $table.validate(selectRecords).catch(errMap => errMap)if (errMap) {VXETable.modal.message({ status: 'error', content: '校验不通过!' })} else {VXETable.modal.message({ status: 'success', content: '校验成功!' })}} else {VXETable.modal.message({ status: 'warning', content: '未选中数据!' })}}const insertEvent = (row: any) => {const $table = xTable.valueconst record = {checked: false}$table.insertAt(record, row).then(({ row }) => {$table.setEditRow(row)})}const getInsertEvent = () => {const $table = xTable.valueconst insertRecords = $table.getInsertRecords()VXETable.modal.alert(insertRecords.length)}const getRemoveEvent = () => {const $table = xTable.valueconst removeRecords = $table.getRemoveRecords()VXETable.modal.alert(removeRecords.length)}const getUpdateEvent = () => {const $table = xTable.valueconst updateRecords = $table.getUpdateRecords()VXETable.modal.alert(updateRecords.length)}findList()return {xTable,demo1,findList,validEvent,fullValidEvent,selectValidEvent,insertEvent,getInsertEvent,getRemoveEvent,getUpdateEvent}}})
后言
创作不易,要是本文章对广大读者有那么一点点帮助 不妨三连支持一下,您的鼓励就是博主创作的动力
相关文章:
Vue的高级表格组件库【vxe-table】
文章目录 前言vxe-table官网实现表头拖拽树形表格全键盘操作后言 前言 hello world欢迎来到前端的新世界 😜当前文章系列专栏:前端系列文章 🐱👓博主在前端领域还有很多知识和技术需要掌握,正在不断努力填补技术短板…...

从0到0.01入门React | 002.精选 React 面试题
🤍 前端开发工程师(主业)、技术博主(副业)、已过CET6 🍨 阿珊和她的猫_CSDN个人主页 🕠 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 🍚 蓝桥云课签约作者、已在蓝桥云课上架的前后端实战课程《Vue.js 和 Egg.js 开发企业级健康管理项目》、《带你从入…...

假冒 Skype 应用程序网络钓鱼分析
参考链接: https://slowmist.medium.com/fake-skype-app-phishing-analysis-35c1dc8bc515 背景 在Web3世界中,涉及假冒应用程序的网络钓鱼事件相当频繁。慢雾安全团队此前曾发表过分析此类网络钓鱼案例的文章。由于Google Play在中国无法访问,许多用户…...

软件外包开发的需求表达方法
软件开发需求的有效表达对于项目的成功至关重要。无论选择哪种需求表达方法,清晰、详细、易于理解是关键。与开发团队建立良好的沟通渠道,确保他们对需求有充分的理解,并随着项目的推进及时调整和更新需求文档。以下是一些常用的需求表达方法…...

详解JS的四种异步解决方案:回调函数、Promise、Generator、async/await
同步&异步的概念 在讲这四种异步方案之前,我们先来明确一下同步和异步的概念: 所谓同步(synchronization),简单来说,就是顺序执行,指的是同一时间只能做一件事情,只有目前正在执行的事情做完之后&am…...

Python进行多线程爬取数据通用模板
首先,我们需要导入所需的库,包括requests和BeautifulSoup。requests库用于发送HTTP请求,BeautifulSoup库用于解析HTML文档。 import requests from bs4 import BeautifulSoup然后,我们需要定义一个函数来发送HTTP请求并返回响应。…...

基于springboot实现沁园健身房预约管理系统【项目源码】
基于springboot实现沁园健身房预约管理系统演示 B/S架构 B/S结构是目前使用最多的结构模式,它可以使得系统的开发更加的简单,好操作,而且还可以对其进行维护。使用该结构时只需要在计算机中安装数据库,和一些很常用的浏览器就可以…...

论文笔记:Deep Trajectory Recovery with Fine-Grained Calibration using Kalman Filter
TKDE 2021 1 intro 1.1 背景 用户轨迹数据对于改进以用户为中心的应用程序很有用 POI推荐城市规划路线规划由于设备和环境的限制,许多轨迹以低采样率记录 采样的轨迹无法详细说明物体的实际路线增加了轨迹中两个连续采样点之间的不确定性——>开发有效的算法以…...

ubuntu下tensorrt环境配置
文章目录 一、Ubuntu18.04环境配置1.1 安装工具链和opencv1.2 安装Nvidia相关库1.2.1 安装Nvidia显卡驱动1.2.2 安装 cuda11.31.2.3 安装 cudnn8.21.2.4 下载 tensorrt8.4.2.4 二、编写CMakeLists.txt三、TensorRT系列教程 一、Ubuntu18.04环境配置 教程同样适用与ubuntu22.04…...

网络安全基础之php开发文件下载的实现
前言 php是网络安全学习里必不可少的一环,简单理解php的开发环节能更好的帮助我们去学习php以及其他语言的web漏洞原理 正文 在正常的开发中,文件下载的功能是必不可少,比如我们在论坛看到好看图片好听的歌时,将其下载下来时就…...

【学习笔记】 - GIT的基本操作,IDEA接入GIT以及上传hub
用github蛮多,但git没怎么用,看着视频对着写点笔记以及操作 一、GIT文件的三种状态和模式 已提交(committed) 已提交表示数据已经安全的保存在本地数据库中。 已修改(modified) 已修改表示修改了文件,但还没保存到数据库中。…...

Antd React Form.Item内部是自定义组件怎么自定义返回值
在线演示https://stackblitz.com/edit/stackblitz-starters-xwtwyz?filesrc%2FSelfTreeSelect.tsx 需求 当我们点击提交,需要返回用户名和选中树的id信息,但是,我不关要返回树的id信息,还需要返回选中树的名称 //默认返回的 {userName:梦洁,treeInfo:leaf1-value } //但是需…...
2023最新ACL大模型论文分类汇总(有代码的)
1 大模型文化道德 Knowledge of cultural moral norms in large language models url:https://aclanthology.org/2023.acl-long.26/code:https://github.com/AidaRamezani/cultural_inference 2 长文本推理 Open-ended Long Text Generation via Mask…...

Java版 招投标系统简介 招投标系统源码 java招投标系统 招投标系统功能设计
功能描述 1、门户管理:所有用户可在门户页面查看所有的公告信息及相关的通知信息。主要板块包含:招标公告、非招标公告、系统通知、政策法规。 2、立项管理:企业用户可对需要采购的项目进行立项申请,并提交审批,查看所…...

Ubuntu 22.04源码安装cmake 3.27.7
安装参考博客是《ubuntu安装cmake》和《Ubuntu 安装CMake》。 https://cmake.org/download是cmake官网下载的网址。 sudo wget -c https://github.com/Kitware/CMake/releases/download/v3.27.7/cmake-3.27.7.tar.gz可以下载源码,最后显示‘cmake-3.27.7.tar.gz’…...

无人地磅称重系统|自助过磅 料仓联动 自助卸料
上海思伟无人地磅系统 自助过磅、 自助卸料 、料仓联动 智能、省人、安全 无人监管过磅 对地磅及其相关的所有硬件进行配置和管理; 支持红外、道闸、车牌识别、AI分析、拍照存档、LED语音播报一体机等设备; 实现稳定可靠的无人监管称重功能…...
冥想第九百七十三天
1.今天周六,很冷的天,上午上了一上午的日语课。 2.下午去看了朋友刚出生的孩子。 3.充实的一天。感谢父母,感谢朋友,感谢家人,感谢不断进步的自己....

ROS 学习应用篇(三)话题Topic学习之自定义话题消息的类型的定义与调用
自定义消息类型的定义 Person.msg文件的定义(数据接口文件的定义) 创建msg文件 首先在功能包下新建msg文件夹,接着在该文件夹下创建文件。 定义msg文件内容 一个消息最重要的就是数据结构类型。这就需要引入一个msg文件,用于…...

财税服务展示预约小程序的作用是什么
财税财政往往困扰着很多公司,尤其是公司里没有相应职员或工作压力大的情况下,不少商家就会寻找代理记账、审计服务、会计代理等服务的机构。 对财政服务代理机构(会计公司)来说,市场企业多而广,理论上来说…...

RT-Thread提供的网络世界入口 -net组件
作为一款在RTOS领域对网络支持很丰富的RT-Thread,对设备联网功能的支持的工具就是net组件。 位于/rt-thread/components/net路劲下,作为一款基础组件,env与Studio的工程配置项界面的配置项都依赖该目录下的Kconfig。 我们对网络功能的选择&am…...
C++:std::is_convertible
C++标志库中提供is_convertible,可以测试一种类型是否可以转换为另一只类型: template <class From, class To> struct is_convertible; 使用举例: #include <iostream> #include <string>using namespace std;struct A { }; struct B : A { };int main…...
连锁超市冷库节能解决方案:如何实现超市降本增效
在连锁超市冷库运营中,高能耗、设备损耗快、人工管理低效等问题长期困扰企业。御控冷库节能解决方案通过智能控制化霜、按需化霜、实时监控、故障诊断、自动预警、远程控制开关六大核心技术,实现年省电费15%-60%,且不改动原有装备、安装快捷、…...

2021-03-15 iview一些问题
1.iview 在使用tree组件时,发现没有set类的方法,只有get,那么要改变tree值,只能遍历treeData,递归修改treeData的checked,发现无法更改,原因在于check模式下,子元素的勾选状态跟父节…...

Keil 中设置 STM32 Flash 和 RAM 地址详解
文章目录 Keil 中设置 STM32 Flash 和 RAM 地址详解一、Flash 和 RAM 配置界面(Target 选项卡)1. IROM1(用于配置 Flash)2. IRAM1(用于配置 RAM)二、链接器设置界面(Linker 选项卡)1. 勾选“Use Memory Layout from Target Dialog”2. 查看链接器参数(如果没有勾选上面…...

Psychopy音频的使用
Psychopy音频的使用 本文主要解决以下问题: 指定音频引擎与设备;播放音频文件 本文所使用的环境: Python3.10 numpy2.2.6 psychopy2025.1.1 psychtoolbox3.0.19.14 一、音频配置 Psychopy文档链接为Sound - for audio playback — Psy…...

微信小程序云开发平台MySQL的连接方式
注:微信小程序云开发平台指的是腾讯云开发 先给结论:微信小程序云开发平台的MySQL,无法通过获取数据库连接信息的方式进行连接,连接只能通过云开发的SDK连接,具体要参考官方文档: 为什么? 因为…...

安宝特案例丨Vuzix AR智能眼镜集成专业软件,助力卢森堡医院药房转型,赢得辉瑞创新奖
在Vuzix M400 AR智能眼镜的助力下,卢森堡罗伯特舒曼医院(the Robert Schuman Hospitals, HRS)凭借在无菌制剂生产流程中引入增强现实技术(AR)创新项目,荣获了2024年6月7日由卢森堡医院药剂师协会࿰…...
Java编程之桥接模式
定义 桥接模式(Bridge Pattern)属于结构型设计模式,它的核心意图是将抽象部分与实现部分分离,使它们可以独立地变化。这种模式通过组合关系来替代继承关系,从而降低了抽象和实现这两个可变维度之间的耦合度。 用例子…...

Netty从入门到进阶(二)
二、Netty入门 1. 概述 1.1 Netty是什么 Netty is an asynchronous event-driven network application framework for rapid development of maintainable high performance protocol servers & clients. Netty是一个异步的、基于事件驱动的网络应用框架,用于…...

GO协程(Goroutine)问题总结
在使用Go语言来编写代码时,遇到的一些问题总结一下 [参考文档]:https://www.topgoer.com/%E5%B9%B6%E5%8F%91%E7%BC%96%E7%A8%8B/goroutine.html 1. main()函数默认的Goroutine 场景再现: 今天在看到这个教程的时候,在自己的电…...