vue编写一个可拖动的模块,并可以和任何其他组件组合使用

实现思路

  1. 使用 Vue 的自定义指令(directive)来处理拖动逻辑。
  2. mounted 钩子中添加鼠标事件监听器,以实现拖动功能。
  3. unmounted 钩子中移除鼠标事件监听器,防止内存泄漏。

代码示例

<template>
  <div v-draggable class="draggable-component">
    <slot></slot>
  </div>
</template>

<script>
export default {
  name: 'DraggableComponent',
  directives: {
    draggable: {
      // 当绑定元素插入到 DOM 中时
      mounted(el) {
        let isDragging = false;
        let initialX, initialY;
        let offsetX = 0, offsetY = 0;

        const handleMouseDown = (e) => {
          isDragging = true;
          initialX = e.clientX - offsetX;
          initialY = e.clientY - offsetY;
        };

        const handleMouseMove = (e) => {
          if (isDragging) {
            offsetX = e.clientX - initialX;
            offsetY = e.clientY - initialY;
            el.style.transform = `translate3d(${offsetX}px, ${offsetY}px, 0)`;
          }
        };

        const handleMouseUp = () => {
          isDragging = false;
        };

        el.addEventListener('mousedown', handleMouseDown);
        document.addEventListener('mousemove', handleMouseMove);
        document.addEventListener('mouseup', handleMouseUp);

        // 在元素销毁时移除事件监听器
        el._removeEventListeners = () => {
          el.removeEventListener('mousedown', handleMouseDown);
          document.removeEventListener('mousemove', handleMouseMove);
          document.removeEventListener('mouseup', handleMouseUp);
        };
      },
      unmounted(el) {
        if (el._removeEventListeners) {
          el._removeEventListeners();
        }
      }
    }
  }
};
</script>

<style scoped>
.draggable-component {
  position: absolute;
  background-color: #f0f0f0;
  border: 1px solid #ccc;
  padding: 10px;
  cursor: move;
}
</style>

代码解释

  • 模板部分

    • <div v-draggable class="draggable-component">:使用自定义指令 v-draggable 使该 div 元素具有可拖动的功能。
    • <slot></slot>:使用插槽,允许其他组件或内容插入到这个可拖动的 div 中,从而实现与其他组件的组合使用。
  • 脚本部分

    • directives:定义了一个名为 draggable 的自定义指令。
      • mounted 钩子:
        • isDragging:用于标记是否正在拖动。
        • initialXinitialY:记录鼠标按下时的初始位置。
        • offsetXoffsetY:存储元素相对于初始位置的偏移量。
        • handleMouseDown:当鼠标按下时触发,标记为正在拖动,并记录初始位置。
        • handleMouseMove:当鼠标移动且正在拖动时触发,计算偏移量并更新元素的 transform 属性,使用 translate3d 进行元素的平移。
        • handleMouseUp:当鼠标松开时触发,标记为停止拖动。
        • 添加鼠标事件监听器:为元素添加 mousedown 事件监听器,为文档添加 mousemovemouseup 事件监听器。
        • el._removeEventListeners:存储一个函数,用于移除事件监听器。
      • unmounted 钩子:在元素销毁时调用 el._removeEventListeners 移除事件监听器,防止内存泄漏。

使用示例

<template>
  <div id="app">
    <DraggableComponent>
      <h1>可拖动的组件</h1>
      <p>这是一个可以拖动的模块,并且可以和任何其他组件组合使用。</p>
      <AnotherComponent></AnotherComponent>
    </DraggableComponent>
  </div>
</template>

<script>
import DraggableComponent from './DraggableComponent.vue';
import AnotherComponent from './AnotherComponent.vue';

export default {
  name: 'App',
  components: {
    DraggableComponent,
    AnotherComponent
  }
};
</script>

