【vue使用echarts实现地图下钻,修改悬浮框样式】

vue使用echarts实现地图下钻,修改悬浮框样式

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

首页安装echarts

安装命令

npm install echarts

我是放在main.js里面,也可单页面引用

import * as echarts from 'echarts';
Vue.prototype.$echarts = echarts;

完整代码

<template>
	<div class="mapEcharts">
		<div class="map-left">
			<svg @click="backMap" t="1681180771137" class="backIcon icon" viewBox="0 0 1024 1024" version="1.1"
				xmlns="http://www.w3.org/2000/svg" p-id="3427" width="100" height="100">
				<path
					d="M426.666667 384V213.333333l-298.666667 298.666667 298.666667 298.666667v-174.933334c213.333333 0 362.666667 68.266667 469.333333 217.6-42.666667-213.333333-170.666667-426.666667-469.333333-469.333333z"
					p-id="3428" fill="#ffffff"></path>
			</svg>
			<div id="mapEchart"></div>
		</div>
		<div class="map-right">
			<!-- 其他区域 -->
			<div class="top-other">
				<div style="text-align: left;display: flex;align-items: center;">
					<div style="width:3px;height:18px;background: #FBA922;"></div>
					<div style="margin-left: 5px;font-size: 16px;">其他区域</div>
				</div>
				<div class="list">
					<div class="list-item">
						<div class="num">12</div>
						<div class="title">本月销售额</div>
					</div>
					<div class="list-item">
						<div class="num">12</div>
						<div class="title">本月客户家数</div>
					</div>
					<div class="list-item">
						<div class="num">12</div>
						<div class="title">本月毛利</div>
					</div>
				</div>
				<div style="text-align: left;display: flex;align-items: center;margin-top: 20px;">
					<div style="width:3px;height:16px;background: #FBA922;"></div>
					<div style="margin-left: 5px;font-size: 18px;">客户数</div>
				</div>
				<div class="list">
					<div class="list-item">
						<div class="num">12</div>
						<div class="title">电商部</div>
					</div>
					<div class="list-item">
						<div class="num">12</div>
						<div class="title">销售部</div>
					</div>
					<div class="list-item">
						<div class="num">12</div>
						<div class="title">未合作</div>
					</div>
				</div>
			</div>

			<!-- 下方列表 -->
			<div class="table-cont">
				<el-table :data="tableData" height="35vh" style="margin-bottom: 10px"
					element-loading-background="rgba(0, 0, 0, 0.7)" element-loading-text="拼命加载中"
					element-loading-spinner="el-icon-loading" v-loading="loading">
					<el-table-column type="index" align="center" label="序号" width="60"></el-table-column>
					<el-table-column prop="hkDate" align="left" label="省" width="120" show-overflow-tooltip v-if="listAdcode == '100000'"></el-table-column>
					<el-table-column prop="hkDate" align="left" label="市" width="120" show-overflow-tooltip v-if="listAdcode == '610000'"></el-table-column>
					<el-table-column prop="hkDate" align="left" label="区/县" width="120" show-overflow-tooltip v-if="listAdcode == '610600'"></el-table-column>
					<el-table-column prop="orderDate" align="left" label="市场客户数" width="120"
						show-overflow-tooltip></el-table-column>
					<el-table-column prop="customerKey" align="left" label="公司客户数" width="120"
						show-overflow-tooltip></el-table-column>
					<el-table-column prop="customerName" align="left" label="未做客户数" width="150"
						show-overflow-tooltip></el-table-column>
					<el-table-column prop="hkAmount" align="center" label="批连客户" width="120" sortable
						show-overflow-tooltip></el-table-column>
					<el-table-column prop="ratio" align="center" label="诊疗客户" width="110" sortable
						show-overflow-tooltip></el-table-column>
					<el-table-column prop="proceedsAmount" align="center" label="药店客户" sortable width="110"
						show-overflow-tooltip></el-table-column>
					<el-table-column prop="day" align="center" label="当月客户数" width="110" sortable
						show-overflow-tooltip></el-table-column>
					<el-table-column prop="zhy" align="left" label="当月销售额" width="200" show-overflow-tooltip></el-table-column>
					<el-table-column prop="salesErpDepart" align="left" label="新开客户数" width="120"
						show-overflow-tooltip></el-table-column>
					<el-table-column prop="salesPersonName" align="left" label="新开销售额" width="110"
						show-overflow-tooltip></el-table-column>
					<el-table-column prop="hkdh" align="left" label="控销销售额" width="150" show-overflow-tooltip></el-table-column>
					<el-table-column prop="salesmanRatio" label="控销客户数" width="135" show-overflow-tooltip></el-table-column>
				</el-table>
				<!-- 分页 -->
				<pageCmp @refurbish="refurbish" :total="total" :current="pageNum" :pageSize="pageSize"></pageCmp>
			</div>

		</div>

	</div>
