一、问题描述
当地图中拖动缩放事件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>