广告

买白酒,找南将

Skip to content

tree-变化多端的树

Vue2 版本

ever-changing-tree.vue

javascript
<template>
  <div>

    <div ref="myPage" class="my-graph" style="height:calc(100vh);">
      <RelationGraph
        ref="graphRef"
        :options="graphOptions">
        <template #graph-plug>
          <div class="c-my-panel">
            <div class="c-option-name">布局方向:</div>
            <el-radio-group size="mini" v-model="layoutFrom" @change="updateLayouterOptions">
              <el-radio-button label="left">left</el-radio-button>
              <el-radio-button label="right">right</el-radio-button>
              <el-radio-button label="top">top</el-radio-button>
              <el-radio-button label="bottom">bottom</el-radio-button>
            </el-radio-group>
            <div class="c-option-name">布局层级内扩展方向:</div>
            <el-radio-group size="mini" v-model="layoutExpansionDirection" @change="updateLayouterOptions">
              <el-radio-button label="center">默认</el-radio-button>
              <el-radio-button label="left">向左</el-radio-button>
              <el-radio-button label="right">向右</el-radio-button>
              <el-radio-button label="top">向上</el-radio-button>
              <el-radio-button label="bottom">向下</el-radio-button>
            </el-radio-group>
            <div class="c-option-name">连线形状:</div>
            <el-radio-group v-model="lineShape" size="mini" @change="tabChange">
              <el-radio-button :label="1">直线</el-radio-button>
              <el-radio-button :label="2" >2</el-radio-button>
              <el-radio-button :label="3" >3</el-radio-button>
              <el-radio-button :label="5" >5</el-radio-button>
              <el-radio-button :label="6" >6</el-radio-button>
              <el-radio-button :label="7" >7</el-radio-button>
              <el-radio-button :label="4" >折线</el-radio-button>
            </el-radio-group>
            <div v-if="lineShape === 4">
              <div class="c-option-name">折线弯角大小:</div>
              <el-radio-group v-model="polyLineRadius" size="mini" @change="tabChange">
                <el-radio-button :label="0">0</el-radio-button>
                <el-radio-button :label="3">3</el-radio-button>
                <el-radio-button :label="5" >5</el-radio-button>
                <el-radio-button :label="8" >8</el-radio-button>
                <el-radio-button :label="10" >10</el-radio-button>
                <el-radio-button :label="15" >15</el-radio-button>
              </el-radio-group>
            </div>
            <div class="c-option-name">连接点:</div>
            <el-radio-group v-model="linePoint" size="mini" @change="tabChange">
              <el-radio-button label="border">边缘</el-radio-button>
              <el-radio-button label="tb">上下</el-radio-button>
              <el-radio-button label="lr">左右</el-radio-button>
              <el-radio-button label="ltrb">ltrb</el-radio-button>
              <el-radio-button label="left">L</el-radio-button>
              <el-radio-button label="top">T</el-radio-button>
              <el-radio-button label="right">R</el-radio-button>
              <el-radio-button label="bottom">B</el-radio-button>
            </el-radio-group>
            <div class="c-option-name">层级距离 & 节点间距:</div>
            <div>
              <div>
                <div>
                  <div class="c-option-name">节点间【横向】距离:</div>
                  <el-slider v-model="range_horizontal" :min="40" :max="500" @change="updateLayouterOptions">
                  </el-slider>
                </div>
                <div>
                  <div class="c-option-name">节点间【纵向】距离:</div>
                  <el-slider v-model="range_vertical" :min="40" :max="500" @change="updateLayouterOptions">
                  </el-slider>
                </div>
              </div>
<!--              <div>-->
<!--                <div style="font-size:14px;">-->
<!--                  <div class="c-option-name">依次设置每个层级的距离:</div>-->
<!--                  <el-input v-model="levelDistance" placeholder="请输入内容" @change="tabChange"></el-input>-->
<!--                </div>-->
<!--              </div>-->
            </div>
          </div>
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

