在数据可视化中,地图是很重要的一个环节,很多时候需要展现的不仅是国家地图,还需要能从国家进入到省市。这个逐级进入的过程就是我们今天说的地图下钻。
地图下钻看起来很屌、很高大上,但是仔细琢磨一下,技术实现上真的很简单。
文章目录
- [1. 本章主要功能介绍](#1. 本章主要功能介绍)
- [1. 地图下钻实现思路(本章核心)](#1. 地图下钻实现思路(本章核心))
- [2. 下钻代码实现(本章核心)](#2. 下钻代码实现(本章核心))
- [3. 完整代码](#3. 完整代码)
-
- [3.1 vue部分](#3.1 vue部分)
- [3.2 mapData.js 部分](#3.2 mapData.js 部分)
- [4. 接口数据](#4. 接口数据)
1. 本章主要功能介绍
- 自定义标注样式&自定义tooltip弹窗样式(上章内容) echarts 自定义标注样式&自定义tooltip弹窗样式
- 自定义地图贴图样式(上章内容)echarts 实现中国geo地图自定义贴图实例
- 下拉菜单选择切换地图省份(本章核心)
- 点击地图切换地图省份(本章核心)
- 进入全屏/退出全屏(其他)
1. 地图下钻实现思路(本章核心)
mounted
里面首先初始化一个中国地图的数据,provinceCode默认为100000
- 创建echarts实例,获取中国地图的
geoJSON
数据 - 拿到geoJSON 数据调用
initEcharts
函数区创建echarts - geoJSON 数据对
echarts
进行渲染上图 - 上图完成后,通过
this.myChart.on("click",()=>{})
事件活动点击的对象,params.name
可以拿到点击的省份名称,修改provinceCode
值为点击的省份 - 监听
provinceCode
改变,重新执行第一递归循环即可
2. 下钻代码实现(本章核心)
3. 完整代码
3.1 vue部分
html
<template>
<div class="map">
<div class="navProvince">
<select
v-model="provinceCode"
@change="changeProvince($event.target.value)"
>
<option
:value="item.code"
v-for="(item, index) in provinceData()"
:key="index"
>
{{ item.name }}
</option>
</select>
<img src="@/assets/img/dataView/down.webp" alt="" srcset="" />
</div>
<div class="navBase" v-if="baseId">
<select v-model="baseId" @change="changeBaseId($event.target.value)">
<option
:value="item.baseId"
v-for="(item, index) in baseData"
:key="index"
>
{{ item.name }}
</option>
</select>
<img src="@/assets/img/dataView/down.webp" alt="" srcset="" />
</div>
<div
class="screen"
@click="toggleFullScreen"
:title="isFullscreen ? '退出全屏' : '进入全屏'"
>
<img src="@/assets/img/dataView/screen.webp" alt="" srcset="" />
<span>{{ isFullscreen ? "退出全屏" : "进入全屏" }}</span>
</div>
<div ref="myEchart" id="personnel"></div>
</div>
</template>
<script>
import "echarts-gl"; //3D地图插件
import screenfull from "screenfull";
import axios from "axios";
import {
provinceObj,
provinceData,
getCodeByName,
formatHtml,
} from "./js/mapData";
import { API_listBase } from "@/api/dataView";
import { mapGetters } from "vuex";
export default {
data() {
return {
provinceData,
isFullscreen: false,
publicUrl: "https://geo.datav.aliyun.com/areas_v3/bound/",
chinaGeoJson: [],
myChart: null,
mapImg: null, //地图底色
mapActiveImg: null, //地图悬浮移入后的底色
baseData: [], //基地数据
provinceCode: 100000,
baseId: "",
};
},
computed: {
...mapGetters({
getProvinceCode: "dataView/getProvinceCode",
getBaseId: "dataView/getBaseId",
}),
},
watch: {
provinceCode(_newData, _oldData) {
this.provinceCode = _newData;
this.$store.commit(
"dataView/SET_PROVINCECODE",
provinceObj[_newData] == "全国" ? "" : _newData
);
this.getList();
},
},
methods: {
// 请求三方地图数据接口
async getGeoJson(jsonName, province) {
let res = await axios.get(`${this.publicUrl}${jsonName}`);
if (res && res.status === 200) {
this.initEcharts(res.data, province);
}
},
// 加载地图数据
initEcharts(geoJson, name) {
this.$echarts.registerMap(name, geoJson);
let option = {
tooltip: {
trigger: "item",
formatter: function (params) {
// 为小图表创建一个容器
if (params.data) {
return params.name + " : " + params.data.data.datas;
}
},
},
geo: {
map: name == "全国" ? "china" : name,
zoom: 1,
roam: false,
itemStyle: {
normal: {
areaColor: {
type: "radial",
x: 0.5,
y: 0.5,
r: 0.8,
colorStops: [
{
offset: 0,
color: "#09132c", // 0% 处的颜色
},
{
offset: 1,
color: "#274d68", // 100% 处的颜色
},
],
globalCoord: true, // 缺省为 false
},
shadowColor: "rgb(13, 48, 92,0.8)", //底层颜色
shadowOffsetX: 10,
shadowOffsetY: 11,
},
},
regions: [
{
show: false,
name: "南海诸岛",
itemStyle: {
areaColor: "rgba(0, 10, 52, 1)",
borderColor: "rgba(0, 10, 52, 1)",
normal: {
opacity: 0,
label: {
show: false,
color: "#009cc9",
},
},
},
},
],
},
series: [
{
type: "map",
roam: false,
label: {
normal: {
show: true,
textStyle: {
color: "#D5E0EE",
},
},
emphasis: {
textStyle: {
color: "rgb(183,185,14)",
},
},
},
itemStyle: {
normal: {
borderColor: "rgb(81, 184, 220)",
borderWidth: 1,
areaColor: {
image: this.mapImg,
repeat: "repeat",
},
},
emphasis: {
areaColor: {
image: this.mapActiveImg,
repeat: "repeat",
},
borderColor: "#2ab8ff",
borderWidth: 1,
shadowColor: "rgba(0, 255, 255, 0.7)",
shadowBlur: 10,
shadowOffsetX: 0,
shadowOffsetY: 1,
label: {
show: false,
},
},
},
zoom: 1,
map: name == "全国" ? "china" : name, //使用
},
{
type: "scatter",
coordinateSystem: "geo",
itemStyle: {
color: "#f00",
},
tooltip: {
trigger: "item",
backgroundColor: "transparent",
formatter: function (params) {
return formatHtml(params.data);
},
},
symbol: `image://${require("@/assets/img/dataView/point.png")}`,
symbolSize: [48, 58],
symbolOffset: [0, 0],
z: 9999,
data: this.baseData,
},
],
};
this.myChart = this.$echarts.init(this.$refs.myEchart);
this.myChart.setOption(option, true);
this.myChart.off("click");
this.myChart.on("click", (params) => {
this.$store.commit("dataView/SET_BASEID", params.data?.baseId);
if (params.data?.provinceCode) {
this.baseId = params.data?.baseId;
}
if (params.data?.provinceCode || getCodeByName(params.name)) {
this.provinceCode =
params.data?.provinceCode || getCodeByName(params.name);
} else {
return this.$message.warning("暂无继续进行地图下钻");
}
});
},
// 切换全屏
toggleFullScreen() {
if (!screenfull.isEnabled) return false;
screenfull.toggle();
},
changeProvince(_data) {
this.baseId = "";
this.$store.commit("dataView/SET_BASEID", "");
},
changeBaseId(_data) {
let params = this.baseData.find((item) => item.baseId == _data);
if (params) {
this.provinceCode = params?.provinceCode;
} else {
return this.$message.warning("切换失败");
}
},
async getList() {
let { code, data } = await API_listBase({
provinceCode: this.getProvinceCode,
baseId: this.getBaseId,
});
if (code === 200) {
this.baseData = data.map((item, _index) => {
return {
...item,
value: item.baseLocation.split(","),
name: item.unitName,
province: item.province,
computilityData: item.computilityData
? item.computilityData
: {
baseCpuserverNumber: 0,
baseCpuScores: 0,
baseIdlecpuScores: 0,
baseMemoryCapacity: 0,
baseIdlememoryCapacity: 0,
baseStorageCapacity: 0,
baseIdlestorageCapacity: 0,
baseUploadBandwidthCapacity: 0,
baseIdleuploadBandwidthCapacity: 0,
baseDownloadBandwidthCapacity: 0,
baseIdledownloadBandwidthCapacity: 0,
baseGpuserverNumber: 0,
baseGpuScores: 0,
baseIdlegpuScores: 0,
baseNpuserverNumber: 0,
baseNpuScores: 0,
baseIdlenpuScores: 0,
baseFp16Computility: 0,
baseIdlefp16Computility: 0,
baseFp32Computility: 0,
baseIdlefp32Computility: 0,
baseGraphicsMemoryCapacity: 0,
baseIdlegraphicsMemoryCapacity: 0,
},
};
});
}
// 初始化中国地图
this.$nextTick(() => {
this.myChart = this.$echarts.init(this.$refs.myEchart);
this.getGeoJson(
`${this.provinceCode}_full.json`,
provinceObj[this.provinceCode]
);
});
},
},
mounted() {
// 地图底色
this.mapImg = document.createElement("img");
this.mapImg.style.height =
this.mapImg.height =
this.mapImg.width =
this.mapImg.style.width =
"100px";
this.mapImg.src =
"";
// 地图悬浮移入后的底色
this.mapActiveImg = document.createElement("img");
this.mapActiveImg.style.height =
this.mapActiveImg.height =
this.mapActiveImg.width =
this.mapActiveImg.style.width =
"100px";
this.mapActiveImg.src =
"";
// 初始化中国地图
this.getList();
},
};
</script>
<style lang="less" scoped>
.map {
flex: 1;
width: 100%;
color: #fff;
position: relative;
.navProvince,
.navBase {
position: absolute;
left: 40px;
width: 160px;
height: 32px;
display: flex;
align-items: center;
justify-content: space-between;
background: url("../../../../assets/img/dataView/nav_bg.webp") no-repeat;
background-size: 160px 32px;
background-position: center;
cursor: pointer;
z-index: 2;
img {
height: 16px;
position: absolute;
top: 50%;
right: 10px;
transform: translateY(-50%);
}
}
.navBase {
left: 220px;
}
.screen {
position: absolute;
right: 40px;
width: 118px;
height: 32px;
display: flex;
align-items: center;
justify-content: center;
background: url("../../../../assets/img/dataView/screen_bg.webp") no-repeat;
background-size: 118px 32px;
background-position: center;
cursor: pointer;
z-index: 2;
img {
height: 24px;
margin-right: 10px;
}
span {
font-weight: normal;
font-size: 16px;
color: #4b94ff;
line-height: 16px;
}
}
}
#personnel {
width: 100%;
height: 100%;
z-index: 1;
}
select {
width: 100%;
padding: 6px 10px 6px 16px;
box-sizing: border-box;
font-size: 16px;
color: #4b94ff;
line-height: 16px;
-webkit-appearance: none;
/* for Chrome, Safari */
-moz-appearance: none;
/* for Firefox */
-ms-appearance: none;
/* for IE10+ */
appearance: none;
-moz-appearance: none;
background: transparent;
-webkit-appearance: none;
-webkit-tap-highlight-color: rgba(0, 0, 0, 0);
-webkit-tap-highlight-color: transparent;
option {
background: transparent;
}
}
</style>
<style scoped lang="less">
/deep/ .tooltip-chart {
background-color: transparent;
width: 520px;
height: 334px;
background: url("../../../../assets/img/dataView/tooltip_bg.webp") no-repeat;
background-size: 100% 100%;
background-position: center;
padding: 16px 25px 16px 20px;
grid-gap: 0 40px;
overflow: auto;
display: grid;
grid-template-columns: repeat(2, 1fr);
.item {
font-weight: 500;
font-size: 13px;
color: #ffffff;
line-height: 20px;
display: flex;
justify-content: space-between;
align-items: center;
span {
&.name{
// color: #4b94ff;
}
&:first-child {
position: relative;
padding-left: 14px;
&::after {
content: "";
position: absolute;
top: 50%;
left: 0;
transform: translateY(-50%);
width: 5px;
height: 5px;
border-radius: 50%;
background: #fff;
}
}
&:last-child {
text-align: left;
}
}
}
}
</style>
3.2 mapData.js 部分
ts
export const provinceObj = {
100000: "全国",
110000: "北京",
120000: "天津",
130000: "河北",
140000: "山西",
150000: "内蒙古",
210000: "辽宁",
220000: "吉林",
230000: "黑龙江",
310000: "上海",
320000: "江苏",
330000: "浙江",
340000: "安徽",
350000: "福建",
360000: "江西",
370000: "山东",
410000: "河南",
420000: "湖北",
430000: "湖南",
440000: "广东",
450000: "广西",
460000: "海南",
500000: "重庆",
510000: "四川",
520000: "贵州",
530000: "云南",
540000: "西藏",
610000: "陕西",
620000: "甘肃",
630000: "青海",
640000: "宁夏",
650000: "新疆",
710000: "台湾",
810000: "香港",
820000: "澳门",
};
export let provinceData = () => {
let arr = [];
for (const key in provinceObj) {
arr.push({
code: key,
name: provinceObj[key],
});
}
return arr;
};
/**
* 根据名称查询对应的键
* @param {*} name
* @returns
*/
export let getCodeByName = (name) => {
for (const [key, value] of Object.entries(provinceObj)) {
if (value === name) {
return key;
}
}
return null; // 如果没有找到
};
/**
* 基地展示数据
* @param {*} param0
* @returns
*/
export let formatHtml = ({ name, computilityData }) => {
return `
<div class="tooltip-chart">
<div class="item">
<span>基地名称</span>
<span class="name">${name}</span>
</div>
<div class="item">
<span>基地CPU服务器数量</span>
<span>${computilityData?.baseCpuserverNumber || 0}</span>
</div>
<div class="item">
<span>基地CPU总核数</span>
<span>${computilityData?.baseCpuScores || 0}</span>
</div>
<div class="item">
<span>基地空闲CPU核数</span>
<span>${computilityData?.baseIdlecpuScores || 0}</span>
</div>
<div class="item">
<span>基地内存空间总容量</span>
<span>${computilityData?.baseMemoryCapacity || 0}</span>
</div>
<div class="item">
<span>基地空闲内存空间容量</span>
<span>${computilityData?.baseIdlememoryCapacity || 0}</span>
</div>
<div class="item">
<span>基地存储空间总容量</span>
<span>${computilityData?.baseStorageCapacity || 0}</span>
</div>
<div class="item">
<span>基地空闲存储空间容量</span>
<span>${computilityData?.baseIdlestorageCapacity || 0}</span>
</div>
<div class="item">
<span>基地上行网络带宽总量</span>
<span>${computilityData?.baseUploadBandwidthCapacity || 0}</span>
</div>
<div class="item">
<span>基地空闲上行网络带宽</span>
<span>${
computilityData?.baseIdleuploadBandwidthCapacity || 0
}</span>
</div>
<div class="item">
<span>基地下行网络带宽总量</span>
<span>${computilityData?.baseDownloadBandwidthCapacity || 0}</span>
</div>
<div class="item">
<span>基地空闲下行网络带宽</span>
<span>${
computilityData?.baseIdledownloadBandwidthCapacity || 0
}</span>
</div>
<div class="item">
<span>基地GPU服务器数量</span>
<span>${computilityData?.baseGpuserverNumber || 0}</span>
</div>
<div class="item">
<span>基地GPU总核数</span>
<span>${computilityData?.baseGpuScores || 0}</span>
</div>
<div class="item">
<span>基地空闲GPU核数</span>
<span>${computilityData?.baseIdlegpuScores || 0}</span>
</div>
<div class="item">
<span>基地NPU服务器数量</span>
<span>${computilityData?.baseNpuServerNumber || 0}</span>
</div>
<div class="item">
<span>基地NPU总核数</span>
<span>${computilityData?.baseNpuScores || 0}</span>
</div>
<div class="item">
<span>基地空闲NPU核数</span>
<span>${computilityData?.baseIdlenpuScores || 0}</span>
</div>
<div class="item">
<span>基地FP16总算力</span>
<span>${computilityData?.baseFp16Computility || 0}</span>
</div>
<div class="item">
<span>基地空闲FP16算力</span>
<span>${computilityData?.baseIdlefp16Computility || 0}</span>
</div>
<div class="item">
<span>基地FP32总算力</span>
<span>${computilityData?.baseFp32Computility || 0}</span>
</div>
<div class="item">
<span>基地空闲FP32算力</span>
<span>${computilityData?.baseIdlefp32Computility || 0}</span>
</div>
<div class="item">
<span>基地显存总量</span>
<span>${computilityData?.baseGraphicsMemoryCapacity || 0}</span>
</div>
<div class="item">
<span>基地空闲显存</span>
<span>${computilityData?.baseIdlegraphicsMemoryCapacity || 0}</span>
</div>
</div>
`;
};
4. 接口数据
- 把moke的接口数据也贴下,方便各位看代码