使用echart实现监控拓扑图

<template>
  <div class="wrap">
    <div ref="main" class="charts"/>
  </div>
</template>

<script>
import echarts from 'echarts'
import { debounce } from '@/utils'

export default {
  name: 'Topology',

  props: {
    title: {
      type: String,
      default: ''
    },
    centerName: {
      type: String,
      default: ''
    },
    centerIcon: {
      type: String,
      default: 'https://www.easyicon.net/api/resizeApi.php?id=1093990&size=48'
    },
    chartData: {
      type: Array,
      default: () => {
        return []
        /*[
          {
              "tps":"14",
              "businessName":"pc端登陆",
              "tp99":"0.000",
              "serviceName":"td-passport",
              "url":"/2login",
              "status":"0"
          }
        ]*/
      }
    },
    autoResize: {
      type: Boolean,
      default: true
    }
  },

  data() {
    return {
      curService: 0,// 第几个服务器被点中。0 没有点击服务器
      preStatus: 2,// 0 或 1
      curService2: 0,// 第几个服务器被点中。0 没有点击服务器
      preStatus2: 2,// 0 或 1
      clickFlag: false,
      chartMain: null,
      chartWidth: 100,
      chartHeight: 100,
      // 服务有问题  展现的红色图
      symbol0: 'image://',
      symbol1: 'image://',
      // Options Begin
      chartOption: {
        title: {
          // text: 'Graph'
        },
        tooltip: {},
        // tooltip: {
        //   formatter: '单击: 看基本数据<br/>双击: 跳业务详情',
        //   textStyle: {
        //     fontSize: 10
        //   }
        // },
        animation: true,
        animationDurationUpdate: 1500,
        animationEasingUpdate: 'quinticInOut',
        backgroundColor: '#242730',
        series: [
          {
            type: 'graph',
            layout: 'none',
            symbolSize: 50,
            symbol: 'image://',
            label: {
              normal: {
                show: true,
                position: 'bottom',
                offset: [0, -6], // [0, -18],
                // backgroundColor: 'rgba(255,255,255,.4)',//改文字背景色
                // borderColor: '#f3f3f3',
                borderWidth: 0, // 1,
                borderRadius: 0, // 3,
                fontSize: 12,
                padding: 4,
                color: '#fff'// '#666'
              }
            },
            tooltip: {
              formatter: '单击预览<br/>双击查看详情',//'{b0}: {c0}<br />{b1}: {c1}',
              backgroundColor: 'rgba(240,182,94,0.5)',
              textStyle: {
                // color: '#ff9000',
                fontSize: 10,
              }
            },
            edgeSymbol: ['circle', 'arrow'],
            edgeSymbolSize: [0, 6],
            edgeLabel: {
              normal: {
                textStyle: {
                  fontSize: 12
                }
              }
            },
            data: [],
            links: [],
            lineStyle: {
              normal: {
                opacity: 1, // 0.7,
                width: 2,
                curveness: 0.1,
                color: '#0a92ff'// '#41b34d'
              }
            }
          }
        ]
      }
      // Options End

    }
  },

  watch: {
    chartData: {
      deep: true,
      handler: function(arg) {
        // console.log('arg==>', arg);
        // console.log('data update');
        if (arg[this.curService -1]) {// 处理异常 arg[this.curService -1] 为undefined
          if (arg[this.curService -1].status != "2") {// 后台忽然返回的数据不是我原来保留的2了, 此时须要记录最新值
            this.preStatus = arg[this.curService -1].status;
          }
          arg[this.curService -1].status = "2";
        }

        if (this.curService2 != 0) {//须要将原来的那个那根线的值放回去
          arg[this.curService2 - 1].status = this.preStatus2;
          this.curService2 = 0;//清空值
          this.preStatus2 = 2;
        }
        this.initData();
        this.initChart();
      }
    }
  },

  created() {
    this.initData()
    this.topoClickShow(1);
  },

  mounted() {
    this.initChart()

    // window.addEventListener('resize', this.chartResize, false)
    if (this.autoResize) {
      this.__resizeHanlder = debounce(() => {
        if (this.chartMain) {
          this.chartMain.resize()
        }
      }, 100)
      window.addEventListener('resize', this.__resizeHanlder)
    }
    // 监听侧边栏的变化
    const sidebarElm = document.getElementsByClassName('sidebar-container')[0]
    sidebarElm.addEventListener('transitionend', this.__resizeHanlder)

    this.chartMain.on('click', (param) => {
      if(this.clickFlag) {//取消上次延时未执行的方法
        this.clickFlag = clearTimeout(this.clickFlag);
      }
      
      this.clickFlag = setTimeout(() => {

        this.chartData.forEach((item, index) => {// index 是0开始,topoClickShow方法传人值1开始
            if (item.businessName == param.name) {
                this.topoClickShow(index+1);
            }
        })

        if ((/\>/gim).test(param.name)) {
          let itemIndex = parseInt((param.name).split('>')[1]);//这里的值是 1开始,不是0开始

          this.topoClickShow(itemIndex);
          
            /*
              if (this.preStatus != "2") {//须要数据移位置 (若是为2说明这个变量还没使用是默认值)
                this.preStatus2 = this.preStatus;
                this.curService2 = this.curService;
              }
              this.curService = itemIndex;
              this.preStatus = this.chartData[itemIndex-1].status;console.log("click---this.preStatus==>", this.preStatus);
              this.chartData[itemIndex-1].status = "2";console.log('this.chartData==>',itemIndex-1,'==>',this.chartData,'==>',this.chartData[itemIndex-1]);
            */
        }

        // click 事件的处理
        this.$emit('serviceChain', param)
      }, 300);//延时300毫秒执行
    })

    this.chartMain.on('dblclick', (param) => {
      if(this.clickFlag) {//取消上次延时未执行的方法
        this.clickFlag = clearTimeout(this.clickFlag);
      }

      // dblclick 事件的处理
      this.$emit('dbService', param)
    })

  },

  destroyed() {
    if (!this.chartMain) {
      return
    }
    // window.removeEventListener('resize', this.chartResize, false);

    if (this.autoResize) {
      window.removeEventListener('resize', this.__resizeHanlder)
    }
    const sidebarElm = document.getElementsByClassName('sidebar-container')[0]
    sidebarElm.removeEventListener('transitionend', this.__resizeHanlder)

    this.chartMain.dispose();
    this.chartMain = null;
  },

  methods: {
    getPosition(size) {
      if (!size) {
        return false
      }

      const result = [],
        centerX = this.chartWidth / 2,
        centerY = this.chartHeight / 2,
        radius = centerX,
        angle = Math.round(360 / size)

      for (let i = 0; i < size; i++) {
        result.push({
          x: centerX + radius * Math.cos(angle * i * 3.14 / 180),
          y: centerY + radius * Math.sin(angle * i * 3.14 / 180)
        })
      }

      return result
    },

    topoClickShow(itemIndex) {
      if (this.preStatus != "2") {//须要数据移位置 (若是为2说明这个变量还没使用是默认值)
        this.preStatus2 = this.preStatus;
        this.curService2 = this.curService;
      }
      this.curService = itemIndex;
      this.preStatus = this.chartData[itemIndex-1].status;//console.log("click---this.preStatus==>", this.preStatus);
      this.chartData[itemIndex-1].status = "2";//console.log('this.chartData==>',itemIndex-1,'==>',this.chartData,'==>',this.chartData[itemIndex-1]);
    },

    chartResize() {
      this.chartMain && this.chartMain.resize()
    },

    initData() {
      if (!this.chartData) return
      const aPosition = this.getPosition(this.chartData.length),
        totalData = [{
          name: this.centerName,
          x: this.chartWidth / 2,
          y: this.chartHeight / 2,
          symbolSize: 80,
          // symbol: 'image://' + this.centerIcon //中心图标
          symbol: 'image://'
        }],
        totalLinks = []
      if (aPosition) {
        for (let i = 0, l = this.chartData.length; i < l; i++) {

          totalData.push(Object.assign(aPosition[i], { 
            name: `${this.chartData[i].businessName}`,
            // symbol: this.chartData[i].status == "0" ? this.symbol : this.symbol0 
            symbol: this.chartData[i].status == "0" ? this.symbol : ( this.chartData[i].status == "1" ? this.symbol0 : this.symbol1 )
          }))

          totalLinks.push({
            source: 0,
            target: i + 1,
            label: {
              show: true,
              color: '#0a92ff', // '#333',
              formatter: this.chartData[i].tps+' tps, '+this.chartData[i].tp99+' tp99'//this.chartData[i].info
              // formatter: (this.chartData[i].tps ? this.chartData[i].tps: '')+' tps, '+(this.chartData[i].tp99 ? this.chartData[i].tp99: '')+' tp99'//this.chartData[i].info
              /*formatter: () => {
                let tps = '';
                let tp99 = '';
                if (this.chartData[i].tps != null) tps = this.chartData[i].tps;
                if (this.chartData[i].tp99 != null) tp99 = this.chartData[i].tp99;
                return tps+' tps, '+tp99+' tp99'
              }*/
            },
            lineStyle: {
              // color: this.chartData[i].status == "0" ? '#373c4f' : '#e68700' //#1bec1e
              color: this.chartData[i].status == "0" ? '#373c4f' : (this.chartData[i].status == "1" ? '#d60037' : '#e68700') 
            }
          })
        }
        // console.log("init--totalData==>", totalData);
        this.chartOption.series[0].data = totalData
        this.chartOption.series[0].links = totalLinks
        // this.chartOption.title.text = this.title
      }
    },
    initChart() {
      if (!this.chartMain) {
        // this.chartMain = this.$echarts.init(this.$refs.main);
        this.chartMain = echarts.init(this.$refs.main)
      } else if (this.chartOption.animation) {
        this.chartOption.animation = false
      }
      this.chartMain.clear()
      this.chartMain.setOption(this.chartOption)

    }
  }
}

</script>

<style scoped>
.wrap{ position:relative; width:100%; height:100%; background:#242730; overflow: hidden; }
.charts{ position:absolute; top:3%; left:3%; width:94%; height:94%; }
</style>