<script>
// 如果您没有在main.js文件中使用Vue.use(RelationGraph); 就需要使用下面这一行代码来引入relation-graph
// import RelationGraph from 'relation-graph';
const graphOptions = {
  allowSwitchLineShape: true,
  allowSwitchJunctionPoint: true,
  defaultLineColor: '#2E74B5',
  defaultNodeColor: '#2E74B5',
  defaultNodeBorderWidth: 0,
  defaultNodeBorderColor: '#2E74B5',
  defaultNodeFontColor: '#ffffff',
  defaultNodeWidth: 40,
  defaultNodeHeight: 40,
  toolBarDirection: 'h',
  toolBarPositionH: 'center',
  toolBarPositionV: 'bottom',
  defaultLineShape: 2,
  defaultJunctionPoint: 'lr',
  // lineUseTextPath: true,
  defaultLineTextOffset_x: 2,
  defaultLineTextOffset_y: -3,
  layout: {
    layoutName: 'tree',
    from: 'left',
    min_per_width: 70, // 根据节点的宽度设置,这个是让图谱看起来偏亮的关键
    min_per_height: 50
  }
  // 这里可以参考"Graph 图谱"中的参数进行设置
};
const __graph_json_data = {
  'rootId': 'a',
  'nodes': [
    { 'id': 'a', 'text': 'a' },
    { 'id': 'b', 'text': 'b' },
    { 'id': 'b1', 'text': 'b1' },
    { 'id': 'b1-1', 'text': 'b1-1' },
    { 'id': 'b1-2', 'text': 'b1-2' },
    { 'id': 'b1-3', 'text': 'b1-3' },
    { 'id': 'b1-4', 'text': 'b1-4' },
    { 'id': 'b1-5', 'text': 'b1-5' },
    { 'id': 'b1-6', 'text': 'b1-6' },
    { 'id': 'b2', 'text': 'b2' },
    { 'id': 'b2-1', 'text': 'b2-1' },
    { 'id': 'b2-2', 'text': 'b2-2' },
    { 'id': 'c', 'text': 'c' },
    { 'id': 'c1', 'text': 'c1' },
    { 'id': 'c2', 'text': 'c2' },
    { 'id': 'c3', 'text': 'c3' }],
  'lines': [
    { 'from': 'a', 'to': 'b', text: 'Text' },
    { 'from': 'b', 'to': 'b1', text: 'Text' },
    { 'from': 'b1', 'to': 'b1-1', text: 'Text' },
    { 'from': 'b1', 'to': 'b1-2', text: 'Text' },
    { 'from': 'b1', 'to': 'b1-3', text: 'Text' },
    { 'from': 'b1', 'to': 'b1-4', text: 'Text' },
    { 'from': 'b1', 'to': 'b1-5', text: 'Text' },
    { 'from': 'b1', 'to': 'b1-6', text: 'Text' },
    { 'from': 'b', 'to': 'b2', text: 'Text' },
    { 'from': 'b2', 'to': 'b2-1', text: 'Text' },
    { 'from': 'b2', 'to': 'b2-2', text: 'Text' },
    { 'from': 'a', 'to': 'c', text: 'Text' },
    { 'from': 'c', 'to': 'c1', text: 'Text' },
    { 'from': 'c', 'to': 'c2', text: 'Text' },
    { 'from': 'c', 'to': 'c3', text: 'Text' }]
};
export default {
  name: 'Demo4AdvLineSlot',
  components: { },
  data() {
    return {
      lineShape: 2,
      polyLineRadius: 5,
      layoutExpansionDirection: 'center',
      linePoint: 'lr',
      layoutFrom: 'left',
      range_horizontal: 300,
      range_vertical: 70,
      levelDistance: '150,200,250,300,350',
      graphOptions
    };
  },
  mounted() {
    this.showGraph();
  },
  methods: {
    showGraph() {
      this.$refs.graphRef.setJsonData(__graph_json_data, (graphInstance) => {
        // 这些写上当图谱初始化完成后需要执行的代码
      });
    },
    tabChange() {
      const graphInstance = this.$refs.graphRef.getInstance();
      const graphOptions = graphInstance.options;
      graphOptions.defaultLineShape = this.lineShape;
      graphOptions.defaultPolyLineRadius = this.polyLineRadius;
      graphOptions.defaultJunctionPoint = this.linePoint;
    },
    async updateLayouterOptions() {
      this.graphOptions.layout.from = this.layoutFrom;
      this.graphOptions.layout.layoutExpansionDirection = this.layoutExpansionDirection;
      this.graphOptions.layout.min_per_width = this.range_horizontal;
      this.graphOptions.layout.max_per_width = this.range_horizontal + 10;
      this.graphOptions.layout.min_per_height = this.range_vertical;
      this.graphOptions.layout.max_per_height = this.range_vertical + 10;
      this.graphOptions.defaultPolyLineRadius = this.polyLineRadius;
      const graphInstance = this.$refs.graphRef.getInstance();
      await graphInstance.setOptions(this.graphOptions);
      this.tabChange();
    }
  }
};
</script>
<style lang="scss" scoped>
.c-my-panel{
  width: 400px;
  position: absolute;
  left: 10px;
  top: 10px;
  border-radius: 10px;
  z-index: 800;
  background-color: #efefef;
  border: #eeeeee solid 1px;
  padding: 10px;
  .c-option-name{
    color: #666666;
    font-size: 14px;
    line-height: 40px;
    padding-left:10px;
    padding-right:10px;
  }
  .c-my-options {
    text-align: center;
    .c-my-option-item {
      text-align: left;
      color: #1da9f5;
      cursor: pointer;
      border-radius: 5px;
      padding-left: 10px;
      margin-top: 5px;
      line-height: 25px;
      &:hover{
        background-color: rgba(29, 169, 245, 0.2);
      }
    }
  }
}
</style>

