微信小程序 拖拽签章

效果

在这里插入图片描述

主要实现的功能点

  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元素只能在当前文件页进行拖拽,上下分页的情况不能直接从上一页拖拽到下一页

版本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('');
}

.pineapple-b {
  width: 100%;
  height: 50rpx;
  overflow: hidden;
  text-overflow: ellipsis;
  display: -webkit-box;
  -webkit-line-clamp: 1;
  line-clamp: 1;
  -webkit-box-orient: vertical;
  border-top: 1rpx solid #c8c7cc;
  text-align: center;
  background-color: #f66e5d;
  color: #fff;
}

.pineapple-b .name {
  font-size: 18rpx;
  padding: 0 10rpx;
}
</style>

z-drag-pineapples组件-编辑、删除操作

z-drag-add组件(印章模板弹层)拖拽一次印模,生成一个印章展示

然后z-drag-pineapples组件生成的每个印章绑定dragStartdragMovedragEnd事件,

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

拖拽过程中是相当于编辑操作,更新坐标位置数据

点击删除图标执行删除操作

<template>
  <view class="pineapple" v-for="item in pineapples" :key="item.uuid" :id="item.uuid"
    @touchstart.stop="dragStart($event, item)" @touchmove.stop="dragMove($event, item)"
    @touchend.stop="dragEnd($event, item)" :style="{ left: item.x + 'px', top: item.y + 'px' }" v-show="item.show">
    <view class="pineapple-del" @touchend.stop="handleDelete(item)">
      x
    </view>
    <view class="pineapple-t">
      <view class="">
        x:{{ item.x }}
      </view>
      <view class="">
        y:{{ item.y }}
      </view>
      <view>
        page:{{ item.page }}
      </view>
    </view>
    <view class="pineapple-b">
      <text class="name">{{ item.name }}</text>
    </view>
  </view>
</template>
<script setup lang="ts">
import { ref, computed, getCurrentInstance, watch, onMounted, } from "vue";
import type { Pineapple } from '@/types/mock'
import { PropType } from 'vue';
import { throttle } from 'lodash-es'
import { getRect } from '@/hooks/helper'

const instance = getCurrentInstance()
const props = defineProps({
  pineapples: {
    type: Array as PropType<Pineapple[]>,
    default: () => [],
  }
});
const emit = defineEmits(["dragStart", "dragMove", "dragEnd", "dragDelete"]);

const pineappleRef = ref({ width: 83, height: 104 })// 凤梨默认尺寸
const isDown = ref(false); // 是否正在拖拽
const pineapplePos = ref({ divX: 0, divY: 0 });

const dragStart = async (e: TouchEvent, item: Pineapple) => {
  console.log('dragStart');

  e.stopPropagation();
  isDown.value = true;
  try {
    const rect = await getRect(`#${item.uuid}`, instance);
    const touch = e.changedTouches[0];
    const { clientX: startX, clientY: startY } = touch;
    const { offsetLeft, offsetTop } = e.currentTarget as HTMLElement;
    pineapplePos.value = {
      divX: startX - offsetLeft,
      divY: startY - offsetTop,
    };
    emit("dragStart", 'update', item, rect, offsetLeft, offsetTop);
  } catch (error) {
  }
};

const dragMove = throttle(async (e: TouchEvent, item: Pineapple) => {
  e.stopPropagation();
  if (!isDown.value) return;
  try {
    const touch = e.changedTouches[0];
    const { divX, divY } = pineapplePos.value;
    // 使用保存的偏移量计算新位置
    const newX = touch.clientX - divX;
    const newY = touch.clientY - divY;
    emit("dragMove", 'update', item, newX, newY);
  } catch (error) {
  }
}, 50);

const dragEnd = async (e: TouchEvent, item: Pineapple) => {
  e.stopPropagation();
  isDown.value = false;
  const rect = await getRect(`#${item.uuid}`, instance);
  emit("dragEnd", 'update', item, rect);
};

const handleDelete = (item: Pineapple) => {
  const index = props.pineapples.findIndex((i: Pineapple) => item.uuid === i.uuid);
  emit("dragDelete", index);
};
</script>
<style>
.pineapple {
  position: absolute;
  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);
  color: #f66e5d;
  margin: 0 20rpx;
}

