广告

买白酒,找南将

Skip to content

展示更多(分页)

Vue2 版本

show-more-nodes-front.vue

javascript
<template>
  <div>
    <div
        v-loading="g_loading"
        element-loading-text="正在加载数据..."
        element-loading-spinner="el-icon-loading"
        element-loading-background="rgba(255, 255, 255, 0.6)"
        style="height:calc(100vh);">
      <RelationGraph
        ref="graphRef"
        :options="graphOptions"
        :on-node-click="onNodeClick"
        :on-line-click="onLineClick"
      >
        <template #node="{node}">
          <div v-if="node.data.type === 'btn-more-from' || node.data.type === 'btn-more-to'" class="my-node more-btn" @click="showMoreNodes(node)">
            {{node.text}}
          </div>
          <div v-else-if="node.data.type === 'root'" class="my-node my-root">
            {{node.text}}
          </div>
          <div v-else-if="node.data.type === 'level1'" class="my-node" style="width: 100px;padding-left:0px;text-align: center;background-color: #2c3e50;color: #ffffff;">
            {{node.text}}
          </div>
          <div v-else class="my-node">
            <span class="c-circle-flag"></span>{{node.text}}
          </div>
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

<script lang="ts">
// 如果您没有在main.js文件中使用Vue.use(RelationGraph); 就需要使用下面这一行代码来引入relation-graph
// import RelationGraph from 'relation-graph';
import {RGNode} from "relation-graph";