Vue3 版本

ever-changing-tree.vue

javascript
<template>
  <div>
    <div ref="myPage" class="my-graph" style="height:calc(100vh);">
      <RelationGraph ref="graphRef" :options="graphOptions">
        <template #graph-plug>
          <div class="c-my-panel">
            <div class="c-option-name">Layout Direction:</div>
            <el-radio-group size="small" v-model="layoutFrom" @change="updateLayouterOptions">
              <el-radio-button label="left">left</el-radio-button>
              <el-radio-button label="right">right</el-radio-button>
              <el-radio-button label="top">top</el-radio-button>
              <el-radio-button label="bottom">bottom</el-radio-button>
            </el-radio-group>
            <div class="c-option-name">Layout Expansion Direction:</div>
            <el-radio-group size="small" v-model="layoutExpansionDirection" @change="updateLayouterOptions">
              <el-radio-button label="center">Default</el-radio-button>
              <el-radio-button label="left">Left</el-radio-button>
              <el-radio-button label="right">Right</el-radio-button>
              <el-radio-button label="top">Top</el-radio-button>
              <el-radio-button label="bottom">Bottom</el-radio-button>
            </el-radio-group>
            <div class="c-option-name">Line Shape:</div>
            <el-radio-group v-model="lineShape" size="small" @change="tabChange">
              <el-radio-button :label="1">Straight</el-radio-button>
              <el-radio-button :label="2">2</el-radio-button>
              <el-radio-button :label="3">3</el-radio-button>
              <el-radio-button :label="5">5</el-radio-button>
              <el-radio-button :label="6">6</el-radio-button>
              <el-radio-button :label="7">7</el-radio-button>
              <el-radio-button :label="4">Polyline</el-radio-button>
            </el-radio-group>
            <div v-if="lineShape === 4">
              <div class="c-option-name">Polyline Radius:</div>
              <el-radio-group v-model="polyLineRadius" size="small" @change="tabChange">
                <el-radio-button :label="0">0</el-radio-button>
                <el-radio-button :label="3">3</el-radio-button>
                <el-radio-button :label="5">5</el-radio-button>
                <el-radio-button :label="8">8</el-radio-button>
                <el-radio-button :label="10">10</el-radio-button>
                <el-radio-button :label="15">15</el-radio-button>
              </el-radio-group>
            </div>
            <div class="c-option-name">Junction Point:</div>
            <el-radio-group v-model="linePoint" size="small" @change="tabChange">
              <el-radio-button label="border">Border</el-radio-button>
              <el-radio-button label="tb">Top-Bottom</el-radio-button>
              <el-radio-button label="lr">Left-Right</el-radio-button>
              <el-radio-button label="ltrb">Left-Top-Right-Bottom</el-radio-button>
              <el-radio-button label="left">L</el-radio-button>
              <el-radio-button label="top">T</el-radio-button>
              <el-radio-button label="right">R</el-radio-button>
              <el-radio-button label="bottom">B</el-radio-button>
            </el-radio-group>
            <div class="c-option-name">Level Distance & Node Spacing:</div>
            <div>
              <div>
                <div>
                  <div class="c-option-name">Node Spacing (Horizontal):</div>
                  <el-slider v-model="range_horizontal" :min="40" :max="500" @change="updateLayouterOptions">
                  </el-slider>
                </div>
                <div>
                  <div class="c-option-name">Node Spacing (Vertical):</div>
                  <el-slider v-model="range_vertical" :min="40" :max="500" @change="updateLayouterOptions">
                  </el-slider>
                </div>
              </div>
            </div>
          </div>
        </template>
      </RelationGraph>
    </div>
  </div>
