可拖拽的流程图/树形图

描述:当前是关于Echarts图表中的 关系图 示例。
 
            //!!!echarts需要是5.0.0.版本以上
const symbol1 = "image://";
const symbol2= 'image://'

var data = [
        { name: '1', value: { serviceId:'crux-gateway', type: 0,category:0,url:'/crux-appmodule/api/appmodule' },x:0,y:100, draggable: true,fixed:true,symbol: symbol2 }, 
        { name: '2', value: { serviceId:'crux-gateway1', type: 1,category:1,url:'/crux-appmodule/api/appmodule'},x:100,y:100, category: 1, draggable: true,fixed:true,symbol: symbol1 }, 
        { name: '3', value: { serviceId:'crux-gateway2', type: 1,category:1,url:'/crux-appmodule/api/appmodule/2' }, x:200,y:100,category: 1, draggable: true,fixed:true,symbol: symbol1 }, 
        { name: '4', value: { serviceId:'crux-gateway', type: 0,category:0,url:'/crux-appmodule/api/appmodule' },x:300,y:0, draggable: true,fixed:true,symbol: symbol2 }, 
        { name: '5', value: { serviceId:'crux-gateway1', type: 1,category:1,url:'/crux-appmodule/api/appmodule'},x:300,y:100, category: 1, draggable: true,fixed:true,symbol: symbol1 }, 
        { name: '6', value: { serviceId:'crux-gateway2', type: 1,category:1,url:'/crux-appmodule/api/appmodule/2' },x:300,y:200, category: 1, draggable: true,fixed:true,symbol: symbol1 }, 
     ]
const positions = {};
var zoomStart = false;
var zoomInfo = {
  zoom:1,
  dx:0,
  dy:0
}
 // tooltip
const tooltip = {
    trigger: 'item',
    textStyle: { fontSize: '100%'},
    formatter: params => {
            return `
                <div>
                    <div>serviceId:${params.value.serviceId}</div>
                    <div>url:${params.value.url}</div>
                </div>
            `
    }
}

const legend = {
    show: true,
    // bottom: '5%',
    // right: '2%',
    // textStyle: { color: '#fff' },
    itemWidth: 20,
    itemHeight: 20,
    // orient: 'vertical',
    data: [
        { name: '成功', icon: symbol2 },
        { name: '失败', icon: symbol1 },
    ]
}
const series = {
    type: 'graph',
    roam:true,
     nodeScaleRatio:0,
    scaleLimit:{
      min:0.5,
      max:3
    },
    symbolSize: 40,
    categories: [ { name: "成功" }, { name: "失败" }],
    label: { 
        show: true,
        position:[-20,40],
        align: 'center',
        textStyle: { fontSize: 12, color: '#fff' },
        // "backgroundColor": "#F4F4F4",
        "borderRadius": [
          0,
          0,
          5,
          5
        ], 
        curveness:true,
        formatter: params => {
            return `{label${params.value.type}|serviceId:}{value|${params.value.serviceId}}\n{label${params.value.type}|url:}{value|${params.value.url}}`
        },
        rich: {
              "label0": {
                "backgroundColor": "#00cc66",
                "color": "#fff",
                "width": 60,
                "height": 30,
                "padding": [
                  0,
                  5,
                  0,
                  10
                ],
                "align":'left'
              },
              "label1": {
                "backgroundColor": "#ff0000",
                "color": "#fff",
                "width": 60,
                "height": 30,
                "padding": [
                  0,
                  5,
                  0,
                  10
                ],
                "align":'left'
              },
              "value": {
                "color": "#888",
                "align": "center",
                "lineHeight": 17,
                "height": 30,
                "borderColor": "#eee",
                "borderWidth": 1,
                "padding": [
                  0,
                  5,
                  0,
                  5
                ]
              }
        }
    },
    labelLayout: function(param){
        positions[param.dataIndex] = param.rect;
        return {
            moveOverlap:true,
            hideOverlap:true
        }
    },
    edgeSymbol: ['none', 'arrow'],
    edgeSymbolSize: [4, 8],
    data: data,
    links: [
        { source: 0, target: 1, value: '',ignoreForceLayout:true }, 
        { source: 1, target: 2, value: '', }, 
        { source: 2, target: 3, value: '', }, 
        { source: 2, target: 4, value: '', }, 
        { source: 2, target: 5, value: '', }, 
    ],
    lineStyle: {
        opacity: 0.9,
        width: 2,
        color: '#333'
    }
}

option = { tooltip, series, legend }
function getGraphicInfo(posMap){
    return Object.keys(posMap).map(function (dataIndex) {
      var item = posMap[dataIndex]
      var system = myChart.getModel().getSeriesByIndex(0).coordinateSystem
      var seriesZoom = system.getZoom();
      var position = myChart.convertToPixel('series',[data[dataIndex].x,data[dataIndex].y])
      return {
        type: 'rect',
        position:[-20,40],
        shape: {
          height:item.height,
          width:item.width
        },
        x:position[0] - 40/2,
        y:position[1] - 40/2,
        scaleX:zoomInfo.zoom,
        scaleY:zoomInfo.zoom,
        // color:'#fff',
        invisible: true,
        draggable: true,
        ondrag: function (dx, dy) {
          onPointDragging(dataIndex, [this.x+20, this.y+20]);
        },
        onmousemove: function () {
          showTooltip(dataIndex);
        },
        onmouseout: function () {
          hideTooltip(dataIndex);
        },
        z: 100
      };
    })
}
setTimeout(function() {
  myChart.setOption({
    graphic:getGraphicInfo(positions)
  });
}, 10);
window.addEventListener('resize', updatePosition);
myChart.on('graphRoam', (e)=>{
  var isZooming = e.hasOwnProperty('zoom')
  if(isZooming){
    zoomInfo.zoom = e.zoom
    zoomInfo.originX= e.originX
    zoomInfo.originY = e.originY
  }else{
    zoomInfo.dy = e.dy
    zoomInfo.dx = e.dx
  }
  zoomStart = true
});
myChart.on('finished', ()=>{
  if(zoomStart){
    zoomStart = false
    updatePosition(true)
  }
});
function updatePosition(isZoom) {
    setTimeout(function() {
      myChart.setOption({
        graphic:getGraphicInfo(positions)
      });
  }, 10);
}
function showTooltip(dataIndex) {
  myChart.dispatchAction({
    type: 'showTip',
    seriesIndex: 0,
    dataIndex: dataIndex
  });
}
function hideTooltip(dataIndex) {
  myChart.dispatchAction({
    type: 'hideTip'
  });
}
function onPointDragging(dataIndex, pos) {
  var position = myChart.convertFromPixel({seriesIndex:0}, pos);
  option.series.data[dataIndex].x = position[0]
  option.series.data[dataIndex].y = position[1]
  myChart.setOption(option)
  updatePosition()
}