解释

  • 首先,将 DraggableComponent 组件导入并注册。
  • 在模板中使用 DraggableComponent 组件,并在其内部使用插槽插入了一个 h1 元素、一个 p 元素和 AnotherComponent 组件,这样就实现了一个可拖动的组件,并且可以方便地与其他组件组合使用。

注意事项

  • 确保在 unmounted 钩子中移除事件监听器,以避免内存泄漏。
  • 该示例只是一个简单的实现,你可以根据需求扩展功能,例如限制拖动范围、添加拖动手柄等。

通过上述方法,你可以在 Vue 中实现一个可拖动的模块,并将其与任何其他组件进行组合使用。如果你需要更复杂的拖动功能,例如与其他组件进行交互或支持多个可拖动元素,可以考虑使用第三方库,如 vue-draggablevue-draggable-resizable。这些库提供了更丰富的功能和更强大的拖动、调整大小等操作。

还有其他方法实现可拖动模块吗?

另一种在 Vue 中实现可拖动模块的方法,使用 Vue 的 ref 和组件方法来处理拖动逻辑:

实现思路

  1. 使用 ref 获取元素引用。
  2. 在组件的 mounted 生命周期中添加鼠标事件监听器。
  3. 在鼠标按下、移动和松开时更新元素的位置。

代码示例

<template>
  <div ref="draggable" class="draggable-component">
    <slot></slot>
  </div>
</template>

<script>
export default {
  name: 'DraggableComponent',
  data() {
    return {
      isDragging: false,
      startX: 0,
      startY: 0,
      offsetX: 0,
      offsetY: 0
    };
  },
  mounted() {
    const draggable = this.$refs.draggable;
    const handleMouseDown = (e) => {
      this.isDragging = true;
      this.startX = e.clientX - this.offsetX;
      this.startY = e.clientY - this.offsetY;
    };
    const handleMouseMove = (e) => {
      if (this.isDragging) {
        this.offsetX = e.clientX - this.startX;
        this.offsetY = e.clientY - this.startY;
        draggable.style.transform = `translate3d(${this.offsetX}px, ${this.offsetY}px, 0)`;
      }
    };
    const handleMouseUp = () => {
      this.isDragging = false;
    };
    draggable.addEventListener('mousedown', handleMouseDown);
    document.addEventListener('mousemove', handleMouseMove);
    document.addEventListener('mouseup', handleMouseUp);
    // 在组件销毁时移除事件监听器
    this.$once('hook:beforeDestroy', () => {
      draggable.removeEventListener('mousedown', handleMouseDown);
      document.removeEventListener('mousemove', handleMouseMove);
      document.removeEventListener('mouseup', handleMouseUp);
    });
  }
};
</script>

<style scoped>
.draggable-component {
  position: absolute;
  background-color: #f0f0f0;
  border: 1px solid #ccc;
  padding: 10px;
  cursor: move;
}
</style>

代码解释

  • 模板部分

    • <div ref="draggable" class="draggable-component">:使用 ref 属性给 div 元素一个引用,方便在组件内部访问。
    • <slot></slot>:使用插槽,允许插入其他组件或内容。
  • 脚本部分

    • data 中存储拖动所需的状态:
      • isDragging:标记是否正在拖动。
      • startXstartY:鼠标按下时的初始位置。
      • offsetXoffsetY:元素相对于初始位置的偏移量。
    • mounted 生命周期钩子:
      • 通过 this.$refs.draggable 获取元素引用。
      • handleMouseDown:鼠标按下时记录初始位置并标记为正在拖动。
      • handleMouseMove:鼠标移动时计算偏移量并更新元素的 transform 属性,使用 translate3d 实现平移。
      • handleMouseUp:鼠标松开时标记为停止拖动。
      • 添加事件监听器:为可拖动元素添加 mousedown 事件监听器,为文档添加 mousemovemouseup 事件监听器。
      • this.$once('hook:beforeDestroy',...):在组件销毁前移除事件监听器,防止内存泄漏。

