广告

买白酒,找南将

Skip to content

节点碰撞检测与位移限制

Vue2 版本

node-dragging.vue

javascript
<template>
  <div>
    <div style="height:calc(100vh);" @mousemove="onMouseMove">
      <RelationGraph
          ref="graphRef"
          :options="graphOptions"
          :on-node-click="onNodeClick"
          :on-line-click="onLineClick"
          :on-node-dragging="onNodeDragging"
          :on-node-drag-start="onNodeDragStart"
          :on-node-drag-end="onNodeDragEnd"
      >
        <template #node="{node}">
          <div><!---------------- if node a ---------------->
            <div style="background-color: #f39930">{{node.text}}</div>
            <div v-if="node.id==='SYS_ROLE'" style="color: #df7f03;padding:10px;">Not allowed to enter</div>
          </div>
        </template>
        <template #graph-plug>
          <!-- To facilitate understanding, the CSS style code is directly embedded here. -->
          <div style="z-index: 300;position: absolute;left:10px; top: calc(100% - 140px);font-size: 12px;background-color: #ffffff;border:#efefef solid 1px;border-radius: 10px;width:250px;height:130px;">
            <div style="padding-left:10px;">
              <div style="line-height: 30px;">The color of the moving over node:</div>
              <div style="margin-top: 10px;">
                Allowed drop
                <div style="height:5px;width:80px;background-color: rgba(212,252,189,0.82);"></div>
              </div>
              <div style="margin-top: 10px;">
                Not allowed drop
                <div style="height:5px;width:80px;background-color: rgba(252,189,189,0.82);"></div>
              </div>
              <div style="margin-top: 10px;">
                Not allowed to enter
                <div style="height:5px;width:80px;background-color: rgba(214,165,246,0.82);"></div>
              </div>
            </div>
          </div>
        </template>
        <template #canvas-plug>
          <!--- 可以将一些不允许被拖动的元素放在这里 --->
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

