ECharts + geoJSON 绘制地图_下钻功能_Vue React 项目组件封装

这篇文章会介绍「 H5 + 原生 JS」和「 封装 React 组件」中两种实现方式。

「 H5 + 原生 JS」

参考文章(1)中的代码已经实现了展示全国地图和点击省市自治区下钻的功能,但是每个区域的颜色是一样的。于是又结合了 参考文章(2)。效果图:

---- 源码地址

参考文章:
HTML5 Canvas实现中国地图 可展开地级市子地图
echarts实现中国地图数据展示

绘制 geoJSON 地址: http://geojson.io/

封装 React 组件

绘制地图调用的 echarts.registerMap(geoJSON) 这一方法,geoJSON 的具体实现请移步至 echarts搞定各种地图(想怎么画就怎么画)

1. 封装 geoJSON 数据,结构大致如下:

{
  "anhui": {
    "type": "FeatureCollection",
    "features": [
      {
        "id": "340100",
        "type": "Feature",
        "geometry": {
          "type": "Polygon",
          "coordinates": [
            "[NGHEDAHAPCFMPGTCFULUNGFELKJGDG@ECKMIGIBE@....."
          ],
          "encodeOffsets": [[119842, 32007]]
        },
        "properties": {
          "cp": [117.283042, 31.86119],
          "name": "合肥市",
          "childNum": 1
        }
      },
      {"id": ........},
    ]
}

2. 创建引入 geoJSON 文件的中间文件:

import {anhui} from './province/anhui';
import {aomen} from './province/aomen';
import {beijing} from './province/beijing';
// etc. 

export default [
  {
    anhui,
    aomen,
    beijing,
   // etc.
  },
];

3. 封装 Map 文件

import React, {Component} from 'react';
import echarts from 'echarts';
import styles from './Map.less';
import * as mapdata from './ProvinceData';
import {china} from './MapData/China';
import * as province from './MapData/province.js';

class CMap extends Component {
  componentDidMount() {
    this.setState({
      province,
    });
    china(echarts);
    this.initEcharts('china', '中国');
  }
  // 初始化echarts
  initEcharts(pName, Chinese_) {
    var myChart = echarts.init(document.getElementById('china-map'));
    var oBack = document.getElementById('back');
    var option = {
      title: {
        text: Chinese_ || pName,
        left: 'center',
      },
      tooltip: {
        trigger: 'item',
        formatter: '{b}<br/>{c}',
      },
      //左侧小导航图标
      visualMap: {
        show: true,
        x: 'left',
        y: 'top',
        splitList: this.props.splitList,
        color: ['#3D74CC', '#5A8EE0', '#6FA4F7', '#80B1FF', '#99C0FF', '#B3D0FF'],
      },

      series: [
        {
          name: Chinese_ || pName,
          type: 'map',
          mapType: pName,
          roam: false, //是否开启鼠标缩放和平移漫游
          data: this.props.mapData,
          top: '3%', //组件距离容器的距离
          zoom: 1.1,
          selectedMode: 'single',

          label: {
            normal: {
              show: true, //显示省份标签
              textStyle: {color: '#585858', fontSize: 12}, //省份标签字体颜色
            },
            emphasis: {
              //对应的鼠标悬浮效果
              show: true,
              textStyle: {color: '#aaa'},
            },
          },
          itemStyle: {
            normal: {
              borderWidth: 0.5, //区域边框宽度
              borderColor: '#A6E1FF', //区域边框颜色
              areaColor: '#fff', //区域颜色
            },

            emphasis: {
              borderWidth: 0.5,
              borderColor: '#FFD1A3',
              areaColor: '#FFDAA6',
            },
          },
        },
      ],
    };

    myChart.setOption(option);

    myChart.off('click');
    let _this = this;
    if (pName === 'china') {
      // 全国时,添加click 进入省级
      myChart.on('click', function(param) {
        for (var i = 0; i < mapdata.provincesText.length; i++) {
          if (param.name === mapdata.provincesText[i]) {
            //显示对应省份的方法
            const pName = mapdata.provinces[i];
            echarts.registerMap(mapdata.provincesText[i], _this.state.province.default[0][pName]);
            _this.initEcharts(mapdata.provincesText[i]);
            break;
          }
        }
        if (param.componentType === 'series') {
          var provinceName = param.name;
        }
      });
    } else {
      // 省份,添加双击 回退到全国
      myChart.on('dblclick', function() {
        _this.initEcharts('china', '中国');
      });
    }
  }

  render() {
    return (
      <div>
        <button
          onClick={() => {
            this.initEcharts('china', '中国');
          }}
        >
          返回全国
        </button>
        <div
          style={{marginTop: '20px', width: '100%', height: document.body.clientHeight * 0.7}}
          id="china-map"
        />
      </div>
    );
  }
}

export default CMap;

4. ProvinceData 数据:

export var provinces = [
  "shanghai",
  "hebei",
 // etc.
];

export var provincesText = [
  "上海",
  "河北",
  "山西",
// etc.
];

export var seriesData = [
  { name: "北京", value: "100" },
  { name: "天津", value: randomData() },
  { name: "上海", value: randomData() },
  { name: "重庆", value: randomData() },
// 其他城市数据.....
}];