使用示例

<template>
  <div id="app">
    <DraggableComponent>
      <h1>可拖动的组件</h1>
      <p>这是另一种实现可拖动模块的方式,可以与其他组件组合使用。</p>
      <AnotherComponent></AnotherComponent>
    </DraggableComponent>
  </div>
</template>

<script>
import DraggableComponent from './DraggableComponent.vue';
import AnotherComponent from './AnotherComponent.vue';

export default {
  name: 'App',
  components: {
    DraggableComponent,
    AnotherComponent
  }
};
</script>

第三种方法:使用 Vue 3 的 Composition API 和 onMountedonUnmounted
如果你使用 Vue 3,还可以使用 Composition API 来实现:

<template>
  <div ref="draggable" class="draggable-component">
    <slot></slot>
  </div>
</template>

<script>
import { ref, onMounted, onUnmounted } from 'vue';

export default {
  name: 'DraggableComponent',
  setup() {
    const draggable = ref(null);
    let isDragging = ref(false);
    let startX = ref(0);
    let startY = ref(0);
    let offsetX = ref(0);
    let offsetY = ref(0);

    const handleMouseDown = (e) => {
      isDragging.value = true;
      startX.value = e.clientX - offsetX.value;
      startY.value = e.clientY - offsetY.value;
    };

    const handleMouseMove = (e) => {
      if (isDragging.value) {
        offsetX.value = e.clientX - startX.value;
        offsetY.value = e.clientY - startY.value;
        draggable.value.style.transform = `translate3d(${offsetX.value}px, ${offsetY.value}px, 0)`;
      }
    };

    const handleMouseUp = () => {
      isDragging.value = false;
    };

    onMounted(() => {
      if (draggable.value) {
        draggable.value.addEventListener('mousedown', handleMouseDown);
        document.addEventListener('mousemove', handleMouseMove);
        document.addEventListener('mouseup', handleMouseUp);
      }
    });

    onUnmounted(() => {
      if (draggable.value) {
        draggable.value.removeEventListener('mousedown', handleMouseDown);
        document.removeEventListener('mousemove', handleMouseMove);
        document.removeEventListener('mouseup', handleMouseUp);
      }
    });

    return {
      draggable
    };
  }
};
</script>

<style scoped>
.draggable-component {
  position: absolute;
  background-color: #f0f0f0;
  border: 1px solid #ccc;
  padding: 10px;
  cursor: move;
}
</style>

代码解释

  • 模板部分
    • 与之前类似,使用 ref 获取元素引用。
  • 脚本部分
    • 使用 ref 来创建响应式变量:draggable 用于存储元素引用,isDraggingstartXstartYoffsetXoffsetY 存储拖动状态。
    • handleMouseDownhandleMouseMovehandleMouseUp 函数处理鼠标事件。
    • onMounted 钩子添加事件监听器。
    • onUnmounted 钩子移除事件监听器。

这些方法都可以实现可拖动模块,你可以根据自己的喜好和项目使用的 Vue 版本选择合适的实现方式。如果你需要更复杂的拖动功能,例如拖动排序、拖动边界限制、与其他组件交互等,可能需要进一步扩展上述代码或使用第三方库,如 vue-draggablevue-draggable-resizable

你可以将上述代码复制到相应的 Vue 文件中进行测试和使用,根据自己的需求进行修改和扩展。如果还有其他需求,例如限制拖动范围或添加更多的交互功能,可以进一步细化上述代码或向我提出更具体的问题。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

5 分钟复刻你的声音,一键实现 GPT-Sovits 模型部署

想象一下&#xff0c;只需简单几步操作&#xff0c;就能生成逼真的语音效果&#xff0c;无论是为客户服务还是为游戏角色配音&#xff0c;都能轻松实现。GPT-Sovits 模型&#xff0c;其高效的语音生成能力为实现自然、流畅的语音交互提供了强有力的技术支持。本文将详细介绍如何…