<script>
// 如果您没有在main.js文件中使用Vue.use(RelationGraph); 就需要使用下面这一行代码来引入relation-graph
// import RelationGraph from 'relation-graph';
import { RGNodesAnalyticUtils } from 'relation-graph';
const { RGNodesAnalytic } = RGNodesAnalyticUtils;
export default {
  name: 'Demo',
  components: {},
  data() {
    return {
      isShowCodePanel: false,
      graphOptions: {
        debug: false,
        // backgroundImage: '/images/graph-grid-bg.png',
        // backgroundColor: '#fce7d3',
        allowSwitchLineShape: true,
        allowSwitchJunctionPoint: true,
        allowShowDownloadButton: true,
        defaultJunctionPoint: 'border',
        placeOtherNodes: false,
        placeSingleNode: false,
        defaultNodeColor: '#ffffff',
        defaultNodeBorderWidth: 1,
        defaultNodeBorderColor: '#f39930',
        defaultNodeWidth: 170,
        defaultNodeHeight: 70,
        defaultLineColor: '#f39930',
        defaultLineWidth: 2,
        defaultLineTextOffset_y: -2,
        layout: {
          layoutName: 'fixed'
        }
        // 这里可以参考"Graph 图谱"中的参数进行设置
      }
    };
  },
  mounted() {
    this.showGraph();
  },
  methods: {
    showGraph() {
      const jsonData = {
        'rootId': 'SYS_ROLE',
        'nodes': [
          { 'id': 'SYS_USER', 'text': 'SYS_USER', 'nodeShape': 1, 'opacity': 1, 'x': -32, 'y': -427, width: 120, height: 80, 'data': { dropAble: true }},
          { 'id': 'SYS_DEPT', 'text': 'SYS_DEPT', 'nodeShape': 0, 'opacity': 1, 'x': -244, 'y': -283, 'data': { dropAble: true }},
          { 'id': 'SYS_ROLE', 'text': 'SYS_ROLE', 'nodeShape': 1, 'opacity': 1, 'x': -21, 'y': -171, width: 300, height: 200, 'data': { allowOverlap: false }},
          { 'id': 'SYS_USER_ROLE', 'text': 'SYS_USER_ROLE', 'nodeShape': 1, 'opacity': 1, 'x': 405, 'y': -174 },
          { 'id': 'SYS_RESOURCE', 'text': 'SYS_RESOURCE', 'nodeShape': 1, 'opacity': 1, 'x': -246, 'y': -80, 'data': { dropAble: true }},
          { 'id': 'SYS_ROLE_RESOURCE', 'text': 'SYS_ROLE_RESOURCE', 'nodeShape': 1, 'opacity': 1, 'x': 338, 'y': -100 }
        ],
        'lines': [
          { from: 'SYS_DEPT', to: 'SYS_USER', text: 'xxxxxxx' },
          { from: 'SYS_DEPT', to: 'SYS_USER' },
          { from: 'SYS_DEPT', to: 'SYS_USER' },
          { from: 'SYS_DEPT', to: 'SYS_USER' },
          { from: 'SYS_DEPT', to: 'SYS_USER' },
          { from: 'SYS_DEPT', to: 'SYS_USER' }
        ]
      };
      this.$refs.graphRef.setJsonData(jsonData, (graphInstance) => {
        // 这些写上当图谱初始化完成后需要执行的代码
      });
    },
    onNodeDragging(nodeObject, newX, newY, $event) {
      console.log('onNodeDragging:', nodeObject);
      const graphInstance = this.$refs.graphRef.getInstance();
      let rejectOverlaped = false;
      let limitedPosition;
      for (const n of graphInstance.getNodes()) {
        if (nodeObject === n) continue;
        const shapeA = nodeObject.nodeShape || graphInstance.options.defaultNodeShape || 0;
        const shapeB = n.nodeShape || graphInstance.options.defaultNodeShape || 0;
        const nodeBox = {
          el: nodeObject.el,
          x: newX,
          y: newY
        };
        const overlap = RGNodesAnalytic.shapesOverlap(nodeBox, n, shapeA, shapeB);
        if (overlap) {
          let newColor = n.data.dropAble ? 'rgba(212,252,189,0.82)' : 'rgba(252,189,189,0.82)';
          if (n.data.allowOverlap === false) {
            rejectOverlaped = true;
            limitedPosition = RGNodesAnalytic.getNoOverlapLimitedPosition(nodeObject, newX, newY, n);
            newColor = 'rgba(214,165,246,0.82)';
          }
          if (!n.data.orignColorSaved) { // Only the initial node color is saved
            n.data.orignColorSaved = true;
            n.data.orignColor = n.color;
          }
          n.color = newColor;
        } else {
          n.color = undefined;
        }
      }
      if (rejectOverlaped) {
        return limitedPosition;
      }
    },
    onNodeDragStart(nodeObject, $event) {
      console.log('onNodeDragStart:', nodeObject);
      nodeObject.data.startX = nodeObject.x;
      nodeObject.data.startY = nodeObject.y;
    },
    onNodeDragEnd(nodeObject, $event) {
      console.log('onNodeDragEnd:', nodeObject);
      const graphInstance = this.$refs.graphRef.getInstance();
      for (const n of graphInstance.getNodes()) {
        if (nodeObject === n) continue;
        if (n.data.allowOverlap === false) continue;
        if (!n.data.dropAble) continue;
        const shapeA = nodeObject.nodeShape || graphInstance.options.defaultNodeShape || 0;
        const shapeB = n.nodeShape || graphInstance.options.defaultNodeShape || 0;
        const overlap = RGNodesAnalytic.shapesOverlap(nodeObject, n, shapeA, shapeB);
        if (overlap) {
          this.updateNodeAsChildren(nodeObject, n);
          break;
        }
      }
      for (const n of graphInstance.getNodes()) {
        n.color = n.data.orignColor; // Restore the original color of the node
      }
    },
    updateNodeAsChildren(node, parentNode) {
      console.log('updateNodeAsChildren:', parentNode.id, '---->', node.id);
      node.x = node.data.startX;
      node.y = node.data.startY;
      const graphInstance = this.$refs.graphRef.getInstance();
      graphInstance.addLines([{
        from: parentNode.id,
        to: node.id,
        text: '新连线',
        color: 'rgba(214,165,246,0.82)',
        lineWidth: 3
      }]);
    },
    onNodeClick(nodeObject, $event) {
      console.log('onNodeClick:', nodeObject);
    },
    onLineClick(lineObject, linkObject, $event) {
      console.log('onLineClick:', lineObject);
    },
    onMouseMove(e) {
      if (e.target) {
        const tagName = e.target.tagName;
        if (tagName === 'path' || tagName === 'text') {
          const graphInstance = this.$refs.graphRef.getInstance();
          const link = graphInstance.isLink(e.target);
          if (link) {
            console.log('Show Detail:', link.fromNode.id, link.toNode.id, link.text);
          }
        } else {
          console.log('Hide Detail');
        }
      }
    }
  }
};

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

