vue2-x6-dag自定义vue组件节点

效果如图

在这里插入图片描述

官方案例

人工智能建模 DAG 图
vue2中自定义节点

代码

1.dag.json

[{"id": "1","shape": "dag-node","x": 290,"y": 110,"data": {"label": "读数据","status": "success"},"ports": [{"id": "1-1","group": "bottom"}]},{"id": "2","shape": "dag-node","x": 290,"y": 225,"data": {"label": "逻辑回归","status": "success"},"ports": [{"id": "2-1","group": "top"},{"id": "2-2","group": "bottom"},{"id": "2-3","group": "bottom"}]},{"id": "3","shape": "dag-node","x": 170,"y": 350,"data": {"label": "模型预测","status": "success"},"ports": [{"id": "3-1","group": "top"},{"id": "3-2","group": "bottom"}]},{"id": "4","shape": "dag-node","x": 450,"y": 350,"data": {"label": "读取参数","status": "success"},"ports": [{"id": "4-1","group": "top"},{"id": "4-2","group": "bottom"}]},{"id": "5","shape": "dag-edge","source": {"cell": "1","port": "1-1"},"target": {"cell": "2","port": "2-1"},"zIndex": 0},{"id": "6","shape": "dag-edge","source": {"cell": "2","port": "2-2"},"target": {"cell": "3","port": "3-1"},"zIndex": 0},{"id": "7","shape": "dag-edge","source": {"cell": "2","port": "2-3"},"target": {"cell": "4","port": "4-1"},"zIndex": 0}]

2.index.html