const graphOptions = {
  debug: false,
  lineUseTextPath: true,
  useAnimationWhenExpanded: true,
  layout:
    {
      layoutName: 'tree',
      from: 'left',
      min_per_width: 400,
      max_per_width: 400,
      min_per_height: 40,
      max_per_height: 50,
      enableGatherNodes: false,
      levelDistance: '200,300,200,300,200,300,200,300,200,300,200,300,200,300,200,300'
    },
  defaultNodeShape:  1,
  defaultLineShape:  4,
  defaultJunctionPoint: 'lr',
  defaultNodeBorderWidth: 0,
  defaultLineColor: '#dddddd',
  defaultNodeColor: 'transparent'
};
export default {
  name: 'InvestmentPenetration',
  components: { },
  data() {
    return {
      currentCase: '横向树状图谱',
      isShowCodePanel: false,
      g_loading: false,
      graphOptions
    };
  },
  mounted() {
    this.showGraph();
  },
  methods: {
    async showGraph() {
      const __graph_json_data =
          { 'rootId': 'root', 'nodes': [
            { 'id': 'root', 'text': '北京天天天科技有限公司', offset_x: 0, data: {type:'root'} },
            { 'id': 'gudong', 'text': '股东', offset_x: -80, data: {type:'level1'} },
            { 'id': 'gaoguan', 'text': '高管', offset_x: -80, data: {type:'level1'} },
            { 'id': 'duiwaitouzi', 'text': '对外投资', offset_x: 80, data: {type:'level1'} },
            { 'id': 'fenzhijigou', 'text': '分支机构', offset_x: 80, data: {type:'level1'} },
          ], lines: [
              {from: 'gudong', to: 'root', lineShape: 3 },
              {from: 'gaoguan', to: 'root', lineShape: 3 },
              {from: 'root', to: 'duiwaitouzi', lineShape: 3 },
              {from: 'root', to: 'fenzhijigou', lineShape: 3 },
            ] };
      const graphInstance = this.$refs.graphRef.getInstance();
      this.generateChildNodes('gudong', 'left', __graph_json_data);
      this.generateChildNodes('gaoguan', 'left', __graph_json_data);
      this.generateChildNodes('fenzhijigou', 'right', __graph_json_data);
      this.generateChildNodes('duiwaitouzi', 'right', __graph_json_data);
      // 以上四行模拟获取股东、高管、分支机构、对外投资的过程可以改成异步从后台获取,在数据全部获取到并放入__graoh_json_data后再执行以下代码:
      await graphInstance.setJsonData(__graph_json_data, true);
      await graphInstance.playShowEffect();
      await this.hideMoreNodes();
    },
    onNodeClick(nodeObject, $event) {
      console.log('onNodeClick:', nodeObject);
    },
    async hideMoreNodes() {
      const graphInstance = this.$refs.graphRef.getInstance();
      //获取所有的节点
      const allNodeData = graphInstance.getNodes();
      let newLines = [];
      let newNodes = [];
      const pageSize = 3;
      allNodeData.forEach(node => {
        if (!node.lot || !node.lot.childs) {
          return;
        }
        // console.log('xxxxx', node.id, node.lot.childs.length);
        //targetTo的节点
        let childNodes = node.targetTo;
        if(childNodes.length>pageSize) {
          const sliceToNode = childNodes.slice(2);
          sliceToNode.forEach(item => {
            item.isHide=true;
          });
          newNodes.push({ id: `${node.id}-to-more`, text: `展开更多T(${childNodes.length - pageSize})`, alignItems: 'left', data: {type: 'btn-more-to' }});
          newLines.push({ from: node.id, to: `${node.id}-to-more`});
        }
        childNodes = node.targetFrom;
        if(childNodes.length>2) {
          const sliceToNode = childNodes.slice(2);
          sliceToNode.forEach(item => {
            item.isHide=true;
          });
          newNodes.push({ id: `${node.id}-from-more`, text: `展开更多F(${childNodes.length - pageSize})`, alignItems: 'right', data: {type: 'btn-more-from' }});
          newLines.push({ from: `${node.id}-from-more`, to: node.id });
        }
      })
      graphInstance.addNodes(newNodes);
      graphInstance.addLines(newLines);
      await graphInstance.doLayout();
      await graphInstance.setZoom(80);
    },
    showMoreNodes(nodeObject) {
      console.log('showMoreNodes');
      const graphInstance = this.$refs.graphRef.getInstance();
      if (nodeObject.data.type === 'btn-more-to') {// 如果点击的树右侧(当前布局from=left时)的节点
        const allSublingNodes = nodeObject.lot.parent.targetTo;
        console.log('showMoreNodes:allSublingNodes', allSublingNodes.length);
        // 【代码片段1】和【代码片段2】一模一样,为了方便理解,这里重复写
        /************分页逻辑【代码片段1】-开始****************************************************************/
        const pageSize = 3;
        let displayedCount = 0; // 本次新显示的节点数
        let newDisplayedCount = 0; // 本次新显示的节点数
        for (const node of allSublingNodes) {
          if (node.isHide === true) {
            node.isHide = false; // 显示节点
            newDisplayedCount++; // 计数,统计当前新显示了几个节点了,假设每页显示3个,那么新显示的节点到3个时就要停止显示新的节点
          }
          if (node.isHide !== true) {
            displayedCount++;
          }
          if (newDisplayedCount >= pageSize) {
            // 新的一个显示够了,停止
            break;
          }
        }
        // 显示的节点数量已经大于等于总节点数时
        if (displayedCount >= allSublingNodes.length) {
          // 如果新显示的节点为0,则表示所有节点都显示了,不能再继续翻页了
          // 此时,需要移除【显示更多】按钮
          graphInstance.removeNodeById(nodeObject.id);
        }
        const moreButton = graphInstance.getNodeById(nodeObject.id);
        moreButton.text = `展示更多(${allSublingNodes.length - displayedCount - 1})`
        /************分页逻辑【代码片段1】- 结束****************************************************************/
        graphInstance.doLayout();
      }
      if (nodeObject.data.type === 'btn-more-from') {// 如果点击的树左侧(当前布局from=left时)的节点,左右处理的主要差别是:一个取targetTo,另一个取targetFrom
        const allSublingNodes = nodeObject.lot.parent.targetFrom;
        console.log('showMoreNodes:allSublingNodes', allSublingNodes.length);
        /************分页逻辑【代码片段2】-开始****************************************************************/
        const pageSize = 3;
        let displayedCount = 0; // 本次新显示的节点数
        let newDisplayedCount = 0; // 本次新显示的节点数
        for (const node of allSublingNodes) {
          if (node.isHide === true) {
            node.isHide = false; // 显示节点
            newDisplayedCount++; // 计数,统计当前新显示了几个节点了,假设每页显示3个,那么新显示的节点到3个时就要停止显示新的节点
          }
          if (node.isHide !== true) {
            displayedCount++;
          }
          if (newDisplayedCount >= pageSize) {
            // 新的一个显示够了,停止
            break;
          }
        }
        const moreButton = graphInstance.getNodeById(nodeObject.id);
        moreButton.text = `展示更多(${allSublingNodes.length - displayedCount - 1})`
        // 显示的节点数量已经大于等于总节点数时
        if (displayedCount >= allSublingNodes.length) {
          // 如果新显示的节点为0,则表示所有节点都显示了,不能再继续翻页了
          // 此时,需要移除【显示更多】按钮
          graphInstance.removeNodeById(nodeObject.id);
        }
        /************分页逻辑【代码片段2】- 结束****************************************************************/
        graphInstance.doLayout();
      }
    },
    onLineClick(lineObject, linkObject, $event) {
      console.log('onLineClick:', lineObject);
    },
    generateChildNodes(parentNodeId:string, newNodesPosition:string, graphJsonData){
      // 以下模拟了从数据库中查询出来的两个投资对象,以及expandedNodeId与他们的投资关系
      const randomChildSize = 1 + Math.floor(Math.random() * 30);
      for (let i = 0 ; i< randomChildSize;i++) {
        const thisNodeIndex = newNodeIndex++;
        const newNodeId = 'N' + thisNodeIndex; // 节点id必须是字符串
        const randomEntName = `北京${randomEntNamePrefix[Math.floor(Math.random() * randomEntNamePrefix.length)]}第${newNodeId}公司`;
        const alignItems = newNodesPosition === 'left' ? 'right' : 'left'; // 新增的节点如果要放在左侧,则这些新增节点显示为右对齐
        const newChild = { id: newNodeId, text: randomEntName, expanded: false, alignItems }
        let newLine;
        if (newNodesPosition === 'left') {
          newLine = { from: newNodeId, to: parentNodeId }
        } else {
          newLine = { from: parentNodeId, to: newNodeId }
        }
        graphJsonData.nodes.push(newChild);
        graphJsonData.lines.push(newLine);
      }
    }
  }
};
let newNodeIndex = 1;
const randomEntNamePrefix = ['', '天星', '星月夜', '风云际会', '金凯宾斯基'];
</script>