::v-deep .relation-graph {
  //customize node
  .rel-node {
    overflow: hidden;
  }
  //customize toolbar style
  .rel-toolbar {
    background-color: #f39930;
    color: #ffffff;
    .c-current-zoom {
      color: #ffffff;
    }
  }
  //customize background image size
  .rel-map {
    background-size: 50px 50px;
    //background-image: radial-gradient(circle,rgba(0,0,0,.1) 3px,transparent 0);
    background-image: linear-gradient(90deg, rgba(243, 153, 48, 0.3) 1px,transparent 0),linear-gradient(180deg,rgba(243, 153, 48, 0.3) 1px,transparent 0);
  }
}
</style>

Vue3 版本

node-dragging.vue

javascript
<template>
  <div>
    <div style="height:calc(100vh);" @mousemove="onMouseMove">
      <RelationGraph

          ref="graphRef"
          :options="graphOptions"
          :on-node-click="onNodeClick"
          :on-line-click="onLineClick"
          :on-node-dragging="onNodeDragging"
          :on-node-drag-start="onNodeDragStart"
          :on-node-drag-end="onNodeDragEnd"
      >
        <template #node="{node}">
          <div><!---------------- if node a ---------------->
            <div style="background-color: #f39930">{{node.text}}</div>
            <div v-if="node.id==='SYS_ROLE'" style="color: #df7f03;padding:10px;">Not allowed to enter</div>
          </div>
        </template>
        <template #graph-plug>
          <!-- To facilitate understanding, the CSS style code is directly embedded here. -->
          <div style="z-index: 300;position: absolute;left:10px; top: calc(100% - 140px);font-size: 12px;background-color: #ffffff;border:#efefef solid 1px;border-radius: 10px;width:250px;height:130px;">
            <div style="padding-left:10px;">
              <div style="line-height: 30px;">The color of the moving over node:</div>
              <div style="margin-top: 10px;">
                Allowed drop
                <div style="height:5px;width:80px;background-color: rgba(212,252,189,0.82);"></div>
              </div>
              <div style="margin-top: 10px;">
                Not allowed drop
                <div style="height:5px;width:80px;background-color: rgba(252,189,189,0.82);"></div>
              </div>
              <div style="margin-top: 10px;">
                Not allowed to enter
                <div style="height:5px;width:80px;background-color: rgba(214,165,246,0.82);"></div>
              </div>
            </div>
          </div>
        </template>
        <template #canvas-plug>
          <!--- 可以将一些不允许被拖动的元素放在这里 --->
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

<script lang="ts" setup>
import { defineComponent, ref, onMounted } from 'vue';
import RelationGraph from 'relation-graph-vue3';
import { RGNodesAnalyticUtils, RGLink, RGNode, RGLine, RGUserEvent, RGJsonData, RGOptions, RelationGraphComponent } from 'relation-graph-vue3';

const { RGNodesAnalytic } = RGNodesAnalyticUtils;

const graphRef = ref<RelationGraphComponent>();
const isShowCodePanel = ref(false);
const graphOptions: RGOptions = {
    debug: false,
    // backgroundImage: '/images/graph-grid-bg.png',
    // backgroundColor: '#fce7d3',
    allowSwitchLineShape: true,
    allowSwitchJunctionPoint: true,
    allowShowDownloadButton: true,
    defaultJunctionPoint: 'border',
    placeOtherNodes: false,
    placeSingleNode: false,
    defaultNodeColor: '#ffffff',
    defaultNodeBorderWidth: 1,
    defaultNodeBorderColor: '#f39930',
    defaultNodeWidth: 170,
    defaultNodeHeight: 70,
    defaultLineColor: '#f39930',
    defaultLineWidth: 2,
    defaultLineTextOffset_y: -2,
    layout: {
        layoutName: 'fixed'
    }
    // 这里可以参考"Graph 图谱"中的参数进行设置

};

