趁着还没有被降本增笑,先做一个很low的组态系统乐呵乐呵

自从上次发布这篇帖子# 还是没记住,浏览器托拽再总结一年多了,一直觉得拖拽没有什么用处。以前项目偶尔要进行组态相关开发,也只是在开源基础上稍微改了改。

项目的需求是这样的,我需要把左边的按钮拖拽到右边背景图上,生成JSON数据保存到数据库。

这个需求看起来很简单,起初的思路是这样。背景框做成一个相对定位,拖拽的内部元素做成绝对定位。只要改变内部元素的相对位置,那么就完成了这个功能。

这么想想挺开心,关键是怎么拖动。主要涉及三个问题

  1. 把元素从左侧拖动到右侧
  2. 元素可以在右侧区域自由滑动
  3. 需要生成json数据存到数据库

解决第一个问题

第一步:创建需要拖动的元素,这里将dragDom设置为false,主要是防止这个dom元素被拖走。

我这里只创建了一个拖动元素,没有创建太多,只为了说明这个过程

html 复制代码
<how-drag-drop :dragDom="false" :data="{ name: '张三', age: 18 }">
   <div class="drag">HOWUSE</div>
</how-drag-drop>

第二步:创建接收区域,背景图所在区,这里就不放背景图片了,这个很简单也不是重点。

draggable 属性设置为false,确保目标区域不是被拖动的。对应js逻辑参考三、四步骤

html 复制代码
  <div class="target-data-wrapper" ref="boundary">
      <!-- 单向拖动,该元素draggable设置为false -->
      <how-drag-drop :draggable="false" @drop="drop" style="position: relative">
        <template v-for="(postion, index) in postions">
          <UseDraggable :boundary="boundary" :position="[postion.x, postion.y]"
            @update-data="(data) => updataData(index, data)">
            <div class="drag">HOWUSE</div>
          </UseDraggable>
        </template>
      </how-drag-drop>
    </div>

这样我们解决掉了第一问题,实现了元素的单项拖拽。

元素自由滑动

第三步:可拖拽容器组件,上边目标区域已经用到了

js 复制代码
// 可拖拽组件容器
const UseDraggable = defineComponent<{ boundary: any; position: [number, number] }>({
  name: 'UseDraggable',
  props: [
    'position',
    'boundary', // 边界元素
  ] as unknown as undefined,
  setup(props, { slots, emit }) {
    const target = ref() // 当前拖动的位置
    const handle = computed(() => target.value)  // 整体元素

    const data = useDraggable(target, {
      handle,
      initialValue: computed(() => props.position ? { x: props.position[0], y: props.position[1] } : { x: 0, y: 0 }),
      containerElement: computed(() => props.boundary),
    })

    watchEffect(() => emit("update-data", data))

    return () => {
      if (slots.default)
        return h('div', { ref: target, style: `touch-action:none;position: absolute;${unref(data.style)}` }, slots.default(reactive(data)))
    }
  },
})

存取JSON数据

第四步:数据接收部分

js 复制代码
const boundary = ref()
const postions = ref<{ x: number, y: number, data: [] }[]>([]) // 数据保存,实际的json数据保存位置

function drop(ev, data) {
  postions.value.push({ x: ev.offsetX, y: ev.offsetY, data: data.dropData })
}

function updataData(index, data) {
  postions.value[index] = { ...postions.value[index], x: data.x, y: data.y}
}

效果如下:

在线地址,参考示例【拖拽元素到另一个可拖拽区域 】:howuse.netlify.app/#/component...

源码地址:github.com/biancangmin...

相关推荐
计算机学姐1 小时前
基于SpringBoot的小型民营加油站管理系统
java·vue.js·spring boot·后端·mysql·spring·tomcat
sunbyte3 小时前
50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | Expanding Cards (展开式卡片)
javascript·vue.js·ecmascript
重生之后端学习4 小时前
02-前端Web开发(JS+Vue+Ajax)
java·开发语言·前端·javascript·vue.js
黄鹂绿柳7 小时前
Vue+Vite学习笔记
vue.js·笔记·学习
来自星星的坤7 小时前
【Vue 3 + Vue Router 4】如何正确重置路由实例(resetRouter)——避免“VueRouter is not defined”错误
前端·javascript·vue.js
清风细雨_林木木13 小时前
Vue 中生成源码映射文件,配置 map
前端·javascript·vue.js
繁依Fanyi15 小时前
ColorAid —— 一个面向设计师的色盲模拟工具开发记
开发语言·前端·vue.js·编辑器·codebuddy首席试玩官
codelxy15 小时前
vue引用cesium,解决“Not allowed to load local resource”报错
javascript·vue.js
Zww089116 小时前
el-dialog鼠标在遮罩层松开会意外关闭,教程图文并茂
javascript·vue.js·计算机外设
sunbyte16 小时前
50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | 页面布局 与 Vue Router 路由配置
前端·javascript·vue.js·tailwindcss