Custom Edge
As mentioned on the home page, the Edge is also Cell. Because of the complexity of the edge graph, we have given Edge
a default Component
, but you can still rely on the EdgeModel's methods and properties to customize Edge's Component. It is also possible to dynamically create different edges according to the status of the connection
import { Flow } from "moa-flow";
import { BizNode, BizNodeModel } from "./BizNode";
import { BizEdge, BizEdgeModel } from "./BizEdge";
export const App = () => {
return (
<Flow
components={{
BizNode: BizNode,
BizEdge: BizEdge,
}}
models={{
BizNode: BizNodeModel,
BizEdge: BizEdgeModel,
}}
linkEdge={(source, target) => {
if (!target.id) return "Edge";
return "BizEdge";
}}
canvasData={{
cells: [
{
component: "BizNode",
nodeName: "node 0",
x: 50,
y: 50,
ports: [{ portType: "in" }, { portType: "out", id: "0-out" }],
},
{
component: "BizNode",
nodeName: "node 1",
x: 500,
y: 200,
ports: [{ portType: "in", id: "1-in" }, { portType: "out" }],
},
{
component: "BizEdge",
nodeName: "node 1",
source: "0-out",
target: "1-in",
},
],
}}
></Flow>
);
};
Dynamically add port
You can dynamically add port, but after modifying the node's ports
data, you need to call syncPorts()
to let moa-flow re-index the ports of node to improve performance
import { Flow } from "moa-flow";
import { BizNode, BizNodeModel } from "./BizNode";
export const App = () => {
return (
<Flow
components={{
BizNode: BizNode,
}}
models={{
BizNode: BizNodeModel,
}}
canvasData={{
cells: [
{
component: "BizNode",
nodeName: "node 0",
x: 50,
y: 50,
ports: [{ portType: "in" }, { portType: "out", id: "0-out" }],
},
{
component: "BizNode",
nodeName: "node 1",
x: 500,
y: 200,
ports: [{ portType: "in", id: "1-in" }, { portType: "out" }],
},
{
component: "Edge",
nodeName: "node 1",
source: "0-out",
target: "1-in",
},
],
}}
></Flow>
);
};
Parent-child node
By setting the parent
attribute in the data
of the node's Model, you can set the parent for the node. When the parent node moves, the child nodes will also move together
import { Flow, NodeModel } from "moa-flow";
import { BizNode, BizNodeModel } from "./BizNode";
import { ParentNode } from "./ParentNode";
export const App = () => {
return (
<Flow
components={{
BizNode,
ParentNode,
}}
models={{
BizNode: BizNodeModel,
ParentNode: NodeModel,
}}
linkEdge="BizEdge"
canvasData={{
x: 50,
y: 50,
cells: [
{
component: "ParentNode",
id: 1,
},
{
component: "BizNode",
nodeName: "child node",
id: 0,
y: 120,
x: 50,
parent: 1,
ports: [{ portType: "in" }, { portType: "out" }],
},
],
}}
/>
);
};
MiniMap
moa-flow has a built-in mini-map plug-in, and detailed usage can be viewed on the API page
import { useRef } from "react";
import { Flow, ContextMenu, FlowModel } from "moa-flow";
import { BizNode } from "./BizNode";
import { BizNodeModel } from "./BizNodeModel";
import { Button } from "antd";
export const App = () => {
const flowModelRef = useRef<FlowModel>();
return (
<Flow
flowModelRef={flowModelRef}
components={{
BizNode: BizNode,
}}
models={{
BizNode: BizNodeModel,
}}
canvasData={{
cells: [
{
component: "BizNode",
nodeName: "node 0",
x: 50,
y: 50,
},
{
component: "BizNode",
nodeName: "node 1",
x: 500,
y: 300,
},
],
}}
>
<ContextMenu>
<Button
onClick={() => {
const flowModel = flowModelRef.current;
flowModel.addCell("BizNode", {
x: 300,
y: 300,
});
}}
>
add node
</Button>
<Button
onClick={() => {
const flowModel = flowModelRef.current;
const { selectCells, deleCell } = flowModel;
deleCell(selectCells[0]);
}}
>
delete
</Button>
</ContextMenu>
</Flow>
);
};
Custom tools
You can get the context of the graph, and on this basis, it is very convenient to customize the expansion tools
import { Flow } from "moa-flow";
import { BizNode, BizNodeModel } from "./BizNode";
import { Toolbar } from "./Toolbar";
export const App = () => {
return (
<Flow
components={{
BizNode: BizNode,
}}
models={{
BizNode: BizNodeModel,
}}
canvasData={{
cells: [
{
component: "BizNode",
nodeName: "node 0",
x: 250,
y: 50,
},
{
component: "BizNode",
nodeName: "node 1",
x: 500,
y: 200,
},
],
}}
>
<Toolbar />
</Flow>
);
};
Set Layout
If you want a graph with a fixed layout (such as a dagre graph), you can also import a third-party layout library such as dagre.js, because the layout library actually just changes the x and y properties of the nodes to make them in the correct position
import { Flow, FlowModel } from "moa-flow";
import { BizNode, BizNodeModel } from "./BizNode";
import dagre from "dagre";
import { useEffect, useRef } from "react";
import { cells } from "./data";
export const App = () => {
const flowModelRef = useRef<FlowModel>();
const dagreGraph = new dagre.graphlib.Graph();
dagreGraph.setDefaultEdgeLabel(() => ({}));
const nodeWidth = 200;
const nodeHeight = 181;
const setLayoutedCells = (nodes, edges, direction = "LR") => {
const context = flowModelRef.current;
dagreGraph.setGraph({ rankdir: direction, ranksep: 150 });
nodes.forEach((node) => {
dagreGraph.setNode(node.id, { width: nodeWidth, height: nodeHeight });
});
edges.forEach((edge) => {
dagreGraph.setEdge(
context.getCellData(edge.source).host,
context.getCellData(edge.target).host
);
});
dagre.layout(dagreGraph);
nodes.forEach((node) => {
const nodeWithPosition = dagreGraph.node(node.id);
context.setCellData(node.id, {
x: nodeWithPosition.x - nodeWidth / 2,
y: nodeWithPosition.y - nodeHeight / 2,
});
});
};
useEffect(() => {
const context = flowModelRef.current;
setLayoutedCells(context.getNodesData(), context.getEdgesData());
});
return (
<Flow
flowModelRef={flowModelRef}
components={{
BizNode,
}}
models={{
BizNode: BizNodeModel,
}}
canvasData={{
x: 100,
y: 100,
scale: 0.7,
cells,
}}
/>
);
};