1、首先下载GOJS

npm install gojs --save

2、Vue中使用

import gojs from "gojs";
var go = gojs;
var $ = go.GraphObject.make;
//代码所需要的JSON格式
import json from "./json";
//项目中应用到的图片
import zhong from "@/assets/basic/zhong.png";
import da from "@/assets/basic/da.png";
import xiao from "@/assets/basic/xiao.png";

3、将注册实例以及业务代码编写再init函数中在mounted挂载中调用

init() {
      // 结合后端数据进行 赋值各项不同节点的与图片文字
      json["nodeDataArray"].forEach((item) => {
        switch (+item.size) {
          case 100:
            item.source = xiao;
            item.stroke = item.type === "0" ? "red" : "#56b1ff";
            break;
          case 150:
            item.source = zhong;
            item.stroke = "#fff";
            break;
          case 200:
            item.source = da;
            item.stroke = "#fff";
            break;
          default:
            item.source = xiao;
            break;
        }
        if (item.type === "0") {
          item.stroke = "red";
        }
      });
      // 判断实例是否存在进行数据变更
      if (this.myDiagram) {
        this.myDiagram.model = go.Model.fromJson(json);

        return;
      }

      this.myDiagram = $(
        go.Diagram,
        this.$refs.myDiagram, //获取原生dom进行gojs的注入
        {
          "undoManager.isEnabled": true, //是否可以进行撤销等操作
          // "undoManager.isEnabled": false,
          initialContentAlignment: go.Spot.Center, //启动视口中间的所有内容
          // "toolManager.mouseWheelBehavior": go.ToolManager.WheelNone, //鼠标滚轮事件禁止
          "toolManager.mouseWheelBehavior": go.ToolManager.WheelZoom, //开启滚轮事件
          // allowHorizontalScroll: false, //禁止水平拖动
          // allowVerticalScroll: false, // 禁止垂直拖动
          // isReadOnly: true, //是否为只读状态
          minScale: 0.2, //  缩小临界值
          maxScale: 1.5, //放大临街值
          scale: 0.38, //初始视图大小比例
          LinkDrawn: this.maybeChangeLinkCategory, // 连线判断
          LinkRelinked: this.maybeChangeLinkCategory, // 连线判断
        }
      );

      const portSize = new go.Size(6, 6); //小节点大小
      var _that = this;
      // 给节点的设置右键按钮  //  nodeMenu上下文菜单
      const nodeMenu = $(
        "ContextMenu",
        this.makeButton("复制", (e, obj) =>
          e.diagram.commandHandler.copySelection()
        ),
        this.makeButton("粘贴", (e, obj) =>
          e.diagram.commandHandler.pasteSelection()
        ),
        this.makeButton("删除", (e, obj) =>
          e.diagram.commandHandler.deleteSelection()
        ),
        this.makeButton("更改背景", (e, obj) =>
          this.changeBackground(obj.part.adornedObject)
        ),
        this.makeButton("添加头部小节点", (e, obj) => this.addPort("top")),
        this.makeButton("添加左侧小节点", (e, obj) => this.addPort("left")),
        this.makeButton("添加右侧小节点", (e, obj) => this.addPort("right")),
        this.makeButton("添加底部小节点", (e, obj) => this.addPort("bottom"))
      );

      // 给小节点(端口)设置删除
      const portMenu = $(
        "ContextMenu",
        this.makeButton("更改小节点颜色", (e, obj) =>
          this.changeColor(obj.part.adornedObject)
        ),
        this.makeButton(
          "删除小节点",
          // 在click事件处理程序中,obj。一部分是装饰;
          // 它的父对象是端口
          (e, obj) => this.removePort(obj.part.adornedObject)
        ),
        this.makeButton("删除全部小节点", (e, obj) =>
          this.removeAll(obj.part.adornedObject)
        )
      );

      //   节点模板
      this.myDiagram.nodeTemplate = $(
        go.Node,
        "Table",
        {
          locationObjectName: "BODY",
          locationSpot: go.Spot.Center, //将节点放置中间
          selectionObjectName: "BODY",
          contextMenu: nodeMenu, // 去除右键
        },
        // 动态绑定位置信息
        new go.Binding("location", "loc", go.Point.parse).makeTwoWay(
          go.Point.stringify
        ),
        // 声明面板 the body
        $(
          go.Panel,
          "Auto",
          //   面板默认设置
          {
            row: 1,
            column: 1,
            name: "BODY",
            stretch: go.GraphObject.Fill,
          },
          $(
            go.Picture,
            {
              name: "Picture",
              source: zhong,
              width: 100,
              height: 100,
              cursor: "pointer",
            },
            //根据JSON格式动态绑定,第三个参数接受一个函数用户业务逻辑的判断
            new go.Binding("source", "source").makeTwoWay(),
            new go.Binding("width", "size", (size) => {
              return +size;
            }),
            new go.Binding("height", "size", (size) => {
              return +size;
            }).makeTwoWay(),
            new go.Binding("name", "type").makeTwoWay()
          ),
            // 这次我使用用的是图片代替节点样式 下面内容是设置节点样式部分
          // $(
          //   go.Shape,
          //   "Circle" , //圆形
          //   // fn( new go.Binding('backgroud').makeTwoWay()),

          //   {
          //     name: "SHAPE",
          //     //   背景色和填充
          //     background: null,

          //     // 内容颜色
          //     fill: "#59b2fb",
          //     //边框色
          //     stroke: '#fcfafa',
          //     strokeWidth: 5,
          //     cursor: "pointer",

          //     // figure: "Circle",
          //     // Ellipse椭圆
          //     // 最小值
          //     minSize: new go.Size(42, 42),
          //   },
          //   //   动态设置
          //   new go.Binding("background", "backgroud").makeTwoWay(),
          //   // new go.Binding("geometryString", "geometryString").makeTwoWay(),
          // ),
            
          $(
            go.TextBlock,
            {
              font: "11pt Helvetica, Arial, sans-serif",
              stroke: "#fff",
              cursor: "pointer",
              editable: false,
            },
            new go.Binding("text", "name").makeTwoWay(),
            new go.Binding("stroke", "stroke").makeTwoWay(),
            new go.Binding("font", "size", (k) => {
              switch (+k) {
                case 100:
                  return "10pt Helvetica, Arial, sans-serif";
                case 150:
                  return "21px Segoe UI,sans-serif";
                case 200:
                  return "24px Segoe UI,sans-serif";
                default:
                  return "11pt Helvetica, Arial, sans-serif";
              }
            })
          )
        ),
        // 设置小节点
        // 绑定 itemArray leftArray
        // 为itemArray中的每个项创建,并绑定到数据。左数组
        $(go.Panel, "Vertical", new go.Binding("itemArray", "leftArray"), {
          row: 1,
          column: 0,
          itemTemplate: $(
            go.Panel,
            {
              _side: "left", //内部属性,以便更容易地判断它位于哪一侧
              fromSpot: go.Spot.Left, //从左
              toSpot: go.Spot.Left, //到左
              fromLinkable: true,
              toLinkable: true,
              cursor: "pointer",
              contextMenu: portMenu,
            },
            new go.Binding("portId", "portId"),
            $(
              go.Shape,
              "Rectangle",
              {
                fill: "#2e68cc",
                stroke: null,
                strokeWidth: 5,
                desiredSize: portSize,
                margin: new go.Margin(10, -3),
              },
              new go.Binding("fill", "portColor")
            )
          ),
        }),
        $(go.Panel, "Horizontal", new go.Binding("itemArray", "topArray"), {
          row: 0,
          column: 1,
          itemTemplate: $(
            go.Panel,
            {
              _side: "top",
              fromSpot: go.Spot.Top,
              toSpot: go.Spot.Top,
              fromLinkable: true,
              toLinkable: true,
              cursor: "pointer",
              contextMenu: portMenu,
            },
            new go.Binding("portId", "portId"),
            $(
              go.Shape,
              "Rectangle",
              {
                stroke: null,
                strokeWidth: 0,
                desiredSize: portSize, //小节点大小
                margin: new go.Margin(0, 10),
              },
              new go.Binding("fill", "portColor")
            )
          ), // end itemTemplate
        }),
        $(go.Panel, "Vertical", new go.Binding("itemArray", "rightArray"), {
          row: 1,
          column: 2,
          itemTemplate: $(
            go.Panel,
            {
              _side: "right",
              fromSpot: go.Spot.Right,
              toSpot: go.Spot.Right,
              fromLinkable: true,
              toLinkable: true,
              cursor: "pointer",
              contextMenu: portMenu,
            },
            new go.Binding("portId", "portId"),
            $(
              go.Shape,
              "Rectangle",
              {
                stroke: null,
                strokeWidth: 0,
                desiredSize: portSize,
                margin: new go.Margin(6, 0),
              },
              new go.Binding("fill", "portColor")
            )
          ), // end itemTemplate
        }),
        $(go.Panel, "Horizontal", new go.Binding("itemArray", "bottomArray"), {
          row: 2,
          column: 1,
          itemTemplate: $(
            go.Panel,
            {
              _side: "bottom",
              fromSpot: go.Spot.Bottom,
              toSpot: go.Spot.Bottom,
              fromLinkable: true,
              toLinkable: true,
              cursor: "pointer",
              contextMenu: portMenu,
            },
            new go.Binding("portId", "portId"),
            $(
              go.Shape,
              "Rectangle",
              {
                stroke: null,
                strokeWidth: 0,
                fill: "#2e68cc",
                desiredSize: portSize,
                margin: new go.Margin(0, 10),
              },
              new go.Binding("fill", "portColor")
            )
          ), // end itemTemplate
        })
      );

      // 实线
      this.myDiagram.linkTemplate = $(
        "Link",
        {
          routing: go.Link.AvoidsNodes,
          adjusting: go.Link.FROM,
          curve: go.Link.JumpGap,
          reshapable: true,
          resegmentable: true,
          relinkableFrom: true,
          relinkableTo: true,
          corner: 4,
          toShortLength: 1,
          cursor: "pointer",
        },
        new go.Binding("points").makeTwoWay(),
        // 线条颜色
        $(
          "Shape",
          {
            name: "LINE",
            isPanelMain: true,
            stroke: "#5aaefd",
            // stroke: "#01C0CA",

            strokeWidth: 4,
          },
          new go.Binding("stroke", "color").makeTwoWay(),
          new go.Binding("strokeWidth", "voltage", (item) => {
            switch (item) {
              case "500kV":
                return 10;
              case "220kV":
                return 10;
              case "110kV":
                return 7;
              default:
                return 7;
            }
          })
        ),
    // 用于动画效果
        // $(
        //   "Shape",
        //   {
        //     isPanelMain: true,
        //     // stroke: "lightblue",
        //     stroke: "#01C0CA",
        //     // stroke: "",
        //     strokeWidth: 8,
        //     name: "DongHua",
        //     // 10 35
        //     strokeDashArray: [10, 35],
        //   },
        //   new go.Binding("stroke", "ShapeColor").makeTwoWay()
        // ),
        // 线路背景设置
        // $(go.Shape, "RoundedRectangle", {
        //   name: "SHAPE",
        //   fill: "rgba(255,255,255,0.4)",
        //   height: 24,
        //   strokeWidth: 1,
        //   // stroke: "rgba(255,255,255,0.8)",
        //   stroke: "rgba(255,255,255,0.8)",
        // }),
        $(
          go.TextBlock,
          "",
          {
            textAlign: "center",
            font: "12pt helvetica, arial, sans-serif",
            stroke: "#A1F9FA",
            editable: true,
          },
          new go.Binding("text", "port1").makeTwoWay()
        )
      );
    //在每条线中心添加小节点以便实现T接线方式
      this.myDiagram.nodeTemplateMap.add(
        "LinkLabel",
        $(
          "Node",
          {
            selectable: false,
            avoidable: false,
            layerName: "Foreground",
          },
          $("Shape", "Ellipse", {
            width: 10,
            height: 10,
            stroke: null,
            fill: null, //去掉小圆点
            portId: "",
            fromLinkable: true,
            toLinkable: true,
            cursor: "pointer",
          })
        )
      );

      // GraphLinkModel对链接标签节点的支持需要指定两个属性
      // 优化连线

      this.myDiagram.model = $(go.GraphLinksModel, {
        linkLabelKeysProperty: "labelKeys",
      });
      // 每当通过链接工具绘制新链接时,它还添加节点数据对象。
      // 用作链接的标签节点的,以允许向链接绘制链接或从链接绘制链接。
      this.myDiagram.toolManager.linkingTool.archetypeLabelNodeData = {
        category: "LinkLabel",
      };

      this.myDiagram.model = go.Model.fromJson(json);

        //动画效果
      // this.changeImg();
    },