function randomData() {
  return Math.round(Math.random() * 500);
}

5. 引用组件

import CMap from './Map.js';

<CMap mapData={mapData} />;

2020 更新升级版:

Vue 组件封装

其实 React 封装也是同理的,之前的版本是简易封装,看着就 low,这一版更新了比较多:

1. 准备 GEOJSON 数据

地图的 JSON 数据可以在这里下载
json-data/map

2. 准备 seriesData 数据

export const seriesData = [
  { name: '北京', value: '100' },
// ...
]

export const provincesdata = [
  { name: '朝阳市', value: randomData() },
//...
}

function randomData() {
  return Math.round(Math.random() * 500)
}

完整代码请参考:json-data/map/emap.js

3. 封装组件

// component/Map.vue

<template>
  <div class="container">
    <div ref="Map" :style="{ height: height, width: width }"></div>
    <el-button
      v-show="ifShowButton"
      class="primary-button"
      type="primary"
      @click="backToWholeNation"
      >返回全国</el-button
    >
  </div>
</template>

<script>
import echarts from 'echarts'
import { mapPath } from '@/static/js/util'
import { getTheme } from '@/static/js/theme'

export default {
  components: {},
// 定义可以接收的属性
  props: {
    lazyResize: {
      type: Number,
      default: 200
    },
    width: {
      type: String,
      default: '80vw'
    },
    height: {
      type: String,
      default: '80vh'
    },
  // map 的 option 都是由此 config 转换
    config: {
      type: Object,
      required: true
    }
  },

  data() {
    return {
      chart: null,
      option: null,
      mapInfo: {},
      mapKey: '中国',
      baseOption: {},
      ifShowButton: false
    }
  },

  mounted() {
    this.restoreChart()
  },

  beforeDestroy() {
    if (this.chart instanceof Object) {
      this.chart.clear()
      this.chart.dispose()
    }
    this.chart = null
    this.baseOption = null
  },

  methods: {
// 返回全国
    backToWholeNation() {
      this.mapKey = '中国'
      this.ifShowButton = false
      this.convertMapOption(this.$props.config, this.mapKey)
      this.getGeoJson().then(() => this.refreshChart())
    },
// 返回默认的 option 配置项
    getBaseOption() {
      return {
        textStyle: {},
        title: {},
        tooltip: {},
        legend: { show: false },
        dataset: [],
        series: []
      }
    },

    // 根据 props 的 config 配置 option
    convertMapOption(config, mapKey) {
      this.baseOption = this.getBaseOption()
      const { seriesData, provincesdata } = config

      // 地图标题
      this.baseOption.title = {
        top: 'top',
        left: 'center',
        text: mapKey,
        textStyle: {
          color: '#f3f3f3'
        }
      }

      // 视觉映射组件配置
      this.baseOption.visualMap = {
        type: 'continuous',
        top: 'center',
        left: 'left',
        calculable: true,
        color: [
          '#3b4994',
          '#8c62aa',
          '#a5add3',
          '#be64ac',
          '#5ac8c8',
          '#ace4e4'
        ]
      }

      // 提示框
      this.baseOption.tooltip = {
        trigger: 'item',
        formatter: '{b}<br/>{c}'
      }

      // 数据
      this.baseOption.series = [
        {
          name: mapKey,
          type: 'map',
          map: mapKey,
          roam: true,
          data: mapKey === '中国' ? seriesData : provincesdata,
          left: mapKey === '海南' ? '80%' : 'center',
          top: mapKey === '海南' ? '215%' : 'center',
          zoom: mapKey === '海南' ? 6 : 1.1,
          scaleLimit: {
            min: 0.5,
            max: 20
          },
          showLegendSymbol: false,
          label: {
            show: true
          },
          emphasis: {
            label: {
              color: '#545454'
            },
            itemStyle: {
              areaColor: null
            }
          },
          nameMap: {
            中华人民共和国: '中国'
          }
        }
      ]
    },

    // 初始化图表
    initChart() {
      this.chart = echarts.init(this.$refs.Map, getTheme(this.$props.config))
     
      this.chart.on('click', (params) => {
        // 这里做了限制,仅可下钻一级,如果有地级市的下属区域数据,可以改造这个方法
        if (this.mapKey === '中国') {
          this.mapKey = params.name
          this.ifShowButton = true
          this.convertMapOption(this.$props.config, this.mapKey)
          this.getGeoJson().then(() => this.refreshChart())
        }
      })
    },

    // 绘制图表
    refreshChart() {
      if (!this.chart) return false
      this.chart.setOption(this.baseOption || {}, true)
      // 添加根据视口缩放重绘功能
      if (this.lazyResize) {
        window.onresize = () => {
          setTimeout(() => {
            this.chart.resize()
          }, this.lazyResize)
        }
      }
    },

    // 根据 mapKey 获取 JSON 数据
    getGeoJson() {
    // 这个方法封装在 一个 util.js 文件,可以参考接下来的一段代码
      const mapInfo = mapPath[this.mapKey]
      return new Promise((resolve, reject) => {
        if (mapInfo instanceof Object) {
          if (mapInfo.registered) {
            resolve(this.mapKey)
            return this.mapKey
          }
        } else {
          return false
        }

        if (mapInfo instanceof Object && mapInfo.key) {
          import(`@/static/json-data/map/${mapInfo.filePath}.json`)
            .then((res) => {
              echarts.registerMap(this.mapKey, res)
              mapInfo.registered = true
              resolve(this.mapKey)
              return this.mapKey
            })
            .catch((error) => {
              throw error
            })
        } else {
          reject(false)
          return false
        }
      })
    },

    async restoreChart() {
      this.convertMapOption(this.$props.config, this.mapKey)
      await this.getGeoJson()
      this.initChart()
      this.refreshChart()
    }
  }
}
</script>
<style lang="less" scoped>
.container {
  height: 80vh;
  margin: 10vh 2vw;
  position: relative;
  .primary-button {
    position: absolute;
    top: 0;
    left: 5vw;
  }
}
</style>

