前端:拖动悬浮小窗

一、效果展示

二、代码

复制代码
<template>
  <div class="contents">
    <div 
      class="contents--menu" 
      :style="style"
      @mousedown="startDrag"
      @touchstart="startDrag"
      @mousemove="onDrag"
      @touchmove="onDrag"
      @mouseup="stopDrag"
      @touchend="stopDrag"
    >
      <m-button v-for="(item, index) in list" :key="index" @click="changeComponent(item)">{{item.name}}</m-button>
    </div>
    <component :is="activeComponent"></component>
  </div>
</template>
<script setup lang="ts">
  // const itemsButton = defineAsyncComponent(() => import('../items/button/index.vue'))
import itemsButton from '../items/button/index.vue'
import modulesButtons from '../modules/buttons/index.vue'
import modulesForm from '../modules/form/index.vue'
import modulesTable from '../modules/table/index.vue'
import pagesList from '../pages/list/index.vue'
const list = [
  { name: '按钮', component: itemsButton },
  { name: '按钮组', component: modulesButtons },
  { name: '表单', component: modulesForm },
  { name: '表格', component: modulesTable },
  { name: '列表', component: pagesList },
]
let activeComponent = ref(list[0].component)
const changeComponent = (item: any) => {
  activeComponent.value = item.component
}


const isDragging = ref(false);
const initialPosition = ref({ x: 0, y: 0 });
const currentPosition = ref({ x: 100, y: 100 }); // 初始位置

const style = computed(() => ({
  position: 'fixed' as const,
  left: `${currentPosition.value.x}px`,
  top: `${currentPosition.value.y}px`,
  cursor: isDragging.value ? 'grabbing' : 'grab',
  userSelect: 'none' as const,
}));

const startDrag = (e: MouseEvent | TouchEvent) => {
  e.preventDefault();
  isDragging.value = true;
  
  // 获取鼠标或触摸位置
  const clientX = e instanceof MouseEvent ? e.clientX : e.touches[0].clientX;
  const clientY = e instanceof MouseEvent ? e.clientY : e.touches[0].clientY;
  
  // 计算鼠标位置与元素左上角的偏移量
  initialPosition.value = {
    x: clientX - currentPosition.value.x,
    y: clientY - currentPosition.value.y
  };
};

const onDrag = (e: MouseEvent | TouchEvent) => {
  if (!isDragging.value) return;
  
  e.preventDefault();
  
  // 获取当前鼠标或触摸位置
  const clientX = e instanceof MouseEvent ? e.clientX : e.changedTouches[0].clientX;
  const clientY = e instanceof MouseEvent ? e.clientY : e.changedTouches[0].clientY;
  
  // 计算新位置
  currentPosition.value = {
    x: clientX - initialPosition.value.x,
    y: clientY - initialPosition.value.y
  };
};

const stopDrag = () => {
  isDragging.value = false;
};
</script>
<style scoped lang="scss">
.contents{
  &--menu{
    z-index: 20;
    border-radius: 8px;
    width: 200px;
    border: 2px solid #ccc;
    padding: 15px;
    background-color: #f9f9f9;
    border-radius: 4px;
    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
    transition: box-shadow 0.2s;
    display: flex;
    flex-wrap: wrap;
    gap: 10px;
    
    &:hover {
      box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
    }
    
    &.dragging {
      z-index: 9999;
    }
    .el-button+.el-button{
      margin: 0;
    }
  }
}
</style>
相关推荐
范纹杉想快点毕业13 小时前
状态机设计模式与嵌入式系统开发完整指南
java·开发语言·网络·数据库·mongodb·设计模式·架构
lly20240613 小时前
移动设备统计:行业趋势与市场洞察
开发语言
专注VB编程开发20年13 小时前
c#模仿内置 Socket.Receive(无需 out/ref,直接写回数据)
开发语言·c#
爱内卷的学霸一枚13 小时前
Python并发编程与性能优化实战指南
开发语言·python·性能优化
pusheng202513 小时前
燃料电池电化学传感器在硫化物固态电池安全监测中的技术优势解析
前端·人工智能·安全
jaysee-sjc13 小时前
【项目二】用GUI编程实现石头迷阵游戏
java·开发语言·算法·游戏
それども13 小时前
Excel文件解析 - SAX和DOM方式的区别
java·前端·excel
それども13 小时前
Excel文件解析 - SAX startRow cell endRow 执行顺序
java·前端·excel
Byron070713 小时前
基于 Vue 的微前端架构落地实战:从 0 到 1 搭建企业级多应用体系
前端·vue.js·架构
一位搞嵌入式的 genius13 小时前
从 URL 到渲染:JavaScript 性能优化全链路指南
开发语言·前端·javascript·性能优化