<style lang="scss">

</style>

<style lang="scss" scoped>
::v-deep .rel-node-checked{
  //box-shadow:none;
  //animation: none;
}
::v-deep .c-collapsed{
  background-color: #888888 !important;
}
::v-deep .c-expanded{
  background-color: #888888 !important;
}
.my-node{
  height:32px;
  text-align: right;
  border: #eeeeee solid 1px;
  border-radius:5px;
  background-color: #ffffff;
  padding-left:5px;
  padding-right:5px;
  line-height: 30px;
  color: #333333;
  &:hover{
    border: rgba(15, 71, 255, 0.5) solid 1px;
  }
  white-space: nowrap;
  text-overflow: ellipsis;
}
.ex-btn {
  border: none;
  &:hover{
    border: none;
  }
}
.more-btn {
  background-color: #efefef;
  cursor: pointer;
  text-align: center;
  padding-left: 0px;
  width: 100px;
  font-size: 12px;
  &:hover{
    color: rgba(15, 71, 255, 0.5);
  }
}
.my-root {
  border: rgba(15, 71, 255, 0.5) solid 1px;
  text-align: center;
  padding-left: 0px;
  width: 180px;
  font-size: 12px;
  &:hover{
    color: rgba(15, 71, 255, 0.5);
  }
}
.c-circle-flag{
  display: inline-block;
  width:12px;
  height:12px;
  border-radius: 50%;
  background-color: #00bb00;
  margin-right:5px;
}
</style>