onMounted(() => {
    showGraph();
});

const showGraph = async() => {
    const jsonData: RGJsonData = {
        rootId: 'SYS_ROLE',
        nodes: [
            { id: 'SYS_USER', text: 'SYS_USER', nodeShape: 1, opacity: 1, x: -32, y: -427, width: 120, height: 80, data: { dropAble: true } },
            { id: 'SYS_DEPT', text: 'SYS_DEPT', nodeShape: 0, opacity: 1, x: -244, y: -283, data: { dropAble: true } },
            { id: 'SYS_ROLE', text: 'SYS_ROLE', nodeShape: 1, opacity: 1, x: -21, y: -171, width: 300, height: 200, data: { allowOverlap: false } },
            { id: 'SYS_USER_ROLE', text: 'SYS_USER_ROLE', nodeShape: 1, opacity: 1, x: 405, y: -174 },
            { id: 'SYS_RESOURCE', text: 'SYS_RESOURCE', nodeShape: 1, opacity: 1, x: -246, y: -80, data: { dropAble: true } },
            { id: 'SYS_ROLE_RESOURCE', text: 'SYS_ROLE_RESOURCE', nodeShape: 1, opacity: 1, x: 338, y: -100 }
        ],
        lines: [
            { from: 'SYS_DEPT', to: 'SYS_USER', text: 'xxxxxxx' },
            { from: 'SYS_DEPT', to: 'SYS_USER' },
            { from: 'SYS_DEPT', to: 'SYS_USER' },
            { from: 'SYS_DEPT', to: 'SYS_USER' },
            { from: 'SYS_DEPT', to: 'SYS_USER' },
            { from: 'SYS_DEPT', to: 'SYS_USER' }
        ]
    };
    const graphInstance = graphRef.value?.getInstance();
    if (graphInstance) {
        await graphInstance.setJsonData(jsonData);
        await graphInstance.moveToCenter();
        await graphInstance.zoomToFit();
    }
};

const onNodeDragging = (nodeObject: RGNode, newX: number, newY: number, $event: RGUserEvent) => {
    console.log('onNodeDragging:', nodeObject);
    const graphInstance = graphRef.value?.getInstance();
    let rejectOverlaped = false;
    let limitedPosition;
    for (const n of graphInstance?.getNodes() || []) {
        if (nodeObject === n) continue;
        const shapeA = nodeObject.nodeShape || graphInstance.options.defaultNodeShape || 0;
        const shapeB = n.nodeShape || graphInstance.options.defaultNodeShape || 0;
        const nodeBox = {
            el: nodeObject.el,
            x: newX,
            y: newY,
        };
        const overlap = RGNodesAnalytic.shapesOverlap(nodeBox, n, shapeA, shapeB);
        if (overlap) {
            let newColor = n.data.dropAble ? 'rgba(212,252,189,0.82)' : 'rgba(252,189,189,0.82)';
            if (n.data.allowOverlap === false) {
                rejectOverlaped = true;
                limitedPosition = RGNodesAnalytic.getNoOverlapLimitedPosition(nodeObject, newX, newY, n);
                newColor = 'rgba(214,165,246,0.82)';
            }
            if (!n.data.orignColorSaved) {// Only the initial node color is saved

                n.data.orignColorSaved = true;
                n.data.orignColor = n.color;
            }
            n.color = newColor;
        } else {
            n.color = undefined;
        }
    }
    if (rejectOverlaped) {
        return limitedPosition;
    }
};

const onNodeDragStart = (nodeObject: RGNode, $event: RGUserEvent) => {
    console.log('onNodeDragStart:', nodeObject);
    nodeObject.data.startX = nodeObject.x;
    nodeObject.data.startY = nodeObject.y;
};