</template>

<script>
	import axios from 'axios';
	//这个是 悬浮上去绿色的框
	var scatterImg =
		'image://';
	import pageCmp from "@/components/page/index.vue";
	export default {
		components: {
			pageCmp
		},
		data() {
			return {
				historyMapData: [{
					name: "map",
					adcode: "100000"
				}],
				provinceInfo: [{
						name: "陕西",
						oredrNum: 599, //订单数量
						goodsNum: 599, //货品数量
					},
					{
						name: "青海",
						oredrNum: 142,
						goodsNum: 599,
					},
					{
						name: "甘肃",
						oredrNum: 44,
						goodsNum: 599,
					},
					{
						name: "宁夏",
						oredrNum: 92,
						goodsNum: 599,
					},
				],

				cityInfo: [{
						name: "西安市",
						oredrNum: 599, //订单数量
						goodsNum: 599, //货品数量
					},
					{
						name: "咸阳市",
						oredrNum: 142,
						goodsNum: 599,
					},
					{
						name: "延安市",
						oredrNum: 44,
						goodsNum: 599,
					},
					{
						name: "宝鸡市",
						oredrNum: 92,
						goodsNum: 599,
					},
				],

				tableData: [],
				loading: false,
				total: 0,
				pageNum: 1,
				pageSize: 30,
				listAdcode: '100000',
			};
		},
		mounted() {
			this.chartMap();
		},
		methods: {
			// 分页
			refurbish(param) {
				this.pageNum = param.current;
				this.pageSize = param.pageSize;
				this.getSalesProceedsDetails();
			},

			chartMap() {
				// 初始化dom
				const myChart = this.$echarts.init(
					document.getElementById("mapEchart")
				);
				// 初始化map
				this.initMap(myChart, "map", "100000");
				// 添加点击事件
				myChart.on("click", (e) => {
					console.log(e)
					const newName = e.name;
					const value = e.data.mapValue;
					if (value.level === "district") return alert("该地区已经无法下钻");
					// 添加历史记录
					this.historyMapData.push(value);
					this.listAdcode = value.adcode;
					// 初始化地图
					this.initMap(myChart, newName, value.adcode);
					
				});

				//让可视化地图跟随浏览器大小缩放
				window.addEventListener("resize", () => {
					myChart.resize();
				});
			},
			async initMap(chartDOM, geoName, adcode) {
				// 清除echarts实例
				chartDOM.clear();
				// 请求map的json
				const mapData = await this.getMapJSON(adcode, geoName);

				console.log(geoName);
				var mapArr = [];
				if (geoName == 'map') {
					mapArr = await this.getMapChart(this.provinceInfo, mapData);
				} else {
					mapArr = await this.getMapChart(this.cityInfo, mapData);
				}

				this.$nextTick(() => {
					console.log(mapArr, '88888888888888')
					// const 

					// 图表配置项
					const option = this.getOption(geoName, mapArr);
					// 渲染配置
					chartDOM.setOption(option);
				})

			},


			// 获取省-》市-》区、县
			async getMapJSON(adcode, geoName) {
				let geoJson = {};
				if (geoName == 'map') {
					geoJson = require("@/common/utils/fourNorthwesternProvinces.json");
				} else {
					const res = await axios.get(
						`https://geo.datav.aliyun.com/areas_v3/bound/${adcode}_full.json`
					);
					// 重新注册地图
					geoJson = res.data;
				}
				this.$echarts.registerMap(geoName, geoJson);
				// 过滤json数据
				const mapData = geoJson.features.map((item) => {
					return {
						value: item.properties,
						name: item.properties.name,
					};
				});
				return mapData;
			},
			getOption(geoName, mapData) {
				// 图表配置项
				const option = {

					tooltip: {
						// 鼠标移到图里面的浮动提示框
						borderWidth: '0',
						textStyle: {
							color: 'rgba(255, 255, 255, 1)' //文字颜色
						},
						className: 'custom-tooltip-boxCont',
						//自定义提示框自动调用函数
						formatter: function(params) {
							let showname = params;
							return (
								`
									<div class="custom-tooltip-box">
										<div style="text-align: left;display: flex;align-items: center;">
												<div style="width: 3px;height:16px;background: #FBA922;"></div>
												<div style="margin-left: 5px;">${showname.data ? showname.data.name : '暂无信息'}</div>
										</div>
										<p style="font-size:14px;display: flex;justify-content: space-between;margin-left: 8px;">
												<span>本月销售额:</span>
												<span>12</span>
										</p>
										<p style="font-size:14px;display: flex;justify-content: space-between;margin-left: 8px;">
												<span>本月客户家数:</span>
												<span>12</span>
										</p>
										<p style="font-size:14px;display: flex;justify-content: space-between;margin-left: 8px;">
												<span>本月毛利:</span>
												<span>12</span>
										</p>
										<div style="text-align: left;display: flex;align-items: center;">
												<div style="width: 3px;height:16px;background: #FBA922;"></div>
												<div style="margin-left: 5px;">客户数:</div>
										</div>
										<p style="font-size:14px;display: flex;justify-content: space-between;margin-left: 8px;">
												<span>电商部:</span>
												<span>12</span>
										</p>
										<p style="font-size:14px;display: flex;justify-content: space-between;margin-left: 8px;">
												<span>销售部:</span>
												<span>12</span>
										</p>
										<p style="font-size:14px;display: flex;justify-content: space-between;margin-left: 8px;">
												<span>未合作:</span>
												<span>12</span>
										</p>
								</div>
					    `
							);
						}
					},
					geo: [{
							zlevel: -100,
							show: true,
							type: "map",
							map: geoName, // 地图类型。echarts-gl 中使用的地图类型同 geo 组件相同
							label: {
								normal: {
									show: true, // 是否显示对应地名
									fontSize: "15",
									textStyle: {
										color: '#fff' //对应地名颜色
									}
								},
								emphasis: {
									textStyle: {
										color: '#fff' //对应地名颜色
									}
								}
							},
							itemStyle: {
								normal: {
									//正常状态下的地图背景色
									borderColor: '#2FC3B3',
									borderWidth: "1",
									areaColor: {
										type: 'radial',
										x: 0.5,
										y: 0.5,
										r: 0.8,
										colorStops: [{
											offset: 0,
											color: '#0198A5' // 0% 处的颜色
										}, {
											offset: 1,
											color: '#033249' // 100% 处的颜色
										}],
										globalCoord: false // 缺省为 false
									},
									shadowColor: '#03495e',
									shadowOffsetX: -2,
									shadowOffsetY: 2,
									shadowBlur: 10
								},
								emphasis: {
									// areaColor: null,
									shadowOffsetX: 0,
									shadowOffsetY: 0,
									shadowBlur: 20, //聚焦时候的阴影范围
									borderWidth: 0,
									shadowColor: "#000000", //聚焦时候的阴影颜色
									// 鼠标放上去地图区域背景颜色
									areaColor: '#389BB7', //聚焦之后的颜色
									borderWidth: 3
								},
							},
						},
						{ // 底部重影层,蓝色偏移形成立体感
							map: geoName,
							roam: false, // 是否允许缩放
							silent: false,
							zoom: 1,
							itemStyle: {
								normal: {
									shadowColor: "#247472",
									shadowOffsetX: 0,
									shadowOffsetY: 12,
									opacity: 0.8,
								},
							},
							zlevel: -111,
						},
					],
					series: [{
						type: "effectScatter", //图例类型
						coordinateSystem: "geo", //使用geo地图
						//这个是位置图标
						symbol: "image://",
						showEffectOn: "emphasis", //涟漪特效何时触发
						symbolSize: 25, //图例大小,
						symbolOffset: [0, -30],
						rippleEffect: { //特效设置
							scale: 2.5,
							color: "rgba(128, 217, 248, 1)",
							number: 3
						},
						showEffectOn: "render",
						data: mapData,
					}, {
						type: "map",
						geoIndex: 0,
						map: geoName, // 地图类型。echarts-gl 中使用的地图类型同 geo 组件相同
						data: mapData, //这里比较重要:获得过滤后的data,这样点击事件时就能获得这个data的值
					}, {
						type: "scatter",
						coordinateSystem: "geo",
						// 地图文字背景框设置
						symbol: scatterImg,
						symbolSize: [80, 30],
						symbolOffset: [0, -70],
						//地图文字设置
						label: {
							normal: {
								show: true,
								formatter: function(params) {
									var name = params.name;
									// var value = params.data.datas;
									// var text = `{fline|${value}}\n{tline|${name}}`;
									var text = `{fline|${name}}`;
									return text;
								},
								rich: {
									fline: {
										padding: [8, 0, 0, 0],
										color: "#fff",
										textShadowColor: "#030615",
										textShadowOffsetX: 1,
										textShadowOffsetY: 1,
										fontSize: 16,
										fontWeight: 550,
									},
								},
								label: {
									show: false,
								},
							},
							emphasis: {
								show: true,
							},
						},
						itemStyle: {
							color: '#1dfd28'
						},
						z: 999,
						data: mapData,
					}, ],
				};
				return option;
			},

			//
			getMapChart(data, mapData) {
				var res = [];
				data.forEach(item => {
					mapData.forEach(row => {
						if (item.name == row.name) {
							res.push({
								name: row.name,
								value: row.value.cp ? row.value.cp : row.value.center,
								oredrNum: item.oredrNum ? item.oredrNum : '暂无数据',
								goodsNum: item.goodsNum ? item.goodsNum : '暂无数据',
								mapValue: row.value,
							})
						} else {
							res.push({
								name: row.name,
								value: item.name ? row.value.cp : '',
								oredrNum: item.oredrNum ? item.oredrNum : '暂无数据',
								goodsNum: item.goodsNum ? item.goodsNum : '暂无数据',
								mapValue: row.value,
							})
						}
					})
				})
				return res;
			},
			backMap() {
				const myChart = this.$echarts.init(
					document.getElementById("mapEchart")
				);
				// 去除当前的地图信息
				this.historyMapData.pop();
				const len = this.historyMapData.length;
				// 获取上一级的地图信息
				const newdata = this.historyMapData[len - 1] ? this.historyMapData[len - 1].name : "map";
				const adcode = this.historyMapData[len - 1] ? this.historyMapData[len - 1].adcode : "100000";
				// 重新渲染地图
				this.initMap(myChart, newdata, adcode);
			},
		},

	};