Vue3 版本

show-more-nodes-front.vue

javascript
<template>
  <div>
    <div

      v-loading="g_loading"
      element-loading-text="Loading..."
      element-loading-spinner="el-icon-loading"
      element-loading-background="rgba(255, 255, 255, 0.6)"
      style="height:calc(100vh);">
      <RelationGraph

        ref="graphRef"
        :options="graphOptions"
        :on-node-click="onNodeClick"
        :on-line-click="onLineClick"
      >
        <template #node="{node}">
          <div v-if="node.data.type === 'btn-more-from' || node.data.type === 'btn-more-to'" class="my-node more-btn" @click="showMoreNodes(node)">
            {{ node.text }}
          </div>
          <div v-else-if="node.data.type === 'root'" class="my-node my-root">
            {{ node.text }}
          </div>
          <div v-else-if="node.data.type === 'level1'" class="my-node" style="width: 100px;padding-left:0px;text-align: center;background-color: #2c3e50;color: #ffffff;">
            {{ node.text }}
          </div>
          <div v-else class="my-node">
            <span class="c-circle-flag" />{{ node.text }}
          </div>
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

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

const graphOptions: RGOptions = {
    debug: false,
    lineUseTextPath: true,
    useAnimationWhenExpanded: true,
    'layouts': [
        {
            'layoutName': 'tree',
            'from': 'left',
            'min_per_width': '400',
            'max_per_width': '400',
            'min_per_height': '40',
            'max_per_height': '50',
            enableGatherNodes: false,
            levelDistance: '200,300,200,300,200,300,200,300,200,300,200,300,200,300,200,300'
        }
    ],
    'defaultNodeShape': 1,
    'defaultLineShape': 4,
    'defaultJunctionPoint': 'lr',
    'defaultNodeBorderWidth': 0,
    'defaultLineColor': '#dddddd',
    'defaultNodeColor': 'transparent'
};

const graphRef = ref<RelationGraphComponent | null>(null);
const g_loading = ref(false);

const showGraph = async () => {
    const __graph_json_data: RGJsonData = {
        'rootId': 'root',
        'nodes': [
            { 'id': 'root', 'text': 'Tian Technology Co., Ltd.', offset_x: 0, data: {type:'root'} },
            { 'id': 'gudong', 'text': 'Shareholders', offset_x: -80, data: {type:'level1'} },
            { 'id': 'gaoguan', 'text': 'Executives', offset_x: -80, data: {type:'level1'} },
            { 'id': 'duiwaitouzi', 'text': 'External Investment', offset_x: 80, data: {type:'level1'} },
            { 'id': 'fenzhijigou', 'text': 'Branches', offset_x: 80, data: {type:'level1'} },
        ],
        lines: [
            {from: 'gudong', to: 'root', lineShape: 3 },
            {from: 'gaoguan', to: 'root', lineShape: 3 },
            {from: 'root', to: 'duiwaitouzi', lineShape: 3 },
            {from: 'root', to: 'fenzhijigou', lineShape: 3 },
        ]
    };

    const graphInstance = graphRef.value!.getInstance();
    generateChildNodes('gudong', 'left', __graph_json_data);
    generateChildNodes('gaoguan', 'left', __graph_json_data);
    generateChildNodes('fenzhijigou', 'right', __graph_json_data);
    generateChildNodes('duiwaitouzi', 'right', __graph_json_data);

    await graphInstance.setJsonData(__graph_json_data, true);
    await graphInstance.playShowEffect();
    await hideMoreNodes();
};

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