const onNodeDragEnd = (nodeObject: RGNode, $event: RGUserEvent) => {
    console.log('onNodeDragEnd:', nodeObject);
    const graphInstance = graphRef.value?.getInstance();
    for (const n of graphInstance?.getNodes() || []) {
        if (nodeObject === n) continue;
        if (n.data.allowOverlap === false) continue;
        if (!n.data.dropAble) continue;
        const shapeA = nodeObject.nodeShape || graphInstance.options.defaultNodeShape || 0;
        const shapeB = n.nodeShape || graphInstance.options.defaultNodeShape || 0;
        const overlap = RGNodesAnalytic.shapesOverlap(nodeObject, n, shapeA, shapeB);
        if (overlap) {
            updateNodeAsChildren(nodeObject, n);
            break;
        }
    }
    for (const n of graphInstance?.getNodes() || []) {
        n.color = n.data.orignColor; // Restore the original color of the node

    }
};

const updateNodeAsChildren = (node: RGNode, parentNode: RGNode) => {
    console.log('updateNodeAsChildren:', parentNode.id, '---->', node.id);
    node.x = node.data.startX;
    node.y = node.data.startY;
    const graphInstance = graphRef.value?.getInstance();
    if (graphInstance) {
        graphInstance.addLines([{
            from: parentNode.id,
            to: node.id,
            text: '新连线',
            color: 'rgba(214,165,246,0.82)',
            lineWidth: 3

        }]);
    }
};

const onNodeClick = (nodeObject: RGNode, $event: RGUserEvent) => {
    console.log('onNodeClick:', nodeObject);
};

const onLineClick = (lineObject: RGLine, linkObject: RGLink, $event: RGUserEvent) => {
    console.log('onLineClick:', lineObject);
};

const onMouseMove = (e: MouseEvent) => {
    if (e.target) {
        const tagName = (e.target as HTMLElement).tagName;
        if (tagName === 'path' || tagName === 'text') {
            const graphInstance = graphRef.value?.getInstance();
            const link: RGLink | null = graphInstance?.isLink(e.target as HTMLElement);
            if (link) {
                console.log('Show Detail:', link.fromNode.id, link.toNode.id, link.text);
            }
        } else {
            console.log('Hide Detail');
        }
    }
};
</script>

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

::v-deep(.relation-graph) {
    //customize node
    .rel-node {
        overflow: hidden;
    }
    //customize toolbar style
    .rel-toolbar {
        background-color: #f39930;
        color: #ffffff;
        .c-current-zoom {
            color: #ffffff;
        }
    }
    //customize background image size
    .rel-map {
        background-size: 50px 50px;
        //background-image: radial-gradient(circle,rgba(0,0,0,.1) 3px,transparent 0);
        background-image: linear-gradient(90deg, rgba(243, 153, 48, 0.3) 1px,transparent 0),linear-gradient(180deg,rgba(243, 153, 48, 0.3) 1px,transparent 0);
    }
}
</style>

React 版本

node-dragging.tsx

javascript
import React, { useRef, useEffect } from 'react';
import RelationGraph, {
  RGLink,
  RGNodesAnalyticUtils,
  RGNodeSlotProps
} from 'relation-graph-react';
import type { RGNode, RGLine, RGUserEvent, RGJsonData, RGOptions, RelationGraphComponent } from 'relation-graph-react';
import './node-dragging.scss';

const { RGNodesAnalytic } = RGNodesAnalyticUtils;

