当前位置: 首页 > news >正文

微信小程序 拖拽签章

微信小程序 拖拽签章

效果

在这里插入图片描述

主要实现的功能点

  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('
http://www.xdnf.cn/news/1302733.html

相关文章:

  • Git版本控制器
  • spring中异步任务注解@Async和@scheduled的使用
  • 2025年机械制造、机器人与计算机工程国际会议(MMRCE 2025)
  • Docker Compose 入门教程
  • MySQL、PolarDB、PolarDB-X、TableStore、MongoDB、TiDB、ClickHouse选型
  • docker入门
  • Java 调用 Python 脚本:实现 HelloWorld
  • 计算机视觉(opencv)实战五——图像平滑处理(均值滤波、方框滤波、高斯滤波、中值滤波)附加:视频逐帧平滑处理
  • 从根本上解决MAC权限问题(关闭sip)
  • SSL和TLS协议的消息认证码(MAC)
  • Android RxJava变换操作符详解
  • 使用SQLALCHEMY的outerjoin时的bug
  • 训练大模型的前提:数据治理工程:从原始数据到高质量语料的系统化治理实践
  • vector接口模拟实现及其原理
  • Redis 官方提供免费的 30 MB 云数据库
  • 阿里云出里两款新的云服务器
  • Uniapp之微信小程序自定义底部导航栏形态
  • 订单簿数据智能解析深度学习算法筛选大单并预测即时价格变动
  • MuMu模拟器Pro Mac 安卓手机平板模拟器(Mac中文)
  • 智能家居【home assistant】(二)-集成xiaomi_home
  • 云原生俱乐部-k8s知识点归纳(3)
  • 【计算机视觉与深度学习实战】02基于形态学的权重自适应图像去噪系统
  • 自学大语言模型之Transformer的Tokenizer
  • Android 欧盟网络安全EN18031 要求对应的基本表格填写
  • 对抗损失(GAN)【生成器+判断器】
  • HarmonyOS 实战:用 List 与 AlphabetIndexer 打造高效城市选择功能
  • 【Java】HashMap的详细介绍
  • PCA降维全解析:从原理到实战
  • JAVA文件管理系统:如何玩转文件操作
  • CUDA中的基本概念