</template>

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

const graphOptions: RGOptions = {
  allowSwitchLineShape: true,
  allowSwitchJunctionPoint: true,
  defaultLineColor: '#2E74B5',
  defaultNodeColor: '#2E74B5',
  defaultNodeBorderWidth: 0,
  defaultNodeBorderColor: '#2E74B5',
  defaultNodeFontColor: '#ffffff',
  defaultNodeWidth: 40,
  defaultNodeHeight: 40,
  toolBarDirection: 'h',
  toolBarPositionH: 'center',
  toolBarPositionV: 'bottom',
  defaultLineShape: 2,
  defaultJunctionPoint: 'lr',
  defaultLineTextOffset_x: 2,
  defaultLineTextOffset_y: -3,
  layout: {
    layoutName: 'tree',
    from: 'left',
    'min_per_width': 70,
    'min_per_height': 50

  }
};

const __graph_json_data: RGJsonData = {
  'rootId': 'a',
  'nodes': [
    { 'id': 'a', 'text': 'a' },
    { 'id': 'b', 'text': 'b' },
    { 'id': 'b1', 'text': 'b1' },
    { 'id': 'b1-1', 'text': 'b1-1' },
    { 'id': 'b1-2', 'text': 'b1-2' },
    { 'id': 'b1-3', 'text': 'b1-3' },
    { 'id': 'b1-4', 'text': 'b1-4' },
    { 'id': 'b1-5', 'text': 'b1-5' },
    { 'id': 'b1-6', 'text': 'b1-6' },
    { 'id': 'b2', 'text': 'b2' },
    { 'id': 'b2-1', 'text': 'b2-1' },
    { 'id': 'b2-2', 'text': 'b2-2' },
    { 'id': 'c', 'text': 'c' },
    { 'id': 'c1', 'text': 'c1' },
    { 'id': 'c2', 'text': 'c2' },
    { 'id': 'c3', 'text': 'c3' }
  ],
  'lines': [
    { 'from': 'a', 'to': 'b', text: 'Text' },
    { 'from': 'b', 'to': 'b1', text: 'Text' },
    { 'from': 'b1', 'to': 'b1-1', text: 'Text' },
    { 'from': 'b1', 'to': 'b1-2', text: 'Text' },
    { 'from': 'b1', 'to': 'b1-3', text: 'Text' },
    { 'from': 'b1', 'to': 'b1-4', text: 'Text' },
    { 'from': 'b1', 'to': 'b1-5', text: 'Text' },
    { 'from': 'b1', 'to': 'b1-6', text: 'Text' },
    { 'from': 'b', 'to': 'b2', text: 'Text' },
    { 'from': 'b2', 'to': 'b2-1', text: 'Text' },
    { 'from': 'b2', 'to': 'b2-2', text: 'Text' },
    { 'from': 'a', 'to': 'c', text: 'Text' },
    { 'from': 'c', 'to': 'c1', text: 'Text' },
    { 'from': 'c', 'to': 'c2', text: 'Text' },
    { 'from': 'c', 'to': 'c3', text: 'Text' }
  ]
};
const lineShape = ref(2);
const polyLineRadius = ref(5);
const layoutExpansionDirection = ref('center');
const linePoint = ref('lr');
const layoutFrom = ref('left');
const range_horizontal = ref(300);
const range_vertical = ref(70);
const graphRef = ref<RelationGraphComponent>();

const showGraph = async () => {
    const graphInstance = graphRef.value!.getInstance();
    await graphInstance.setJsonData(__graph_json_data);
    await graphInstance.moveToCenter();
    await graphInstance.zoomToFit();
};

const tabChange = () => {
    const graphInstance = graphRef.value!.getInstance();
    const graphOptions = graphInstance.options;
    graphOptions.defaultLineShape = lineShape.value;
    graphOptions.defaultPolyLineRadius = polyLineRadius.value;
    graphOptions.defaultJunctionPoint = linePoint.value;
};