const hideMoreNodes = async () => {
    const graphInstance = graphRef.value!.getInstance();
    const allNodeData = graphInstance.getNodes();
    let newLines: RGLine[] = [];
    let newNodes: RGNode[] = [];
    const pageSize = 3;

    allNodeData.forEach(node => {
        if (!node.lot || !node.lot.childs) {
            return;
        }

        let childNodes = node.targetTo;
        if(childNodes.length>pageSize) {
            const sliceToNode = childNodes.slice(2);
            sliceToNode.forEach(item => {
                item.isHide=true;
            });
            newNodes.push({ id: `${node.id}-to-more`, text: `Show More T(${childNodes.length - pageSize})`, alignItems: 'left', data: {type: 'btn-more-to' }});
            newLines.push({ from: node.id, to: `${node.id}-to-more`});
        }

        childNodes = node.targetFrom;
        if(childNodes.length>2) {
            const sliceToNode = childNodes.slice(2);
            sliceToNode.forEach(item => {
                item.isHide=true;
            });
            newNodes.push({ id: `${node.id}-from-more`, text: `Show More F(${childNodes.length - pageSize})`, alignItems: 'right', data: {type: 'btn-more-from' }});
            newLines.push({ from: `${node.id}-from-more`, to: node.id });
        }
    });

    graphInstance.addNodes(newNodes);
    graphInstance.addLines(newLines);
    await graphInstance.doLayout();
    await graphInstance.setZoom(80);
};

const showMoreNodes = (nodeObject: RGNode) => {
    const graphInstance = graphRef.value!.getInstance();

    if (nodeObject.data.type === 'btn-more-to') {
        const allSublingNodes = nodeObject.lot.parent.targetTo;
        const pageSize = 3;
        let displayedCount = 0;
        let newDisplayedCount = 0;

        for (const node of allSublingNodes) {
            if (node.isHide === true) {
                node.isHide = false;
                newDisplayedCount++;
            }
            if (node.isHide !== true) {
                displayedCount++;
            }
            if (newDisplayedCount >= pageSize) {
                break;
            }
        }

        if (displayedCount >= allSublingNodes.length) {
            graphInstance.removeNodeById(nodeObject.id);
        }

        const moreButton = graphInstance.getNodeById(nodeObject.id);
        moreButton.text = `Show More (${allSublingNodes.length - displayedCount - 1})`;

        graphInstance.doLayout();
    }

    if (nodeObject.data.type === 'btn-more-from') {
        const allSublingNodes = nodeObject.lot.parent.targetFrom;
        const pageSize = 3;
        let displayedCount = 0;
        let newDisplayedCount = 0;

        for (const node of allSublingNodes) {
            if (node.isHide === true) {
                node.isHide = false;
                newDisplayedCount++;
            }
            if (node.isHide !== true) {
                displayedCount++;
            }
            if (newDisplayedCount >= pageSize) {
                break;
            }
        }

        const moreButton = graphInstance.getNodeById(nodeObject.id);
        moreButton.text = `Show More (${allSublingNodes.length - displayedCount - 1})`;

        if (displayedCount >= allSublingNodes.length) {
            graphInstance.removeNodeById(nodeObject.id);
        }

        graphInstance.doLayout();
    }
};

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

const generateChildNodes = (parentNodeId: string, newNodesPosition: string, graphJsonData: RGJsonData) => {
    const randomChildSize = 1 + Math.floor(Math.random() * 30);

    for (let i = 0 ; i< randomChildSize;i++) {
        const thisNodeIndex = newNodeIndex++;
        const newNodeId = 'N' + thisNodeIndex;
        const randomEntName = `Beijing ${randomEntNamePrefix[Math.floor(Math.random() * randomEntNamePrefix.length)]} No.${newNodeId} Company`;
        const alignItems = newNodesPosition === 'left' ? 'right' : 'left';
        const newChild: RGNode = { id: newNodeId, text: randomEntName, expanded: false, alignItems };
        let newLine: RGLine;

        if (newNodesPosition === 'left') {
            newLine = { from: newNodeId, to: parentNodeId };
        } else {
            newLine = { from: parentNodeId, to: newNodeId };
        }

        graphJsonData.nodes.push(newChild);
        graphJsonData.lines.push(newLine);
    }
};

