以前写过 vue项目中封装echarts的比较优雅的方式,vue2项目中封装echarts地图比较优雅的方式,这次对地图做一个小拓展,在vue2项目中封装echarts地图比较优雅的方式 的基础上进行。
能学到的知识
- 1、echarts地图背景图设置,可以设置随意的地图纹理
- 2、echarts地图打点,显示数据
效果图
虽然不明显,但是打点是真的在闪动的
实现
在 vue2项目中封装echarts地图比较优雅的方式 的基础上稍微做点改变。
echarts地图模块封装
|-- src
|-- components
|-- chart
|-- options // 存放各种图表的option
|-- map // 地图option
|-- index.js
具体代码如下:
// import echarts from 'echarts'
import * as echarts from 'echarts'
import mapbg from '@/assets/images/map-bg.png'
const getSimpleMap = (jsonMap, data, config) => {
if (!echarts.getMap(jsonMap.mark)) {
echarts.registerMap(jsonMap.mark, jsonMap.json)
}
const convertData = (data) => {
console.log(data)
var res = []
for (var i = 0; i < data.length; i++) {
const mapData = data[i].hoverObj
const center = data[i].mapProperty.center
if (center && mapData) {
res.push({
name: mapData.name,
value: center.concat(mapData.num)
})
}
}
return res
}
const defaultConfig = {
tooltip: {
// 窗口外框
trigger: 'item',
padding: 0,
borderWidth: 0,
borderColor: '#FFFFFF',
backgroundColor: '#FFFFFF',
formatter: (params) => {
const { data } = params
const str = `<div style="width:300px;height: 98px;box-shadow: 0px 4px 20px 0px rgba(0, 0, 0, 0.8);
color: #fff;text-align:left;border-radius: 6px;">
<div style="background-color: rgba(102, 182, 255, 1);height: 44px;line-height: 44px;font-size:14px;font-weight:400;border-top-left-radius: 6px;border-top-right-radius: 6px;display: flex;align-items: center;">
<img style="width: 13px;height:16px;margin-left: 24px;margin-right: 3px;" src="images/map-ic.png">${data.name}
</div>
<div style="width: 100%;height:54px;display: flex;flex-wrap: wrap;">
<div style="display: flex;justify-content: space-between;width: 100%;padding-left:15px;padding-right: 15px;">
<div style="display:flex;align-items:center;width:132px;">
<div style="font-size: 12px;color: #555555;margin-right:10px;">区号:</div>
<div style="font-size: 14px;color: #333333;">${data.hoverObj == null ? '' : data.hoverObj.adcode}</div>
</div>
</div>
</div>
</div>`
return str
}
},
geo: {
map: jsonMap.mark,
type: 'map',
layoutCenter: ['50%', '50%'],
layoutSize: '150%',
zoom: 0.65,
roam: false,
// 样式一
// itemStyle: {
// normal: {
// areaColor: 'rgba(13, 110, 191, 1)',
// shadowColor: 'rgba(13, 110, 191, 0.5)',
// shadowOffsetX: 6,
// shadowOffsetY: 12
// }
// }
// 样式二
itemStyle: {
normal: {
borderColor: '#AEF3FF',
borderWidth: 2,
areaColor: {
image: mapbg, //
repeat: 'repeat' // // 是否平铺,可以是 'repeat-x', 'repeat-y', 'no-repeat'
},
shadowColor: 'rgba(13, 110, 191, 0.5)',
shadowOffsetX: 6,
shadowOffsetY: 12
}
}
},
series: [
{
type: 'map',
map: jsonMap.mark, // 自定义扩展图表类型
zoom: 0.65, // 缩放
animationDuration: 1200,
// 点击选中后的效果
select: {
label: {
color: '#FFFFFF',
fontSize: 12,
fontWeight: 400
},
// 清除点击选中后的背景色
itemStyle: {
color: null
}
},
// 样式一
// itemStyle: {
// // 地图样式
// normal: {
// borderColor: '#FFFFFF',
// borderWidth: 3,
// areaColor: 'rgba(201, 229, 255, 1)'
// }
// },
// 样式二
itemStyle: {
// 地图样式
normal: {
borderColor: '#AEF3FF',
borderWidth: 2,
areaColor: {
image: mapbg, //
repeat: 'repeat' // // 是否平铺,可以是 'repeat-x', 'repeat-y', 'no-repeat'
}
}
},
// 样式一
// label: {
// show: true,
// color: '#666666',
// fontSize: 12,
// fontWeight: 400
// },
// 样式二
label: {
show: true,
color: '#FFFFFF',
fontSize: 12,
fontWeight: 400
},
emphasis: {
// 鼠标移入动态的时候显示的默认样式
label: {
show: true,
color: '#FFFFFF',
fontSize: 15,
fontWeight: 600
},
itemStyle: {
areaColor: 'rgba(102, 182, 255, 0.7)',
borderColor: '#FFFFFF',
borderWidth: 2
}
},
layoutCenter: ['50%', '50%'],
layoutSize: '150%',
data: data
},
{
type: 'effectScatter',
coordinateSystem: 'geo',
symbol: 'image://',
symbolSize: [16, 16],
// symbolOffset: [],
label: {
normal: {
show: true,
offset: [0, -18],
textStyle: {
color: '#fff',
fontSize: 12,
fontWeight: 500
},
formatter(value) {
return '{a|' + value.data.value[2] + '}'
},
rich: {
a: {
color: '#FFFFFF',
backgroundColor: '#06406A',
width: 28,
height: 16,
lineHeight: 16,
align: 'center',
borderWidth: 1,
borderColor: '#79D6FE',
borderRadius: 2,
padding: [0, 4, 0, 4]
}
}
}
},
itemStyle: {
normal: {
color: 'rgba(255, 178, 76, 1)'
}
},
data: convertData(data),
showEffectOn: 'render',
rippleEffect: {
brushType: 'stroke'
},
hoverAnimation: true,
zlevel: 9
}
]
}
const opt = Object.assign({}, defaultConfig, config)
const { legend, tooltip, series, geo, grid } = opt
const chartOpt = {
grid,
legend,
tooltip,
geo,
series
}
return chartOpt
}
export default {
getSimpleMap
}
新增的代码,最好在Visual Studio Code 的Timeline 里面对比查看,只是添加了地图纹理和地图打点。
地图纹理 (components–chart–options–map–index.js)
实现代码在:
geo 样式二的itemStyle ,这里控制鼠标移入地图模块时的纹理series 样式二的itemStyle ,这里控制地图默认状态下的纹理
地图打点 (components–chart–options–map–index.js)
实现代码在effectScatter 模块,主要是自定义了打点的图标,还有自定义了显示信息的label 的样式,这里可以根据设计随意实现。
打点数据在convertData 方法里面处理,可以根据接口数据看着办。
数据 (views–echartMapTest–index.js)
因为要在地图上打点,需要坐标,加载数据的时候,需要加上地图的坐标,主要是加上这一行,properties 包含了地图所有基本数据了
mapProperty: item.properties
具体如下:
initMap(url) {
mapRequest(url).then((res) => {
const mapData = res.data
const jsonMap = { mark: this.mapName, json: mapData }
const data = mapData.features.map((item) => {
const { name, adcode } = item.properties
let hoverObj = {}
const objIndex = this.$utils.findElem(this.mapPopData, 'adcode', adcode)
if (objIndex !== -1) {
hoverObj = this.mapPopData[objIndex]
} else {
hoverObj = null
}
return {
name,
mapProperty: item.properties,
hoverObj: hoverObj
}
})
this.mapOpt = this.$eChartFn.getSimpleMap(jsonMap, data)
}).catch((err) => {
console.log(err, '加载地图失败')
})
}
代码总览
涉及的文件如下(具体参考代码 ):
|-- public
|-- data
|-- 4401.json
|-- mapdata.json
|-- src
|-- api
|-- map.js // 获取地图geojson数据、地图弹窗接口模拟数据
|-- components
|-- chart
|-- index.vue // 图表单文件组件,供界面调用
|-- index.js // 实现自动化导入options里的图表option
|-- options // 存放各种图表的option
|-- map // 地图option
|-- index.js
|-- views
|-- echartMapTest // 实例所在
|-- index.vue
|-- index.scss
|-- index.js
|-- utils
|---utils.js
|-- main.js // 全局引入echarts图表
代码
按代码总览 的目录去代码里找着看就行了。
总结
echarts 地图的纹理不只可以纯色或者渐变色的,还可以根据设计随意设置。项目中UI设计的很好看,这里只是随意截取了纹理的一部分做了个例子。
以上,基本只要UI设计的出来,就可以实现任意纹理样式的地图。
|