const updateLayouterOptions = async () => {
    const treeLayoutOptions = graphOptions.layout as RGTreeLayoutOptions;
    treeLayoutOptions.from = layoutFrom.value;
    treeLayoutOptions.layoutExpansionDirection = layoutExpansionDirection.value;
    treeLayoutOptions.min_per_width = range_horizontal.value;
    treeLayoutOptions.max_per_width = range_horizontal.value + 10;
    treeLayoutOptions.min_per_height = range_vertical.value;
    treeLayoutOptions.max_per_height = range_vertical.value + 10;
    graphOptions.defaultPolyLineRadius = polyLineRadius.value;
    const graphInstance = graphRef.value!.getInstance();
    await graphInstance.setOptions(graphOptions);
    tabChange();
};

onMounted(() => {
    showGraph();
});
</script>

<style lang="scss" scoped>
.c-my-panel {
  width: 400px;
  position: absolute;
  left: 10px;
  top: 10px;
  border-radius: 10px;
  z-index: 800;
  background-color: #efefef;
  border: #eeeeee solid 1px;
  padding: 10px;
  .c-option-name {
    color: #666666;
    font-size: 14px;
    line-height: 40px;
    padding-left: 10px;
    padding-right: 10px;
  }
  .c-my-options {
    text-align: center;
    .c-my-option-item {
      text-align: left;
      color: #1da9f5;
      cursor: pointer;
      border-radius: 5px;
      padding-left: 10px;
      margin-top: 5px;
      line-height: 25px;
      &:hover {
        background-color: rgba(29, 169, 245, 0.2);
      }
    }
  }
}
</style>

React 版本

ever-changing-tree.tsx

javascript
import React, {useEffect, useRef, useState} from 'react';
import RelationGraph, {
  RGOptions,
  RGJsonData,
  RGNode,
  RGLine,
  RGLink,
  RGUserEvent,
  RelationGraphComponent,
  RGTreeLayoutOptions,
  RGLineShape, RGJunctionPoint
} from 'relation-graph-react';
import {MySelector} from "./RGDemoComponents/MyUIComponents";