const NodeDragging = () => {
  const graphRef = useRef<RelationGraphComponent>();

  useEffect(() => {
    showGraph();
  }, []);

  const graphOptions: RGOptions = {
    debug: true,
    allowSwitchLineShape: true,
    allowSwitchJunctionPoint: true,
    allowShowDownloadButton: true,
    defaultJunctionPoint: 'border',
    placeOtherNodes: false,
    placeSingleNode: false,
    defaultNodeColor: '#ffffff',
    defaultNodeBorderWidth: 1,
    defaultNodeBorderColor: '#f39930',
    defaultNodeWidth: 170,
    defaultNodeHeight: 70,
    defaultLineColor: '#f39930',
    defaultLineWidth: 2,
    defaultLineTextOffset_y: -2,
    layout: {
      layoutName: 'fixed'
    }
  };

  const showGraph = async () => {
    const jsonData: RGJsonData = {
      rootId: 'SYS_ROLE',
      nodes: [
        { id: 'SYS_USER', text: 'SYS_USER', nodeShape: 1, opacity: 1, x: -32, y: -427, width: 120, height: 80, data: { dropAble: true } },
        { id: 'SYS_DEPT', text: 'SYS_DEPT', nodeShape: 0, opacity: 1, x: -244, y: -283, data: { dropAble: true } },
        { id: 'SYS_ROLE', text: 'SYS_ROLE', nodeShape: 1, opacity: 1, x: 0, y: 0, width: 300, height: 200, data: { allowOverlap: false } },
        { id: 'SYS_USER_ROLE', text: 'SYS_USER_ROLE', nodeShape: 1, opacity: 1, x: 405, y: -174 },
        { id: 'SYS_RESOURCE', text: 'SYS_RESOURCE', nodeShape: 1, opacity: 1, x: -246, y: -80, data: { dropAble: true } },
        { id: 'SYS_ROLE_RESOURCE', text: 'SYS_ROLE_RESOURCE', nodeShape: 1, opacity: 1, x: 338, y: -100 }
      ],
      lines: [
        { from: 'SYS_DEPT', to: 'SYS_USER', text: 'xxxxxxx' },
        { from: 'SYS_DEPT', to: 'SYS_USER' },
        { from: 'SYS_DEPT', to: 'SYS_USER' },
        { from: 'SYS_DEPT', to: 'SYS_USER' },
        { from: 'SYS_DEPT', to: 'SYS_USER' },
        { from: 'SYS_DEPT', to: 'SYS_USER' }
      ]
    };
    graphRef.current!.setJsonData(jsonData, () => {
      // set data ok.
    });
  };

  const onNodeDragging = (nodeObject: RGNode, newX: number, newY: number, _$event: RGUserEvent) => {
    console.log('onNodeDragging:', nodeObject);
    const graphInstance = graphRef.current!.getInstance();
    let rejectOverlaped = false;
    let limitedPosition;
    for (const n of graphInstance?.getNodes() || []) {
      if (nodeObject === n) continue;
      const shapeA = nodeObject.nodeShape || graphInstance.options.defaultNodeShape || 0;
      const shapeB = n.nodeShape || graphInstance.options.defaultNodeShape || 0;
      const nodeBox = {
        el: nodeObject.el,
        x: newX,
        y: newY
      };
      const overlap = RGNodesAnalytic.shapesOverlap(nodeBox, n, shapeA, shapeB);
      if (overlap) {
        let newColor = n.data!.dropAble ? 'rgba(212,252,189,0.82)' : 'rgba(252,189,189,0.82)';
        if (n.data!.allowOverlap === false) {
          rejectOverlaped = true;
          limitedPosition = RGNodesAnalytic.getNoOverlapLimitedPosition(nodeObject, newX, newY, n);
          newColor = 'rgba(214,165,246,0.82)';
        }
        if (!n.data!.orignColorSaved) {
          n.data!.orignColorSaved = true;
          n.data!.orignColor = n.color;
        }
        n.color = newColor;
      } else {
        n.color = undefined;
      }
    }
    if (rejectOverlaped) {
      return limitedPosition;
    }
  };

  const onNodeDragStart = (nodeObject: RGNode, _$event: RGUserEvent) => {
    console.log('onNodeDragStart:', nodeObject);
    nodeObject.data!.startX = nodeObject.x;
    nodeObject.data!.startY = nodeObject.y;
  };

  const onNodeDragEnd = (nodeObject: RGNode, _$event: RGUserEvent) => {
    console.log('onNodeDragEnd:', nodeObject);
    const graphInstance = graphRef.current!.getInstance();
    for (const n of graphInstance?.getNodes() || []) {
      if (nodeObject === n) continue;
      if (n.data!.allowOverlap === false) continue;
      if (!n.data!.dropAble) continue;
      const shapeA = nodeObject.nodeShape || graphInstance.options.defaultNodeShape || 0;
      const shapeB = n.nodeShape || graphInstance.options.defaultNodeShape || 0;
      const overlap = RGNodesAnalytic.shapesOverlap(nodeObject, n, shapeA, shapeB);
      if (overlap) {
        updateNodeAsChildren(nodeObject, n);
        break;
      }
    }
    for (const n of graphInstance?.getNodes() || []) {
      n.color = n.data!.orignColor;
    }
  };

  const updateNodeAsChildren = (node: RGNode, parentNode: RGNode) => {
    console.log('updateNodeAsChildren:', parentNode.id, '---->', node.id);
    node.x = node.data!.startX;
    node.y = node.data!.startY;
    const graphInstance = graphRef.current?.getInstance();
    if (graphInstance) {
      graphInstance.addLines([{
        from: parentNode.id,
        to: node.id,
        text: 'New Line',
        color: 'rgba(214,165,246,0.82)',
        lineWidth: 3

      }]);
    }
  };

  const onNodeClick = (nodeObject: RGNode, _$event: RGUserEvent) => {
    console.log('onNodeClick:', nodeObject);
  };

  const onLineClick = (lineObject: RGLine, _linkObject: RGLink, _$event: RGUserEvent) => {
    console.log('onLineClick:', lineObject);
  };

  const onMouseMove = (e: React.MouseEvent) => {
    if (e.target) {
      const tagName = (e.target as HTMLElement).tagName;
      if (tagName === 'path' || tagName === 'text') {
        const graphInstance = graphRef.current!.getInstance();
        const link: RGLink | undefined = graphInstance.isLink(e.target as HTMLElement);
        if (link) {
          console.log('Show Detail:', link.fromNode.id, link.toNode.id);
        }
      } else {
        console.log('Hide Detail');
      }
    }
  };
  const MyNodeSlot:React.FC<RGNodeSlotProps> = ({ node }) => {
    return (
      <div className="h-full">
        <div className="bg-yellow-500">{node.text}</div>
        {node.id === 'SYS_ROLE' && <div style={{ color: '#df7f03', padding: '10px' }}>Not allowed to enter</div>}
      </div>
    );
  };
  const MyGraphSlot:React.FC = () => {
    return (
      <>
        <div>
          <div style={{ zIndex: 300, position: 'absolute', left: '10px', top: 'calc(100% - 140px)', fontSize: '12px', backgroundColor: '#ffffff', border: '#efefef solid 1px', borderRadius: '10px', width: '250px', height: '130px' }}>
            <div style={{ paddingLeft: '10px' }}>
              <div style={{ lineHeight: '30px' }}>The color of the moving over node:</div>
              <div style={{ marginTop: '10px' }}>
                Allowed drop

                <div style={{ height: '5px', width: '80px', backgroundColor: 'rgba(212,252,189,0.82)' }}></div>
              </div>
              <div style={{ marginTop: '10px' }}>
                Not allowed drop

                <div style={{ height: '5px', width: '80px', backgroundColor: 'rgba(252,189,189,0.82)' }}></div>
              </div>
              <div style={{ marginTop: '10px' }}>
                Not allowed to enter

                <div style={{ height: '5px', width: '80px', backgroundColor: 'rgba(214,165,246,0.82)' }}></div>
              </div>
            </div>
          </div>
        </div>
      </>
    );
  };
  return (
    <div>
      <div style={{ height: '100vh' }} onMouseMove={onMouseMove}>
        <RelationGraph
          ref={graphRef}
          options={graphOptions}
          onNodeDragging={onNodeDragging}
          onNodeDragStart={onNodeDragStart}
          onNodeDragEnd={onNodeDragEnd}
          onNodeClick={onNodeClick}
          onLineClick={onLineClick}
          graphPlugSlot={MyGraphSlot}
          nodeSlot={MyNodeSlot}
        >
        </RelationGraph>
      </div>
    </div>
  );
};

export default NodeDragging;

node-dragging.scss

scss
//customize node
.relation-graph {
  .rel-node {
    overflow: hidden;
  }
  .rel-toolbar {
    background-color: #f39930;
    color: #ffffff;
    .c-current-zoom {
      color: #ffffff;
    }
  }
  .rel-map {
    background-size: 50px 50px;
    //background-image: radial-gradient(circle,rgba(0,0,0,.1) 3px,transparent 0);
    background-image: linear-gradient(90deg, rgba(0, 0, 0, 0.1) 2px, transparent 0), linear-gradient(180deg, rgba(
            0,
            0,
            0,
            0.1
          ) 2px, transparent 0);
  }
}

本站搭建特别鸣谢【茂神大佬】