.pineapple-t {
  width: 100%;
  font-size: 24rpx;
  flex: 1;
  display: flex;
  flex-direction: column;
  justify-content: center;
  align-items: center;
  border-radius: 12rpx;
  color: #f90511;
  background-size: cover;
  background-image: url('');
}

.pineapple-b {
  width: 100%;
  height: 50rpx;
  overflow: hidden;
  text-overflow: ellipsis;
  display: -webkit-box;
  -webkit-line-clamp: 1;
  line-clamp: 1;
  -webkit-box-orient: vertical;
  border-top: 1rpx solid #c8c7cc;
  text-align: center;
  background-color: #f66e5d;
  color: #fff;
  border-bottom-left-radius: 12rpx;
  border-bottom-right-radius: 12rpx;
}

.pineapple-b .name {
  font-size: 18rpx;
  padding: 0 10rpx;
}

.pineapple-del {
  position: absolute;
  top: -15rpx;
  right: -15rpx;
  z-index: 1;
  width: 30rpx;
  height: 30rpx;
  border-radius: 50%;
  background-color: #fff;
  display: flex;
  justify-content: center;
  border: 1rpx solid #c8c7cc;
  font-size: 20rpx;
  line-height: 25rpx;
}
</style>

index页面

子组件z-drag-add拖拽添加
通过这个index父组件通信传输,处理更新数据,并传给另一个子组件z-drag-pineapples

子组件z-drag-pineapples编辑、删除操作
通过这个index父组件通信传输,处理更新数据,并传给另一个子组件z-drag-pineapples

index父组件处理数据后,通过Pinia状态管理,更新数据

<template>
  <view class="content">
    <z-drag-files :files="files" />
    <z-drag-add :pineapples="pineapples" :files="files" :scrollTopHeight="scrollTopHeight" @dragStart="dragStart"
      @dragMove="dragMove" @dragEnd="dragEnd" />
    <z-drag-pineapples :pineapples="pineapplesPosition" @dragStart="dragStart" @dragMove="dragMove" @dragEnd="dragEnd"
      @dragDelete="dragDelete" />
    <z-drag-dom />
  </view>
</template>

<script setup lang="ts">
import { ref, computed, getCurrentInstance, watch, onMounted } from "vue";
import { onPageScroll, onLoad } from "@dcloudio/uni-app";
import type { PageScrollOption } from '@dcloudio/uni-app';
import { filesMock, pineapplesMock, } from '@/mock/index'
import type { File, Pineapple } from '@/types/mock'
import { spaceHeight, marginWidth, marginTop } from '@/config/index'
import { checkPineapplePage, checkPineappleBoundary } from '@/utils/tool'
import { buildShortUUID } from "@/utils/uuid";
import { useAppStore } from "@/store/modules/app";

const appStore = useAppStore();
const files = ref<File[]>([])
const pineapples = ref(pineapplesMock.pineapples)
const pineapplesPosition = ref([])
const windowInfo = ref()
const dragHeight = ref(0)
const scrollTopHeight = ref(0)

const { proxy } = getCurrentInstance() as any;

// 创建通用的拖动payload生成函数
const createDragPayload = (
  item: Pineapple,
  position: { x: number; y: number; page: number },
  rect: { width: number; height: number },
  isDown: boolean,
  uuid?: string
) => ({
  ...item,
  ...position,
  ...rect,
  uuid: uuid || item.uuid || buildShortUUID("pineapple"),
  isDown
});

// 开始拖动
const dragStart = (type: string, item: Pineapple, rect: DOMRect, left: number, top: number) => {
  // 校验拖动位置是否超出页面范围(两页之间)
  const position = checkPineapplePage(
    { x: left, y: Math.max(top, marginTop), width: rect.width, height: rect.height },
    files.value
  );
  const payload = createDragPayload(item, position, rect, true);
  if (type === 'add') {
    pineapplesPosition.value.push({ ...payload, show: false });
  } else {
    pineapplesPosition.value = pineapplesPosition.value.map(v =>
      v.uuid === payload.uuid ? { ...v, ...position, show: false } : v
    );
  }
  // 实时更新拖动层坐标
  appStore.setPineapple(payload);
};