const MyComponent = () => {
  const graphOptions: RGOptions = {
    allowSwitchLineShape: true,
    allowSwitchJunctionPoint: true,
    defaultLineColor: '#2E74B5',
    defaultNodeColor: '#2E74B5',
    defaultNodeBorderWidth: 0,
    defaultNodeBorderColor: '#2E74B5',
    defaultNodeFontColor: '#ffffff',
    defaultNodeWidth: 40,
    defaultNodeHeight: 40,
    toolBarDirection: 'h',
    toolBarPositionH: 'center',
    toolBarPositionV: 'bottom',
    defaultLineShape: 2,
    defaultJunctionPoint: 'lr',
    defaultLineTextOffset_x: 2,
    defaultLineTextOffset_y: -3,
    layout: {
      layoutName: 'tree',
      from: 'left',
      'min_per_width': 70,
      'min_per_height': 50

    }
  };

  const __graph_json_data: RGJsonData = {
    'rootId': 'a',
    'nodes': [
      { 'id': 'a', 'text': 'a' },
      { 'id': 'b', 'text': 'b' },
      { 'id': 'b1', 'text': 'b1' },
      { 'id': 'b1-1', 'text': 'b1-1' },
      { 'id': 'b1-2', 'text': 'b1-2' },
      { 'id': 'b1-3', 'text': 'b1-3' },
      { 'id': 'b1-4', 'text': 'b1-4' },
      { 'id': 'b1-5', 'text': 'b1-5' },
      { 'id': 'b1-6', 'text': 'b1-6' },
      { 'id': 'b2', 'text': 'b2' },
      { 'id': 'b2-1', 'text': 'b2-1' },
      { 'id': 'b2-2', 'text': 'b2-2' },
      { 'id': 'c', 'text': 'c' },
      { 'id': 'c1', 'text': 'c1' },
      { 'id': 'c2', 'text': 'c2' },
      { 'id': 'c3', 'text': 'c3' }
    ],
    'lines': [
      { 'from': 'a', 'to': 'b', text: 'Text' },
      { 'from': 'b', 'to': 'b1', text: 'Text' },
      { 'from': 'b1', 'to': 'b1-1', text: 'Text' },
      { 'from': 'b1', 'to': 'b1-2', text: 'Text' },
      { 'from': 'b1', 'to': 'b1-3', text: 'Text' },
      { 'from': 'b1', 'to': 'b1-4', text: 'Text' },
      { 'from': 'b1', 'to': 'b1-5', text: 'Text' },
      { 'from': 'b1', 'to': 'b1-6', text: 'Text' },
      { 'from': 'b', 'to': 'b2', text: 'Text' },
      { 'from': 'b2', 'to': 'b2-1', text: 'Text' },
      { 'from': 'b2', 'to': 'b2-2', text: 'Text' },
      { 'from': 'a', 'to': 'c', text: 'Text' },
      { 'from': 'c', 'to': 'c1', text: 'Text' },
      { 'from': 'c', 'to': 'c2', text: 'Text' },
      { 'from': 'c', 'to': 'c3', text: 'Text' }
    ]
  };

  const [lineShape, setLineShape] = useState(2);
  const [polyLineRadius, setPolyLineRadius] = useState(5);
  const [layoutExpansionDirection,setLayoutExpansionDirection] = useState('center');
  const [linePoint, setLinePoint] = useState('lr');
  const [layoutFrom, setLayoutFrom] = useState('left');
  const [range_horizontal, setRange_horizontal] = useState(300);
  const [range_vertical, setRange_vertical] = useState(70);
  const graphRef = useRef<RelationGraphComponent|null>(null);

  const showSeeksGraph = async () => {
    const graphInstance = graphRef.current?.getInstance();
    await graphInstance?.setJsonData(__graph_json_data);
    await graphInstance?.moveToCenter();
    await graphInstance?.zoomToFit();
  };

  const updateOptions = () => {
    const graphInstance = graphRef.current?.getInstance();
    const graphOptions = graphInstance?.options!;
    graphOptions.defaultLineShape = lineShape as RGLineShape;
    graphOptions.defaultPolyLineRadius = polyLineRadius;
    graphOptions.defaultJunctionPoint = linePoint as RGJunctionPoint;
    graphInstance?.dataUpdated();
  };

  const updateLayouterOptions = async () => {
    const treeLayoutOptions = graphOptions.layout as RGTreeLayoutOptions;
    treeLayoutOptions.from = layoutFrom;
    treeLayoutOptions.layoutExpansionDirection = layoutExpansionDirection;
    treeLayoutOptions.min_per_width = range_horizontal;
    treeLayoutOptions.max_per_width = range_horizontal + 10;
    treeLayoutOptions.min_per_height = range_vertical;
    treeLayoutOptions.max_per_height = range_vertical + 10;
    const graphInstance = graphRef.current?.getInstance();
    await graphInstance?.setOptions(graphOptions);
    await graphInstance?.doLayout();
  };

  useEffect(() => {
    showSeeksGraph();
  }, []);
  useEffect(() => {
    updateOptions();
  }, [lineShape, polyLineRadius, linePoint]);
  useEffect(() => {
    updateLayouterOptions();
  }, [layoutExpansionDirection, layoutFrom, range_horizontal, range_vertical]);

  return (
    <div>
      <div className="my-graph" style={{ height: '100vh' }}>
        <div className="w-96 rounded-lg absolute left-20 top-20 z-20 p-4 bg-white border-solid border-2 border-black shadow-lg">
          <div className="c-option-name">Layout Direction:</div>
          <MySelector

            data={[
              { value: 'left', text: 'left' },
              { value: 'right', text: 'right' },
              { value: 'top', text: 'top' },
              { value: 'bottom', text: 'bottom' }
            ]}
            currentValue={layoutFrom}
            onChange={(newValue: string, label) => { setLayoutFrom(newValue) }}
          />
          <div className="c-option-name">Layout Expansion Direction:</div>
          <MySelector

            data={[
              { value: 'center', text: 'Default' },
              { value: 'left', text: 'Left' },
              { value: 'right', text: 'Right' },
              { value: 'top', text: 'Top' },
              { value: 'bottom', text: 'Bottom' }
            ]}
            currentValue={layoutExpansionDirection}
            onChange={(newValue: string, label) => { setLayoutExpansionDirection(newValue); }}
          />
          <div className="c-option-name">Line Shape:</div>
          <MySelector

            data={[
              { value: 1, text: 'Straight' },
              { value: 2, text: '2' },
              { value: 3, text: '3' },
              { value: 5, text: '5' },
              { value: 6, text: '6' },
              { value: 7, text: '7' },
              { value: 4, text: 'Polyline' }
            ]}
            currentValue={lineShape}
            onChange={(newValue: number, label) => { setLineShape(newValue); }}
          />
          {lineShape === 4 && (
            <div>
              <div className="c-option-name">Polyline Radius:</div>
              <MySelector

                data={[
                  { value: 0, text: '0' },
                  { value: 3, text: '3' },
                  { value: 5, text: '5' },
                  { value: 8, text: '8' },
                  { value: 10, text: '10' },
                  { value: 15, text: '15' }
                ]}
                currentValue={polyLineRadius}
                onChange={(newValue: number, label) => { setPolyLineRadius(newValue); }}
              />
            </div>
          )}
          <div className="c-option-name">Junction Point:</div>
          <MySelector

            data={[
              { value: 'border', text: 'Border' },
              { value: 'tb', text: 'Top-Bottom' },
              { value: 'lr', text: 'Left-Right' },
              { value: 'ltrb', text: 'Left-Top-Right-Bottom' },
              { value: 'left', text: 'L' },
              { value: 'top', text: 'T' },
              { value: 'right', text: 'R' },
              { value: 'bottom', text: 'B' }
            ]}
            currentValue={linePoint}
            onChange={(newValue: string, label) => { setLinePoint(newValue); }}
          />
          <div className="c-option-name">Level Distance & Node Spacing:</div>
          <div>
            <div>
              <div>
                <div className="c-option-name">Node Spacing (Horizontal):</div>
                <input

                  type="range"
                  min={40}
                  max={500}
                  value={range_horizontal}
                  onChange={(e) => { setRange_horizontal(parseFloat(e.target.value)); }}
                />
              </div>
              <div>
                <div className="c-option-name">Node Spacing (Vertical):</div>
                <input

                  type="range"
                  min={40}
                  max={500}
                  value={range_vertical}
                  onChange={(e) => { setRange_vertical(parseFloat(e.target.value)); }}
                />
              </div>
            </div>
          </div>
        </div>
        <RelationGraph ref={graphRef} options={graphOptions}>
        </RelationGraph>
      </div>
    </div>
  );
};