<html>
<head><meta charset="utf-8" /><meta name="viewport" content="width=device-width, initial-scale=1" /><title>vue2-x6-DAG自定义vue组件节点</title><script type="text/javascript" src="https://cdn.jsdelivr.net/npm/vue@2.6.14/dist/vue.js"></script><script type="text/javascript" src="https://unpkg.com/@antv/x6@1.34.14/dist/x6.js"></script><script type="text/javascript" src="https://unpkg.com/@antv/x6-vue-shape@1.1.4/dist/x6-vue-shape.js"></script><style>.node {display: flex;align-items: center;width: 100%;height: 100%;background-color: #fff;border: 1px solid #c2c8d5;border-left: 4px solid #5F95FF;border-radius: 4px;box-shadow: 0 2px 5px 1px rgba(0, 0, 0, 0.06);}.node img {width: 20px;height: 20px;flex-shrink: 0;margin-left: 8px;}.node .label {display: inline-block;flex-shrink: 0;width: 104px;margin-left: 8px;color: #666;font-size: 12px;}.node .status {flex-shrink: 0;}.node.success {border-left: 4px solid #52c41a;}.node.failed {border-left: 4px solid #ff4d4f;}.node.running .status img {animation: spin 1s linear infinite;}.x6-node-selected .node {border-color: #1890ff;border-radius: 2px;box-shadow: 0 0 0 4px #d4e8fe;}.x6-node-selected .node.success {border-color: #52c41a;border-radius: 2px;box-shadow: 0 0 0 4px #ccecc0;}.x6-node-selected .node.failed {border-color: #ff4d4f;border-radius: 2px;box-shadow: 0 0 0 4px #fedcdc;}.x6-edge:hover path:nth-child(2) {stroke: #1890ff;stroke-width: 1px;}.x6-edge-selected path:nth-child(2) {stroke: #1890ff;stroke-width: 1.5px !important;}@keyframes running-line {to {stroke-dashoffset: -1000;}}@keyframes spin {from {transform: rotate(0deg);}to {transform: rotate(360deg);}}</style>
</head><body><div id="workflow"><h5>注意事项:</h5><h6>1.必须启动前端服务访问,使用hbuilderx运行内置浏览器或者vscode安装Live Server插件,打开文件右键,选择“open with five server”即可,否则跨域!!!</h6><h6>2.cdn或本地引入方式x6-vue-shape版本必须和x6同一个大版本!!!</h6><button @click="add">外部点击Add第一个元素num</button><div id="container"></div></div>
</body></html>
<script>// 自定义vue组件节点const nodeBox = Vue.component('node-box', {template: `<div :class="['node', status]"  @click="nodeDrag()"><img :src="image.logo" /><span class="label">{{label}}内部Add:{{ num }}</span><span class="status"><img v-if="status === 'success'" :src="image.success" /><img v-if="status === 'failed'" :src="image.failed" /><img v-if="status === 'running'" :src="image.running" /></span></div>`,inject: ["getGraph", "getNode"],data() {return {// 自定义测试数字num: 0,// 自定义节点名称label: '',// 自定义节点状态status: '',// 状态数据iconimage: {logo: 'https://gw.alipayobjects.com/mdn/rms_43231b/afts/img/A*evDjT5vjkX0AAAAAAAAAAAAAARQnAQ',success:'https://gw.alipayobjects.com/mdn/rms_43231b/afts/img/A*6l60T6h8TTQAAAAAAAAAAAAAARQnAQ',failed:'https://gw.alipayobjects.com/mdn/rms_43231b/afts/img/A*SEISQ6My-HoAAAAAAAAAAAAAARQnAQ',running:'https://gw.alipayobjects.com/mdn/rms_43231b/afts/img/A*t8fURKfgSOgAAAAAAAAAAAAAARQnAQ',}}},mounted() {const node = this.getNode();// 监听数据改变事件,通过外部控制节点数据node.on("change:data", ({ current }) => {this.num = current.num;this.label = current.label;this.status = current.status;});},methods: {// 内部控制节点数据,节点拖动后触发,拿到x,y数据回传后端nodeDrag() {const node = this.getNode();const { num } = node.getData();node.setData({num: num + 1,});console.log('节点坐标:', node.position())},},})// 父页面const workflow = new Vue({el: '#workflow',data() {return {graph: null,};},mounted() {// 使用 CDN 引入时暴露了 X6 全局变量const { Graph, Path } = X6// 自定义vue节点Graph.registerNode("dag-node", {inherit: "vue-shape",x: 200,y: 150,width: 180,height: 36,component: {template: `<node-box />`,components: {nodeBox,},},ports: {groups: {top: {position: 'top',attrs: {circle: {r: 4,magnet: true,stroke: '#C2C8D5',strokeWidth: 1,fill: '#fff',},},},bottom: {position: 'bottom',attrs: {circle: {r: 4,magnet: true,stroke: '#C2C8D5',strokeWidth: 1,fill: '#fff',},},},},},});// 注册连接线Graph.registerEdge('dag-edge',{inherit: 'edge',attrs: {line: {stroke: '#C2C8D5',strokeWidth: 1,targetMarker: null,},},},true,)// 注册连接线弧度Graph.registerConnector('algo-connector',(s, e) => {const offset = 4const deltaY = Math.abs(e.y - s.y)const control = Math.floor((deltaY / 3) * 2)const v1 = { x: s.x, y: s.y + offset + control }const v2 = { x: e.x, y: e.y - offset - control }return Path.normalize(`M ${s.x} ${s.y}L ${s.x} ${s.y + offset}C ${v1.x} ${v1.y} ${v2.x} ${v2.y} ${e.x} ${e.y - offset}L ${e.x} ${e.y}`,)},true,)// 定义画布const graph = new Graph({container: document.getElementById('container'),// 对齐线snapline: true,width: 800,height: 600,panning: {enabled: true,eventTypes: ['leftMouseDown', 'mouseWheel'],},mousewheel: {enabled: true,modifiers: 'ctrl',factor: 1.1,maxScale: 1.5,minScale: 0.5,},highlighting: {magnetAdsorbed: {name: 'stroke',args: {attrs: {fill: '#fff',stroke: '#31d0c6',strokeWidth: 4,},},},},connecting: {snap: true,allowBlank: false,allowLoop: false,highlight: true,connector: 'algo-connector',connectionPoint: 'anchor',anchor: 'center',validateMagnet({ magnet }) {return magnet.getAttribute('port-group') !== 'top'},createEdge() {return graph.createEdge({shape: 'dag-edge',attrs: {line: {strokeDasharray: '5 5',},},zIndex: -1,})},},selecting: {enabled: true,multiple: true,rubberEdge: true,rubberNode: true,modifiers: 'shift',rubberband: true,},});this.graph = graph// 连接线,连接完成事件graph.on('edge:connected', ({ edge }) => {edge.attr({line: {strokeDasharray: '',},})})// 监听子节点数据改变graph.on('node:change:data', ({ node }) => {const edges = graph.getIncomingEdges(node)const { status } = node.getData()edges?.forEach((edge) => {if (status === 'running') {edge.attr('line/strokeDasharray', 5)edge.attr('line/style/animation', 'running-line 30s infinite linear')} else {edge.attr('line/strokeDasharray', '')edge.attr('line/style/animation', '')}})})// 数据状态const nodeStatusList = [[{id: '1',status: 'running',},{id: '2',status: 'default',},{id: '3',status: 'default',},{id: '4',status: 'default',},],[{id: '1',status: 'success',},{id: '2',status: 'running',},{id: '3',status: 'default',},{id: '4',status: 'default',},],[{id: '1',status: 'success',},{id: '2',status: 'success',},{id: '3',status: 'running',},{id: '4',status: 'running',},],[{id: '1',status: 'success',},{id: '2',status: 'success',},{id: '3',status: 'success',},{id: '4',status: 'failed',},],]// 初始化节点/边const init = (data) => {const cells = []data.forEach((item) => {if (item.shape === 'dag-node') {cells.push(graph.createNode(item))} else {cells.push(graph.createEdge(item))}})graph.resetCells(cells)}// 显示节点状态const showNodeStatus = async (statusList) => {const status = statusList.shift()status?.forEach((item) => {const { id, status } = itemconst node = graph.getCellById(id)const data = node.getData()node.setData({...data,status: status,num: 0,})})setTimeout(() => {showNodeStatus(statusList)}, 3000)}// 模拟数据fetch('./dag.json').then((response) => response.json()).then((data) => {// 加载节点init(data)// 节点状态showNodeStatus(nodeStatusList)// 将画布内容中心与视口中心对齐graph.centerContent()})},methods: {// 控制子组件内部变量add() {const nodes = this.graph.getNodes();console.log(nodes)if (nodes.length) {nodes.forEach((node) => {// 1 为该组件id,正式环境根据传参id来控制具体某个元素内部if (node.id == 1) {const { num } = node.getData();node.setData({num: num + 1,});}});}},}});
</script>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/111950.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

爬虫 — 验证码反爬

目录 一、超级鹰二、图片验证模拟登录1、页面分析1.1、模拟用户正常登录流程1.2、识别图片里面的文字 2、代码实现 三、滑块模拟登录1、页面分析2、代码实现&#xff08;通过对比像素获取缺口位置&#xff09; 四、openCV1、简介2、代码3、案例 五、selenium 反爬六、百度智能云…

力扣刷题19-删除链表的倒数第N个节点

题目来源 题目描述&#xff1a; class Solution {public ListNode removeNthFromEnd(ListNode head, int n) {//为了删除的格式一样&#xff0c;引入虚拟头节点ListNode dummyNodenew ListNode(1);dummyNode.nexthead;ListNode slowdummyNode;ListNode fastdummyNode;for(int…

【Linux网络编程】Socket-TCP实例

该代码利用socket套接字建立Tcp连接&#xff0c;包含服务器和客户端。当服务器和客户端启动时需要把端口号或ip地址以命令行参数的形式传入。服务器启动如果接受到客户端发来的请求连接&#xff0c;accept函数会返回一个打开的socket文件描述符&#xff0c;区别于监听连接的lis…

Stable Diffusion - 采样器 DPM++ 3M SDE Karras 与 SDXL Refiner 测试

欢迎关注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://spike.blog.csdn.net/article/details/132978866 Paper: DPM-Solver: Fast Solver for Guided Sampling of Diffusion Probabilistic Models 扩散概率模型&#xff08;DPMs&#xff09;…

Linux集群时间同步方法

参考&#xff1a;https://www.cnblogs.com/felixzh/p/10638399.html

Qt(day1)

思维导图 Qt实现第一个web的基础页面 #include "mywnd.h" #include "ui_mywnd.h" #include<QLabel> #include<QLineEdit> #include<qpushbutton.h>MyWnd::MyWnd(QWidget *parent): QMainWindow(parent), ui(new Ui::MyWnd) {ui->set…

C语言的文件操作(炒详解)

⭐回顾回顾文件操作的相关细节⭐ 欢迎大家指正错误 &#x1f4dd;在之前的学习中&#xff0c;不管增加数据&#xff0c;减少数据&#xff0c;当程序退出时&#xff0c;所有的数据都会销毁&#xff0c;等下次运行程序时&#xff0c;又要重新输入相关数据&#xff0c;如果一直像这…

现在做抖音小店还来得及吗?新手正确入局抖店的流程,醒醒团队

我是王路飞。 现在很多入局抖音的&#xff0c;都意识到了一件事&#xff0c;那就是做店要比做账号容易。 做账号&#xff0c;意味着你选择了成为抖音网红-卖货的路线&#xff0c;看似光鲜亮丽&#xff0c;实则都是团队内容运气等多重因素结合得到的结果。 但是&#xff0c;做…

mysql MVCC多版本并发控制

mvcc的概念 mvcc 的实现依赖于&#xff1a; 隐藏字段 行格式&#xff08;row_id,trx_id,roll_ponter&#xff09;UndologRead view innodb 存储引擎的表来说&#xff0c;聚集索引记录中都包含两个必要的隐藏字段&#xff0c;row_id(如果没有聚集索引&#xff0c;才会创建的) …

【JDK 8-函数式编程】4.6 方法引用与构造函数引用

一、 方法引用与构造函数引用 1. 说明 2. 语法: 二、静态方法 三、实例方法 四、构造函数 4.1 单个参数 4.2 2个参数 五、执行结果 一、 方法引用与构造函数引用 以前调用&#xff1a;对象.方法名、类名.方法名 jdk1.8提供了另外一种调用方式 :: 1. 说明 用来直接访…

Jmeter —— 常用的几种断言方法(基本用法)

在使用JMeter进行性能测试或者接口自动化测试工作中&#xff0c;经常会用到的一个功能&#xff0c;就是断言&#xff0c;断言相当于检查点&#xff0c;它是用来判断系统返回的响应结果是否正确&#xff0c;以此帮我们判断测试是否通过&#xff0c;本文 主要介绍几种常用的断言&…

数据结构上机练习——单链表的基本操作、头文件、类定义、main函数、多种链表算法的实现,含注释

文章目录 单链表的基本操作实现1.头文件2.类定义和多种算法的实现2.1创建空表2.2头插法创建n个元素的线性链表2.3一个带头节点的链表存放一组整数&#xff0c;设计一个算法删除值等于x的所有节点。2.4计算线性表中值为偶数的节点个数2.5一个带头节点的单链表heada存放一组整数&…