// 拖动中
const dragMove = (type: string, item: Pineapple, newX: number, newY: number, width: number, height: number) => {
  const effectiveWidth = width || item.width || 0;
  const effectiveHeight = height || item.height || 0;
  // 校验拖动位置是否超出页面范围(4个边)
  const position = checkPineappleBoundary(
    newX, newY,
    effectiveWidth, effectiveHeight,
    windowInfo.value.windowWidth,
    dragHeight.value,
    files.value
  );
  const payload = createDragPayload(
    item,
    position,
    { width: effectiveWidth, height: effectiveHeight },
    true,
    item.uuid || appStore.pineapple.uuid
  );
  // 实时更新拖动层坐标
  appStore.setPineapple(payload);
};

// 拖动结束
const dragEnd = (type: string, item: Pineapple, rect: DOMRect) => {
  const pineapple = appStore.pineapple;
  const effectiveWidth = rect?.width || item.width || 0;
  const effectiveHeight = rect?.height || item.height || 0;
  // 校验拖动位置是否超出页面范围(两页之间)
  const position = checkPineapplePage(
    { x: pineapple.x, y: pineapple.y, width: effectiveWidth, height: effectiveHeight },
    files.value
  );
  const payload = createDragPayload(
    item,
    position,
    { width: effectiveWidth, height: effectiveHeight },
    false,
    item.uuid || appStore.pineapple.uuid
  );

  pineapplesPosition.value = pineapplesPosition.value.map(v =>
    v.uuid === payload.uuid ? { ...v, ...position, show: true } : v
  );
  pineapplesPosition.value.forEach(v => v.show = true);
  // 实时更新拖动层坐标
  appStore.setPineapple(payload);
};
// 删除
const dragDelete = (index: number) => {
  pineapplesPosition.value.splice(index, 1);
  appStore.setPineapple({ x: 0, y: 0, page: 0, uuid: '', isDown: false, })
};

watch(
  () => files.value,
  (newVal) => {
    // 当文件列表变化时,重新计算凤梨的位置
    if (newVal && newVal.length > 0) {
      let currentTop = 0;
      for (let i = 0; i < newVal.length; i++) {
        const rect = newVal[i];
        currentTop = currentTop + rect.height + spaceHeight;
      }
      dragHeight.value = currentTop;
    }
  },
  { immediate: true, deep: true }
);
// 页面滚动
onPageScroll((e: PageScrollOption) => {
  scrollTopHeight.value = e.scrollTop;
});
// 页面加载
onMounted(() => {
  windowInfo.value = uni.getWindowInfo();
  let { windowWidth } = windowInfo.value
  files.value = filesMock.files.map((v) => {
    let width = windowWidth
    let scale = width / v.width
    let height = v.height * scale
    return {
      ...v,
      width: width - marginWidth,
      height: height,
    }
  })
})
// 页面加载
onLoad(() => {
  wx.showShareMenu({
    menus: ['shareAppMessage', 'shareTimeline'],
  });
}) 
</script>

<style>
.content {
  display: flex;
  flex-direction: column;
  align-items: center;
  justify-content: center;
}
</style>

z-drag-dom拖拽dom组件

通过pinia,获取数据,computed计算属性实时更新拖动层坐标,并跟随手指移动

<template>
  <view class="drag-dom pineapple" :style="{
    left: pineapple.x + 'px',
    top: pineapple.y + 'px',
    opacity: pineapple.isDown ? 1 : 0
  }">
    <view class="pineapple-t">
      {{ pineapple.typeName }}
    </view>
    <view class="pineapple-b">
      <text class="name">{{ pineapple.name }}</text>
    </view>
  </view>
</template>
<script setup lang="ts">
import { ref, computed, watch, } from "vue";
import { useAppStore } from "@/store/modules/app";
const appStore = useAppStore();
const pineapple = computed(() => appStore.pineapple)
</script>
<style>
.drag-dom {
  position: absolute;
  pointer-events: none;
  z-index: 999;
}

.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('');
}

.pineapple-b {
  width: 100%;
  height: 50rpx;
  overflow: hidden;
  text-overflow: ellipsis;
  display: -webkit-box;
  -webkit-line-clamp: 1;
  line-clamp: 1;
  -webkit-box-orient: vertical;
  border-top: 1rpx solid #c8c7cc;
  text-align: center;
  background-color: #f66e5d;
  color: #fff;
}

.pineapple-b .name {
  font-size: 18rpx;
  padding: 0 10rpx;
}
</style>

store/modules/app.ts 状态库

import { defineStore } from 'pinia';
import { store } from '@/store';

