echarts实现图表标签(label)可拖拽,以及保存拖拽后的位置

news2025/1/11 13:59:38

需求背景:
当echarts图表中像素点非常多,或者有像素点重合的时候,标签就会被覆盖或者重叠。为了解决这个问题,让用户体验更加友好,于是就实现了对label进行拖拽。用户可以把label拖拽到任何他想要的位置,并且能够将位置保存下来,下次还能进行编辑回显。(下文将以折线图作为示范)

实现思路:
1、首先需要实现label可拖拽
2、将拖拽后的label坐标记录下来,回传到服务端进行保存
3、编辑回显的时候将保存过的位置取出来赋值给对应的label

实现拖拽前的效果展示:
我们可以看到,线条之间重合的部分和值很接近的部分label都重叠在一起了,非常影响观看效果。
拖拽前.jpg
实现拖拽后的效果展示:
按照自己的审美,将label拖拽到旁边的空白区域后看起来就要好很多了。
拖拽后png

具体实现

实现label可拖拽

我们通过对echarts官网文档的查阅,可以发现能通过配置series-line.labelLayout. draggable实现拖拽效果

注意:
1、echarts从 v5.0.0 开始支持 labelLayout
2、仅仅设置daggable:true是不会生效的,daggable需要跟x和y同时设置才能生效(备注:这点官方文档并没有说明,导致浪费了很多时间)
3、labelLayout在本地运行正常,但是打包后失效了:
LabelLayout需手动引入,否则打包会被tree shaking。
import { LabelLayout } from ‘echarts/features’;

我们已经知道了如果要实现拖拽,draggable需要设为true,那x和y需要设置成什么值呢,初始化的时候建议设置成label的默认位置,比如折线图的label默认展示在当前坐标点的上方,那这就是折线图标签的默认位置。可以通过labelLayout回调函数第一个参数的labelRect获取到当前label的默认位置:

 labelLayout: (e:any) => {
      return {
        draggable: true,
        y: e.labelRect.y,
        x: e.labelRect.x
      }
    }

获取拖拽后的坐标位置并保存

通过上一步的配置,我们已经能够实现label的拖拽了,那拖拽后的坐标点的位置我们要怎么获取呢?把echarts文档上上下下看了好多遍也没有发现他们给标签拖拽提供了drag回调,所以只能想办法自己实现了。
首先我们可以通过监听图表的mouseup事件来获取到鼠标松开时的坐标点位置。

  myChart.value.on('mouseup', function (params: any) {
      if (params.event.target.style.text) {
        const offsetX = params.event.offsetX / echartMain.value.offsetWidth
        const offsetY = params.event.offsetY / echartMain.value.offsetHeight
        emit('dragEnd', params, offsetX, offsetY)
      }
    })

注意:
1、我们可以看到上面代码中有一个 params.event.offsetX / echartMain.value.offsetWidth,为啥要这么写呢?
主要是为了适配不同大小的屏幕。因为图表一般都是用做大屏展示,会面临缩放,以及适配不同屏幕大小的问题,如果label位置保存的是一个固定值,那在不同的屏幕下始终在相同坐标位置展示的话就会有问题,所以这里建议保存一个相对值。我这里保存的是当前鼠标松开时的坐标点与图表区域宽高的一个比值,这样在不同的屏幕下,只需要用当前的图表宽高*这个比值就能完美的还原label的位置了。
2、mouseup能被多个元素触发,那怎样判断当前触发mouseup的元素就是label呢?
当params.event.target.style.text有值的时候说明点击的是label。因为echart的监听的mouseup方法只对图表中的元素有效果,对于横纵轴等其他地方的元素不会触发回调。如果要获取其他地方的点击回调,需要使用getZr()。而图表中的点线柱等元素都没有text属性,只有label有这个属性,所以能准确的拿到label拖拽时松开鼠标的事件。

编辑回显

此时我们已经获取到每个label移动后的位置并保存下来了,当点编辑的时候,需要将之前保存的标签位置在图表中回显出来继续编辑。我们将labelLayout改造下:

labelLayout: (e:any) => {
     const field = item.data[e.dataIndex]
     const cachePos = position?.find((item:any) => item.name === field.name && item.value === field.value)
     let x = e.labelRect.x
     let y = e.labelRect.y
     if (cachePos) {
       x = (cachePos.offsetX) * echartMain.offsetWidth
       y = (cachePos.offsetY) * echartMain.offsetHeight
     }
     return {
       draggable: true,
       y,
       x
     }
   }

前两行是取出当前标签对应的label位置,这两行代码不用关注,因为每个项目上存储的方式都不一样。当我们找到当前标签对应的位置后,用这个位置乘以图表区域的宽高,就得到了横纵轴的偏移量,再把这个值返回出去就大功告成啦!

可是真的大功告成了吗?