export default MyComponent;

📂 RGDemoComponents

MyUIComponents.tsx

javascript
import React from "react";

export interface MySelectorProps {
  small?: boolean
  currentValue: string|number
  data:{value: string|number, text:string}[]
  onChange: (newValue:string|number, label:string) => void
}
export const MySelector:React.FC<MySelectorProps> = ({small, data, onChange, currentValue}) => {
  return (
    <div className="flex flex-wrap justify-center rounded-lg border border-gray-900 overflow-hidden">
      {
        data.map(item =>
          <div key={item.value}
               className={`border-r w-auto text-xs cursor-pointer whitespace-nowrap ${currentValue === item.value && 'bg-blue-500 text-white'} ${small?' px-2 h-6 leading-6':'h-8 px-3 leading-8'}`}
               onClick={() => {onChange(item.value, item.text);}}
          >
          {item.text}
        </div>)
      }
    </div>
  );
};
export interface MySwitchProps {
  currentValue: boolean
  onChange: (newValue:boolean) => void
}
export const MySwitch:React.FC<MySwitchProps> = ({onChange, currentValue}) => {
  return (
    <div className={`w-14 flex rounded-full border p-0.5 ${currentValue ? 'justify-end border-blue-500' : 'justify-start border-gray-500'}`}>
      <div
        className={`w-8 h-5 leading-8 rounded-full w-auto px-3 text-xs cursor-pointer whitespace-nowrap ${currentValue ? 'bg-blue-500' : 'bg-gray-500'}`} onClick={() => {onChange(!currentValue);}}>
      </div>
    </div>
  );
};
export interface MySliderProps {
  min: number
  max: number
  step: number
  currentValue: number
  onChange: (newValue:number) => void
}
export const MySlider:React.FC<MySliderProps> = ({min, max, step, currentValue, onChange}) => {
  return (
    <div>
      <input
        type="range"
        className="w-72"
        min={min}
        max={max}
        step={step}
        value={currentValue}
        onChange={(e) => { onChange(parseFloat(e.target.value))}}
      />
    </div>
  );
};

