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

需求背景:
当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。本文使用的是(echarts v5.4.0)
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
 }
}

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

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

  • 3
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
Echarts 提供了图表实例的 `setOption` 方法中可以配置事件处理函数,通过监听鼠标事件实现拖动功能。 具体实现方法如下: 1. 创建一个全局变量 `mousedown`,用于标记鼠标左键是否按下,初始值为 `false`。 2. 在 `option` 中配置 `grid` 的 `containLabel` 属性为 `true`,表示网格区域包含坐标轴标签。 3. 在 `option` 中配置 `dataZoom` 组件,设置 `type` 为 `'inside'`,表示内置的数据区域缩放组件。 4. 在 `option` 中配置 `toolbox` 工具箱组件,设置 `feature` 属性中的 `dataZoom` 为 `{}`,表示数据区域缩放工具。 5. 在 `option` 中配置 `series` 的 `itemStyle` 属性的 `normal` 属性中,设置 `opacity` 属性为 `0.8`,表示图表项的透明度。 6. 在 `option` 中配置 `series` 的 `label` 属性的 `normal` 属性中,设置 `show` 属性为 `true`,表示显示图表项的标签。 7. 在 `option` 中配置 `series` 的 `emphasis` 属性的 `label` 属性中,设置 `show` 属性为 `false`,表示不显示图表项的标签。 8. 在 `option` 中配置 `series` 的 `itemStyle` 属性的 `emphasis` 属性中,设置 `opacity` 属性为 `1`,表示图表项的透明度。 9. 在 `option` 中配置 `series` 的 `markPoint` 属性,用于标记拖动区域的起始点和结束点。 10. 在 `option` 中配置 `tooltip` 组件,设置 `triggerOn` 属性为 `'mousemove'`,表示鼠标移动时触发提示框。 11. 在 `option` 中配置 `brush` 组件,用于选择图表项或拖动区域。 12. 在 `option` 中配置 `brush` 组件的 `toolbox` 属性,表示在工具箱中显示选择和清除按钮。 13. 在 `option` 中配置 `brush` 组件的 `brushLink` 属性,表示与数据区域缩放组件联动。 14. 在 `option` 中配置 `brush` 组件的 `seriesIndex` 属性,表示选择或拖动的系列索引。 15. 在 `option` 中配置 `brush` 组件的 `throttleType` 属性为 `'debounce'`,表示拖动过程中节流处理。 16. 通过监听 `mousedown`、`mousemove` 和 `mouseup` 事件实现鼠标左右键的拖动功能。 具体代码实现如下: ```javascript var myChart = echarts.init(document.getElementById('myChart')); var mousedown = false; // 鼠标左键是否按下 var start, end; // 拖动区域的起始点和结束点 myChart.setOption({ grid: { containLabel: true }, dataZoom: { type: 'inside' }, toolbox: { feature: { dataZoom: {} } }, series: [{ type: 'scatter', data: [ [10.0, 8.04], [8.0, 6.95], [13.0, 7.58], [9.0, 8.81], [11.0, 8.33], [14.0, 9.96], [6.0, 7.24], [4.0, 4.26], [12.0, 10.84], [7.0, 4.82], [5.0, 5.68] ], itemStyle: { normal: { opacity: 0.8 }, emphasis: { opacity: 1 } }, label: { normal: { show: true }, emphasis: { show: false } }, markPoint: { symbolSize: 50, label: { formatter: function(params) { var x = params.value[0]; var y = params.value[1]; return '(' + x.toFixed(2) + ', ' + y.toFixed(2) + ')'; } }, data: [{ name: '起始点', coord: [0, 0] }, { name: '结束点', coord: [0, 0] }] } }], tooltip: { triggerOn: 'mousemove' }, brush: { toolbox: ['rect', 'polygon', 'lineX', 'lineY', 'keep', 'clear'], brushLink: 'all', seriesIndex: 0, throttleType: 'debounce' } }); myChart.on('mousedown', function(params) { if (params.event.which === 1) { // 鼠标左键按下 mousedown = true; start = [params.event.offsetX, params.event.offsetY]; end = [params.event.offsetX, params.event.offsetY]; myChart.setOption({ series: [{ markPoint: { data: [{ name: '起始点', coord: myChart.convertFromPixel('grid', start) }, { name: '结束点', coord: myChart.convertFromPixel('grid', end) }] } }] }); } else if (params.event.which === 3) { // 鼠标右键按下 mousedown = false; } }); myChart.on('mousemove', function(params) { if (mousedown) { // 鼠标左键按下 end = [params.event.offsetX, params.event.offsetY]; myChart.setOption({ series: [{ markPoint: { data: [{ name: '起始点', coord: myChart.convertFromPixel('grid', start) }, { name: '结束点', coord: myChart.convertFromPixel('grid', end) }] } }] }); } }); myChart.on('mouseup', function(params) { if (params.event.which === 1) { // 鼠标左键释放 mousedown = false; end = [params.event.offsetX, params.event.offsetY]; myChart.dispatchAction({ type: 'brush', areas: [{ brushType: 'rect', coordRange: [ myChart.convertFromPixel('grid', start), myChart.convertFromPixel('grid', end) ] }] }); myChart.setOption({ series: [{ markPoint: { data: [{ name: '起始点', coord: [0, 0] }, { name: '结束点', coord: [0, 0] }] } }] }); } }); ``` 上述代码实现了鼠标左键拖动选择区域,在选择区域内可以拖动鼠标右键进行平移,也可以使用内置的数据区域缩放组件或工具箱中的数据区域缩放工具进行缩放。
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值