Echarts中拖动缩放事件georoam影响click事件的解决办法

一、问题描述

当地图中拖动缩放事件georoam与点击事件同时存在时,拖动事件会影响click事件,可能导致点击地图区域时,触发不了click事件,而是触发georoam事件;或者点击地图上的选点时能够触发click事件,但是点击区域时触发不了的情况

二、问题视频

地图事件

三、解决办法

1、添加全局控制
2、添加鼠标按下和抬起事件,并控制全局参数
3、在拖动缩放事件georoam中控制全局参数,使点击时不触发georoam事件

// 添加全局控制
let isMouseDown = ref(false)
// 添加鼠标按下和抬起事件,并控制全局参数
 myChart.on('mousedown', () => (isMouseDown.value = true))
 myChart.on('mouseup', (e) => {
   isMouseDown.value = false
   moveEcharts(e)
 })
// 在拖动缩放事件georoam中控制全局参数,使点击时不触发georoam事件
 myChart.on('georoam', function (params) {
   if (isMouseDown.value) return // 过滤通过鼠标拖拽触发的georoam
     moveEcharts(params)
 })
// click事件与moveEcharts方法
 myChart.on('click', function (params) {
      console.log(params)

      if (!params) return
      let obj = params
      if (obj.componentType == 'geo' && !curzone.value) {
        myChart.dispose()
        console.log('点击省份')
        initechart(obj.name)
        curzone.value = obj.name
        getArreaList('city', obj.name)
      } else if (obj.componentType == 'series') {
        cityTitle.value = obj.name
        console.log('点击城市', obj.name)
        listQuery.value.county = cityList.value.find((item) => item.name == obj.name).code
        showPop.value = true
        getList(cityTitle.value)
      }
    })
function moveEcharts(params) {
    var option = myChart.getOption() //获得option对象
    // console.log('option的值:', option)
    // 获取当前的缩放和中心位置
    const currentZoom = option.geo[0].zoom
    const currentCenter = option.geo[0].center

    // 遍历所有geo组件,更新它们的缩放和中心位置
    option.geo.forEach((geo, index) => {
      // 保持geo组件的可交互性
      geo.silent = false // 确保geo组件可以接收事件
      geo.selectedMode = 'single' // 保持选中模式

      if (params.zoom != null && params.zoom != undefined) {
        // 捕捉到缩放时
        geo.zoom = currentZoom // 所有geo组件的缩放等级跟着一起改变
      }
      geo.center = currentCenter // 所有geo组件的中心位置随着一起改变
    })
    isMouseDown.value = false
    myChart.setOption(option, { silent: true }) //设置option
  }

四、全部代码

<script setup>
import {
  getMapNewsList,
  getMapNewsByType,
  getMapNewsByPCode,
  getMapNewsBypermission
} from '@/api/screen'
import * as echarts from 'echarts'
import axios from 'axios'
import { useStore } from 'vuex'

const store = useStore()
const cityTitle = ref('')
const fglTotal = ref('1.3W')
const showPop = ref(false)
const loading = ref(false)
const paginationParam = ref({
  currentPage: 1,
  pageSize: 10,
  total: 0
})
const listQuery = ref({
  current: 1,
  size: 10,
  province: '370000',
  city: null,
  county: null,
  system: null
})
const cityList = ref([])
const tableColumns = reactive([
  {
    align: 'left',
    prop: 'title',
    label: '稿件标题',
    width: 383,
    showOverflowTooltip: true
  },

  {
    align: 'left',
    prop: 'createTime',
    label: '发布时间',
    width: 188,
    showOverflowTooltip: true
  },
  {
    align: 'center',
    prop: 'createName',
    width: 98,
    label: '签发人'
  }
])
const tableData = ref([])

watch(
  () => store.state.users.screenZfType,
  (val) => {
    switch (val) {
      case 'systhese':
        listQuery.value.system = null
        break
      case 'zfw':
        listQuery.value.system = 'pal'
        break
      case 'fy':
        listQuery.value.system = 'court'
        break
      case 'jcy':
        listQuery.value.system = 'ppo'
        break
      case 'ga':
        listQuery.value.system = 'psb'
        break
      case 'sfxz':
        listQuery.value.system = 'doj'
        break
      default:
        listQuery.value.system = null
        break
    }
    if (tableData.value && tableData.value.length > 0) {
      getList(cityTitle.value)
    }
  },
  {
    immediate: true
  }
)

