vue3+echarts5 飞线图绘制

679 阅读5分钟
  • 上demo展示 Filmage 2024-04-17_104920.gif
  • vue代码
<template>
  <div class="h-full flex flex-col">
    <div class="chart_title">飞线图</div>
    <div ref="chartRef" class="flex-1 mt-4"></div>
  </div>
</template>

<script setup>
import echarts from "~/utils/echarts.config";
import useEcharts from "@/hooks/useEcharts";
// 数据源下面有写
const props = defineProps({
  datas: {
    type: Object,
    required: true,
  },
  geoJSONData: {
    type: Object,
    required: true,
  },
});
const { echartResize } = useEcharts();
const chartRef = ref();
const myChart = ref();

// * 用于转换成需要的特定格式数据
const convertData = data => {
  const res = [];
  for (let i = 0; i < data.length; i++) {
    const dataItem = data[i];
    const fromCoord = props.datas.chinaGeoCoordMap[dataItem[0].name];
    const toCoords = props.datas.toCoords; //被攻击点
    if (fromCoord && toCoords[i]) {
      // * 3.2.0 之前
      // res.push([{ coord: toCoords[i] }, { coord: fromCoord, value: dataItem[0].value, visualMap: false }]);
      res.push({ coords: [toCoords[i], fromCoord] });
      // * 从 3.2.0 起改为如下配置
      // {
      //     coords: [
      //         [120, 66],  // 起点
      //         [122, 67]   // 终点
      //         ...         // 如果 polyline 为 true 还可以设置更多的点
      //     ],
      //     // 统一的样式设置
      //     lineStyle: {
      //     }
      // }
    }
  }
  return res;
};
// 图标初始化
const chartInit = () => {
  const series = [];
  [["四川", props.datas.chinaDatas]].forEach(item => {
    series.push(
      {
        type: "map", // 图表类型为地图
        map: "chinaGeoCoordMap", // 使用 registerMap 注册的地图名称
        aspectScale: 0.85, // 地图纵横比
        layoutCenter: ["50%", "50%"], // 地图位置居中
        layoutSize: "140%", // 地图尺寸充满整个容器
        zoom: 1, // 当前视角的缩放比例
        // roam: true, // 是否开启平游或缩放(如果开启,可以拖动和缩放地图)
        scaleLimit: {
          // 滚轮缩放的极限控制
          min: 1, // 最小缩放比例
          max: 2, // 最大缩放比例
        },
        itemStyle: {
          areaColor: "#12235c",
          borderColor: "#2ab8ff",
          borderWidth: 0.5,
          shadowColor: "rgba(0,54,255, 0.4)",
          shadowBlur: 100,
        },
        emphasis: {
          // 高亮样式
          areaColor: "#02102b", // 高亮区域颜色
          label: {
            color: "#fff", // 高亮区域标签颜色
          },
        },
        label: {
          show: true, // 显示省份名称标签
          color: "#fff", // 省份名称标签颜色
          fontSize: 12, // 省份名称标签字体大小
          fontWeight: "bold", // 省份名称标签字体粗细
        },
      },
      {
        type: "lines", // 路径图
        zlevel: 2,
        effect: {
          show: true,
          period: 3, //箭头指向速度,值越小速度越快
          trailLength: 0.02, //特效尾迹长度[0,1]值越大,尾迹越长重
          symbol: "arrow", //箭头图标
          symbolSize: 5, //图标大小
        },
        lineStyle: {
          color: "#00eaff",
          width: 1, //尾迹线条宽度
          opacity: 0.7, //尾迹线条透明度
          curveness: 0.3, //尾迹线条曲直度
        },
        data: convertData(item[1]),
      },
      {
        type: "effectScatter", // 带有涟漪特效动画的散点(气泡)图。利用动画特效可以将某些想要突出的数据进行视觉突出
        coordinateSystem: "geo", // 使用地理坐标系,通过 geoIndex 指定相应的地理坐标系组件
        zlevel: 2,
        rippleEffect: {
          //涟漪特效
          period: 4, //动画时间,值越小速度越快
          brushType: "stroke", //波纹绘制方式 stroke, fill
          scale: 4, //波纹圆环最大限制,值越大波纹越大
        },
        emphasis: {
          show: true,
          position: "right", //显示位置
          offset: [5, 0], //偏移设置
          formatter: params => params.data.name, //圆环显示文字
          fontSize: 13,
        },
        symbol: "circle",
        symbolSize: val => 5 + val[2] * 5, //圆环大小
        itemStyle: {
          show: true,
          color: "#00eaff",
        },
        data: item[1].map(dataItem => {
          // 系列中的数据内容数组。数组项通常为具体的数据项
          return props.datas.chinaGeoCoordMap[dataItem[0].name].concat([dataItem[0].value]);
        }),
      },
      //被攻击点
      {
        type: "scatter", // 散点(气泡)图
        coordinateSystem: "geo", // 使用地理坐标系,通过 geoIndex 指定相应的地理坐标系组件
        zlevel: 2,
        emphasis: {
          // show: false,   //定位标记
          color: "#f60",
          label: {
            normal: {
              show: false, //定位点名字
              position: "right",
              // offset:[5, 0],
              color: "#0f0",
              formatter: "{b}",
              textStyle: {
                color: "#0f0",
              },
            },
          },
        },
        symbol: "pin", //定位图标样式
        // 标记的大小,可以设置成诸如 10 这样单一的数字,也可以用数组分开表示宽和高,例如 [20, 10] 表示标记宽为20,高为10
        symbolSize: 50,
        // 系列中的数据内容数组。数组项通常为具体的数据项
        data: [props.datas.chinaGeoCoordMap[item[0]].concat([10])],
      }
    );
  });

  const option = {
    // backgroundColor: "#061d4d", // 背景颜色
    geo: {
      map: "chinaGeoCoordMap", // 使用 registerMap 注册的地图名称
      aspectScale: 0.85, // 地图纵横比
      layoutCenter: ["50%", "50%"], // 地图位置
      layoutSize: "100%", // 地图尺寸
      itemStyle: {
        shadowColor: "#276fce",
        shadowOffsetX: 0,
        shadowOffsetY: 15,
        opacity: 0.5,
      },
      emphasis: {
        itemStyle: {
          // 默认样式
          shadowColor: "#276fce", // 阴影颜色
          shadowOffsetX: 0, // 阴影水平偏移量
          shadowOffsetY: 15, // 阴影垂直偏移量
          opacity: 0.5, // 地图区域透明度
          // 高亮样式
          areaColor: "#276fce", // 高亮区域颜色
        },
      },
      regions: [
        // 自定义区域样式
        {
          name: "南海诸岛", // 区域名称
          itemStyle: {
            areaColor: "rgba(0, 10, 52, 1)", // 区域颜色
            borderColor: "rgba(0, 10, 52, 1)", // 区域边框颜色
            opacity: 0, // 区域透明度
            label: {
              show: false, // 是否显示标签
              color: "#009cc9", // 标签颜色
            },
          },
          label: {
            show: false, // 是否显示标签
            color: "#FFFFFF", // 标签颜色
            fontSize: 12, // 标签字体大小
          },
        },
      ],
    },
    series,
  };
  option && myChart.value.setOption(option);
};