自动化办公|xlwings简介

xlwings 是一个开源的 Python 库&#xff0c;旨在实现 Python 与 Microsoft Excel 的无缝集成。它允许用户使用 Python 脚本自动化 Excel 操作&#xff0c;读取和写入数据&#xff0c;执行宏&#xff0c;甚至调用 VBA 脚本。这使得数据分析、报告生成和其他与 Excel 相关的任务…

JTAG调试器的使用方法

概述: JTAG(联合测试行动组)是基于IEEE 1149.1标准的专用硬件接口。该接口旨在将复杂的芯片和设备连接到标准测试和调试硬件。目前JTAG主要用于: Output control of microcircuits; Testing of printed circuit boards; Flashing of microchips with memory; Chip softwa…

第三次作业:源码编译安装httpd 2.4,提供系统服务管理脚本并测试(两种方法实现)

准备工作 下载httpd软件包进行源码安装 [rootrocky ~]# wget https://archive.apache.org/dist/httpd/httpd-2.4.46.tar.gz 由于进行源码安装&#xff0c;需要对安装对应的编译工具 [rootopen2 ~]# yum install gcc gcc-c make -y 解压文件到指定的目录/usr/local/src下 [root…

Web自动化之验证码识别彻底解决方案

验证码识别解决方案 对于web应用程序来讲&#xff0c;处于安全性考虑&#xff0c;在登录的时候&#xff0c;都会设置验证码&#xff0c;验证码的类型种类繁多&#xff0c;有图片中辨别数字字母的&#xff0c;有点击图片中指定的文字的&#xff0c;也有算术计算结果的&#xff…

第34天:Web开发-PHP应用鉴别修复AI算法流量检测PHP.INI通用过滤内置函数

#知识点 1、安全开发-原生PHP-PHP.INI安全 2、安全开发-原生PHP-全局文件&单函数 3、安全开发-原生PHP-流量检测&AI算法 一、通用-PHP.INI设置 参考&#xff1a; https://www.yisu.com/ask/28100386.html https://blog.csdn.net/u014265398/article/details/109700309 …

C++多态的认识与理解

多态的定义 多态其实就是同一操作在不同的对象上可以有不同的实现方式。 多态的类型 多态分为静态多态和动态多态两种&#xff0c;而静态多态其实我们之前就了解过&#xff0c;今天主要是讲解一下动态多态。 静态多态&#xff08;编译时多态&#xff09;:静态多态其实就是在…

Ubuntu安装搜狗输入法

Ubuntu安装搜狗输入法 官网完整安装步骤调整默认输入法 官网 搜狗输入法 - 官网&#xff1a;https://shurufa.sogou.com/ 搜狗输入法Linux版 - 首页&#xff1a;https://shurufa.sogou.com/linux 搜狗输入法Linux版 - 安装指南&#xff1a;https://shurufa.sogou.com/linux/gu…

ElasticSearch下

DSL查询 叶子查询&#xff1a;在特定字段里查询特定值&#xff0c;属于简单查询&#xff0c;很少单独使用复合查询&#xff1a;以逻辑方式组合多个叶子查询或更改叶子查询的行为方式 在查询后还可以对查询结果做处理&#xff1a; 排序&#xff1a;按照1个或多个字段做排序分页…

【个人开发】nginx域名映射及ssl证书配置踩坑记录

1. 背景 买了个域名&#xff0c;申请了ssl证书&#xff0c;买都买了&#xff0c;不得部署点东西。于是开始一堆踩坑之旅。 2. 踩过的坑 2.1 报错1:域名访问 Invalid Host header 项目是一个简单的vue项目&#xff0c;启动方式如下 npm run serve但浏览器出现报错&#xff…

Go-Zero整合Goose实现MySQL数据库版本管理