const curzone = ref()
//获取稿件列表
function getList(city) {
  console.log(cityList.value)

  cityList.value.forEach((item) => {
    if (item.type == 'county' && item.name == city) {
      listQuery.value.city = item.pcode
      listQuery.value.county = item.code
    } else if (item.type == 'city' && item.name == city) {
      listQuery.value.city = item.code
      listQuery.value.county = null
    }
  })

  getMapNewsList(listQuery.value)
    .then((res) => {
      tableData.value = res.data.list || []
      paginationParam.value.total = res.data.total
    })
    .catch(() => {
      tableData.value = []
      paginationParam.value.total = 0
    })
}
//根据类型获取区域列表
function getArreaList(type, name) {
  getMapNewsByType(type).then((res) => {
    if (type === 'province') {
      listQuery.value.province = res.data[0].code
    }

    let cityCode = res.data[0].code
    let PCode = type === 'city' ? cityList.value.find((item) => item.name == name).code : cityCode

    getArreaListByPCode(PCode)
  })
}
//根据父级code获取区域列表
function getArreaListByPCode(pCode) {
  getMapNewsByPCode(pCode).then((res) => {
    cityList.value = res.data || []
    // listQuery.value.province = res.data[0].pcode
  })
}

//关闭弹窗
function closePop() {
  showPop.value = false
  listQuery.value.current = 1
  paginationParam.value.currentPage = 1
}
let isMouseDown = ref(false)
function initechart(site) {
  var chartDom = document.getElementById('myechart')
  var myChart = echarts.init(chartDom)
  var customerBatteryCityData = [
    { name: '济南市', value: 5 },
    { name: '济宁市', value: 4 },
    { name: '青岛市', value: 3 },
    { name: '威海市', value: 2 },
    { name: '烟台市', value: 1 },
    { name: '菏泽市', value: 1 },
    { name: '日照市', value: 1 },
    { name: '临沂市', value: 1 },
    { name: '枣庄市', value: 1 },
    { name: '滨州市', value: 0 },
    { name: '德州市', value: 1 },
    { name: '聊城市', value: 1 },
    { name: '泰安市', value: 1 },
    { name: '淄博市', value: 0 },
    { name: '潍坊市', value: 0 }
  ]
  var img2 =
    'image://'

  var option

  axios.get(`./static/map/${site}.json`).then((res) => {
    echarts.registerMap('Shandong', res.data)
    // 获取标记城市的坐标点
    let allpoint = res.data.features
    let citydata = allpoint.map((item, idx) => {
      return {
        name: item.properties.name,
        value: item.properties.center
      }
    })

    option = {
      // tooltip: {
      //   trigger: 'item',
      //   showDelay: 0,
      //   transitionDuration: 0.2
      // },
      geo: [
        {
          map: 'Shandong',
          zoom: 1,
          silent: false, // 允许接收事件
          select: {
            disabled: false, // 启用选中
            itemStyle: {
              areaColor: '#389BB7' // 选中样式
            }
          },
          layoutCenter: ['50%', '50%'], //位置
          layoutSize: '100%', //大小
          label: {
            show: curzone.value ? true : false,
            show: false,
            color: '#fff',
            fontSize: 16
          },
          itemStyle: {
            borderColor: 'rgba(147, 235, 248, 1)',
            borderWidth: 1,
            areaColor: {
              type: 'radial',
              x: 0.5,
              y: 0.5,
              r: 0.8,
              colorStops: [
                {
                  offset: 0,
                  color: 'RGBA(11, 75, 231, 1)' // 0% 处的颜色
                },
                {
                  offset: 1,
                  color: 'RGBA(6, 146, 243, 1)' // 100% 处的颜色
                }
              ],
              globalCoord: false // 缺省为 false
            },
            shadowBlur: 10,
            shadowColor: 'rgba(128, 217, 248, 1)',
            shadowOffsetX: -2,
            shadowOffsetY: 2
          },
          emphasis: {
            label: {
              show: curzone.value ? true : false,
              color: '#fff'
            },
            itemStyle: {
              areaColor: '#389BB7'
            }
          },
          top: 0,
          roam: true // 'scale' 或 'move'
        },
        // 重影
        {
          map: 'Shandong',
          zlevel: -1,
          zoom: 1,
          roam: true,
          layoutCenter: ['50%', '52%'],
          layoutSize: '100%', //大小
          select: {
            disabled: true
          },
          silent: true,
          itemStyle: {
            areaColor: '#031525',
            borderWidth: 0,
            shadowColor: '#D79D3D',
            shadowBlur: 30,
            shadowOffsetX: -5,
            shadowOffsetY: 10
          }
        }
      ],
      series: [
        {
          name: '城市坐标点',
          type: 'scatter',
          coordinateSystem: 'geo',
          symbolSize: 10,
          label: {
            formatter: '{b}',
            position: 'top',
            color: '#fff',
            fontSize: 16,
            // show: curzone.value ? false : true
            show: false
          },

          itemStyle: {
            borderColor: 'RGBA(255, 214, 44, 1)',
            color: '#Fff',
            borderWidth: 3,
            shadowColor: 'RGBA(255, 214, 44, 1)',
            shadowOffsetY: 0,
            shadowBlur: 10,
            areaColor: 'RGBA(255, 214, 44, 1)'
          },
          emphasis: {
            areaColor: 'rgba(29,85,139,.6)'
          },

          data: citydata
        },
        //柱状体的主干
        {
          type: 'lines',
          effect: {
            show: false,
            symbolSize: 5 // 图标大小
          },
          lineStyle: {
            width: 6, // 尾迹线条宽度
            color: 'rgb(22,255,255, .6)',
            opacity: 1, // 尾迹线条透明度
            curveness: 0 // 尾迹线条曲直度
          },
          label: {
            show: 0,
            position: 'end',
            formatter: '245'
          },
          silent: true,
          data: lineData()
        },
        // 柱状体的顶部
        {
          type: 'scatter',
          coordinateSystem: 'geo',
          geoIndex: 0,
          label: {
            normal: {
              show: true,
              formatter: function (params) {
                var name = params.data[2].name
                // var text = `{tline|${name}}:{fline|${value}}`
                var text = `{tline|城市}:{fline|${name}}`
                return text
              },
              color: '#fff',
              rich: {
                fline: {
                  // padding: [0, 25],
                  color: '#fff',
                  fontSize: 14,
                  fontWeight: 400
                },
                tline: {
                  // padding: [0, 27],
                  color: '#ABF8FF',
                  fontSize: 12
                }
              }
            },
            emphasis: {
              show: true
            }
          },
          itemStyle: {
            color: '#00FFF6',
            opacity: 1
          },
          symbol: img2,
          symbolSize: [100, 50],
          symbolOffset: [0, -20],
          data: scatterData()
        },
        // 柱状体的底部
        {
          geoIndex: 0,
          type: 'effectScatter',
          coordinateSystem: 'geo',
          rippleEffect: {
            scale: 10,
            brushType: 'stroke'
          },
          showEffectOn: 'render',
          itemStyle: {
            color: '#00FFFF'
          },
          label: {
            formatter: '{b}',
            position: 'bottom',
            color: '#fff',
            fontSize: 12,
            distance: 10,
            show: false
          },
          symbol: 'circle',
          symbolSize: [10, 5],
          itemStyle: {
            // color: '#F7AF21',
            color: 'rgb(22,255,255, 1)',
            opacity: 1
          },
          data: scatterData2()
        }
      ]
    }
    myChart.setOption(option)

    // 动态计算柱形图的高度(定一个max)
    function lineMaxHeight() {
      const maxValue = Math.max(...customerBatteryCityData.map((item) => item.value))
      return 0.9 / maxValue
    }
    // 柱状体的主干
    function lineData() {
      return customerBatteryCityData
        .map((item) => {
          const cityInfo = citydata.find((city) => city.name === item.name)
          if (!cityInfo) {
            console.error(`未找到 ${item.name} 的城市信息`)
            return null
          }
          const coord = cityInfo.value
          return {
            coords: [coord, [coord[0], coord[1] + item.value * lineMaxHeight()]]
          }
        })
        .filter(Boolean)
    }
    // 柱状体的顶部
    function scatterData() {
      return customerBatteryCityData
        .filter((item) => item.value > 0)
        .map((item) => {
          const cityInfo = citydata.find((city) => city.name === item.name)
          if (!cityInfo) {
            console.error(`未找到 ${item.name} 的城市信息`)
            return null
          }
          const coord = cityInfo.value
          return [coord[0], coord[1] + item.value * lineMaxHeight(), item]
        })
        .filter(Boolean)
    }
    // 柱状体的底部
    function scatterData2() {
      return customerBatteryCityData
        .filter((item) => item.value > 0)
        .map((item) => {
          const cityInfo = citydata.find((city) => city.name === item.name)
          if (!cityInfo) {
            console.error(`未找到 ${item.name} 的城市信息`)
            return null
          }
          return {
            name: item.name,
            value: cityInfo.value
          }
        })
        .filter(Boolean)
    }

    myChart.on('mousedown', () => (isMouseDown.value = true))
    myChart.on('mouseup', (e) => {
      console.log('抬起', e)
      isMouseDown.value = false
      moveEcharts(e)
    })
    myChart.on('click', function (params) {
      console.log(params)

      if (!params) return
      let obj = params
      if (obj.componentType == 'geo' && !curzone.value) {
        myChart.dispose()
        console.log('点击省份')
        initechart(obj.name)
        curzone.value = obj.name
        getArreaList('city', obj.name)
      } else if (obj.componentType == 'series') {
        cityTitle.value = obj.name
        console.log('点击城市', obj.name)
        listQuery.value.county = cityList.value.find((item) => item.name == obj.name).code
        showPop.value = true
        getList(cityTitle.value)
      }
    })
    // georoam事件处理

    myChart.on('georoam', function (params) {
      if (isMouseDown.value) return // 过滤通过鼠标拖拽触发的georoam
      console.log(params)
      moveEcharts(params)
    })
  })
  function moveEcharts(params) {
    var option = myChart.getOption() //获得option对象
    // console.log('option的值:', option)
    // 获取当前的缩放和中心位置
    const currentZoom = option.geo[0].zoom
    const currentCenter = option.geo[0].center

    // 遍历所有geo组件,更新它们的缩放和中心位置
    option.geo.forEach((geo, index) => {
      // 保持geo组件的可交互性
      geo.silent = false // 确保geo组件可以接收事件
      geo.selectedMode = 'single' // 保持选中模式

      if (params.zoom != null && params.zoom != undefined) {
        // 捕捉到缩放时
        geo.zoom = currentZoom // 所有geo组件的缩放等级跟着一起改变
      }
      geo.center = currentCenter // 所有geo组件的中心位置随着一起改变
    })
    isMouseDown.value = false
    myChart.setOption(option, { silent: true }) //设置option
  }
}
function goall() {
  initechart('山东省')
  curzone.value = null
  getArreaList('province', '')
}
onMounted(() => {
  initechart('山东省')
  getArreaList('province', '')
})

function changeCurrentPage(page) {
  paginationParam.value.currentPage = page
  listQuery.value.current = page
  getList(cityTitle.value)
}
</script>

<template>
  <div class="screwh">
    <p class="city">
      <span class="all" @click="goall">全部</span>
      <span v-if="curzone" class="curCity">> {{ curzone }}</span>
    </p>
    <div id="myechart"></div>
  </div>
  <popup-template
    :showPop="showPop"
    :loading="loading"
    :cityTitle="cityTitle"
    :baseColumns="tableColumns"
    :tableData="tableData"
    :paginationParam="paginationParam"
    @closePop="closePop"
    @changeCurrentPage="changeCurrentPage"
  >
    <!-- <template #header>
      <div class="popup-tips">平台发稿量:{{ fglTotal }}</div>
    </template> -->
  </popup-template>
</template>


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

岩岩很哇塞!

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值