此时才发现一个很严肃的问题,因为用户在拖拽标签的时候,有可能拖动的是标签的头部,也有可能是中间或者尾部的位置,而labelLayout中设置的x和y是相对于标签的左上角进行设置的。比如用户拖拽的是标签的尾部,那最后记录的位置就是标签尾部拖拽完成后所在的位置。当编辑回显的时候,又相对于标签左上角来进行赋值,那么最后的效果就会产生一个标签长度的误差。
所以我们需要计算出鼠标拖拽时点击的位置相对于标签头部的偏移量,然后保存offsetX和offsetY的时候需要减去这个偏移量再保存,这样存下来的值就是相对于标签头部的坐标位置。
首先我们可以通过监听mousedown事件来获取当前鼠标按下时的位置,然后再减去上一次的位置(如果是第一次拖拽的话,就减去标签的初始位置,如果是第二次及以上次数拖拽的话,就减去上一次存储的标签位置),就得到了鼠标点击的位置相对于标签头部的偏移量:

// 监听鼠标按下事件
myChart.value.on('mousedown', function (params: any) {
     if (params.event.target.style.text) {
       const startOffsetX = params.event.offsetX / echartMain.value.offsetWidth
       const startOffsetY = params.event.offsetY / echartMain.value.offsetHeight
       emit('dragStart', params, startOffsetX, startOffsetY)
     }
   })
// 获取鼠标按下的位置相对于标签头部位置的偏移量
const dragStart = (params:any, startOffsetX:number, startOffsetY:number) => {
 const cachePos = labelPosition?.find((item:any) => item.name === params.data.name && item.value === params.data.value )
 let distX = startOffsetX
 let distY = startOffsetY
 if (cachePos) {
   distX = Math.max(startOffsetX - cachePos.offsetX, 0)
   distY = Math.max(startOffsetY - cachePos.offsetY, 0)
 }
 cachePos.distX = distX
 cachePos.distY = distY
}
// 在计算拖拽后的位置时,减去这个偏移量
const dragEnd = (params:any, offsetX:number, offsetY:number) => {
 const cachePos = props.data.chartStyle.labelPosition?.find((item:any) => item.name === params.data.name && item.value === params.data.value)
 if (cachePos) {
   cachePos.offsetX = offsetX - cachePos.distX
   cachePos.offsetY = offsetY - cachePos.distY
 }
}

到这里就真的大功告成啦~

感谢你能看到这里,有任何问题或者建议都可以与我联系,如果我的文章对你有一点点帮助或者启发,可以给我点一个小小的赞或者关注,这将是对我最大的鼓励!

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

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

相关文章

从0开始做yolov5模型剪枝

文章目录 从0开始做yolov5模型剪枝 ****1 前言2 GitHub取源码3 原理3.1 原理3.2 network slimming过程 4 具体实施步骤4.1 安装虚拟环境4.2 配置参数4.2.1 数据集参数4.2.2 模型结构参数4.2.3 train.py中的参数 4.3 正常训练4.3.1 准备4.3.2 训练及问题解决 4.4 稀疏化训练4.4.…

kali 2023.3新增工具

在终端模拟器中运行 sudo apt update && sudo apt full-upgrade 命令来更新其安装 Kali Linux 2023.3 发布中包含了九个新工具,分别是: Calico:云原生网络和网络安全。 cri-tools:用于Kubelet容器运行时接口的命令行界面…

【无标题】AI在开发工具中的应用:自动代码生成

AI辅助开发工具的概述 随着人工智能(AI)技术的发展,越来越多的开发工具开始利用AI来辅助程序员的工作。其中一项重要的应用是自动代码生成。AI辅助开发工具能够通过学习大量的代码库和规范,分析程序员的需求,并自动生成…

论文阅读_模型结构_LoRA

name_en: LoRA: Low-Rank Adaptation of Large Language Models name_ch: LORA:大语言模型的低阶自适应 paper_addr: http://arxiv.org/abs/2106.09685 date_read: 2023-08-17 date_publish: 2021-10-16 tags: [‘深度学习’,‘大模型’] author: Edward J. Hu cita…

leetcode76. 最小覆盖子串(滑动窗口-java)

滑动窗口 最小覆盖子串滑动窗口代码 上期经典 最小覆盖子串 难度 - 困难 原题链接 - 最小覆盖字串 给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串,则返回空字符串 “” 。 注意: 对于 t…

【TypeScript】never 类型

在 TypeScript 中,never 是一种特殊的类型,表示永远不会发生的值或类型。它通常用于表示一些绝对不可能出现的情况,例如永远不会返回的函数类型或在某些条件下绝对不会发生的值。 以下是一些关于 never 类型的情况: 函数返回值…

大红喜庆版UI猜灯谜小程序源码/猜字谜微信小程序源码

今天给大家带来一款UI比较喜庆的猜灯谜小程序,大家看演示图的时候当然也是可以看得到那界面是多么的喜庆,而且新的一年也很快就来了,所以种种的界面可能都比较往喜庆方面去变吧。 这款小程序搭建是免服务器和域名的,只需要使用微信开发者工具…

