记录使用echarts的graph类型绘制流程图全过程(一)-x,y位置的计算

  • 2019 年 10 月 8 日
  • 筆記

先说下本次案例业务需求,输入2个节点,获取数据后绘制出2个节点间的路径,之前使用的是网状图,但是网状图的效果不佳,需要转换成流程图的模式:

那么如何在不修改数据的情况下,实现类似效果尼?

看了下echarts的graph类型,可以实现类似的,下面是官方的实例

从实例中可以看出,难点在于节点的显示位置x,y和曲线的设置。业务数据中:
1、节点的数量不定,关系的数量不定,
2、后台返回的数据只有单独的节点信息和关系信息

实现思路:

1、分析数据,获取前后节点关系,获得行数位置(节点的xIndex信息)

在节点数组中找到开始节点并设置xIndex 为1,然后从它开始找第二层的节点

// 获取节点的x轴顺序      setNodeOrder () {        this.entityList.forEach(item => {          if (item.id === this.startNode) {            item.xIndex = 1;          }            // 设置一个对象,记录节点信息,用于后面查询信息,比数组查询来的快          this.nodesObj[item.id] = item;        });          this.findNodeOrder(2, this.startNode);      }        // 广度遍历--按顺序找到第index层的数据,并设置对应的参数,再层层递进      findNodeOrder (xIndex, sId){        }

想一下,如果是第二层的节点,那么应该是关系中,源是startNode或者目的节点是startNode,如此类推,层层递进,
这里需要使用广度遍历而不是深度遍历,设定2个数组,currentQueue记录当前层的节点,afterQueue记录下一层的节点,当currentQueue层遍历完后,将afterQueue变成currentQueue,继续遍历,直至结束,核心代码如下:

......           let nextIds = [];          this.relationList.forEach(item => {            // 源            if (item.source === sId) {              if (!this.nodesObj[item.target].xIndex) {                this.nodesObj[item.target].xIndex = xIndex;                nextIds.push(item.target);              }            }            // 目的            if (item.target === sId) {              if (!this.nodesObj[item.source].xIndex) {                this.nodesObj[item.source].xIndex = xIndex;                nextIds.push(item.source);              }            }          });            let nextIdsLen = nextIds.length;            // 1、没有当前的队列,没有后续的队列,有nextIds          if (            !this.currentQueue.length &&            !this.afterQueue.length &&            nextIdsLen          ) {            this.currentQueue = nextIds;            this.currentXindex = this.currentXindex + 1;            this.setNextOrder();          } else if (this.currentQueue.length && nextIdsLen) {            // 2、有当前的队列在遍历,排队            this.afterQueue = this.afterQueue.concat(nextIds);          } else if (!this.currentQueue.length && this.afterQueue.length) {            // 3、没有当前的队列了,有排队的队列,则排队的进去            if (nextIdsLen) {              this.afterQueue = this.afterQueue.concat(nextIds);            }            this.currentQueue = JSON.parse(JSON.stringify(this.afterQueue));            this.afterQueue = [];            this.currentXindex = this.currentXindex + 1;          }

setNextOrder函数:
js// 设置下个 setNextOrder () { while (this.currentQueue.length) { let id = this.currentQueue.shift(); this.findNodeOrder(this.currentXindex, id); } }

2、根据行数信息设置列数的位置(节点的yIndex层的信息)

先排序第一层和第二层的顺序,因为第一层就一个节点,第二层和第一层的关系都是一样的,所以顺序无所谓,可以直接遇到就叠加

// 排完了x,再排列y          // 先排第一和二行的元素的y,按顺序排          let maxXindex = 1;          let secondYIndexObj = {};            for (let i in this.nodesObj) {            let item = this.nodesObj[i];            if (!item.yIndex) {              maxXindex = item.xIndex > maxXindex ? item.xIndex : maxXindex;              if (item.xIndex === 1) {                item.yIndex = 1;                this.yIndexObj[item.id] = {                  xIndex: 1,                  yIndex: 1                };              }              if (item.xIndex === 2) {                if (!secondYIndexObj[item.xIndex]) {                  item.yIndex = 1;                  // 记录当前的y轴上的节点个数                  secondYIndexObj[item.xIndex] = 1;                  this.yIndexObj[item.id] = {                    xIndex: 2,                    yIndex: 1                  };                } else {                  item.yIndex = secondYIndexObj[item.xIndex] + 1;                  secondYIndexObj[item.xIndex] = item.yIndex;                    this.yIndexObj[item.id] = {                    xIndex: 2,                    yIndex: item.yIndex                  };                }              }            }          }

但是从第三层开始就要注意了,不能再按照第二层的递增顺序的方法来控制。因为第二层与第三层的关系和数量都不一定,如果随机排列会导致线很乱,最好的方法是根据第二层的顺序,获取第三层的顺序。即将第二层第N个节点有关的的节点设置在第N层。这样如果第二层的数量与第三层的数量相等,那么就完全在一条直线上了。如果数量不等,也不至于那么混乱~但是如果第三层有多个节点与第二层的同一个节点有关系,这个时候我选择的是随机选择层级了~

          // 从第三层开始          if (maxXindex > 2) {            // 后面列的排序根据前一列为准(尽量保证在一条直线上)            for (let j = 3; j <= maxXindex; j++) {              for (let i in this.nodesObj) {                let item = this.nodesObj[i];                while (item.xIndex === j && !item.yIndex) {                  // 先看跟它一层的节点有多少个                  if (!this.nodeOrders[j]) {                    let totals = this.findYtotal(j);                    this.nodeOrders[j] = [];                    for (let i = 1; i <= totals; i++) {                      this.nodeOrders[j].push(i);                    }                  }                    // 找第二层中的对应的层级                  let findX = j - 1;                    // 找到所有的层级                  let findYs = this.findLinkNode(item.id, findX);                    // 找到还有的层级                  let sameArr = findYs.filter(x =>                    this.nodeOrders[j].includes(x)                  );                  let findY;                    // 找不到按顺序抽取了                  if (!sameArr.length) {                    // 只能随机选择一个了                    let ran = Math.floor(                      Math.random() * this.nodeOrders[j].length                    );                    findY = this.nodeOrders[j][ran];                      this.nodeOrders[j].splice(ran, 1);                  } else {                    findY = sameArr[0];                    // 去除该顺序                    let order;                    this.nodeOrders[j].find((num, k) => {                      if (num === findY) {                        order = k;                        return true;                      }                    });                    this.nodeOrders[j].splice(order, 1);                  }                    this.yIndexObj[item.id] = {                    xIndex: j,                    yIndex: findY                  };                  item.yIndex = findY;                }              }            }          }

3、设置具体的位置信息

获取图表的中心位置centerY,将起点和终点的y位置放在中间,其他的根据上面获取的xIndex和yIndex的还有根据情况设置的间距(gapX,gapY)及节点的大小(size)计算出每个节点的x和y信息

// 设置节点的位置x,y      setNodesPositon () {        for (let i in this.nodesObj) {          let item = this.nodesObj[i];          if (item.id === this.startNode) {            item.y = this.centerY;            item.x = 1;          }            if (!item.x) {            item.x = this.gapX * (item.xIndex - 1) + this.size / 2;          }            if (!item.y && !item.total) {            item.total = this.findYtotal(item.xIndex);              if (item.total === 1) {              item.y = this.centerY;            } else {              let middleNum = item.total / 2;              let ceilNum = Math.ceil(middleNum);                let topGap;              let bottomGap;                let ty = (ceilNum - item.yIndex) * (this.gapY + this.size);              let by = (item.yIndex - ceilNum) * (this.gapY + this.size);                if (item.total % 2 === 1) {                topGap = this.centerY - ty;                bottomGap = this.centerY + by;              } else {                topGap = this.centerY - (ty + this.gapY + 0.5 * this.size);                bottomGap = this.centerY + (by - 0.5 * this.size);              }                if (item.yIndex <= middleNum) {                item.y = topGap;              } else {                item.y = bottomGap;              }                // 奇数              if (item.total % 2 === 1) {                if (item.yIndex === ceilNum) {                  item.y = this.centerY;                }              }            }          }        }      }