</script>

<style lang="scss" scoped>
	.mapEcharts {
		position: relative;
		overflow: hidden;
		width: 100%;
		height: 80vh;
		padding: 0px;
		margin: 0px;
		// background-image: url('../../../assets/images/index/bg.jpg');
		// background-size: 100% 100%;
		background-image: radial-gradient(#073C53, #02273F);
		display: flex;
		box-sizing: border-box;
		font-size: 14px;

		.map-left {
			width: 40%;

			.backIcon {
				position: absolute;
				left: 20px;
				top: 20px;
				cursor: pointer;
				z-index: 1;
			}

			#mapEchart {
				width: 100%;
				height: 80vh;
			}
		}

		.map-right {
			color: #fff;
			width: 60%;
			font-size: 14px;

			.top-other {
				width: 90%;
				background-image: url('../../../assets/images/index/tabbg.png');
				background-size: 100% 100%;
				padding: 45px 55px 55px;
				margin: 25px auto;
				box-sizing: border-box;


				.list {
					display: flex;

					.list-item {
						width: calc(100% / 3);

						.num {
							font-size: 24px;
							margin-top: 20px;
							margin-bottom: 10px;
						}
					}
				}
			}


		}
	}

	:deep(.map-right .table-cont) {
		width: 90%;
		margin: 0 auto;

		.el-table {
			color: #fff;
			background-color: transparent !important;

			thead tr>th {
				background: #056f9370 !important;
			}
		}

		.el-table tr {
			background-color: transparent !important;
		}

		.el-table th.el-table__cell.is-leaf,
		.el-table td.el-table__cell {
			border-bottom: 0.0625rem solid #ebeef561;
		}

		.el-table__body-wrapper::-webkit-scrollbar {
			height: 6px !important;
			cursor: pointer;
		}

		.el-pagination.is-background .btn-prev,
		.el-pagination.is-background .btn-next{
			background-color: #ecf5ff69;
		}
		.el-pagination.is-background .el-pager li {
			background-color: #056f9370;
		}
	}

	// 提示框样式
	/* // 给父盒子清除默认已有样式 */
	:deep(.custom-tooltip-boxCont) {
		// padding: 10px !important;
		border: none !important;
		box-shadow: none !important;
		background-color: transparent !important;
		background-image: url('../../../assets/images/index/toopImage.png');
		background-size: 100% 100%;
		width: 350px;
		height: auto;
		padding: 17px 23px !important;
		box-sizing: border-box;
		color: #fff;

		.custom-tooltip-box {
			padding: 15px;
			background-color: #01192d91;
		}

	}
