核心概念
节点
节点即画布中的一系列形状,可以是矩形,圆形,多边形等,我们可以根据业务需要自定义节点的形状。
内置节点
为了方便业务方的需求,更快的接入画布到业务中,我们提供了多种内置的节点,可以直接通过数据配置的方式直接接入。
基础节点
通用属性配置
// 所有节点通用属性
interface NodeData extends NodeConfig {
type: string; // 节点类型
id: string; // 节点的唯一标识
clientX?: number;
clientY?: number;
x?: number; // 节点位置的 x 值
y?: number; // 节点位置的 y 值
color: string; // 默认绘制的 hover, port高亮的值
fillColor: string; // 节点填充颜色
strokeColor: string; // 节点边的颜色
title: string; // 节点名称
outputs?: number[][]; // 出口节点位置
inputs?: number[][]; // 入口节点位置
width?: number; // 节点宽度
height?: number; // 节点高度
outPortFillColor: string; // port的填充色
portColor: string; // port默认颜色
baseNodeCfg: Partial<baseNodeCfg>; // baseNode属性
cardNodeCfg: Partial<cardNodeCfg>; // cardNode属性
[key: string]: any; // 可能存在一些自定义的属性值
}
基础节点配置
// base节点属性
interface baseNodeCfg {
type: 'block' | 'icon' | 'blockText';
status: 'notReady' | 'success' | 'inProgress' | 'failure' | 'stop' | 'lock' | 'special';
icon: string; // icon的类
blockTitle: string; //
subTitle: string;// 第二行的文案 title对应第一行的文案
statusIcon: string; // 状态icon
statusFillColor: string; // 状态颜色
blockFillColor: string; // 左侧块状填充色
description: string;
leftFillWidth: number;
leftIconColor: string;
statusDesc: string;
}
卡片节点
卡片节点配置
interface cardNodeCfg {
panels: object[] | string[]; // 下面展示文案
icon: string; // 头部icon
iconColor: string; // icon颜色
subTitle: string; // 右侧文案
color: string; // 文字颜色
}
如何使用 icon
icon 我们是使用的公司提供的 iconfont,具体的项目地址是iconfont.sankuai.com,具体的使用方式跟其他字体图标一致,直接使用 class 即可。
如何自定义节点
首先,我们需要绘制出节点的形状,以下面的节点为例,我们分步演示如何画出一个节点。
第一步:定义节点的关键形状。
function createNode (group, config, width, height) {
const { strokeColor, fillColor } = cfg;
// 外边框
const keyShape = group.addShape('rect', {
attrs: {
x: 0, // 相对坐标
y: 0, // 相对坐标
width: width, // 节点宽
height: height, // 节点高
stroke: strokeColor, // 外边框颜色
fill: fillColor, // 填充颜色
radius: 2 // 半径
}
});
return keyShape;
}
G6 中,每个节点都必须有一个关键形状,我们可以将其理解为节点的外部轮廓。我们通过group.addShape()
方法来定义关键形状,这里我们定义了矩形react
,也可以定义其他形状,然后在第二个参数中传入配置即可。
createNode
函数中的前两个参数group
与config
会在此函数执行的时候由 G6 传入相关配置,group
为 G6 的概念,代表一组图形,config
代表我们配置节点的参数,比如我们配置节点的时候这样配置{strokeColor: '#ff0000', fillColor: '#00ff00'}
,则渲染的时候就会渲染出外边框是红色,填充色为绿色的图形。
第二步:在 group 里定义左边底色
group.addShape('rect', {
attrs: {
x: 0,
y: 0,
width: h - 1.5,
height: h,
fill: strokeColor,
radius: [2, 0, 0, 2]
},
draggable: true
});
第三步,我们给节点加上文字描述,作为节点名称
// 文字
const textShape = group.addShape('text', {
attrs: {
x: (height - 14) / 2 + height,
y: 9,
width: width - height - (height - 14) / 2,
fill: 'rgba(0, 0, 0, 0.92)',
fontSize: 13,
textBaseline: 'top',
text: name
}
});
第四步,我们给节点加上一个图片标签,美观的同时也让其更有意义
// 增加左侧图片
const image =
'';
group.addShape('image', {
attrs: {
x: (height - 16) / 2,
y: (height - 16) / 2,
width: 16,
height: 16,
img: image
},
draggable: true
});
至此,我们就创建了一个节点,只要配置好数据,我们就能在画布上展示一系列这样的节点了。
port
节点与节点之间,如果要产生关系,需要通过连线来表示,要想将两个节点相连,需要通过 port 进行操作,所以一个节点需要分别有入线port
以及出线port
,否则其就是一个孤立的节点,不会与其他节点产生关系。
如何绘制 port
绘制 port 非常简单,只需要在节点数据中配置即可,我们通过下面的例子讲解:
上面的例子中,将鼠标移到节点上,你会发现上边有两个入线port
,下边有三个出线port
,那如何设置呢?其实很简单,只需要如下配置即可:
const config = {
type: 'rectNode',
id: 'node1', // String,该节点存在则必须,节点的唯一标识
x: 100, // Number,可选,节点位置的 x 值
y: 20, // Number,可选,节点位置的 y 值
color: '#1ea565',
fillColor: '#ffffffc0',
strokeColor: '#1ea565',
name: '节点1',
inputs: [[0, 0.5]], // 以左上角坐标为(0,0),右下角坐标为(1,1)
outputs: [
[1, 0],
[1, 0.5],
[1, 1],
],
}
可以看到,我们在component
对象里配置了两个数组,分别为inputs
与outputs
,每个数组项就是一个 port,这里只需要传入portId
以及name
即可,port
的位置会以节点边框的宽度进行平分,达到视觉上的美观。
边
上一节讲到,节点与之间的关系通过连线来表示,那如何配置连线呢?我们通过以下例子进行说明。
可以看到,节点 1 与节点 2 通过连线表明他们之间有某种关系,要实现连线,我们可以如下配置:
{
edges: [
{
source: "node4",
// String,必须,起始点 id
target: "node1",
// String,必须,目标点 id
sourceAnchorIndex: 0,
// 所在outputs的下标
targetAnchorIndex: 0,
// outputs的长度 + 所在inputs的下标
style: {
endArrow: {
path: "M 0,0 L 20,10 L 20,-10 Z",
d: 5,
fill: "#f00",
stroke: "#0f0",
opacity: 0.5,
lineWidth: 3
// ...
}
},
type: "cubic"
}
]
}
flow-graph-g6
内置的连线,我们只需要简单配置即可实现常规的业务需求。
内置连线
如何配置内置连线
线段通用配置
interface EdgeData
extends EdgeConfig {
type: string; // 边类型
source: string; // 起始节点 id
target: string; // 目标节点 id
sourceAnchorIndex?: number; // 起始点port索引
targetAnchorIndex?: number; // 目标点port索引
style?: EdgeStyle; // 线的样式
label?: string; // 线的文本
labelCfg: labelCfg; // 文本配置项
[key
:
string
]:
any; // 可能存在一些自定义的属性值
baseLineCfg?: Partial<baseLineCfg>; // 自定义线段配置
}
自定义线段配置
interface baseLineCfg {
type: 'rect' | 'rectText' | 'rectIcon' | 'icon' | 'dash' | 'statusText' | 'normal'; // 边类型
rectStrokeColor: string; // 矩形border颜色
rectFillColor: string; // 矩形填充色
rectRadius: number; // 默认为10 rect、rectText、rectIcon下生效
rectWidth: number; // 不建议使用
rectHeight: number; // 不建议使用
iconColor: string;
textColor: string;
icon: string; // icon的类
label: string; // 展示文案
showDashed: boolean; // 是否展示虚线
statusFillColor: string; // 状态颜色 type为statusText时生效
}
除了初始化配置,我们还可以手动将两个节点相连,现在您就可以用鼠标将节点 2 与节点 3 相连,动手体验感受更加深刻。
如何自定义边
除了内置的线段 line、Polyline、Quadratic、Cubic、Arc、Loop 之外,我们还可以自定义边,那如何进行边的自定义,我们通过以下例子进行说明
第一步:定义边的关键形状。
G6.registerEdge('hvh', {
draw (cfg, group) {
const startPoint = cfg.startPoint;
const endPoint = cfg.endPoint;
const shape = group.addShape('path', {
attrs: {
stroke: '#333',
path: [
['M', startPoint.x, startPoint.y],
['L', endPoint.x / 2 + (1 / 2) * startPoint.x, startPoint.y], // 三分之一处
['L', endPoint.x / 2 + (1 / 2) * startPoint.x, endPoint.y], // 三分之二处
['L', endPoint.x, endPoint.y],
],
},
// must be assigned in G6 3.3 and later versions. it can be any value you want
name: 'path-shape',
});
return shape;
},
});
第二步:指定边的类型。
{
edges: [
{
id: "edge1",
target: "node2",
source: "node1",
type: "hvh"
},
{
id: "edge2",
target: "node3",
source: "node1",
type: "hvh"
}
],
}
事件系统
与 flow-graph-g6 的交互过程中,所有数据都是通过事件传递的。flow-graph-g6 封装了 11 种事件,每种事件对应着不同的动作,详情见下表:
动作 | 事件 |
---|---|
拖动节点开始 | nodeDragStart |
拖动节点结束 | nodeDragEnd |
鼠标悬浮节点 | nodeHover |
鼠标从节点上移除 | nodeUnHover |
右键单击节点 | nodeRightClick |
选中节点(左键单击节点) | nodeSelect |
节点由选中态变为非选中态 | nodeUnselect |
鼠标悬浮 port | portHover |
创建连线 | edgeCreate |
左键单击连线 | edgeClick |
右键单击连线 | edgeRightClick |
鼠标悬浮连线 | edgeHover |
鼠标离开连线 | edgeUnHover |
点击画布 | canvasClick |
监听事件
flow-graph-g6
通过 graphUtil
导出所有事件的事件名,每一个Flow
实例都会维护一个自己的eventemitter3
实例,因此我们可以使用eventemitter3
的方法来绑定以及解绑事件。我们可以通过flow.bus
获取flow
实例的eventemitter3
实例,所以我们就可以通过flow.bus.on
方法监听事件,通过flow.bus.off
方法解绑事件。
我们通过给节点加一个选中事件的例子进行演示:
上面的例子中,用鼠标左键点击节点,就会弹出一个浏览器原生 alert 框,里面显示点击的节点名称。要配置这样的功能,核心代码只要以下几行:
import { graphUtil } from '@ss/mtdv-graph-flow';
const flow = new Flow({/*...*/ })
const { events } = graphUtil;
flow.bus.on(events.nodeSelect, e => {
alert(`您点击了${e.model.name}`);
});
插件机制
G6 中,如果想要改变节点或边的状态,一种方式是在注册节点的时候声明状态,然后在需要改变状态的地方调用graph.setItemState
方法。flow-graph-g6 中分别对节点和边进行了一些状态的声明,具体如下:
node:
状态 | 描述 |
---|---|
selected | 节点被选择后的状态 |
hover | 节点被 hover 的状态 |
highlight | 节点高亮态,用来区分正常节点 |
failHighlight | 另一种高亮态,颜色与 highlight 有所区分 |
portHighlight | port 高亮状态 |
default | 正常状态 |
edge:
状态 | 描述 |
---|---|
selected | 边被选中后的状态 |
hover | 边被 hover 后的状态 |
unHover | 边取消选中后的状态 |
highlight | 边高亮状态,用来区分正常节点 |
default | 正常状态 |
这些状态主要是改变节点或边的颜色,用来区分业务上不同的类型。我们可通过初始化节点的时候对节点的颜色进行配置;对于边的颜色,目前只支持通过注册复写的形式进行配置。
下面演示如何通过配置区分节点的状态展示:
如上面的例子所示,当我们用鼠标点击节点,会发现节点展示的状态并不一致,节点 1 展示偏绿色,节点 2 展示偏蓝色。
这里我们只需要在配置节点的时候声明不同的颜色即可。
{
nodes: [
// 节点1
{
// ...
color: '#1ea565',
// ...
},
// 节点2
{
// ...
color: '#1239C7',
// ...
}
]
}
但是有时候我们希望做到更加精细的配置,此时就可以通过插件的方式注册或重写状态,这样你就可以随心所欲地定制不同状态的展现形式了。