ant design实现圈选功能

2024-02-06 07:48
文章标签 实现 功能 ant design 圈选

本文主要是介绍ant design实现圈选功能,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

ant design下运用js实现框选功能

由于同事离职,公司缺人,他的工作便交接到我的手里了,我一个android开发者,以前也从来没做过web端开发啊,没办法,领导交代的任务硬着头皮也得接下来啊!拿到手上,做的第一个功能,便是存储计划,需要实现可按照天、周、月存储,并且以鼠标圈选的形式实现,接下来附上自己的实现效果图:

在这里插入图片描述

实现流程

本来拿到这个任务的时候,自己是想用Grid实现的,但是看到官网上面的一句话,直接打消了我的念头,官网是这么说的:
在这里插入图片描述
也就是说用Grid每一行最多显示24个单元格,这个完全达不到我的要求,因为我每行需要显示25个单元格(每行的title+24小时),我决定还是自己用div画吧。

1.先画单元格

画单元格分成第一行和剩余的行两种:
在这里插入图片描述
第一行组件我们定义为ColumsTitle
循环里面的每一个div其实代表的是每一个单元格。

//标题列组件
const ColumsTitle = () => {const colums = [];for (let i = 0; i < 25; i++) {if (i == 0) {colums.push(<div key={i} className={styles["columns-title-border"]}></div>);} else {colums.push(<div key={i} className={styles["columns-border-none"]}>{i - 1}</div>);}}return colums;
}

剩余行组件,我们定义为Colums:

//列组件
class Colums extends PureComponent {render() {const colums = [];for (let i = 0; i < 25; i++) {if (i == 0) {colums.push(<div key={i} className={styles["columns-title-border"]}>{this.props.rowName}</div>);} else {colums.push(<div id={this.props.rowName + i} key={this.props.rowName + i} className={styles["columns-border"]} name="chooseDiv"></div>);}}return colums;}
}

最后一个就是整体上的组件了,我们叫做Rows

// 行组件
const Rows = (props) => {const rows = [];var rowLength = 1;var rowName = "";if (props.saveType == "1") {rowLength = 2;} else if (props.saveType == "2") {rowLength = 8;} else if (props.saveType == "3") {rowLength = 32;}for (let i = 0; i < rowLength; i++) {rowName = formatRowName(props, i);if (i == 0) {rows.push(<Row key={i}><div className={styles["columns-title-out-margin"]}><ColumsTitle/></div></Row>);} else {rows.push(<Row key={i}><div className={styles["columns-title-out"]}><Colums saveType={props.saveType} rowName={rowName}/></div></Row>);}}return rows;
};

我们渲染到SavePlan这个组件里面:

export default class SavePlan extends PureComponent {constructor(props) {super(props);this.state = {saveType: "1"//1 按天存储 2 按周存储 3 按月存储}}handleRadioChange = e => {this.setState({saveType: e.target.value});};onChange(value) {console.log('changed', value);}render() {return (<PageHeaderWrapper><div><h1>存储计划</h1><div className={styles["title-row"]}><Radio.Group defaultValue="1" size="large" onChange={this.handleRadioChange}><Radio.Button value="1">天存储</Radio.Button><Radio.Button value="2">周存储</Radio.Button><Radio.Button value="3">月存储</Radio.Button></Radio.Group><div className={styles["right-div"]}>存储周期:<InputNumber min={1} max={10} defaultValue={3} onChange={this.onChange}/>(天)<div className={styles["title-row"]}><Button type="primary" className={styles.btn}>确定</Button><Button className={styles.btn}>取消</Button></div></div></div><Rows saveType={this.state.saveType}></Rows></div></PageHeaderWrapper>);}}

到这一步,我们在页面其实已经可以看到整个布局了,但是还没有添加鼠标事件,还没有圈选功能,接下来我们看鼠标事件。

2.鼠标事件

我们这里主要用到了鼠标的三个事件:onmousedown、onmousemove、onmouseup。
我们首先设定可选的单元格,标题设定为不可选:

 -webkit-user-select: none; /* 禁止 DIV 中的文本被鼠标选中 */-moz-user-select: none; /* 禁止 DIV 中的文本被鼠标选中 */-ms-user-select: none; /* 禁止 DIV 中的文本被鼠标选中 */user-select: none; /* 禁止 DIV 中的文本被鼠标选中 */

思路就是:获取鼠标按下时的坐标,并判断是否在可选区域内,若在那么就添加一个div(也就是我们的圈选框),图解如下:
在这里插入图片描述