</style>

在这里插入图片描述
在这里插入图片描述

西北四省数据 fourNorthwesternProvinces.json
在最上面的文件里

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
首先,你需要在 Vue 项目中安装 echartsecharts-gl: ``` npm install echarts echarts-gl --save ``` 然后,在需要使用地图的组件中引入 echarts: ```javascript import echarts from 'echarts' import 'echarts-gl' ``` 接着,你可以使用以下代码来绘制地图: ```vue <template> <div id="chart" style="width: 100%; height: 500px;"></div> </template> <script> import echarts from 'echarts' import 'echarts-gl' export default { data() { return { chart: null } }, mounted() { this.chart = echarts.init(document.getElementById('chart')) this.chart.on('click', this.handleChartClick) this.drawMap() }, methods: { drawMap() { const option = { series: [ { type: 'map3D', map: 'china', nameMap: { china: '中国' }, label: { show: true, textStyle: { color: '#fff' } }, itemStyle: { color: '#1E90FF', opacity: 0.7 }, emphasis: { label: { show: true }, itemStyle: { color: '#FFA500' } }, data: this.getData() } ] } this.chart.setOption(option) }, handleChartClick(params) { // 处理点击事件 console.log(params) }, getData() { // 返回地图数据 } } } </script> ``` 其中,`drawMap` 方法用来绘制地图,`handleChartClick` 方法用来处理点击事件,`getData` 方法用来获取地图数据。 需要注意的是,地图需要在 `series` 中使用 `map3D` 类型,并且需要设置 `map` 属性为地图名称(如 `'china'`)。 当点击地图区域时,会触发 `handleChartClick` 方法,并且参数中会包含点击区域的信息。你可以根据这些信息来实现取功能,例如重新加载新的地图数据并更新地图

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值