Hi~ 大家好,我是小鑫同学,一位长期从事前端开发的编程爱好者,我将使用更为实用的案例输出更多的编程知识,同时我信奉分享是成长的唯一捷径,在这里也希望我的每一篇文章都能成为你技术落地的参考~
使用
pnpm create vite
创建项目,选择Vue
框架及TypeScript
变体;
在完成了前面的准备工作后,需要安装非核心的插件依赖模块:
pnpm add @logicflow/core
pnpm add @logicflow/extension
import { BpmnElement } from '@logicflow/extension';
LogicFlow.use(BpmnElement);
import LogicFlow from "@logicflow/core";
import { DndPanel, SelectionSelect } from "@logicflow/extension";
import "@logicflow/core/dist/style/index.css";
import "@logicflow/extension/lib/style/index.css";
const lf = new LogicFlow({
container: document.querySelector("#app"),
grid: true,
plugins: [DndPanel, SelectionSelect]
});
拖拽面板的插件安装完成后就可以为拖拽面板添加元素了,也就是被拖拽的内容,每个元素都具备以下7
个可选选项进行控制:
名称 | 类型 | 描述 |
---|---|---|
type | string | 指定用户鼠标可以拖入的节点类型,包括自定义节点类型, 不传则不会创建节点,只会触发callback |
text | string | 创建节点的文本 |
properties | object | 创建节点的properties |
label | string | 拖拽面板节点文本描述 |
icon | string | 拖拽面试上显示的图标,可以传入图标url地址或base64编码 |
className | string | 额外传入可以拖拽项的class, 用于自定义拖拽项的样式 |
callback | fn | 用户鼠标按下拖拽项后触发的回调 |
<script setup lang="ts">
import LogicFlow from "@logicflow/core";
import { DndPanel } from "@logicflow/extension";
import "@logicflow/core/dist/style/index.css";
import "@logicflow/extension/lib/style/index.css";
import { onMounted, ref } from "vue";
const container = ref();
const lf = ref<LogicFlow>();
const patternItems = [
{
type: 'circle',
text: '开始',
label: '开始节点',
icon: '',
},
{
type: 'rect',
label: '用户任务',
icon: '',
},
{
type: 'circle',
text: '结束',
label: '结束节点',
icon: '',
}
]
onMounted(() => {
lf.value = new LogicFlow({
container: container.value,
grid: true,
plugins: [DndPanel]
})
lf.value.extension.dndPanel.setPatternItems(patternItems);
lf.value.render();
})
</script>
框选和拖动画布两个功能不能同时支持,stopMoveGraph开启时(禁止拖动画布)则支持框选,反之失效;
默认配置:
new LogicFlow({stopMoveGraph: true});
开启方式:
lf.extension.selectionSelect.openSelectionSelect();
关闭方式:
lf.extension.selectionSelect.closeSelectionSelect();
使用setSelectionSense
函数调整:
lf.extension.selectionSelect.setSelectionSense(false, true);
参数 | 默认值 | 描述 |
---|---|---|
isWholeEdge | true | 是否要边的起点终点都在选区范围才算选中 |
isWholeNode | true | 是否要节点的全部点都在选区范围才算选中 |
框选功能和拖拽画布虽然不能同时支持,但可以在使用时灵活控制启用和禁用来切换;
通过在Dndpanel
示例的patternItems
数据中新增下面的配置,通过点击元素后触发的callback
事件启用框选,并监听选中事件后关闭框选来完成功能的切换;
const patternItems = [{
label: '选区',
icon: '',
callback: () => {
lf.value!.extension.selectionSelect.openSelectionSelect();
lf.value!.once('selection:selected', () => {
lf.value!.extension.selectionSelect.closeSelectionSelect();
});
}
}]
LF渲染下面的graphData
数据将得到一个默认的分组区域,它很像一个rect节点;
const graphData = {
nodes: [
{
type: 'group',
x: 300,
y: 300
}
]
}
group
节点通过特有的children
属性可以管理其该包含的子节点;
const graphData = {
nodes: [
{
type: 'group',
x: 300,
y: 300,
children: ["node_2"]
},
{
id: 'node_2',
type: 'circle',
x: 300,
y: 300
}
]
}
准备自定义分组模板:
import {GroupNode} from "@logicflow/extension";
class MyGroup extends GroupNode.view { }
class MyGroupModel extends GroupNode.model { }
export default {
type: "my-group",
model: MyGroupModel,
view: MyGroup
};
优先注册&使用:
<script setup lang="ts">
import LogicFlow from "@logicflow/core";
import { Group } from "@logicflow/extension";
import "@logicflow/core/dist/style/index.css";
import "@logicflow/extension/lib/style/index.css";
// 导入模块
import MyGroup from "./MyGroup";
import { onMounted, ref } from "vue";
const container = ref();
const lf = ref<LogicFlow>();
const graphData = {
nodes: [
{
type: 'my-group',
x: 300,
y: 300,
children: ["node_2"]
},
{
id: 'node_2',
type: 'circle',
x: 300,
y: 300
}
]
}
onMounted(() => {
lf.value = new LogicFlow({
container: container.value,
grid: true,
plugins: [Group],
})
// 注册自定义分组
lf.value.register(MyGroup);
lf.value.render(graphData);
})
</script>
自定义样式,重写getNodeStyle
函数:
getNodeStyle() {
const style = super.getNodeStyle();
style.stroke = "#AEAFAE";
style.strokeWidth = 1;
return style;
}
自定义形状,重写initNodeData
函数:
initNodeData(data: any) {
super.initNodeData(data);
this.isRestrict = true;
this.resizable = true;
this.width = 480;
this.height = 280;
}
groupModel除节点通用的属性外,还有下面这些状态属性的支持,建议在
setAttributes
函数配置;
setAttributes(): void {
this.isRestrict = true;
this.resizable = true;
this.foldable = true;
}
名称 | 类型 | 描述 |
---|---|---|
isRestrict | boolean | 是否限制分组子节点拖出分组,默认false |
resizable | boolean | 分组是否支持手动调整大小,默认false |
foldable | boolean | 分组是否显示展开收起按钮,默认false |
width | number | 分组宽度 |
height | number | 分组高度 |
foldedWidth | number | 分组折叠后的宽度 |
foldedHeight | number | 分组折叠后的高度 |
isFolded | boolean | 只读,表示分组是否被折叠。 |
isGroup | boolean | 只读,永远为true, 用于识别model为group。 |
addChild:
将某个节点设置为分组的子节点。注意,此方法只会添加关系,不会自动将节点移动到分组里面。
const groupModel = lf.getNodeModelById('group_id');
const node = lf.addNode({
type: 'rect',
x: groupModel.x,
y: groupModel.y
});
groupModel.addChild(node.id);
removeChild:
从分组中移除某个子节点。
const groupModel = lf.getNodeModelById('group_id');
groupModel.removeChild('node_id_1');
foldGroup:
收起分组, 参数为true表示收起分组、false表示展开分组
const groupModel = lf.getNodeModelById('group_id');
groupModel.foldGroup(true);
isAllowAppendIn(nodeData):
校验是否允许传入节点添加到此分组中,默认所有的节点都可以。
class MyGroupModel extends GroupNode.model {
isAllowAppendIn(nodeData: any): boolean {
// 设置只允许circle节点被添加到此分组中
return nodeData.type === 'circle'
}
}
getAddableOutlineStyle:
设置拖动节点到分组上时,分组高亮的提示效果样式。
class MyGroupModel extends GroupNode.model {
getAddableOutlineStyle() {
const style = super.getAddableOutlineStyle();
style.stroke = '#AEAFAE';
style.strokeDasharray = '3 3';
return style;
}
}
如果看完觉得有收获,欢迎点赞、评论、分享支持一下。你的支持和肯定,是我坚持写作的动力~