长按拖拽移动的vue3组件

复制代码
<template>
  <div
    ref="draggableElement"
    class="draggable-element"
    :style="elementStyle"
    @mousedown="startDrag"
    @touchstart="startDrag"
  >
    <slot></slot>
  </div>
</template>

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

export default {
  name: 'PressAndDrag',
  setup() {
    const draggableElement = ref(null);
    const isDragging = ref(false);
    const startPos = ref({ x: 0, y: 0 });
    const currentPos = ref({ x: 0, y: 0 });
    const pressTimer = ref(null);
    const elementStyle = ref({
      position: 'fixed',
      left: '100px',
      top: '100px',
      cursor: 'pointer',
      userSelect: 'none',
      touchAction: 'none'
    });

    // 长按开始
    const startPress = (event) => {
      // 阻止默认行为,防止触摸设备上的滚动等
      event.preventDefault();
      
      // 设置定时器,500ms后开始拖动
      pressTimer.value = setTimeout(() => {
        isDragging.value = true;
        
        // 获取初始位置
        const clientX = event.clientX || event.touches[0].clientX;
        const clientY = event.clientY || event.touches[0].clientY;
        
        startPos.value = {
          x: clientX - currentPos.value.x,
          y: clientY - currentPos.value.y
        };
      }, 500); // 500ms长按时间
    };

    // 开始拖动(鼠标/触摸按下)
    const startDrag = (event) => {
      startPress(event);
      
      // 添加移动和结束事件的监听
      window.addEventListener('mousemove', handleMove);
      window.addEventListener('touchmove', handleMove, { passive: false });
      window.addEventListener('mouseup', endDrag);
      window.addEventListener('touchend', endDrag);
    };

    // 处理移动
    const handleMove = (event) => {
      if (!isDragging.value) return;
      event.preventDefault();
      
      const clientX = event.clientX || event.touches[0].clientX;
      const clientY = event.clientY || event.touches[0].clientY;
      
      currentPos.value = {
        x: clientX - startPos.value.x,
        y: clientY - startPos.value.y
      };
      
      updatePosition();
    };

    // 更新元素位置
    const updatePosition = () => {
      elementStyle.value.left = `${currentPos.value.x}px`;
      elementStyle.value.top = `${currentPos.value.y}px`;
    };

    // 结束拖动
    const endDrag = () => {
      // 清除长按定时器
      if (pressTimer.value) {
        clearTimeout(pressTimer.value);
        pressTimer.value = null;
      }
      
      isDragging.value = false;
      
      // 移除事件监听
      window.removeEventListener('mousemove', handleMove);
      window.removeEventListener('touchmove', handleMove);
      window.removeEventListener('mouseup', endDrag);
      window.removeEventListener('touchend', endDrag);
    };

    // 组件卸载时清理
    onUnmounted(() => {
      endDrag();
    });

    return {
      draggableElement,
      elementStyle,
      startDrag,
      handleMove,
      endDrag
    };
  }
};
</script>

<style scoped>
.draggable-element {
  background-color: #42b983;
  padding: 20px;
  border-radius: 8px;
  color: white;
  box-shadow: 0 2px 8px rgba(0, 0, 0, 0.2);
}
</style>

使用示例

复制代码
<template>
  <div class="container">
    <PressAndDrag>
      长按并拖动我
    </PressAndDrag>
  </div>
</template>

<script>
import PressAndDrag from './PressAndDrag.vue';

export default {
  components: {
    PressAndDrag
  }
};
</script>

<style>
.container {
  height: 100vh;
  position: relative;
}
</style>
相关推荐
东风西巷1 分钟前
Atlantis Word Processor:全方位的文字处理专家
前端·学习·word·软件需求
今天不要写bug13 分钟前
基于elementUI实现一个可编辑的表格(简洁版)
前端·javascript·elementui
上优16 分钟前
Vue3纯前端同源跨窗口通信移动AGV小车
前端·vue.js·状态模式
h_k1008616 分钟前
Chrome 插件开发入门技术文章大纲
前端·chrome
一只小阿乐17 分钟前
vue-router 的实现原理
前端·javascript·vue.js·路由·vue-router
小圣贤君17 分钟前
小说写作中的时间轴管理:基于 Vue 3 的事序图技术实现
vue.js·electron·写作·甘特图·时间轴·事序图
Zz_waiting.17 分钟前
案例开发 - 日程管理 - 第七期
开发语言·前端·javascript·vue.js·html·路由
writeone18 分钟前
9-10关于JS初学产生的问题
开发语言·javascript·ecmascript
一只小风华~21 分钟前
Vue:事件处理机制详解
前端·javascript·vue.js·typescript·前端框架
dy17174 小时前
element-plus表格默认展开有子的数据
前端·javascript·vue.js