获取 mapPath 的完整代码请参考 js/util.js

// util.js
export const mapPath = {
  中国: {
    key: 'china',
    name: '中国',
    filePath: 'china',
    registered: false
  },
//...
}

4. 引用组件

// pages/Map.vue
<template>
    <Map :config="config" :lazy-resize="lazyResize" />
</template>

<script>
import { seriesData, provincesdata } from '@/static/json-data/map/emap.js'
import Map from '@/components/Map.vue'

export default {
  components: {
    Map
  },

  data() {
    return {
      config: { seriesData, provincesdata, theme: 'default', mapKey: '中国' },
      lazyResize: 200,
    }
  },
}
</script>
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352

推荐阅读更多精彩内容

  • echarts提供了全国各省市及区县的js和json文件,但是并没有细化到区域内各街道范围,有时候项目中又有这种需...
    前端很忙阅读 157,148评论 40 291
  • 如何清除浮动,有哪几种方法,区别是什么 1、移动端你遇到过什么兼容问题? 1、如果在input设置边框颜色在ios...
    崽崽不哭阅读 787评论 0 1
  • 有多少人因为自己的阅读速度不够理想,忍受着龟速的折磨? 又有多少人为了提高自己的速度,愿意放弃熟悉的阅读方式,去追...
    梁宝沧阅读 244评论 0 1
  • 堵车 今天中考,玉峰路上车真多,一辆挨看一辆。 整个大街上六条车道被堵得满满的。汽车...
    耕耘人生阅读 330评论 2 7
  • 这是王倩的第七次相亲,她心里是记得这个数的,一张张脸又在她的眼前浮现,她的脸颊早在第三次的时候就不会发烫了。出...
    园林鸟阅读 467评论 1 0