推荐阅读 【系列好文】go-zero从入门到精通&#xff08;看了就会&#xff09; 教程地址&#xff1a;https://blog.csdn.net/u011019141/article/details/139619172 Go-Zero整合Goose实现MySQL数据库版本管理的教程 在开发中&#xff0c;数据库迁移和版本管理是必不可少的工作。…

每日一题洛谷P1427 小鱼的数字游戏c++

#include<iostream> using namespace std; int main() {long long s[100] { 0 };int i 0;while (1) {cin >> s[i];if (s[i]0) {break;}i;}for (i; i > 0;i--) {if(s[i]!0)cout << s[i] << " ";}return 0; }

力扣 子集

回溯基础&#xff0c;一题多解&#xff0c;不同的回朔过程。 题目 求子集中&#xff0c;数组的每种元素有选与不选两种状态。因此在使用dfs与回溯时把每一个元素分别进行选与不选的情况考虑即可。可以先用dfs跳过当前元素即不选然后一直深层挖下去&#xff0c;直到挖到最深了即…

宇泰串口卡驱动在Ubuntu22.04编译、安装汇总

从官网下载驱动官网地址 上传到Ubuntu, 目录结构如下&#xff1a; 驱动源代码: 驱动代码是基于开源项目编译来的 编译路径不能有中文路径&#xff0c;否则可能有类似错误 源码是基于Linux2.3内核编译&#xff0c;我当前是6.8.0-51&#xff0c;数据结构有升级&#xff0c;需要调…

LSA更新、撤销

LSA的新旧判断&#xff1a; 1.seq&#xff0c;值越大越优先 2.chksum&#xff0c;值越大越优先 3.age&#xff0c;本地的LSA age和收到的LSA age作比较 如果差值<900s&#xff0c;认为age一致&#xff0c;保留本地的&#xff1a;我本地有一条LSA是100 你给的是400 差值小于…

运行fastGPT 第四步 配置ONE API 添加模型

上次已经装好了所有的依赖和程序。 下面在网页中配置One API &#xff0c;这个是大模型的接口。配置好了之后&#xff0c;就可以配置fastGPT了。 打开 OneAPI 页面 添加模型 这里要添加具体的付费模型的API接口填进来。 可以通过ip:3001访问OneAPI后台&#xff0c;**默认账号…

强化学习-蒙特卡洛方法

强化学习-数学理论 强化学习-基本概念强化学习-贝尔曼公式强化学习-贝尔曼最优公式强化学习-值迭代与策略迭代强化学习-蒙特卡洛方法 文章目录 强化学习-数学理论一、蒙特卡洛方法理论(Monte Carlo, MC)二、MC Basic2.1 算法拆解2.2 MC Basic算法 三、MC Exploring Starts3.1 …

【专题一 递归】21. 合并两个有序链表

1.题目解析 2.讲解算法原理 解法:递归-> 重复的子问题 重复子问题 ->函数头的设计 合并两个有序链表--->Node dfs(l1&#xff0c;l2) 只关心某一个子问题在做什么事情 ->函数体的设计 比大小l1→next dfs( l1.next, l2)return l1 递归的出口 if(l1null)return l2…

企业级NoSQL数据库Redis

1.浏览器缓存过期机制 1.1 最后修改时间 last-modified 浏览器缓存机制是优化网页加载速度和减少服务器负载的重要手段。以下是关于浏览器缓存过期机制、Last-Modified 和 ETag 的详细讲解: 一、Last-Modified 头部 定义:Last-Modified 表示服务器上资源的最后修改时间。 作…

FPGA车牌识别

基于FPGA的车牌识别主要包含以下几个步骤&#xff1a;图像采集、颜色空间转换、边缘检测、形态学处理&#xff08;腐蚀和膨胀&#xff09;、特征值提取、模板匹配、结果显示。先用matlab对原理进行仿真&#xff0c;后用vivado和modelsim进行设计和仿真。 一、1.图像采集采用ov…