export interface MyRangeSliderProps {
  min: number
  max: number
  step: number
  currentValue: [number, number]
  onChange: (newValue:[number, number]) => void
}
export const MyRangeSlider:React.FC<MyRangeSliderProps> = ({min, max, step, currentValue, onChange}) => {
  return (
    <div className="w-72">
      <div>Min:</div>
      <input
        type="range"
        className="w-full"
        min={min}
        max={max}
        step={step}
        value={currentValue[0]}
        onChange={(e) => { if (parseFloat(e.target.value) < currentValue[1]) onChange([parseFloat(e.target.value), currentValue[1]])}}
      />
      <div>Max:</div>
      <input
        type="range"
        className="w-full"
        min={min}
        max={max}
        step={step}
        value={currentValue[1]}
        onChange={(e) => { if (parseFloat(e.target.value) > currentValue[0]) onChange([currentValue[0], parseFloat(e.target.value)])}}
      />
    </div>
  );
};
export interface MyButtonProps {
  onClick: () => void
  disabled?: boolean
}
export const MyButton:React.FC<MyButtonProps> = ({children, onClick, disabled}) => {
  return (
    <button className={`mr-2 px-2 py-1 rounded ${disabled===true ? 'bg-gray-300 text-black cursor-not-allowed':'bg-blue-500 hover:bg-blue-700 text-white'}`}
            onClick={()=>{onClick();}}>{children}</button>
  );
};
export interface MyLinkButtonProps {
  onClick: () => void
}
export const MyLinkButton:React.FC<MyLinkButtonProps> = ({children, onClick}) => {
  return (
    <div className="text-blue-600 cursor-pointer underline decoration-1" onClick={()=>{onClick();}}>
      {children}
    </div>
  );
};
export interface MyCheckBoxProps {
  currentValue: string|number
  data:{value: string|number, text:string}[]
  onChange: (newValue:string|number, label:string) => void
}
export const MyCheckBox:React.FC<MyCheckBoxProps> = ({data, onChange, currentValue}) => {
  // console.log(data);
  return (
    <div className="flex gap-2 flex-wrap">
      {
        data.map(thisItem =>
          <div
            key={thisItem.value}
            className={`px-1 py-0.5 flex justify-center place-items-center rounded-sm text-sm cursor-pointer  hover:bg-gray-300 ${currentValue === thisItem.value ? 'text-blue-600':'text-gray-500'}`}
            onClick={()=>{onChange(thisItem.value, thisItem.text);}}
          >
            <div className={`w-4 h-4 mr-1 rounded-full ${currentValue === thisItem.value ? 'border border-blue-500 bg-blue-500 text-blue-600':'border border-gray-500 text-gray-500'}`}></div>
            {thisItem.text}
          </div>
        )
      }
    </div>
  );
};
export interface CheckboxOption {
  value: string | number;
  text: string;
}

export interface MyMultiCheckBoxProps {
  currentValue: (string | number)[];
  checkboxOptions: CheckboxOption[];
  onChange: (newValue: (string | number)[]) => void;
}
export const MyMultiCheckBox:React.FC<MyMultiCheckBoxProps> = ({data, onChange, currentValue}) => {
  // console.log(data);
  const onClickItem = (item: string | number) => {
    const newValue = currentValue.includes(item)
      ? currentValue.filter((value) => value !== item)
      : [...currentValue, item];
    onChange(newValue);
  };
  return (
    <div className="flex gap-2 flex-wrap">
      {
        data.map(thisItem =>
          <div
            key={thisItem.value}
            className={`px-1 py-0.5 flex justify-center place-items-center rounded-sm text-sm cursor-pointer  hover:bg-gray-300 ${currentValue === thisItem.value ? 'text-blue-600':'text-gray-500'}`}
            onClick={()=>{onClickItem(thisItem.value);}}
          >
            <div className={`w-4 h-4 mr-1 rounded-full ${currentValue.includes(thisItem.value) ? 'border border-blue-500 bg-blue-500 text-blue-600':'border border-gray-500 text-gray-500'}`}></div>
            {thisItem.text}
          </div>
        )
      }
    </div>
  );
};
export const ElMessage = (messageObject) => {
  console.warn(messageObject);
}
export const ElNotification = (messageObject) => {
  console.warn(messageObject);
}

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