onMounted(() => {
    showGraph();
});
let newNodeIndex = 1;
const randomEntNamePrefix = ['', 'Tianxing', 'Starry Night', 'Gathering Storm', 'Jin Kaibinski'];
</script>

<style lang="scss">

</style>

<style lang="scss" scoped>
::v-deep(.relation-graph) {
    .rel-node-checked{
        //box-shadow:none;
        //animation: none;
    }
    .c-collapsed{
        background-color: #888888 !important;
    }
    .c-expanded{
        background-color: #888888 !important;
    }
}
.my-node{
  height:32px;
  text-align: right;
  border: #eeeeee solid 1px;
  border-radius:5px;
  background-color: #ffffff;
  padding-left:5px;
  padding-right:5px;
  line-height: 30px;
  color: #333333;
  &:hover{
    border: rgba(15, 71, 255, 0.5) solid 1px;
  }
  white-space: nowrap;
  text-overflow: ellipsis;
}
.ex-btn {
  border: none;
  &:hover{
    border: none;
  }
}
.more-btn {
  background-color: #efefef;
  cursor: pointer;
  text-align: center;
  padding-left: 0px;
  width: 100px;
  font-size: 12px;
  &:hover{
    color: rgba(15, 71, 255, 0.5);
  }
}
.my-root {
  border: rgba(15, 71, 255, 0.5) solid 1px;
  text-align: center;
  padding-left: 0px;
  width: 180px;
  font-size: 12px;
  &:hover{
    color: rgba(15, 71, 255, 0.5);
  }
}
.c-circle-flag{
  display: inline-block;
  width:12px;
  height:12px;
  border-radius: 50%;
  background-color: #00bb00;
  margin-right:5px;
}
</style>

React 版本

show-more-nodes-front.tsx

javascript
import React, { useRef, useEffect } from 'react';
import RelationGraph, {RGNodeSlotProps} from 'relation-graph-react';
import { RGNode, RGLine, RGLink, RGUserEvent, RGJsonData, RGOptions, RelationGraphComponent } from 'relation-graph-react';
import './show-more-nodes-front.scss';

