vue3 + antv/x6 实现拖拽侧边栏节点到画布

前篇:vue3+ts使用antv/x6 + 自定义节点
前篇:vue3+antv x6自定义节点样式

1、创建侧边栏

  • 用antd的menu来做侧边栏
sql 复制代码
npm i --save ant-design-vue@4.x
js 复制代码
//入口文件main.js内
import Antd from 'ant-design-vue';
import App from './App';
import 'ant-design-vue/dist/reset.css';

const app = createApp(App);
app.use(Antd).mount('#app');
  • 侧边栏结构
json 复制代码
// index.js内,拖拽节点侧边栏
export const MENU_TREE = [
    {
        key: '1',
        label: 'type1',
        title: 'drag node 1',
    },
]
  • 画布展示
js 复制代码
//index.vue内
<template>
    <div class="box">
        <a-menu
            class="menutree"
            :items="menutree"
        />
        <div id="container"></div>
    </div>
    <TeleportContainer/>
</template> 
<script setup lang='ts'>
import {MENU_TREE} from "./index";
let menutree = MENU_TREE
</script>
<style scoped>
.menutree{
    display: flex;
    position: relative;
}
#container{
    height: 100vh;
}
.menutree{
    width: 200px;
    height: 100vh;
    position: absolute;
    left: 0;
    top: 0;
    flex-direction: column;
    z-index: 2;
  	user-select: none;/* 鼠标按下时没有复制等操作 */
}
</style>    

2、拖拽交互

  • 安装
sql 复制代码
  npm install @antv/x6-plugin-dnd --save
  • 引入并使用(index.vue)
js 复制代码
import { Dnd } from '@antv/x6-plugin-dnd'
//...
let dnd:Dnd
//开始拖拽
const startDrag = (e,nodevo)=>{
    const node = graph.createNode(formatData(nodevo))
    dnd.start(node, e)
}
//初始化画布
const graphInit = ()=>{
    //...
    dnd = new Dnd({
        target: graph,
        getDragNode: (node) => node.clone({ keepId: true }),//拖拽开始时,获取被拖拽的节点,默认克隆 dnd.start 传入的节点。
        getDropNode: (node) => {//拖拽结束时,获取放置到目标画布的节点,默认克隆被拖拽的节点。
            return graph.createNode(formatData(node.getData()))
        }
    })
}

3、侧边栏和拖拽建立联系

  • 在侧边栏menu中添加拖拽节点信息(index.js)
json 复制代码
MENU_TREE = [{
    key: '1',
    label: 'type1',
    title: 'drag node 1',
    ports:[{
        id: 'port-1',
        name: 'drag1-port1',
    }]
}]
  • fomatMenu处理menu内需要拖拽到节点信息(index.js)
js 复制代码
export function fomatMenu (menu: any, dragStart) {
    let tempmenu = menu.map(ele => {
        const nodeVO = {
            id: `node-${Math.random()}`,
            nodeName: ele.title,
            x: 0,
            y: 0,
            ports: ele.ports
        }
        return {
            key: ele.key,
            label: ele.label,
            onMousedown: (e: Event) => {
                dragStart(e, nodeVO)
            },
        }
    });
    return tempmenu
}
  • 主画布渲染menu时先处理menu信息
js 复制代码
import {MENU_TREE,fomatMenu} from "./index";
let menutree = fomatMenu(MENU_TREE,startDrag) //放在startDrag之后
  • 效果
相关推荐
SuperEugene11 小时前
表单最佳实践:从 v-model 到自定义表单组件(含校验)
前端·javascript·vue.js
昨晚我输给了一辆AE8611 小时前
为什么现在不推荐使用 React.FC 了?
前端·react.js·typescript
不会敲代码111 小时前
深入浅出 React 闭包陷阱:从现象到原理
前端·react.js
不会敲代码111 小时前
React性能优化:深入理解useMemo和useCallback
前端·javascript·react.js
Dilettante25811 小时前
我的 Monorepo 实践经验:从基础概念到最佳实践
前端·前端工程化
只会cv的前端攻城狮11 小时前
Elpis-Core — 融合 Koa 洋葱圈模型实现服务端引擎
前端·后端
Java小卷12 小时前
流程设计器为啥选择diagram-js
前端·低代码·工作流引擎
HelloReader12 小时前
Isolation Pattern(隔离模式)在前端与 Core 之间加一道“加密网关”,拦截与校验所有 IPC
前端
兆子龙13 小时前
从 float 到 Flex/Grid:CSS 左右布局简史与「刁钻」布局怎么搞
前端·架构
YukiMori2313 小时前
一个有趣的原型继承实验:为什么“男人也会生孩子”?从对象赋值到构造函数继承的完整推演
前端·javascript