微信小程序 拖拽签章

发布于:2025-08-16 ⋅ 阅读:(15) ⋅ 点赞:(0)

微信小程序 拖拽签章

效果

在这里插入图片描述

主要实现的功能点

  1. 文件按比例加载图片(宽高设定拖拽范围)
  2. 弹层展示印章模板
  3. 模板拖拽到文件图片上
  4. 实时获取拽拽位置

难点 弹层中的元素如何拖拽到文件图片上

实现历程

版本1.0

以前我们拖拽一个图层到另一个图层上,pc端使用的是mousedown mousemove mouseup事件

在这里插入图片描述

如上图:

  1. 給物料区域,图片A绑定点击事件,点击图片A,传输数据

  2. 生成一个拖拽dom元素,追加div元素到文件区域中心

  3. 然后绑定鼠标事件,鼠标事件实时获取坐标位置

为什么不直接使用 dragstart dragend drop, 当时文件时pdf,pdf展现的形式是canvas

  • 缺点:点击添加拖拽dom元素,然后再点击元素进行拖拽,多页文件得点击按钮翻页进行拖拽

版本2.0

现在文件的展示形式是图片的形式,然后使用的是dragstart dragend drop事件

在这里插入图片描述

如上图:

现在是多页滚动展示文件

  1. 給物料区域,图片A绑定drag事件,点击图片A,触发drag事件,按住鼠标左键,拖动图片A元素,图片A跟随鼠标到拖拽区域

  2. 蒙层也就是div图层绑定drop事件,用于接收数据(img只是用来展示,img元素和蒙层div是兄弟关系)

  3. 接收数据后,拖拽div数组添加元素,通过for循环展示拖拽div(这个拖拽div蒙层div的子元素),并设定当前文件页的拖拽范围

  4. 拖拽div绑定mouse事件,实时获取坐标位置

鼠标移动过程中,拖拽div跟随鼠标移动

鼠标松开,拖拽div固定在当前位置

  • 优点:文件多页是滚动展示,拖拽添加

  • 缺点:拖拽添加拖拽dom元素,然后再点击元素进行拖拽还是是两步操作

虽然文件多页是滚动展示,但是拖拽dom元素只能在当前文件页进行拖拽,上下分页的情况不能直接从上一页拖拽到下一页

版本3.0

h5版本,使用的是touchstart touchmove touchend事件

在这里插入图片描述

移动端不支持dragstart dragend dropmousedown mousemove mouseup事件,所以h5版本使用的是touchstart touchmove touchend事件,但是同理:

  1. 还是先给物料区域的图片A绑定点击事件,点击图片A,传输数据

  2. 在点击图片A事件中,生成一个拖拽dom元素,追加div元素到文件区域中心

  3. 然后给拖拽dom元素绑定touchstart touchmove touchend事件

touchstart事件中,获取拖拽dom元素的坐标位置

touchmove事件中,实时获取拖拽dom元素的坐标位置

touchend事件中,拖拽dom元素固定在当前位置

touchmove事件中,实时获取拖拽dom元素的坐标位置,计算拖拽dom元素的位置,实时更新拖拽dom元素的位置

  • 优点:增加了拖拽元素可以从上一页直接拖拽到下一页

  • 缺点:还是点击添加拖拽dom元素,然后再点击元素进行拖拽

版本4.0

微信小程序版本,使用的是touchstart touchmove touchend事件

在这里插入图片描述

项目目录

─src
├─components
│ ├─z-drag-add 印章模板弹层-添加操作
│ ├─z-drag-dom 拖拽dom组件
│ ├─z-drag-files 拖拽区域(文件展示)
│ ├─z-drag-pineapples 展示印模组件-编辑、删除操作
├─config
├─hooks
├─mock
├─pages
│ ├─index
├─plugins
├─static
│ └─images
├─store
│ └─modules
├─types
└─utils

准备阶段:印章模板弹层,拖拽区域(文件展示),拖拽dom组件,拖拽到文件区域展示印模组件

他们之间的关系

在这里插入图片描述
看不清可以看这个链接

如上图:

z-drag-files组件

只负责接收父组件index的数据,展示文件图片

<template>
  <view class="files-box">
    <view class="files" v-for="item in files" :key="item.id"
      :style="{ width: item.width + 'px', height: item.height + 'px' }">
      <image class="file-img" :src="item.url" :style="{ width: item.width + 'px', height: item.height + 'px' }"></image>
    </view>
  </view>
</template>
<script setup lang="ts">
import type { File } from '@/types/mock'
const props = defineProps<{
  files: File[]
}>();
</script>
<style>
.files-box {
  margin-top: 27px;
}

.files {
  margin: 0 auto 20rpx;
}

.file-img {
  box-shadow: 0 5rpx 10rpx rgba(0, 0, 0, 0.3);
  border-radius: 16rpx;
}
</style>

z-drag-add组件-添加操作

印章模板弹层,每一个印模添加dragStartdragMovedragEnd事件

实时传输拖拽dom元素的坐标位置,印章高宽及业务数据

父组件index负责接收数据,添加数据,z-drag-pineapples增加拖拽dom元素

<template>
  <view class="custom-collapse">
    <view class="content" :class="isOpen ? 'show-content' : ''">
      <view class="content-box">
        <view class="pineapple" v-for="(item, index) in pineapples" :key="index"
          @touchstart.stop="dragStart($event, item, index)" @touchmove.stop="dragMove($event, item)"
          @touchend.stop="dragEnd($event, item)">
          <view class="pineapple-t">
            {
  
  { item.typeName }}
          </view>
          <view class="pineapple-b">
            <text class="name">{
  
  { item.name }}</text>
          </view>
        </view>
      </view>
    </view>
    <view class="custom-collapse-header" v-show="isOpen">
      请添加签署区拖拽到需要签字盖章的位置
    </view>
    <view class="content-header" :class="isOpen ? 'is-open' : ''">
      <view class="content-title" @click.stop="handleVisible">
        <text class="arrow-icon" :class="isOpen ? 'is-open' : ''"></text>
      </view>
    </view>
  </view>