const ShowMoreNodesFront = () => {
  const graphRef = useRef<RelationGraphComponent | null>(null);

  const graphOptions: RGOptions = {
    debug: false,
    lineUseTextPath: true,
    useAnimationWhenExpanded: true,
    layout: {
      layoutName: 'tree',
      from: 'left',
      min_per_width: '400',
      max_per_width: '400',
      min_per_height: '40',
      max_per_height: '50',
      enableGatherNodes: false,
      levelDistance: '200,300,200,300,200,300,200,300,200,300,200,300,200,300,200,300'
    },
    defaultNodeShape: 1,
    defaultLineShape: 4,
    defaultJunctionPoint: 'lr',
    defaultNodeBorderWidth: 0,
    defaultLineColor: '#dddddd',
    defaultNodeColor: 'transparent'
  };

  const showGraph = async () => {
    const __graph_json_data: RGJsonData = {
      rootId: 'root',
      nodes: [
        { id: 'root', text: 'Tian Technology Co., Ltd.', offset_x: 0, data: { type: 'root' } },
        { id: 'gudong', text: 'Shareholders', offset_x: -80, data: { type: 'level1' } },
        { id: 'gaoguan', text: 'Executives', offset_x: -80, data: { type: 'level1' } },
        { id: 'duiwaitouzi', text: 'External Investment', offset_x: 80, data: { type: 'level1' } },
        { id: 'fenzhijigou', text: 'Branches', offset_x: 80, data: { type: 'level1' } },
      ],
      lines: [
        { from: 'gudong', to: 'root', lineShape: 3 },
        { from: 'gaoguan', to: 'root', lineShape: 3 },
        { from: 'root', to: 'duiwaitouzi', lineShape: 3 },
        { from: 'root', to: 'fenzhijigou', lineShape: 3 },
      ]
    };

    const graphInstance = graphRef.current!.getInstance();
    generateChildNodes('gudong', 'left', __graph_json_data);
    generateChildNodes('gaoguan', 'left', __graph_json_data);
    generateChildNodes('fenzhijigou', 'right', __graph_json_data);
    generateChildNodes('duiwaitouzi', 'right', __graph_json_data);

    await graphInstance.setJsonData(__graph_json_data, true);
    await graphInstance.playShowEffect();
    await hideMoreNodes();
  };

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

  const hideMoreNodes = async () => {
    const graphInstance = graphRef.current!.getInstance();
    const allNodeData = graphInstance.getNodes();
    let newLines: RGLine[] = [];
    let newNodes: RGNode[] = [];
    const pageSize = 3;

    allNodeData.forEach(node => {
      if (!node.lot || !node.lot.childs) {
        return;
      }

      let childNodes = node.targetTo;
      if (childNodes.length > pageSize) {
        const sliceToNode = childNodes.slice(2);
        sliceToNode.forEach(item => {
          item.isHide = true;
        });
        newNodes.push({ id: `${node.id}-to-more`, text: `Show More T(${childNodes.length - pageSize})`, alignItems: 'left', data: { type: 'btn-more-to' } });
        newLines.push({ from: node.id, to: `${node.id}-to-more` });
      }

      childNodes = node.targetFrom;
      if (childNodes.length > 2) {
        const sliceToNode = childNodes.slice(2);
        sliceToNode.forEach(item => {
          item.isHide = true;
        });
        newNodes.push({ id: `${node.id}-from-more`, text: `Show More F(${childNodes.length - pageSize})`, alignItems: 'right', data: { type: 'btn-more-from' } });
        newLines.push({ from: `${node.id}-from-more`, to: node.id });
      }
    });

    graphInstance.addNodes(newNodes);
    graphInstance.addLines(newLines);
    await graphInstance.doLayout();
    await graphInstance.setZoom(80);
  };

  const showMoreNodes = (nodeObject: RGNode) => {
    const graphInstance = graphRef.current!.getInstance();

    if (nodeObject.data.type === 'btn-more-to') {
      const allSublingNodes = nodeObject.lot.parent.targetTo;
      const pageSize = 3;
      let displayedCount = 0;
      let newDisplayedCount = 0;

      for (const node of allSublingNodes) {
        if (node.isHide === true) {
          node.isHide = false;
          newDisplayedCount++;
        }
        if (node.isHide !== true) {
          displayedCount++;
        }
        if (newDisplayedCount >= pageSize) {
          break;
        }
      }

      if (displayedCount >= allSublingNodes.length) {
        graphInstance.removeNodeById(nodeObject.id);
      }

      const moreButton = graphInstance.getNodeById(nodeObject.id);
      moreButton.text = `Show More (${allSublingNodes.length - displayedCount - 1})`;

      graphInstance.doLayout();
    }

    if (nodeObject.data.type === 'btn-more-from') {
      const allSublingNodes = nodeObject.lot.parent.targetFrom;
      const pageSize = 3;
      let displayedCount = 0;
      let newDisplayedCount = 0;

      for (const node of allSublingNodes) {
        if (node.isHide === true) {
          node.isHide = false;
          newDisplayedCount++;
        }
        if (node.isHide !== true) {
          displayedCount++;
        }
        if (newDisplayedCount >= pageSize) {
          break;
        }
      }

      const moreButton = graphInstance.getNodeById(nodeObject.id);
      moreButton.text = `Show More (${allSublingNodes.length - displayedCount - 1})`;

      if (displayedCount >= allSublingNodes.length) {
        graphInstance.removeNodeById(nodeObject.id);
      }

      graphInstance.doLayout();
    }
  };

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

  const generateChildNodes = (parentNodeId: string, newNodesPosition: string, graphJsonData: RGJsonData) => {
    const randomChildSize = 1 + Math.floor(Math.random() * 30);

    for (let i = 0; i < randomChildSize; i++) {
      const thisNodeIndex = newNodeIndex++;
      const newNodeId = 'N' + thisNodeIndex;
      const randomEntName = `Beijing ${randomEntNamePrefix[Math.floor(Math.random() * randomEntNamePrefix.length)]} No.${newNodeId} Company`;
      const alignItems = newNodesPosition === 'left' ? 'right' : 'left';
      const newChild: RGNode = { id: newNodeId, text: randomEntName, expanded: false, alignItems };
      let newLine: RGLine;

      if (newNodesPosition === 'left') {
        newLine = { from: newNodeId, to: parentNodeId };
      } else {
        newLine = { from: parentNodeId, to: newNodeId };
      }

      graphJsonData.nodes.push(newChild);
      graphJsonData.lines.push(newLine);
    }
  };

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

  let newNodeIndex = 1;
  const randomEntNamePrefix = ['', 'Tianxing', 'Starry Night', 'Gathering Storm', 'Jin Kaibinski'];
  const MyNodeSlot: React.FC<RGNodeSlotProps> = ({node}) => {
    return (
      <div>
        {node.data?.type === 'btn-more-from' || node.data?.type === 'btn-more-to' ? (
          <div className="my-node more-btn" onClick={() => showMoreNodes(node)}>
            {node.text}
          </div>
        ) : node.data?.type === 'root' ? (
          <div className="my-node my-root">
            {node.text}
          </div>
        ) : node.data?.type === 'level1' ? (
          <div className="my-node" style={{ width: '100px', paddingLeft: '0px', textAlign: 'center', backgroundColor: '#2c3e50', color: '#ffffff' }}>
            {node.text}
          </div>
        ) : (
          <div className="my-node">
            <span className="c-circle-flag" />{node.text}
          </div>
        )}
      </div>
    );
  };
  return (
    <div>
      <div style={{ height: '100vh' }}>
        <RelationGraph
          ref={graphRef}
          options={graphOptions}
          onNodeClick={onNodeClick}
          onLineClick={onLineClick}
          nodeSlot={MyNodeSlot}
        >
        </RelationGraph>
      </div>
    </div>
  );
};