export interface Pineapple {
  x: number;
  y: number;
  page: number; // 页码
  uuid: string; // 唯一标识符
  isDown: boolean; // 是否按下
  typeName?: string; // 类型名称
  name?: string;
}
export const useAppStore = defineStore('app', {
  state: () => ({
    pineapple: {
      x: 0,
      y: 0,
      page: 1, // 默认页码为1
      uuid: '', // 默认唯一标识符为空
      isDown: false, // 默认未按下
      typeName: '', // 默认类型名称为空
      name: '', // 默认名称为空
    } as Pineapple
  }),
  getters: {
    getPineapple(state) {
      return state.pineapple
    },
  },
  actions: {
    setPineapple(pineapple: Pineapple) {
      this.pineapple = pineapple;
    },
  }
});

// Need to be used outside the setup
export function useAppStoreWithOut() {
  return useAppStore(store);
}

tool.ts工具函数

checkPineappleBoundary 检查元素是否超出边界

checkPineapplePage 校验凤梨页码是否在文件上下页中间

import { marginTop, marginWidth, spaceHeight } from '@/config/index';
import type { File } from '@/mock/index'
// 边界计算器类,用于处理元素拖拽时的边界计算
class BoundaryCalculator {
  private readonly maxX: number;// X轴最大允许拖拽距离
  private readonly maxY: number;// Y轴最大允许拖拽距离

  constructor(
    private readonly options: {
      dragWidth: number// 拖拽区域宽度
      dragHeight: number// 拖拽区域高度
      marginWidth: number// 元素左侧间距
      marginTop: number// 元素顶部间距
      spaceHeight: number// 元素之间间距
    }
  ) {
    // 计算X轴最大允许拖拽距离
    this.maxX = options.dragWidth - options.marginWidth
    // 计算Y轴最大允许拖拽距离
    this.maxY = options.dragHeight + options.marginTop - options.spaceHeight
  }

  // 数值范围限制工具
  private clamp(val: number, min: number, max: number) {
    return Math.max(min, Math.min(val, max))
  }

  // 核心计算方法
  calculate(
    moveX: number, // 元素X轴移动距离
    moveY: number, // 元素Y轴移动距离
    width: number, // 元素宽度
    height: number, // 元素高度
    files: File[] // 所有文件
  ) {
    // 限制X轴移动距离在合理范围内
    const clampedX = this.clamp(moveX, 0, this.maxX - width)
    // 限制Y轴移动距离在合理范围内
    const clampedY = this.clamp(moveY, this.options.marginTop, this.maxY - height)

    return {
      x: clampedX,
      y: clampedY,
      page: this.getPage(clampedY, files) // 当前所在页码
    }
  }

  // 分页计算逻辑(根据垂直位置确定页码)
  private getPage(y: number, files: File[]) {
    let currentTop = 0  // 当前累计高度
    for (const [index, file] of files.entries()) {
      currentTop += file.height + this.options.spaceHeight
      if (y < currentTop) return index + 1 // 返回第一个超过当前高度的页码
    }
    return files.length  // 默认返回最后一页
  }
}

// 边界检查入口函数
export const checkPineappleBoundary = (
  moveX: number,
  moveY: number,
  width: number,
  height: number,
  dragWidth: number, // 拖拽区域实际宽度
  dragHeight: number, // 拖拽区域实际高度
  files: File[]
) => {
  // 创建边界计算器实例
  const calculator = new BoundaryCalculator({
    dragWidth,
    dragHeight,
    marginWidth,
    marginTop,
    spaceHeight
  })
  return calculator.calculate(moveX, moveY, width, height, files)
}
// 页面边界定义
interface PageBoundary {
  top: number; // 页面上边界
  bottom: number; // 页面下边界
}

