使用 Sigma.js 布局算法
Vue2 版本
use-sigma-layout.vue
javascript
<template>
<div>
<div style="height:calc(100vh);">
<RelationGraph
ref="graphRef"
:options="graphOptions"
:on-node-click="onNodeClick"
:on-line-click="onLineClick"
>
<template #graph-plug>
<div class="c-my-panel">
<div class="c-option-name" style="line-height: 25px;padding:10px;">
此布局使用到了第三方布局算法:Sigma,同时relation-graph的支持插槽等强大功能您依然可以使用,完美融合。Sigma力导向图布局是经典的布局算法。
</div>
</div>
</template>
</RelationGraph>
</div>
</div>
</template>
<script>
// 如果您没有在main.js文件中使用Vue.use(RelationGraph); 就需要使用下面这一行代码来引入relation-graph
// import RelationGraph from 'relation-graph';
import Graph from "graphology";
import circular from "graphology-layout/circular";
import forceAtlas2 from "graphology-layout-forceatlas2";
export default {
name: 'Demo',
components: { },
data() {
return {
currentCase: '横向树状图谱',
isShowCodePanel: false,
graphOptions: {
debug: false,
lineUseTextPath: true,
layout: {
layoutName: 'fixed'
},
defaultNodeShape: 0,
defaultLineShape: 1,
defaultJunctionPoint: 'border',
defaultNodeBorderWidth: 0,
defaultNodeWidth: 30,
defaultNodeHeight: 30,
defaultLineColor: 'rgba(0, 186, 189, 1)',
defaultNodeColor: 'rgba(0, 206, 209, 1)'
}
};
},
mounted() {
this.useGraphologyLayout();
},
methods: {
async useGraphologyLayout() {
// const __graph_json_data = {"rootId":"root","nodes":[{"id":"root","text":"主题"},{"id":"N2","text":"New-N2"},{"id":"N3","text":"New-N3"},{"id":"N4","text":"New-N4"},{"id":"N5","text":"New-N5"},{"id":"N6","text":"New-N6"},{"id":"N7","text":"New-N7"},{"id":"N8","text":"New-N8"}],"lines":[{"from":"root","to":"N2","text":"新连线1"},{"from":"root","to":"N3","text":"新连线2"},{"from":"N3","to":"N4","text":"新连线4"},{"from":"N4","to":"N5","text":"新连线5"},{"from":"N5","to":"N6","text":"新连线6"},{"from":"N6","to":"N2","text":"新连线7"},{"from":"N2","to":"N8","text":"新连线8"},{"from":"N2","to":"N7","text":"新连线9"},{"from":"N7","to":"N8","text":"新连线10"},{"from":"root","to":"N8","text":"新连线11"}]};
let __graph_json_data =
{ 'rootId': 'root', 'nodes': [{ 'id': 'root', 'text': '节点' }, { 'id': 'N2', 'text': '新节点N2' }, { 'id': 'N3', 'text': '新节点N3' }, { 'id': 'N4', 'text': '新节点N4' }, { 'id': 'N5', 'text': '新节点N5' }, { 'id': 'N6', 'text': '新节点N6' }, { 'id': 'N7', 'text': '新节点N7' }, { 'id': 'N8', 'text': '新节点N8' }, { 'id': 'N9', 'text': '新节点N9' }, { 'id': 'N10', 'text': '新节点N10' }, { 'id': 'N11', 'text': '新节点N11' }, { 'id': 'N12', 'text': '新节点N12' }, { 'id': 'N13', 'text': '新节点N13' }, { 'id': 'N14', 'text': '新节点N14' }, { 'id': 'N15', 'text': '新节点N15' }, { 'id': 'N16', 'text': '新节点N16' }, { 'id': 'N17', 'text': '新节点N17' }, { 'id': 'N18', 'text': '新节点N18' }, { 'id': 'N19', 'text': '新节点N19' }, { 'id': 'N20', 'text': '新节点N20' }, { 'id': 'N21', 'text': '新节点N21' }, { 'id': 'N22', 'text': '新节点N22' }, { 'id': 'N23', 'text': '新节点N23' }, { 'id': 'N24', 'text': '新节点N24' }, { 'id': 'N25', 'text': '新节点N25' }, { 'id': 'N26', 'text': '新节点N26' }, { 'id': 'N27', 'text': 'New-N27' }, { 'id': 'N28', 'text': 'New-N28' }, { 'id': 'N29', 'text': 'New-N29' }, { 'id': 'N30', 'text': 'New-N30' }, { 'id': 'N31', 'text': 'New-N31' }, { 'id': 'N32', 'text': 'New-N32' }, { 'id': 'N33', 'text': 'New-N33' }, { 'id': 'N34', 'text': 'New-N34' }, { 'id': 'N35', 'text': 'New-N35' }, { 'id': 'N36', 'text': 'New-N36' }, { 'id': 'N37', 'text': 'New-N37' }, { 'id': 'N38', 'text': 'New-N38' }, { 'id': 'N39', 'text': 'New-N39' }], 'lines': [{ 'from': 'N3', 'to': 'N2', 'text': '新连线1' }, { 'from': 'N2', 'to': 'root', 'text': '新连线1' }, { 'from': 'root', 'to': 'N4', 'text': '新连线2' }, { 'from': 'N4', 'to': 'N5', 'text': '新连线3' }, { 'from': 'N6', 'to': 'N7', 'text': '新连线2' }, { 'from': 'N7', 'to': 'root', 'text': '新连线3' }, { 'from': 'N8', 'to': 'N9', 'text': '新连线4' }, { 'from': 'N9', 'to': 'root', 'text': '新连线5' }, { 'from': 'N10', 'to': 'N11', 'text': '新连线6' }, { 'from': 'N11', 'to': 'root', 'text': '新连线7' }, { 'from': 'N13', 'to': 'N12', 'text': '新连线8' }, { 'from': 'N12', 'to': 'root', 'text': '新连线9' }, { 'from': 'N18', 'to': 'N17', 'text': '新连线10' }, { 'from': 'N17', 'to': 'N14', 'text': '新连线11' }, { 'from': 'N15', 'to': 'N14', 'text': '新连线12' }, { 'from': 'N16', 'to': 'N15', 'text': '新连线13' }, { 'from': 'N12', 'to': 'N14', 'text': '新连线14' }, { 'from': 'N20', 'to': 'N19', 'text': '新连线15' }, { 'from': 'N21', 'to': 'N19', 'text': '新连线16' }, { 'from': 'N19', 'to': 'N15', 'text': '新连线17' }, { 'from': 'N26', 'to': 'N22', 'text': '新连线18' }, { 'from': 'N24', 'to': 'N25', 'text': '新连线19' }, { 'from': 'N24', 'to': 'N22', 'text': '新连线20' }, { 'from': 'N22', 'to': 'N23', 'text': '新连线21' }, { 'from': 'N23', 'to': 'N14', 'text': '新连线22' }, { 'from': 'root', 'to': 'N30', 'text': '新连线1' }, { 'from': 'root', 'to': 'N27', 'text': '新连线2' }, { 'from': 'N30', 'to': 'N33', 'text': '新连线3' }, { 'from': 'N30', 'to': 'N29', 'text': '新连线4' }, { 'from': 'N27', 'to': 'N28', 'text': '新连线5' }, { 'from': 'N27', 'to': 'N31', 'text': '新连线6' }, { 'from': 'N27', 'to': 'N32', 'text': '新连线7' }, { 'from': 'N4', 'to': 'N34', 'text': '新连线8' }, { 'from': 'N28', 'to': 'N35', 'text': '新连线9' }, { 'from': 'N28', 'to': 'N36', 'text': '新连线12' }, { 'from': 'N28', 'to': 'N37', 'text': '新连线13' }, { 'from': 'N36', 'to': 'N39', 'text': '新连线14' }, { 'from': 'N36', 'to': 'N38', 'text': '新连线15' }] };
__graph_json_data.nodes.forEach(node => { // 随机设置节点宽高
// node.width = 40 + Math.floor(Math.random() * 200)
// node.height = 40
node.text = 'N';
})
const graph = new Graph();
const graphInstance = this.$refs.graphRef.getInstance();
await graphInstance.setJsonData(__graph_json_data);
graphInstance.getNodes().forEach(node => {
graph.addNode(node.id, {
text: node.text,
width: node.el.offsetWidth,
height: node.el.offsetHeight,
});
});
graphInstance.getLinks().forEach(link => {
link.relations.forEach((line) => {
graph.addEdge(link.fromNode.id, link.toNode.id, {
id: line.id, // 设置id,到会儿通过id找到jsonData中的line,设置line.data.points
weight: 1 // 这是一个非常重要的参数,尝试设置为0或者1或者中间值查看效果
});
})
});
circular.assign(graph);
forceAtlas2.assign(graph, 50);
graph.nodes().forEach(nodeId => { // 拷贝布局结果到relation-graph对应的节点中
const node = graph.getNodeAttributes(nodeId);
console.log(nodeId, );
const rgNode = graphInstance.getNodeById(nodeId);
rgNode.x = node.x * 10; // 通过乘以一个变量,来调节节点之间的距离
rgNode.y = node.y * 10;
})
},
onNodeClick(nodeObject, $event) {
console.log('onNodeClick:', nodeObject);
},
onLineClick(lineObject, linkObject, $event) {
console.log('onLineClick:', lineObject);
}
}
};
</script>
<style lang="scss">
</style>
<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 版本
use-sigma-layout.vue
javascript
<template>
<div>
<div style="height:calc(100vh);">
<RelationGraph
ref="graphRef"
:options="graphOptions"
:on-node-click="onNodeClick"
:on-line-click="onLineClick"
>
<template #graph-plug>
<div class="c-my-panel">
<div class="c-option-name" style="line-height: 25px;padding:10px;">
This layout uses a third-party layout algorithm: Sigma, and you can still use powerful features such as slot support in relation-graph, perfect integration. Sigma force-directed graph layout is a classic layout algorithm.
</div>
</div>
</template>
</RelationGraph>
</div>
</div>
</template>
<script lang="ts" setup>
import { ref, onMounted } from 'vue';
import RelationGraph, {RGJsonData} from 'relation-graph-vue3';
import { RelationGraphComponent, RGOptions, RGNode, RGLine, RGLink, RGUserEvent } from 'relation-graph-vue3';
import Graph from "graphology";
import circular from "graphology-layout/circular";
import forceAtlas2 from "graphology-layout-forceatlas2";
const graphRef = ref<RelationGraphComponent>();
const graphOptions: RGOptions = {
debug: false,
lineUseTextPath: true,
layout: {
layoutName: 'fixed'
},
defaultNodeShape: 0,
defaultLineShape: 1,
defaultJunctionPoint: 'border',
defaultNodeBorderWidth: 0,
defaultNodeWidth: 30,
defaultNodeHeight: 30,
defaultLineColor: 'rgba(0, 186, 189, 1)',
defaultNodeColor: 'rgba(0, 206, 209, 1)'
};
onMounted(async () => {
await useGraphologyLayout();
});
const useGraphologyLayout = async () => {
let __graph_json_data: RGJsonData = {
rootId: 'root',
nodes: [
{ id: 'root', text: '节点' },
{ id: 'N2', text: '新节点N2' },
{ id: 'N3', text: '新节点N3' },
{ id: 'N4', text: '新节点N4' },
{ id: 'N5', text: '新节点N5' },
{ id: 'N6', text: '新节点N6' },
{ id: 'N7', text: '新节点N7' },
{ id: 'N8', text: '新节点N8' },
{ id: 'N9', text: '新节点N9' },
{ id: 'N10', text: '新节点N10' },
{ id: 'N11', text: '新节点N11' },
{ id: 'N12', text: '新节点N12' },
{ id: 'N13', text: '新节点N13' },
{ id: 'N14', text: '新节点N14' },
{ id: 'N15', text: '新节点N15' },
{ id: 'N16', text: '新节点N16' },
{ id: 'N17', text: '新节点N17' },
{ id: 'N18', text: '新节点N18' },
{ id: 'N19', text: '新节点N19' },
{ id: 'N20', text: '新节点N20' },
{ id: 'N21', text: '新节点N21' },
{ id: 'N22', text: '新节点N22' },
{ id: 'N23', text: '新节点N23' },
{ id: 'N24', text: '新节点N24' },
{ id: 'N25', text: '新节点N25' },
{ id: 'N26', text: '新节点N26' },
{ id: 'N27', text: 'New-N27' },
{ id: 'N28', text: 'New-N28' },
{ id: 'N29', text: 'New-N29' },
{ id: 'N30', text: 'New-N30' },
{ id: 'N31', text: 'New-N31' },
{ id: 'N32', text: 'New-N32' },
{ id: 'N33', text: 'New-N33' },
{ id: 'N34', text: 'New-N34' },
{ id: 'N35', text: 'New-N35' },
{ id: 'N36', text: 'New-N36' },
{ id: 'N37', text: 'New-N37' },
{ id: 'N38', text: 'New-N38' },
{ id: 'N39', text: 'New-N39' }
],
lines: [
{ from: 'N3', to: 'N2', text: '新连线1' },
{ from: 'N2', to: 'root', text: '新连线1' },
{ from: 'root', to: 'N4', text: '新连线2' },
{ from: 'N4', to: 'N5', text: '新连线3' },
{ from: 'N6', to: 'N7', text: '新连线2' },
{ from: 'N7', to: 'root', text: '新连线3' },
{ from: 'N8', to: 'N9', text: '新连线4' },
{ from: 'N9', to: 'root', text: '新连线5' },
{ from: 'N10', to: 'N11', text: '新连线6' },
{ from: 'N11', to: 'root', text: '新连线7' },
{ from: 'N13', to: 'N12', text: '新连线8' },
{ from: 'N12', to: 'root', text: '新连线9' },
{ from: 'N18', to: 'N17', text: '新连线10' },
{ from: 'N17', to: 'N14', text: '新连线11' },
{ from: 'N15', to: 'N14', text: '新连线12' },
{ from: 'N16', to: 'N15', text: '新连线13' },
{ from: 'N12', to: 'N14', text: '新连线14' },
{ from: 'N20', to: 'N19', text: '新连线15' },
{ from: 'N21', to: 'N19', text: '新连线16' },
{ from: 'N19', to: 'N15', text: '新连线17' },
{ from: 'N26', to: 'N22', text: '新连线18' },
{ from: 'N24', to: 'N25', text: '新连线19' },
{ from: 'N24', to: 'N22', text: '新连线20' },
{ from: 'N22', to: 'N23', text: '新连线21' },
{ from: 'N23', to: 'N14', text: '新连线22' },
{ from: 'root', to: 'N30', text: '新连线1' },
{ from: 'root', to: 'N27', text: '新连线2' },
{ from: 'N30', to: 'N33', text: '新连线3' },
{ from: 'N30', to: 'N29', text: '新连线4' },
{ from: 'N27', to: 'N28', text: '新连线5' },
{ from: 'N27', to: 'N31', text: '新连线6' },
{ from: 'N27', to: 'N32', text: '新连线7' },
{ from: 'N4', to: 'N34', text: '新连线8' },
{ from: 'N28', to: 'N35', text: '新连线9' },
{ from: 'N28', to: 'N36', text: '新连线12' },
{ from: 'N28', to: 'N37', text: '新连线13' },
{ from: 'N36', to: 'N39', text: '新连线14' },
{ from: 'N36', to: 'N38', text: '新连线15' }
]
};
__graph_json_data.nodes.forEach(node => {
node.text = 'N';
});
const graph = new Graph();
const graphInstance = graphRef.value!.getInstance();
await graphInstance.setJsonData(__graph_json_data);
graphInstance.getNodes().forEach(node => {
graph.addNode(node.id, {
text: node.text,
width: node.el.offsetWidth,
height: node.el.offsetHeight
});
});
graphInstance.getLinks().forEach(link => {
link.relations.forEach((line) => {
graph.addEdge(link.fromNode.id, link.toNode.id, {
id: line.id,
weight: 1
});
});
});
circular.assign(graph);
forceAtlas2.assign(graph, 50);
graph.nodes().forEach(nodeId => {
const node = graph.getNodeAttributes(nodeId);
const rgNode = graphInstance.getNodeById(nodeId);
rgNode.x = node.x * 10;
rgNode.y = node.y * 10;
});
};
const onNodeClick = (nodeObject: RGNode, $event: RGUserEvent) => {
console.log('onNodeClick:', nodeObject);
};
const onLineClick = (lineObject: RGLine, linkObject: RGLink, $event: RGUserEvent) => {
console.log('onLineClick:', lineObject);
};
</script>
<style lang="scss">
</style>
<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 版本
use-sigma-layout.tsx
javascript
import React, { useEffect, useRef } from 'react';
import RelationGraph, { RGJsonData, RelationGraphComponent, RGOptions, RGNode, RGLine, RGLink, RGUserEvent } from 'relation-graph-react';
import Graph from "graphology";
import circular from "graphology-layout/circular";
import forceAtlas2 from "graphology-layout-forceatlas2";
const MyComponent = () => {
const graphRef = useRef<RelationGraphComponent|null>(null);
const graphOptions: RGOptions = {
debug: false,
lineUseTextPath: true,
layout: {
layoutName: 'fixed'
},
defaultNodeShape: 0,
defaultLineShape: 1,
defaultJunctionPoint: 'border',
defaultNodeBorderWidth: 0,
defaultNodeWidth: 30,
defaultNodeHeight: 30,
defaultLineColor: 'rgba(0, 186, 189, 1)',
defaultNodeColor: 'rgba(0, 206, 209, 1)'
};
const useGraphologyLayout = async () => {
let __graph_json_data: RGJsonData = {
rootId: 'root',
nodes: [
{ id: 'root', text: '节点' },
{ id: 'N2', text: '新节点N2' },
{ id: 'N3', text: '新节点N3' },
{ id: 'N4', text: '新节点N4' },
{ id: 'N5', text: '新节点N5' },
{ id: 'N6', text: '新节点N6' },
{ id: 'N7', text: '新节点N7' },
{ id: 'N8', text: '新节点N8' },
{ id: 'N9', text: '新节点N9' },
{ id: 'N10', text: '新节点N10' },
{ id: 'N11', text: '新节点N11' },
{ id: 'N12', text: '新节点N12' },
{ id: 'N13', text: '新节点N13' },
{ id: 'N14', text: '新节点N14' },
{ id: 'N15', text: '新节点N15' },
{ id: 'N16', text: '新节点N16' },
{ id: 'N17', text: '新节点N17' },
{ id: 'N18', text: '新节点N18' },
{ id: 'N19', text: '新节点N19' },
{ id: 'N20', text: '新节点N20' },
{ id: 'N21', text: '新节点N21' },
{ id: 'N22', text: '新节点N22' },
{ id: 'N23', text: '新节点N23' },
{ id: 'N24', text: '新节点N24' },
{ id: 'N25', text: '新节点N25' },
{ id: 'N26', text: '新节点N26' },
{ id: 'N27', text: 'New-N27' },
{ id: 'N28', text: 'New-N28' },
{ id: 'N29', text: 'New-N29' },
{ id: 'N30', text: 'New-N30' },
{ id: 'N31', text: 'New-N31' },
{ id: 'N32', text: 'New-N32' },
{ id: 'N33', text: 'New-N33' },
{ id: 'N34', text: 'New-N34' },
{ id: 'N35', text: 'New-N35' },
{ id: 'N36', text: 'New-N36' },
{ id: 'N37', text: 'New-N37' },
{ id: 'N38', text: 'New-N38' },
{ id: 'N39', text: 'New-N39' }
],
lines: [
{ from: 'N3', to: 'N2', text: '新连线1' },
{ from: 'N2', to: 'root', text: '新连线1' },
{ from: 'root', to: 'N4', text: '新连线2' },
{ from: 'N4', to: 'N5', text: '新连线3' },
{ from: 'N6', to: 'N7', text: '新连线2' },
{ from: 'N7', to: 'root', text: '新连线3' },
{ from: 'N8', to: 'N9', text: '新连线4' },
{ from: 'N9', to: 'root', text: '新连线5' },
{ from: 'N10', to: 'N11', text: '新连线6' },
{ from: 'N11', to: 'root', text: '新连线7' },
{ from: 'N13', to: 'N12', text: '新连线8' },
{ from: 'N12', to: 'root', text: '新连线9' },
{ from: 'N18', to: 'N17', text: '新连线10' },
{ from: 'N17', to: 'N14', text: '新连线11' },
{ from: 'N15', to: 'N14', text: '新连线12' },
{ from: 'N16', to: 'N15', text: '新连线13' },
{ from: 'N12', to: 'N14', text: '新连线14' },
{ from: 'N20', to: 'N19', text: '新连线15' },
{ from: 'N21', to: 'N19', text: '新连线16' },
{ from: 'N19', to: 'N15', text: '新连线17' },
{ from: 'N26', to: 'N22', text: '新连线18' },
{ from: 'N24', to: 'N25', text: '新连线19' },
{ from: 'N24', to: 'N22', text: '新连线20' },
{ from: 'N22', to: 'N23', text: '新连线21' },
{ from: 'N23', to: 'N14', text: '新连线22' },
{ from: 'root', to: 'N30', text: '新连线1' },
{ from: 'root', to: 'N27', text: '新连线2' },
{ from: 'N30', to: 'N33', text: '新连线3' },
{ from: 'N30', to: 'N29', text: '新连线4' },
{ from: 'N27', to: 'N28', text: '新连线5' },
{ from: 'N27', to: 'N31', text: '新连线6' },
{ from: 'N27', to: 'N32', text: '新连线7' },
{ from: 'N4', to: 'N34', text: '新连线8' },
{ from: 'N28', to: 'N35', text: '新连线9' },
{ from: 'N28', to: 'N36', text: '新连线12' },
{ from: 'N28', to: 'N37', text: '新连线13' },
{ from: 'N36', to: 'N39', text: '新连线14' },
{ from: 'N36', to: 'N38', text: '新连线15' }
]
};
__graph_json_data.nodes.forEach(node => {
node.text = 'N';
});
const graph = new Graph();
const graphInstance = graphRef.current!.getInstance();
await graphInstance.setJsonData(__graph_json_data);
graphInstance.getNodes().forEach(node => {
graph.addNode(node.id, {
text: node.text,
width: node.el.offsetWidth,
height: node.el.offsetHeight
});
});
graphInstance.getLinks().forEach(link => {
link.relations.forEach((line) => {
graph.addEdge(link.fromNode.id, link.toNode.id, {
id: line.id,
weight: 1
});
});
});
circular.assign(graph);
forceAtlas2.assign(graph, 50);
graph.nodes().forEach(nodeId => {
const node = graph.getNodeAttributes(nodeId);
const rgNode = graphInstance.getNodeById(nodeId);
rgNode.x = node.x * 10;
rgNode.y = node.y * 10;
});
};
const onNodeClick = (nodeObject: RGNode, $event: RGUserEvent) => {
console.log('onNodeClick:', nodeObject);
};
const onLineClick = (lineObject: RGLine, linkObject: RGLink, $event: RGUserEvent) => {
console.log('onLineClick:', lineObject);
};
useEffect(() => {
useGraphologyLayout();
}, []);
return (
<div>
<div 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 leading-12 p-3">
This layout uses a third-party layout algorithm: Sigma, and you can still use powerful features such as slot support in relation-graph, perfect integration. Sigma force-directed graph layout is a classic layout algorithm.
</div>
</div>
<RelationGraph
ref={graphRef}
options={graphOptions}
onNodeClick={onNodeClick}
onLineClick={onLineClick}
>
</RelationGraph>
</div>
</div>
);
};
export default MyComponent;