  1. 获取可选单元格数组
 //可选单元格var fileNodes = document.getElementsByName("chooseDiv");
  1. 获取鼠标点击位置的坐标
var evt = window.event||arguments[0];//加上滚动距离var scrollX = document.documentElement.scrollLeft || document.body.scrollLeft;var scrollY = document.documentElement.scrollTop || document.body.scrollTop;var startX =evt.pageX || evt.clientX + scrollX;var startY =evt.pageY || evt.clientY + scrollY;
  1. 判断可选框坐标范围
//判断鼠标点击的点是否在可选框内部,主要是判断第一个可选框的左上角坐标和最后一个圈选框的右下角坐标if ((startX >= firstDivOffsetLeft && startY >= firstDivOffsetTop) && (startX <= lastDivOffsetLeft && startY <= lastDivOffsetTop))
  1. 判断鼠标点击在哪一个单元格里面,并获取该单元格左上角坐标
//判断鼠标点击的点在哪一个div里面,然后更改圈选框的左上角坐标为该div的左上角坐标for (var i = 0; i < fileNodes.length; i++) {if ((startX >= getOffsetLeft(fileNodes[i]) && startX <= getOffsetLeft(fileNodes[i]) + fileNodes[i].offsetWidth) && (startY >= getOffsetTop(fileNodes[i]) && startY <= getOffsetTop(fileNodes[i]) + fileNodes[i].offsetHeight)) {console.log("在内部");startX = getOffsetLeft(fileNodes[i]);startY = getOffsetTop(fileNodes[i]);break;} else {console.log("不在内部");}}
  1. 创建圈选框,并更改圈选框的左上角坐标为该单元格的左上角坐标
 //创建选择框selDiv = document.createElement("div");selDiv.style.cssText = "position:absolute;width:0px;height:0px;font-size:0px;margin:0px;padding:0px;border:1px dashed #0099FF;background-color:#C3D5ED;z-index:1000;filter:alpha(opacity:60);opacity:0.6;display:none;";selDiv.id = "selectDiv";document.body.appendChild(selDiv);selDiv.style.left = startX + "px";selDiv.style.top = startY + "px";
  1. 鼠标移动过程中,改变圈选框的宽高
 evt = window.event || arguments[0];if (isSelect) {if (selDiv.style.display == "none") {selDiv.style.display = "";}//加上鼠标滚动距离var _scrollX = document.documentElement.scrollLeft || document.body.scrollLeft;var _scrollY = document.documentElement.scrollTop || document.body.scrollTop;_x = evt.pageX || evt.clientX + _scrollX;_y = evt.pageY || evt.clientY + _scrollY;selDiv.style.left = Math.min(_x, startX) + "px";selDiv.style.top = Math.min(_y, startY) + "px";selDiv.style.width = Math.abs(_x - startX) + "px";selDiv.style.height = Math.abs(_y - startY) + "px";
  1. 鼠标抬起的时候,计算被圈选的单元格并更改样式
 var _l = selDiv.offsetLeft, _t = selDiv.offsetTop;var _w = selDiv.offsetWidth, _h = selDiv.offsetHeight;for (var i = 0; i < selList.length; i++) {var sl = selList[i].offsetWidth + getOffsetLeft(selList[i]);var st = selList[i].offsetHeight + getOffsetTop(selList[i]);if (sl > _l && st > _t && getOffsetLeft(selList[i]) < _l + _w && getOffsetTop(selList[i]) < _t + _h) {if (selList[i].className.indexOf("seled") == -1) {selList[i].className = styles["columns-borderseled"];}else {selList[i].className = styles["columns-border"];}}}
  1. 其他工具方法
const getOffsetLeft = function (obj) {var tmp = obj.offsetLeft;var node = obj.offsetParent;while (node != null) {tmp += node.offsetLeft;node = node.offsetParent;}return tmp;
}const getOffsetTop = function (obj) {var tmp = obj.offsetTop;var node = obj.offsetParent;while (node != null) {tmp += node.offsetTop;node = node.offsetParent;}return tmp;
}

这篇关于ant design实现圈选功能的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/683617

相关文章

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P

工厂ERP管理系统实现源码(JAVA)

工厂进销存管理系统是一个集采购管理、仓库管理、生产管理和销售管理于一体的综合解决方案。该系统旨在帮助企业优化流程、提高效率、降低成本,并实时掌握各环节的运营状况。 在采购管理方面,系统能够处理采购订单、供应商管理和采购入库等流程,确保采购过程的透明和高效。仓库管理方面,实现库存的精准管理,包括入库、出库、盘点等操作,确保库存数据的准确性和实时性。 生产管理模块则涵盖了生产计划制定、物料需求计划、

C++——stack、queue的实现及deque的介绍

目录 1.stack与queue的实现 1.1stack的实现  1.2 queue的实现 2.重温vector、list、stack、queue的介绍 2.1 STL标准库中stack和queue的底层结构  3.deque的简单介绍 3.1为什么选择deque作为stack和queue的底层默认容器  3.2 STL中对stack与queue的模拟实现 ①stack模拟实现