// watch(() => props.datas, chartInit);

onMounted(() => {
  // 5.0依赖删除了内置 geoJSON 数据 需自行注册
  const geoJSON = props.geoJSONData;
  echarts.registerMap("chinaGeoCoordMap", geoJSON);
  myChart.value = echarts.init(chartRef.value);
  echartResize(myChart.value);  // myChart.resize(); 监听浏览器resize事件执行或者卸载echarts的resize事件,只是统一封装了个hooks
  chartInit();
});
</script>

<style lang="scss" scoped></style>

  • 后台数据
// props.geoJSONData
async getGeoJSONData() {
    if (fs.existsSync(cacheFilePath)) {
      // 如果本地缓存文件存在,则读取缓存数据
      const cacheData = fs.readFileSync(cacheFilePath, "utf-8");
      return JSON.parse(cacheData);
    }
    // 数据源
    const url = "https://geojson.cn/api/data/china.json"; // 替换为实际的 API URL
    try {
      const response = await axios.get(url); // 进行axios转发获取数据源
      const data = response.data;
      fs.writeFileSync(cacheFilePath, JSON.stringify(data)); // 将数据写入本地缓存文件
      return data;
    } catch (error) {
      throw Error(error);
    }
  }
// props.data 替换为实际的数据源
 getChinaData() {
    return {
      chinaGeoCoordMap: {
        四川: [103.9526, 30.7617],
        黑龙江: [127.9688, 45.368],
        内蒙古: [110.3467, 41.4899],
        吉林: [125.8154, 44.2584],
        北京市: [116.4551, 40.2539],
        辽宁: [123.1238, 42.1216],
        河北: [114.4995, 38.1006],
        天津: [117.4219, 39.4189],
        山西: [112.3352, 37.9413],
        陕西: [109.1162, 34.2004],
        甘肃: [103.5901, 36.3043],
        宁夏: [106.3586, 38.1775],
        青海: [101.4038, 36.8207],
        新疆: [87.611053, 43.828171],
        西藏: [91.117212, 29.646922],
        重庆: [108.384366, 30.439702],
        山东: [117.1582, 36.8701],
        河南: [113.4668, 34.6234],
        江苏: [118.8062, 31.9208],
        安徽: [117.29, 32.0581],
        湖北: [114.3896, 30.6628],
        浙江: [119.5313, 29.8773],
        福建: [119.4543, 25.9222],
        江西: [116.0046, 28.6633],
        湖南: [113.0823, 28.2568],
        贵州: [106.6992, 26.7682],
        云南: [102.9199, 25.4663],
        广东: [113.12244, 23.009505],
        广西: [108.479, 23.1152],
        海南: [110.3893, 19.8516],
        台湾: [120.702967, 24.123621],
        上海: [121.4648, 31.2891],
      },
      chinaDatas: [
        [{ name: "北京市", value: 0 }, { name: "黑龙江", value: 0 }, { name: "上海市" }],
        [{ name: "内蒙古", value: 0 }],
        [{ name: "吉林", value: 0 }],
        [{ name: "辽宁", value: 0 }],
        [{ name: "河北", value: 0 }],
        [{ name: "天津", value: 0 }],
        [{ name: "山西", value: 0 }],
        [{ name: "陕西", value: 0 }],
        [{ name: "甘肃", value: 0 }],
        [{ name: "新疆", value: 0 }],
        [{ name: "西藏", value: 0 }],
        [{ name: "台湾", value: 0 }],
        [{ name: "黑龙江", value: 0 }],
        [{ name: "云南", value: 1 }],
        [{ name: "宁夏", value: 0 }],
        [{ name: "青海", value: 0 }],
        [{ name: "四川", value: 2 }],
        [{ name: "重庆", value: 1 }],
        [{ name: "山东", value: 0 }],
        [{ name: "河南", value: 0 }],
        [{ name: "江苏", value: 0 }],
        [{ name: "安徽", value: 0 }],
        [{ name: "湖北", value: 0 }],
        [{ name: "浙江", value: 0 }],
        [{ name: "福建", value: 0 }],
        [{ name: "江西", value: 0 }],
        [{ name: "湖南", value: 0 }],
        [{ name: "贵州", value: 1 }],
        [{ name: "广西", value: 0 }],
        [{ name: "海南", value: 0 }],
        [{ name: "上海", value: 0 }],
      ],
      // 攻击起始点
      toCoords: [
        [103.9526, 30.7617],
        [103.9526, 30.7617],
        [108.384366, 30.439702],
        [108.384366, 30.439702],
        [108.384366, 30.439702],
        [108.384366, 30.439702],
        [108.384366, 30.439702],
        [102.9199, 25.4663],
        [102.9199, 25.4663],
        [102.9199, 25.4663],
        [102.9199, 25.4663],
        [102.9199, 25.4663],
        [106.6992, 26.7682],
        [106.6992, 26.7682],
        [106.6992, 26.7682],
        [106.6992, 26.7682],
        [106.6992, 26.7682],
        [103.9526, 30.7617],
        [103.9526, 30.7617],
        [103.9526, 30.7617],
        [103.9526, 30.7617],
        [103.9526, 30.7617],
        [103.9526, 30.7617],
        [103.9526, 30.7617],
        [103.9526, 30.7617],
        [103.9526, 30.7617],
        [103.9526, 30.7617],
        [103.9526, 30.7617],
      ],
    };
  }