export default ShowMoreNodesFront;

show-more-nodes-front.scss

scss
.relation-graph {
  .rel-node-checked {
    //box-shadow:none;
    //animation: none;
  }
  .c-collapsed {
    background-color: #888888 !important;
  }
  .c-expanded {
    background-color: #888888 !important;
  }
}
.my-node {
  height: 32px;
  text-align: right;
  border: #eeeeee solid 1px;
  border-radius: 5px;
  background-color: #ffffff;
  padding-left: 5px;
  padding-right: 5px;
  line-height: 30px;
  color: #333333;
  &:hover {
    border: rgba(15, 71, 255, 0.5) solid 1px;
  }
  white-space: nowrap;
  text-overflow: ellipsis;
}
.ex-btn {
  border: none;
  &:hover {
    border: none;
  }
}
.more-btn {
  background-color: #efefef;
  cursor: pointer;
  text-align: center;
  padding-left: 0px;
  width: 100px;
  font-size: 12px;
  &:hover {
    color: rgba(15, 71, 255, 0.5);
  }
}
.my-root {
  border: rgba(15, 71, 255, 0.5) solid 1px;
  text-align: center;
  padding-left: 0px;
  width: 180px;
  font-size: 12px;
  &:hover {
    color: rgba(15, 71, 255, 0.5);
  }
}
.c-circle-flag {
  display: inline-block;
  width: 12px;
  height: 12px;
  border-radius: 50%;
  background-color: #00bb00;
  margin-right: 5px;
}

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