4、判断连线是否为小黑点(并不是节点,而是用于T接线的黑点)

maybeChangeLinkCategory(e) {
      var link = e.subject;
      var linktolink = link.fromNode.isLinkLabel || link.toNode.isLinkLabel;
        //在线中判断是否连接小黑点用于个人业务逻辑
      e.diagram.model.setCategoryForLinkData(
        link.data,
        linktolink ? "linkToLink" : ""
      );
    },

5、右键按钮事件封装成公共方法

//上下文button
    makeButton(text, action, visiblePredicate) {
      return $(
        "ContextMenuButton",
        $(go.TextBlock, text, {
          alignment: go.Spot.Left,
          margin: 5,
          textAlign: "center",
          font: "14px sans-serif",
          opacity: 0.85,
          stroke: "#404040",
          width: 100,
        }),
        { click: action },
        // don't bother with binding GraphObject.visible if there's no predicate
        visiblePredicate
          ? new go.Binding("visible", "", (o, e) =>
              o.diagram ? visiblePredicate(o, e) : false
            ).ofObject()
          : {}
      );
    },

6.对应的右键事件处理函数封装

// 添加头部小节点
    addPort(side) {
      this.myDiagram.startTransaction("addPort");
      this.myDiagram.selection.each((node) => {
        // skip any selected Links
        if (!(node instanceof go.Node)) return;
        // compute the next available index number for the side
        let i = 0;
        while (node.findPort(side + i.toString()) !== node) i++;
        // now this new port name is unique within the whole Node because of the side prefix
        const name = side + i.toString();
        // get the Array of port data to be modified
        const arr = node.data[side + "Array"];
        if (arr) {
          // create a new port data object
          const newportdata = {
            portId: name,
            portColor: "#2e68cc",
          };
          // and add it to the Array of port data
          this.myDiagram.model.insertArrayItem(arr, -1, newportdata);
        }
      });
      this.myDiagram.commitTransaction("addPort");
    },
    // 删除小节点
    removePort(port) {
      this.myDiagram.startTransaction("removePort");
      const pid = port.portId;
      const arr = port.panel.itemArray;
      for (let i = 0; i < arr.length; i++) {
        if (arr[i].portId === pid) {
          this.myDiagram.model.removeArrayItem(arr, i);
          break;
        }
      }
      this.myDiagram.commitTransaction("removePort");
    },
    // 删除全部小节点
    removeAll(port) {
      this.myDiagram.startTransaction("removePorts");
      const nodedata = port.part.data;
      const side = port._side; // there are four property names, all ending in "Array"
      this.myDiagram.model.setDataProperty(nodedata, side + "Array", []); // an empty Array
      this.myDiagram.commitTransaction("removePorts");
    },
    // 更改小节点颜色
    changeColor(port) {
      this.myDiagram.startTransaction("colorPort");
      const data = port.data;
      this.myDiagram.model.setDataProperty(data, "portColor", "#ddd");
      this.myDiagram.commitTransaction("colorPort");
    },
    // 更改背景色
    changeBackground(port) {
      // this.myDiagram.startTransaction("colorPort");
      const data = port.data;
      this.myDiagram.model.setDataProperty(data, "source", error2);
      // this.myDiagram.commitTransaction("colorPort");
    },

7.动画效果的实现

AnimationYes() {
      var animation = new go.Animation();
      animation.easing = go.Animation.EaseLinear;
      this.myDiagram.links.each(function (link) {
        // 60 20
        animation.add(link.findObject("DongHua"), "strokeDashOffset", 60, 20);
      });
      animation.runCount = Infinity;
      animation.start();
    },


模拟电流