243:vue+Openlayers 更改鼠标滚轮缩放地图大小,每次缩放小一点

第243个 点击查看专栏目录 本示例的目的是介绍如何在vue+openlayers项目中设置鼠标滚轮缩放地图大小,每次滑动一格滚轮,设定的值非默认值1。具体的设置方法,参考源代码。 直接复制下面的 vue+openlayers源代码,操作2分钟即可运行实现效果 文章目录 示例效果配置方式示例源…

Java快速入门体验

Java快速入门体验 一、环境信息1.1 硬件信息1.2 软件信息 二、Maven安装2.1 Maven介绍2.2 Maven安装包下载2.3 Maven安装2.4 Maven初始化 三、Java安装3.1 JDK下载3.2 JDK安装3.3 JDK初始化 四、开发环境搭建4.1 安装开发工具4.2 关联Maven环境4.2.1 新建JAVA项目4.2.2 Maven与…

5G网关如何提升智慧乡村农业生产效率

得益于我国持续推进5G建设,截至今年5月,我国5G基站总数已达284.4万个,覆盖全国所有地级市、县城城区和9成以上的乡镇镇区,实现“镇镇通5G”,全面覆盖了从城市到农村的延伸。 依托5G网络的技术优势,智慧乡村…

有没有好用的微信管理软件?解决企业营销管理痛点

企业营销管理痛点: 1、如何提高员工跟进客户的能力和效率? 2、怎么杜绝飞单私单工作怠慢等问题? 3、微信好友太多无法实现精准营销? 4、如何第一时间知道员工的违规行为? 多微信聚合聊天 多个微信号聚合在一个界面…

Linux 打开U盘硬盘等报错 file type exfat not configured in kernel

目录 原因: 查看系统文件系统和当前系统版本 回归正题,如何解决报错 在centons 7中打开U盘,报错file type exfat not configured in kernel。 原因: 这是因为Linux采用的文件系统和我U盘的文件系统不一致引起。如下图&#xf…

在线ppt转pdf如何转换?用这一个方法就够了

在线PPT转PDF是一种将PPT文件转换为PDF格式的便捷且常用的工具。随着科技的发展,PPT已经成为了商务、教育等领域中最常用的演示工具之一。PDF格式具有较好的稳定性和兼容性。PPT文件可能因为不同的操作系统、软件版本或字体缺失等原因而导致显示不一致或乱码等问题&…

【数据结构练习】单链表OJ题(二)

目录 一、相交链表二、环形链表1三、环形链表2四、链表分割五、复制带随机指针的链表 一、相交链表 题目: 示例: 注意:不能根据节点的值来比较是否相交,而是根据节点在内存中是否指向相同的位置。 例如以上图: 链表…

ATFX汇市:美元指数疯狂上涨,英镑单日贬值近1%

环球汇市行情摘要—— 昨日,美元指数上涨0.55%,收盘在103.95点, 欧元贬值0.50%,收盘价1.0810点; 日元贬值0.69%,收盘价145.84点; 英镑贬值0.98%,收盘价1.2600点; 瑞…

火山引擎边缘云,助你沉浸式回忆童年

发现了吗?在抖音、西瓜视频上能观看4K修复的经典港片了!得益于抖音、中国电影资料馆、火山引擎共同发起的“经典香港电影修复计划”,我们童年时期看过的《大话西游之大圣娶亲》《武状元苏乞儿》等22部港片以更清晰、流畅、颜色饱满的状态回归…

基于Java+SpringBoot+vue前后端分离可盈保险合同管理系统设计实现

博主介绍:✌全网粉丝30W,csdn特邀作者、博客专家、CSDN新星计划导师、Java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ 🍅文末获取源码联系🍅 👇🏻 精彩专…

【数据库】使用ShardingSphere+Mybatis-Plus实现读写分离

书接上回:数据库调优方案中数据库主从复制,如何实现读写分离 ShardingSphere 实现读写分离的方式是通过配置数据源的方式,使得应用程序可以在执行读操作和写操作时分别访问不同的数据库实例。这样可以将读取操作分发到多个从库(从…

十二、pikachu之URL重定向

文章目录 1、URL重定向概述2、实战3、URL跳转的几种方式:3.1 META标签内跳转3.2 javascript跳转3.3 header头跳转 1、URL重定向概述 不安全的url跳转问题可能发生在一切执行了url地址跳转的地方。如果后端采用了前端传进来的(可能是用户传参,或者之前预埋…

Allegro平台如何利用测评打造爆款产品,优化listing提升曝光度!

Allegro是波兰本土最大的电商平台,1999年由波兰人自己创造成立,75%的波兰人都知道该网站,Allegro的品牌认知度在波兰高达98%。也是东欧“最大”拍卖网站。 波兰的消费者非常熟悉Allegro平台,大部分波兰人如果想要购买商品&#x…