// 生成页面边界
const generatePageBoundaries = (files: File[]) => {
  return files.reduce<PageBoundary[]>((acc, file, index) => {
    const prev = acc[acc.length - 1]?.bottom || marginTop // 获取前一页底部位置
    const spacing = index < files.length - 1 ? spaceHeight : 0 // 最后一页不加间距
    return [...acc,
    {
      top: prev,
      bottom: prev + file.height + spacing  // 当前页底部 = 前一页底部 + 元素高度 + 间距
    }]
  }, [])
}
// 二分查找当前页(根据元素Y轴位置)
const findCurrentPage = (pineappleY: number) => (pages: PageBoundary[]) => {
  let low = 0, high = pages.length - 1
  while (low <= high) {
    const mid = (low + high) >> 1 // 位运算取中间索引
    pineappleY >= pages[mid].top - 1e-6 ? (low = mid + 1) : (high = mid - 1) // 比较元素Y轴位置与中间页边界
  }
  return Math.min(high + 1, pages.length) // 返回合法页码
}
// 页码有效性验证
const validatePageRange = (currentPage: number) => (pages: PageBoundary[]) => {
  if (currentPage <= 0) return 1 // 不能小于第一页
  if (currentPage > pages.length) return pages.length // 不能超过最后一页
  return currentPage
}
// 吸附位置计算(处理跨页吸附逻辑)
const calculateSnapPosition = (pineapple: Pineapple) =>
  (ctx: { pages: PageBoundary[]; currentPage: number }) => {
    const { pages, currentPage } = ctx
    const currentPageInfo = pages[currentPage - 1]
    const nextPageInfo = pages[currentPage]
    const pineappleBottom = pineapple.y + pineapple.height
    // 计算与下一页的重叠阈值
    const overlapThreshold = currentPageInfo.bottom - spaceHeight
    const overlap = pineappleBottom - overlapThreshold

    if (overlap > 0 && nextPageInfo) {
      // 当重叠超过元素高度的50%时吸附到下一页
      const shouldSnapNext = overlap > pineapple.height * 0.5
      return {
        x: pineapple.x,
        y: Math.max(
          shouldSnapNext ? nextPageInfo.top : overlapThreshold - pineapple.height,
          marginTop
        ),
        page: currentPage + (shouldSnapNext ? 1 : 0)
      }
    }
    return { ...pineapple, page: currentPage }
  }


// 校验凤梨页码是否在文件上下页中间
export const checkPineapplePage = (pineapple: Pineapple, files: File[]) => {
  // 1. 生成页面边界
  const pages = generatePageBoundaries(files);

  // 2. 查找当前页
  let currentPage = findCurrentPage(pineapple.y)(pages);

  // 3. 验证页码有效性
  currentPage = validatePageRange(currentPage)(pages);

  // 4. 计算吸附位置
  const ctx = { pages, currentPage };
  const result = calculateSnapPosition(pineapple)(ctx);

  // 5. 最终位置修正
  return {
    ...result,
    y: Math.max(result.y, marginTop).toFixed(2),
  };
}

uuid.ts 生成唯一标识符

const hexList: string[] = [];
for (let i = 0; i <= 15; i++) {
  hexList[i] = i.toString(16);
}

export function buildUUID(): string {
  let uuid = '';
  for (let i = 1; i <= 36; i++) {
    if (i === 9 || i === 14 || i === 19 || i === 24) {
      uuid += '-';
    } else if (i === 15) {
      uuid += 4;
    } else if (i === 20) {
      uuid += hexList[(Math.random() * 4) | 8];
    } else {
      uuid += hexList[(Math.random() * 16) | 0];
    }
  }
  return uuid.replace(/-/g, '');
}

let unique = 0;
export function buildShortUUID(prefix = ''): string {
  const time = Date.now();
  const random = Math.floor(Math.random() * 1000000000);
  unique++;
  return prefix + '_' + random + unique + String(time);
}

这次版本的升级

  1. 突破弹层元素拖拽交互瓶颈:实现了弹层内元素向文件图片的直接拖拽,解决了此前操作链路不畅的问题。
  2. 简化印模拖拽流程:升级物料区印模交互逻辑,支持印模直接跟随手指移动完成添加,省去了 “点击添加后再拖拽调整” 的两步操作,提升操作效率。
  3. 优化多页文件展示体验:针对多页文件滚动场景,封装并升级印章拖拽至上下页时的吸顶算法,确保印章定位精准、交互流畅。
  4. 提升代码可维护性:对组件进行精细化拆分,明确业务组件与通用组件的职责边界,增强代码可读性与后续迭代的便捷性。

还需要优化的点

  1. 通用印模css
  2. 通用拖拽事件
  3. 业务数据处理的逻辑
    gitee地址https://gitee.com/sleepkele/uniapp-wechat-drag
Logo

为武汉地区的开发者提供学习、交流和合作的平台。社区聚集了众多技术爱好者和专业人士,涵盖了多个领域,包括人工智能、大数据、云计算、区块链等。社区定期举办技术分享、培训和活动,为开发者提供更多的学习和交流机会。

更多推荐