</template>

<script setup lang="ts">
import { ref, computed, getCurrentInstance, watch, } from "vue";
import type { File, Pineapple } from '@/types/mock'
import { throttle } from 'lodash-es'
import { getRect } from '@/hooks/helper'

const instance = getCurrentInstance();
const props = defineProps({
  pineapples: {
    type: Array,
    default: () => [],
  },
  files: {
    type: Array as () => File[],
    default: () => [],
  },
  scrollTopHeight: {
    type: Number,
    default: 0,
  },
});

const emit = defineEmits(["dragStart", "dragMove", "dragEnd"]);
defineExpose({ open: () => isOpen.value = true, close: () => isOpen.value = false });
const handleVisible = () => {
  isOpen.value = !isOpen.value;
};

const isOpen = ref(true); // 控制折叠状态的变量
const isDown = ref(false); // 是否正在拖拽
const pineapplePos = ref({ divX: 0, divY: 0 });

const dragStart = async (e: TouchEvent, item: Pineapple) => {
  e.stopPropagation();
  isDown.value = true;
  isOpen.value = false;
  try {
    const rect = await getRect('.pineapple', instance);
    const touch = e.changedTouches[0];
    const { clientX: startX, clientY: startY } = touch;
    let { offsetLeft: left, offsetTop: top } = e.currentTarget as HTMLElement;
    pineapplePos.value = {
      divX: startX - left,
      divY: startY - top,
    };
    top = props.scrollTopHeight + top;
    emit("dragStart", 'add', item, rect, left, top);

  } catch (error) {
  }
};

const dragMove = async (e: TouchEvent, item: Pineapple) => {
  e.stopPropagation();
  if (!isDown.value) return;
  try {
    const rect = await getRect('.pineapple', instance);
    const touch = e.changedTouches[0];
    const { divX, divY } = pineapplePos.value;
    // 使用保存的偏移量计算新位置
    const newX = touch.clientX - divX;
    let newY = touch.clientY - divY;
    newY = newY + props.scrollTopHeight; // 添加滚动高度; 
    const { width, height } = rect;
    emit("dragMove", 'add', item, newX, newY, width, height);
  } catch (error) {
  }
}

const dragEnd = async (e: TouchEvent, item: Pineapple) => {
  e.stopPropagation();
  isDown.value = false;
  const rect = await getRect('.pineapple', instance);
  emit("dragEnd", 'add', item, rect,);
};
</script>
<style>
.custom-collapse {
  position: fixed;
  top: 0;
  left: 0;
  width: 100vw;
  overflow: hidden;
  margin-bottom: 20rpx;
  background: #fff;
  z-index: 11000;
  box-shadow: 0 -4rpx 15rpx rgba(0, 0, 0, 0.3);
  border-bottom-left-radius: 32rpx;
  border-bottom-right-radius: 32rpx;
}

.content {
  width: 95vw;
  max-height: 0;
  overflow-y: hidden;
  transition: max-height 0.3s cubic-bezier(0.4, 0, 0.2, 1);
  margin: 0 auto;
  /* 添加硬件加速 */
  transform: translateZ(0);
  will-change: max-height;
}

.content-header {
  padding: 0 24rpx 12rpx 24rpx;
  display: flex;
  align-items: center;
  justify-content: space-between;
  transition: all 0.3s;
}

.is-open {
  border-bottom: 1px solid #eee;
}

.content-title {
  position: relative;
  display: flex;
  align-items: center;
  justify-content: center;
  flex: 1;
  padding-top: 12rpx;
}

.arrow-icon {
  display: inline-block;
  width: 30rpx;
  height: 30rpx;
  transition: transform 0.3s ease;
}

.arrow-icon::after {
  content: "";
  position: absolute;
  top: 60%;
  left: 50%;
  width: 16rpx;
  height: 16rpx;
  border-left: 2rpx solid #666;
  border-bottom: 2rpx solid #666;
  transform: translate(-50%, -70%) rotate(-45deg);
  transition: transform 0.3s ease;
}

.arrow-icon.is-open::after {
  transform: translate(-50%, -30%) rotate(135deg);
}

.show-content {
  max-height: 560rpx;
  /* 添加以下属性改善动画性能 */
  transform: translateZ(0);
  backface-visibility: hidden;
  perspective: 1000;
  will-change: transform;
}

.content-box {
  display: flex;
  width: 100%;
  padding-top: 24rpx;
}

.custom-collapse-header {
  width: 90vw;
  margin: 0 auto;
  font-size: 24rpx;
  color: #999;
  padding: 20rpx 0;
}
</style>
<style>
.pineapple {
  display: flex;
  flex-direction: column;
  width: 160rpx;
  height: 200rpx;
  border-radius: 12rpx;
  background-color: #faf6f5;
  box-shadow: 0 15rpx 23rpx rgba(0, 0, 0, 0.3);
  overflow: hidden;
  color: #f66e5d;
  margin: 0 20rpx;
}

.pineapple-t {
  width: 100%;
  font-size: 24rpx;
  flex: 1;
  display: flex;
  justify-content: center;
  align-items: center;
  background-size: cover;
  background-image: url('