Flex中List组件拖动后错位的解决方法

本文探讨了Flex中ItemRenderer组件在List等组件中使用时出现错位问题的原因及解决办法,介绍了ItemRenderer的工作原理及其特点,并提供了两种常见问题的解决方案。

在List中使用了ItemRenderer组件,结果在拖动的过程中老是出现错位的问题,这个问题困扰了我半天,google了好多资料,终于找到一个比较好的解决方案,特转过来分享给大家

使用Flex的组件时,大伙不可避免要和itemRenderer打交道。Flex组件中,支持自定义itemRenderer功能的组件不少,常见的有List,TileList,ComboBox,Tree,DataGrid等。这类组件具有一个共性:显示一组数据,并具备交互功能。

在使用这些组件时,我们只需要把数据按照一定的格式组织好,然后赋予给组件作为dataProvider,剩下的事就由组件来完成。处于需要,有时候我们需要个性化数据的显示方式,比如List组件,默认的itemRenderer是一个类似Label的元件,只显示每一条数据中的固定字段,如果想要显示更多信息,就必须自定义一个itemRenderer。

itemRenderer的实现原理

itemRenderer的实现依赖于 AS3的一个重要特性:反射。反射 (Reflection) 是指在程序在运行时 (run-time) 获取类信息的方式,比如实现动态创建类实例、方法等。示例如下:

import flash.utils.getQualifiedClassName;var cls:Class = flash.display.MovieClip;var mc:MovieClip = new cls();</p>
<p>trace(getQualifiedClassName(mc));</p>
<p>//输出:flash.display::MovieClip

itemRenderer的本质是Class类型,组件获取数据后,动态生成itemRenderer的实例。

itemRenderer 的特点 可以被当作 itemRenderer的组件,都实现了mx.core.IDataRenderer接口或mx.controls.listClasses.IListItemRenderer接口,还有mx.core.listClasses.IDropInListItemRenderer接口,用来实现有下拉框组件的itemRenderer。

不管怎样,组件都有data属性,这是一个存取器性质的函数。 这个函数很有用。

itemRenderer的工作方式 首先要明白一点,组件并不是很简单地把数据一次显示。为了提高运行效率,组件在处理数据时,只是创建了适当数量的itemRenderer,来完成显示区域的数据显示,这样,当数据源的数据量很大时,也不会耗费系统的资源。而且,itemRenderer是根据需要创建,创建后又是反复使用,很绿色,很环保。当显示区域发生变化或者拉动滚动条时,组件只是更新itemRenderer的数据。

比如,TileList、List组件,如果数据源只有一条数据,显示区域可以同时显示十条,则开始时只创建一个itemRenderer,添加一条数据,再创建一个新的itemRenderer。当数据量超过显示区域的最大值10时,就不再创建新的itemRenderer,而是回收利用现有资源。

另外,TileList、List还会创建一个多余的 itemRenderer,用来检测itemRenderer的尺寸等,方便定位。

虽然itemRenderer是真正的环保战士,但如果我们使用不当,就会出现很多奇怪的现象,比如内存泄露就是一个很常见的问题。在Adobe官方的论坛上,就经常见到这类讨论。其实,理解了itemRenderer的动作原理,这些问题就不难找到解决方法。

自定义itemRenderer是个强大的功能,但如果使用不当,就会出现很多奇怪的现象,比如内存泄露就是一个很常见的问题。

常见的两种问题: 1.在自定义itemRenderer中使用creationComplete事件来处理数据 有些朋友习惯在组件中把初始化处理都写在creationComplete事件处理函数中,一般情况下,这没有问题,然而在自定义itemRendere,却不合适。 因为itemRenderer是循环使用的,也就是说,itemRendere在第一次创建后,可能用来显示不同的数据,而本身只被创建一次,creationComplete事件只发生一次,自然就会出问题。 这个问题很常见,可能引发很多奇怪的现象,比如对象无法被清除,资源释放不完全,内存泄露等等。 那如何避免?我们可以覆盖data存取器的setter方法,比如:

override public function set data(value:Object):void{
	//加上自己的代码
}

2.删除数据后系统资源仍被占用 在解决了第一个问题后,还是有可能出现资源无法正确释放的问题。比如TileList组件可显示40条数据,每一条数据都包括一张位图。当删除数据,从50减少4时,会发现显示的数目确实减少了,但系统消耗的资源却不变。数据量越大,这个现象就越明显。 问题出在哪里?根据现象,从逻辑上推测:数据量减少,也就是已有的itemRenderer实例被“删除”的时候,并没有释放资源。 这就是问题所在,已有的itemRenderer并不会被删除,一旦创建就一直存在,所谓的“删除”,仅仅是将它变为不可见,而占有的资源不会自动释放,如果其中包括了位图、声音、动画,将一直存在,成为系统资源杀手。 我提供的解决方法:监听itemRenderer的hide事件,在变为不可见的同时,手动释放资源。也许有其它更好的方法,就等着你去发现了

<template> <view class="buttom-sort bgc-fff"> <view class="dragSortBox"> <view class="dragSortBox-btns flex-center "> <view class="fw-600 color-0000" v-if="isEdit">请长按拖拽调整商家配送顺序</view> <view v-else></view> <view> <template v-if="isEdit"> <text class="dragSortBox-btn" @click="toggleEdit('finish')">完成</text> <text class="dragSortBox-btn" @click="toggleEdit('cancel')">取消</text> </template> <text v-else class="dragSortBox-btn" @click="toggleEdit('edit')">编辑</text> </view> </view> <movable-area id="dragSortArea" class="dragSort-area bgc-f4f4" :style="{height: boxHeight + 'px'}"> <!-- :style="{height:cloneList.length>20 ? '500rpx' : '300rpx'}" --> <scroll-view scroll-y="true" class="scroll-Y"> <view v-for="(item,index) in cloneList" :key="item.id"> <movable-view class="dragSort-view " direction="all" :class="{'is-touched': item.isTouched}" :x="item.x" :y="item.y" :damping="40" :disabled="!isEdit" @change="onChange($event, item)" @touchstart="onTouchstart(item)" @touchend="onTouchend(item, index)" :style="{width: columnWidth + 'px', height: rpxTopx(rowHeight) + 'px', zIndex: item.zIndex}"> <view class="dragSort-view__con "> <view class="dragSort-view__header flex-no-center f24 color-fff"> <view class="dragSort-view__send_sort is-ellipsis ">{{item.send_sort}}</view> <view class="dragSort-view__tag is-ellipsis "> <span v-if="item.tag===1">送返</span> <span v-if="item.tag===2">送</span> <span v-if="item.tag===3">返</span> <span v-if="item.tag===0">无</span> </view> </view> <view class="dragSort-view__label f26 limited-two pd-10">{{item.user_name}}</view> <!-- <view class="dragSort-view__label is-ellipsis">{{item.user_name}} {{item.user_name}}</view> --> <!-- <text class="dragSort-view__btn-del" v-if="isEdit" @click="handleDel(item)">X</text> --> </view> </movable-view> </view> </scroll-view> </movable-area> </view> </view> </template> <script> export default { data() { return { showPop: false, cloneList: [], //用来展示的数据列表 cacheList: [], //用来在点击“编辑”文字按钮的时候,将当前list的数据缓存,以便在取消的时候用到 positionList: [], //用来存储xy坐标定位的列表 columnWidth: 0, //列宽,单位px rowNum: 1, //行数 boxHeight: 10, //可拖动区域的高度,单位px windowWidth: 750, //系统获取到的窗口宽度,单位px curTouchPostionIndex: 0, //当前操作的移动块在positionList队列里的索引 xMoveUnit: 0, //沿x轴移动时的单位距离,单位px yMoveUnit: 0, //沿y轴移动时的单位距离,单位px clearT: '', //onChange事件中使用 clearF: '', //点击“完成”文字按钮时使用 isEdit: false, //是否在编辑状态 } }, props: { //props里属性Number的单位都为rpx,在操作的时候需要用rpxTopx进行转换 list: { //源数据列表 type: Array, default () { return [{ "name": "互联网", "id": 1, //id必传且唯一 }, { "name": "古董", "id": 20 }] } }, label: { //list队列中的对象中要用来展示的key名 type: String, default: 'name' }, rowHeight: { //行高,单位rpx type: Number, default: 60 }, rowSpace: { //行间距,单位rpx type: Number, default: 15 }, columnSpace: { //列间距,单位rpx type: Number, default: 15 }, columnNum: { //列数 type: Number, default: 4 }, zIndex: { //可移动项的默认z-index type: Number, default: 100 } }, computed: { btnText() { return this.isEdit ? '完成' : '编辑' } }, created() { this.windowWidth = uni.getSystemInfoSync().windowWidth; }, mounted() { const query = uni.createSelectorQuery().in(this); query.select('#dragSortArea').boundingClientRect(data => { console.log("得到布局位置信息" + JSON.stringify(data)); this.columnWidth = (data.width - (this.columnNum - 1) * this.rpxTopx(this.columnSpace)) / this .columnNum this.handleListData(); }).exec(); }, methods: { openSort() { this.showPop = !this.showPop }, /* 切换编辑状态 * [type] String 参数状态 */ toggleEdit(type) { if (type == 'finish') { //点击“完成” this.isEdit = false; console.log(this.getSortedIdArr(), 'sortedIdArr') this.$emit('changeMarks', this.getSortedIdArr()) } else if (type == 'cancel') { //点击“取消”,将数据恢复到最近一次编辑时的状态 this.isEdit = false; this.updateList(this.cacheList); } else if (type == 'edit') { //点击“编辑” this.isEdit = true; this.cacheList = JSON.parse(JSON.stringify(this.list)); } }, /* 更新父组件list,并重新渲染布局 * 有改变数组长度的操作内才需要调用此方法进行重新渲染布局进行更新, * 否则直接$emit('update:list')进行更新,无须调用此方法 */ updateList(arr) { this.$emit('update:list', arr); setTimeout(() => { this.handleListData() }, 100) }, /* 删除某项 */ handleDel(obj) { for (var i = 0, len = this.list.length; i < len; i++) { var item = this.list[i]; if (obj.id == item.id) { var theList = JSON.parse(JSON.stringify(this.list)); theList.splice(i, 1); this.updateList(theList); break; } } }, /* 处理源数据列表,生成展示用的cloneList和positionList布局位置信息 */ handleListData() { this.cloneList = JSON.parse(JSON.stringify(this.list)); this.positionList = []; this.rowNum = Math.ceil(this.cloneList.length / this.columnNum); this.boxHeight = this.rowNum * this.rpxTopx(this.rowHeight) + (this.rowNum - 1) * this.rpxTopx(this .rowSpace); this.xMoveUnit = this.columnWidth + this.rpxTopx(this.columnSpace); this.yMoveUnit = this.rpxTopx(this.rowHeight) + this.rpxTopx(this.rowSpace); this.cloneList.forEach((item, index) => { item.sortNumber = index; item.zIndex = this.zIndex; item.x = this.xMoveUnit * (index % this.columnNum); //单位px item.y = Math.floor(index / this.columnNum) * this.yMoveUnit; //单位px this.positionList.push({ x: item.x, y: item.y, id: item.id, }) }) }, /* 找到id在位置队列positionList里对应的索引 */ findPositionIndex(id) { var resultIndex = 0; for (var i = 0, len = this.positionList.length; i < len; i++) { var item = this.positionList[i]; if (item.id == id) { resultIndex = i; break; } } return resultIndex }, /* 触摸开始 */ onTouchstart(obj) { if (!this.isEdit) { return false }; this.curTouchPostionIndex = this.findPositionIndex(obj.id); // 将当前拖动的模块zindex调成当前队列里的最大 this.cloneList.forEach((item, index) => { if (item.id == obj.id) { item.zIndex = item.zIndex + 100000; item.isTouched = true; } else { item.zIndex = this.zIndex + index + 1; item.isTouched = false; } }) this.$set(this.cloneList, 0, this.cloneList[0]) }, /* 触摸结束 */ onTouchend(obj) { if (!this.isEdit) { return false }; this.startSort(this.curTouchPostionIndex, 'onTouchend'); //再次调用并传参数‘onTouchend’,使拖动后且没有找到目标位置的滑块归位 }, /* 移动过程中触发的事件(所有移动块只要一有移动都会触发) */ onChange(e, obj) { if (!this.isEdit) { return false }; var theX = e.detail.x, theY = e.detail.y, curCenterX = theX + this.columnWidth / 2, curCenterY = theY + this.rpxTopx(this.rowHeight) / 2; if (e.detail.source === 'touch') { //表示由“拖动”触发 var targetIndex = this.findTargetPostionIndex({ curCenterX, curCenterY, }) clearTimeout(this.clearT) this.clearT = setTimeout(() => { this.$nextTick(() => { this.startSort(targetIndex); //根据targetIndex将队列进行排序 }) }, 100) } }, /* 根据targetIndex将cloneList进行排序 * [targetIndex] Number 当前拖动的模块拖动到positionList队列里的目标位置的索引 * [type] String 值为onTouchend时,再次调用set方法 */ startSort(targetIndex, type) { var curTouchId = this.positionList[this.curTouchPostionIndex].id; if (this.curTouchPostionIndex < targetIndex) { for (var i = 0, len = this.positionList.length; i < len; i++) { var curItem = this.positionList[i]; var nextItem = this.positionList[i + 1] || this.positionList[this.positionList.length - 1]; if (i >= this.curTouchPostionIndex && i <= targetIndex) { //找到要进行位移的索引集 if (i == targetIndex) { curItem.id = curTouchId; } else { curItem.id = nextItem.id; } } } } else { var clonePostionList = JSON.parse(JSON.stringify(this.positionList)); for (var i = 0, len = this.positionList.length; i < len; i++) { var curItem = this.positionList[i]; var preItem = this.positionList[i - 1] || this.positionList[0]; if (i >= targetIndex && i <= this.curTouchPostionIndex) { //找到要进行位移的索引集 if (i == targetIndex) { curItem.id = curTouchId; } else { curItem.id = clonePostionList[i - 1].id; } } } } this.cloneList.forEach(item => { item.x += 0.001; item.y += 0.001; }) if (type == 'onTouchend') { this.$set(this.cloneList, 0, this.cloneList[0]) } this.$nextTick(() => { this.cloneList.forEach(item => { for (var i = 0, len = this.positionList.length; i < len; i++) { var item02 = this.positionList[i]; if (item.id == item02.id) { item.send_sort = i + 1; item.x = item02.x; item.y = item02.y; } } }) this.$set(this.cloneList, 0, this.cloneList[0]) this.curTouchPostionIndex = targetIndex this.handleEmitData(); //需要在onChange事件里发射信息出去最稳妥,因为在快速拖动释放鼠标的时候该事件会再onTouchend后执行 }) }, /* 处理要发射出去的数据队列,将排序后的结果同步到父组件list */ handleEmitData() { var idArr = this.getSortedIdArr(); var emitList = []; idArr.forEach(id => { for (var i = 0, len = this.list.length; i < len; i++) { var item = this.list[i]; // console.log('item00',item) if (id == item.id) { emitList.push(item); break; } } }) this.$emit('update:list', emitList); }, /* 获取最后的排序完的id队列集 */ getSortedIdArr() { return this.positionList.map(item => item.id) }, /* 找出拖动到positionList队列里的哪个目标索引 * [curObj.curCenterX], Number 当前拖动的模块的中心点x轴坐标 * [curObj.curCenterY], Number 当前拖动的模块的中心点y轴坐标 * return 返回拖动到的目标索引 */ findTargetPostionIndex(curObj) { var resultIndex = this.curTouchPostionIndex; for (var i = 0, len = this.positionList.length; i < len; i++) { var item = this.positionList[i]; if (curObj.curCenterX >= item.x && curObj.curCenterX <= item.x + this.columnWidth) { if (curObj.curCenterY >= item.y && curObj.curCenterY <= item.y + this.rpxTopx(this.rowHeight)) { resultIndex = i; break; } } } return resultIndex; }, /* prx转换成px,返回值还是不带px单位的Number */ rpxTopx(v) { return this.windowWidth * v / 750 } }, watch: { list() { console.log('list change') } } } </script> <style scoped> /* 排序 */ .buttom-sort { position: fixed; bottom: 0; width: 100%; /* height: 400rpx; */ /* padding: 0 30rpx; */ z-index: 999; } .scroll-Y { height: 90%; } .is-ellipsis { white-space: nowrap; overflow: hidden; text-overflow: ellipsis; } .dragSort-wrap {} .dragSortBox { padding: 20rpx; z-index: 99999; } .dragSortBox-btns { padding: 20rpx; margin-bottom: 20rpx; text-align: right; color: red; } .dragSortBox-btn { margin-left: 20rpx; } .dragSort-area { width: 100%; /* background-color: skyblue; */ overflow-y: auto; } .dragSort-view { /* background-color: pink; */ text-align: center; /* overflow-y: auto; */ background: linear-gradient(to bottom, #fff7ec 0%, #fefdf9 90%); } .dragSort-view.is-touched { opacity: .9; } .dragSort-view__con { position: relative; height: 100%; overflow: hidden; /* margin: 20rpx 10rpx; */ } .dragSort-view__header { /* position: absolute; */ top: 10rpx; } .dragSort-view__send_sort { background-color: #3b2720; width: 50rpx; border-radius: 20rpx 0rpx 0rpx 0rpx; } .dragSort-view__tag { width: 60rpx; background-color: #fd3b0a; } .dragSort-view__label { /* position: absolute; */ /* left: 50%; top: 50%; transform: translate(-50%, -50%); */ height: 100%; color: #3b2720 !important; max-width: 100%; } .dragSort-view__btn-del { position: absolute; right: 0; top: 0; transform: translate(50%, -50%); } </style> 要求每行展示5的数据 目前是4
最新发布
09-26
<template> <Teleport to="body"> <div v-if="visible" class="pdf-mask" @mousedown="startDrag"> <div ref="modalRef" class="pdf-modal" :style="modalStyle"> <!-- PDF 头部:标题 + 操作按钮 --> <div class="pdf-header"> <span class="title">报关资料</span> <div class="tools"> <el-button size="small" @click="zoomIn">放大</el-button> <el-button size="small" @click="zoomOut">缩小</el-button> <el-button size="small" @click="close">关闭</el-button> </div> </div> <!-- PDF 标签页 + 内容区域 --> <el-tabs v-model="activeTab" class="pdf-tabs" @tab-click="handleTabClick"> <el-tab-pane v-for="tab in tabs" :key="tab.id" :label="tab.name" :name="tab.id.toString()" <!-- 统一为字符串,避免 el-tabs 类型警告 --> > <!-- 单个 PDF 滚动容器(每个 tab 独立 ref,修复类型断言) --> <div class="pdf-body" :ref="(el) => el && scrollRefMap.set(tab.id, el as HTMLDivElement)" <!-- 显式类型断言 --> @wheel="(e) => onWheel(e, tab.id)" @mousedown="(e) => onPdfMouseDown(e, tab.id)" > <!-- PDF 渲染容器(每个 tab 独立存储,修复类型断言) --> <div :ref="(el) => el && setPdfContainer(tab.id, el as HTMLDivElement)" /> </div> </el-tab-pane> </el-tabs> </div> </div> </Teleport> </template> <script setup lang="ts"> import { ref, reactive, watch, nextTick, onUnmounted } from 'vue' import type { CSSProperties } from 'vue' // 导入 CSS 类型 import * as pdfjsLib from 'pdfjs-dist' import pdfjsWorker from 'pdfjs-dist/build/pdf.worker?worker&url' // 导入 pdfjs 核心类型(解决 PDFLoadingTask/promise 类型错误) import type { PDFLoadingTask, PDFDocumentProxy, PageViewport, RenderTask } from 'pdfjs-dist' // 导入 Element Plus 组件(确保类型正确) import { ElButton, ElTabs, ElTabPane } from 'element-plus' import type { TabsPaneContext } from 'element-plus' // PDF.js 初始化:设置 Worker 路径 pdfjsLib.GlobalWorkerOptions.workerSrc = pdfjsWorker /* ---------------- Props & Emits 定义 ---------------- */ const props = defineProps<{ visible: boolean url?: string highlight?: { x0: number; y0: number; x1: number; y1: number } | null }>() const emit = defineEmits<{ (e: 'update:visible', value: boolean): void (e: 'tabschange', tabId: string | number): void }>() /* ---------------- 状态管理(每个 tab 独立数据) ---------------- */ // 标签页数据:id/名称/PDF 地址 const tabs = ref<Array<{ id: string | number; name: string; url: string }>>([]) // 当前激活的标签页(统一为字符串类型,适配 el-tabs) const activeTab = ref<string>('') // 每个 tab 的 PDF 渲染容器(DOM 引用,明确类型) const pdfContainerMap = new Map<string | number, HTMLDivElement>() // 每个 tab 的滚动容器(DOM 引用,明确类型) const scrollRefMap = new Map<string | number, HTMLDivElement>() // 每个 tab 的缩放比例(0.5~3 范围) const tabScale = new Map<string | number, number>() // 每个 tab 的 PDF 原始视口(用于计算坐标,明确类型) const tabBaseViewport = new Map<string | number, PageViewport>() // PDF 真实渲染尺寸 const pdfRealSize = ref({ width: 0, height: 0 }) // 弹窗 DOM 引用(明确类型) const modalRef = ref<HTMLDivElement | null>(null) // 异步渲染中断控制器(防止切换 tab 时重复渲染) let abortController: AbortController | null = null // 当前 PDF 拖拽对应的滚动容器(解决事件参数不匹配问题) let currentScrollContainer: HTMLDivElement | null = null /* ---------------- 弹窗位置 & 样式(修复 cursor 类型) ---------------- */ // 显式声明 modalStyle 类型,cursor 支持所有 CSS cursor 值 const modalStyle = reactive<{ left: string top: string width: string height: string cursor: CSSProperties['cursor'] // 关键:使用 CSS 原生 cursor 类型 }>({ left: '0px', top: '0px', width: '800px', height: '600px', cursor: 'default' // 初始值符合类型 }) // 拖拽起始坐标 let dragStart: { x: number; y: number } | null = null /* ---------------- 标签页初始化 & 渲染 ---------------- */ /** * 打开 PDF 标签页(外部调用入口) * @param list 标签页数据:code(唯一ID)/name(标签名)/url(PDF地址) */ async function openWithTabs( list: Array<{ code: string; name: string; url: string }> ) { // 1. 清空历史数据(防止前一次残留导致重复) tabs.value = [] pdfContainerMap.clear() scrollRefMap.clear() tabScale.clear() tabBaseViewport.clear() if (abortController) { abortController.abort() abortController = null } // 2. 格式化标签页数据(过滤 all_files 项) tabs.value = list .filter(item => item.name !== 'all_files') .map(item => ({ id: item.code, name: item.name, url: item.url })) // 3. 激活第一个标签页(转为字符串,适配 el-tabs) const firstTabId = tabs.value[0]?.id.toString() ?? '' activeTab.value = firstTabId emit('update:visible', true) // 4. 等待 DOM 更新后,渲染第一个标签页的 PDF await nextTick() const firstTab = tabs.value[0] if (firstTab) { const container = pdfContainerMap.get(firstTab.id) if (container) { await renderSinglePDF(firstTab.url, container, firstTab.id) } } // 5. 弹窗居中显示 const windowWidth = window.innerWidth const windowHeight = window.innerHeight modalStyle.left = `${(windowWidth - 800) / 2}px` modalStyle.top = `${(windowHeight - 600) / 2}px` } /** * 绑定 PDF 容器到 Map(明确类型) * @param tabId 标签页唯一ID * @param el DOM 元素(已断言为 HTMLDivElement) */ function setPdfContainer(tabId: string | number, el: HTMLDivElement) { pdfContainerMap.set(tabId, el) } /** * 渲染单个标签页的 PDF(核心渲染逻辑,修复 pdfjs 类型) * @param pdfUrl PDF 文件地址 * @param container 渲染容器(明确 HTMLDivElement 类型) * @param currentTabId 当前标签页ID(防止异步渲染错位) */ async function renderSinglePDF( pdfUrl: string, container: HTMLDivElement, currentTabId: string | number ) { // 校验:当前标签页是否已切换,若切换则终止渲染 if (activeTab.value !== currentTabId.toString()) return // 校验:容器是否合法 if (!(container instanceof HTMLDivElement)) return // 1. 中断前一次未完成的渲染(解决重复加载) if (abortController) { abortController.abort() abortController = null } abortController = new AbortController() // 2. 清空容器(删除所有旧内容,防止残留) container.innerHTML = '' // 获取当前标签页的滚动容器 const scrollContainer = scrollRefMap.get(currentTabId) if (!scrollContainer) return try { // 3. 加载 PDF 文件(显式声明 PDFLoadingTask 类型,解决 promise 类型错误) const pdfLoadingTask: PDFLoadingTask<PDFDocumentProxy> = pdfjsLib.getDocument({ url: pdfUrl, signal: abortController.signal }) const pdfDoc = await pdfLoadingTask.promise // 现在 TS 能识别 promise 属性 // 渲染第一页(显式声明 Page 类型) const pdfPage = await pdfDoc.getPage(1) // 再次校验标签页(异步延迟防护) if (activeTab.value !== currentTabId.toString()) return // 4. 计算渲染尺寸(基于当前标签页的缩放比例) const baseViewport = pdfPage.getViewport({ scale: 1 }) tabBaseViewport.set(currentTabId, baseViewport) const currentScale = tabScale.get(currentTabId) ?? 1 // 默认缩放 1 // 基准宽度 800px,适配弹窗初始宽度 const renderScale = currentScale * (800 / baseViewport.width) const renderViewport = pdfPage.getViewport({ scale: renderScale }) // 5. 创建 Canvas 并渲染 PDF const canvas = document.createElement('canvas') const ctx = canvas.getContext('2d') if (!ctx) { container.innerHTML = '<div style="padding:20px;text-align:center;color:#f56c6c;">Canvas 初始化失败</div>' return } canvas.width = renderViewport.width canvas.height = renderViewport.height canvas.dataset.scale = String(currentScale) // 存储当前缩放比例 container.appendChild(canvas) // 渲染 PDF 内容(显式声明 RenderTask 类型,解决参数类型警告) const renderTask: RenderTask = pdfPage.render({ canvasContext: ctx, viewport: renderViewport, signal: abortController.signal } as Parameters<typeof pdfPage.render>[0]) // 类型断言匹配渲染参数 await renderTask.promise // 第三次校验标签页(渲染完成后防护) if (activeTab.value !== currentTabId.toString()) { container.innerHTML = '' return } // 6. 设置滚动容器尺寸(适配 PDF 渲染尺寸) scrollContainer.style.width = `${renderViewport.width}px` scrollContainer.style.height = `${renderViewport.height}px` pdfRealSize.value = { width: renderViewport.width, height: renderViewport.height } // 7. 若有高亮需求,渲染高亮区域 if (props.highlight) { drawHighlight(props.highlight, currentTabId) } } catch (error) { // 忽略中断错误,其他错误提示 if ((error as Error).name !== 'AbortError') { console.error(`PDF 加载失败(标签页: ${currentTabId}):`, error) container.innerHTML = `<div style="padding:20px;text-align:center;color:#f56c6c;">PDF 加载失败,请重试</div>` } } } /* ---------------- 标签页切换 & 缩放 ---------------- */ /** * 切换标签页时触发(防止 el-tabs 自带事件与 watch 冲突) */ function handleTabClick(tab: TabsPaneContext) { emit('tabschange', tab.paneName) } /** * 监听 activeTab 变化:切换标签页时重新渲染 PDF */ watch(activeTab, (newTabIdStr) => { // 将字符串 tabId 转回原始类型(适配 Map 存储的 key) const newTabId = tabs.value.find(tab => tab.id.toString() === newTabIdStr)?.id if (!newTabId) return const targetTab = tabs.value.find(tab => tab.id === newTabId) const targetContainer = pdfContainerMap.get(newTabId) if (targetTab && targetContainer) { renderSinglePDF(targetTab.url, targetContainer, newTabId) } }, { immediate: false }) /** * 放大 PDF(当前激活标签页) */ const zoomIn = () => { const currentTabIdStr = activeTab.value const currentTabId = tabs.value.find(tab => tab.id.toString() === currentTabIdStr)?.id if (!currentTabId) return // 缩放范围:0.5 ~ 3 const newScale = Math.min((tabScale.get(currentTabId) ?? 1) + 0.2, 3) tabScale.set(currentTabId, newScale) rerenderActiveTab() } /** * 缩小 PDF(当前激活标签页) */ const zoomOut = () => { const currentTabIdStr = activeTab.value const currentTabId = tabs.value.find(tab => tab.id.toString() === currentTabIdStr)?.id if (!currentTabId) return // 缩放范围:0.5 ~ 3 const newScale = Math.max((tabScale.get(currentTabId) ?? 1) - 0.2, 0.5) tabScale.set(currentTabId, newScale) rerenderActiveTab() } /** * 重新渲染当前激活的标签页 */ async function rerenderActiveTab() { const currentTabIdStr = activeTab.value const currentTabId = tabs.value.find(tab => tab.id.toString() === currentTabIdStr)?.id if (!currentTabId) return const targetTab = tabs.value.find(tab => tab.id === currentTabId) const targetContainer = pdfContainerMap.get(currentTabId) if (targetTab && targetContainer) { await renderSinglePDF(targetTab.url, targetContainer, currentTabId) } } /* ---------------- 弹窗拖拽功能 ---------------- */ /** * 开始拖拽弹窗(仅头部和遮罩区域触发) */ function startDrag(e: MouseEvent) { const target = e.target as HTMLElement // 仅允许点击遮罩或头部拖拽 if (!target.classList.contains('pdf-mask') && !target.closest('.pdf-header')) { return } if (!modalRef.value) return dragStart = { x: e.clientX, y: e.clientY } modalStyle.cursor = 'grabbing' // 现在类型兼容(CSSProperties['cursor'] 支持) // 绑定全局拖拽/停止事件(使用同一函数引用) document.addEventListener('mousemove', onDrag) document.addEventListener('mouseup', stopDrag) e.preventDefault() } /** * 拖拽中:更新弹窗位置 */ function onDrag(e: MouseEvent) { if (!dragStart) return const dx = e.clientX - dragStart.x const dy = e.clientY - dragStart.y // 更新弹窗位置(基于当前位置累加偏移) modalStyle.left = `${parseFloat(modalStyle.left) + dx}px` modalStyle.top = `${parseFloat(modalStyle.top) + dy}px` // 更新拖拽起始点 dragStart = { x: e.clientX, y: e.clientY } } /** * 停止拖拽:清理事件和状态 */ function stopDrag() { dragStart = null modalStyle.cursor = 'default' // 解绑全局事件(与绑定的函数一致) document.removeEventListener('mousemove', onDrag) document.removeEventListener('mouseup', stopDrag) } /* ---------------- PDF 内容拖拽(放大后支持,修复事件参数) ---------------- */ let isPdfDragging = false // 是否处于 PDF 拖拽状态 let pdfDragStart = { x: 0, y: 0 } // PDF 拖拽起始坐标 let scrollStart = { left: 0, top: 0 } // 滚动容器起始位置 /** * PDF 内容鼠标按下:开始拖拽(仅放大时支持,存储当前滚动容器) */ function onPdfMouseDown(e: MouseEvent, tabId: string | number) { const scrollContainer = scrollRefMap.get(tabId) if (!scrollContainer) return // 仅当缩放比例 >1 时允许拖拽 const currentScale = tabScale.get(tabId) ?? 1 if (currentScale <= 1) return isPdfDragging = true pdfDragStart = { x: e.clientX, y: e.clientY } scrollStart = { left: scrollContainer.scrollLeft, top: scrollContainer.scrollTop } // 存储当前拖拽对应的滚动容器(解决事件参数不匹配) currentScrollContainer = scrollContainer // 绑定全局拖拽/停止事件(使用无额外参数的函数) document.addEventListener('mousemove', onPdfMouseMove) document.addEventListener('mouseup', onPdfMouseUp) e.preventDefault() } /** * PDF 内容拖拽中:更新滚动位置(不依赖外部参数,从状态获取容器) */ function onPdfMouseMove(e: MouseEvent) { if (!isPdfDragging || !currentScrollContainer) return const dx = e.clientX - pdfDragStart.x const dy = e.clientY - pdfDragStart.y // 反向滚动(拖拽方向与滚动方向相反) currentScrollContainer.scrollTo({ left: scrollStart.left - dx, top: scrollStart.top - dy, behavior: 'auto' // 即时滚动,无动画 }) } /** * PDF 内容拖拽停止:清理事件和状态 */ function onPdfMouseUp() { isPdfDragging = false currentScrollContainer = null // 清空当前滚动容器 // 解绑全局事件(与绑定的函数一致) document.removeEventListener('mousemove', onPdfMouseMove) document.removeEventListener('mouseup', onPdfMouseUp) } /* ---------------- 滚轮事件(防止冒泡影响页面) ---------------- */ function onWheel(e: WheelEvent, tabId: string | number) { e.stopPropagation() // 阻止滚轮事件冒泡到页面 const scrollContainer = scrollRefMap.get(tabId) if (scrollContainer) { // 自定义滚轮行为(横向+纵向滚动) scrollContainer.scrollLeft -= e.deltaX scrollContainer.scrollTop -= e.deltaY e.preventDefault() // 防止页面整体滚动 } } /* ---------------- 高亮区域绘制 ---------------- */ /** * 绘制 PDF 高亮区域(支持外部调用) * @param rect 高亮坐标(PDF 原始坐标) * @param tabId 标签页ID(默认当前激活标签页) */ function drawHighlight( rect: { x0: number; y0: number; x1: number; y1: number }, tabId?: string | number ) { const targetTabId = tabId ?? ( tabs.value.find(tab => tab.id.toString() === activeTab.value)?.id ?? '' ) const pdfContainer = pdfContainerMap.get(targetTabId) const scrollContainer = scrollRefMap.get(targetTabId) if (!pdfContainer || !scrollContainer) return // 1. 清除旧的高亮区域 pdfContainer.querySelectorAll('.pdf-highlight').forEach(el => el.remove()) // 2. 获取 Canvas 和基础数据 const canvas = pdfContainer.querySelector('canvas') const baseViewport = tabBaseViewport.get(targetTabId) if (!canvas || !baseViewport) return const currentScale = parseFloat(canvas.dataset.scale || '1') // 计算 PDF 原始坐标到 Canvas 渲染坐标的比例 const scaleX = canvas.width / baseViewport.width const scaleY = canvas.height / baseViewport.height // 3. 计算高亮区域在 Canvas 上的位置和尺寸 const highlightStyle = { left: `${rect.x0 * scaleX}px`, top: `${rect.y0 * scaleY}px`, width: `${(rect.x1 - rect.x0) * scaleX}px`, height: `${(rect.y1 - rect.y0) * scaleY}px` } // 4. 创建高亮元素 const highlightEl = document.createElement('div') highlightEl.className = 'pdf-highlight' Object.assign(highlightEl.style, { ...highlightStyle, position: 'absolute', background: 'rgba(0, 255, 0, 0.4)', border: '2px dashed #ff4d4f', pointerEvents: 'none', // 不影响鼠标事件 zIndex: 9999 // 确保高亮在最上层 }) // 5. 添加到容器并自动滚动到高亮区域 pdfContainer.style.position = 'relative' pdfContainer.appendChild(highlightEl) // 等待 DOM 更新后执行滚动 nextTick(() => { const containerWidth = scrollContainer.clientWidth const containerHeight = scrollContainer.clientHeight // 滚动到高亮区域中心 scrollContainer.scrollTo({ left: Math.max(0, parseFloat(highlightStyle.left) + parseFloat(highlightStyle.width) / 2 - containerWidth / 2), top: Math.max(0, parseFloat(highlightStyle.top) + parseFloat(highlightStyle.height) / 2 - containerHeight / 2), behavior: 'smooth' // 平滑滚动 }) }) } /** * 外部调用的高亮方法(适配暴露接口) */ function drawHighlightByCoords(rect: { x0: number; y0: number; x1: number; y1: number }) { drawHighlight(rect) } /* ---------------- 监听 Props 变化 ---------------- */ // 监听 visible 变化:弹窗显示/隐藏时的处理 watch( () => props.visible, async (isVisible, oldVisible) => { // 弹窗隐藏时:重置缩放和中断渲染 if (!isVisible) { tabScale.clear() if (abortController) { abortController.abort() abortController = null } currentScrollContainer = null // 清空拖拽容器状态 return } // 弹窗从隐藏变为显示时:居中弹窗 await nextTick() if (oldVisible === false) { const windowWidth = window.innerWidth const windowHeight = window.innerHeight modalStyle.left = `${(windowWidth - 800) / 2}px` modalStyle.top = `${(windowHeight - 600) / 2}px` } }, { immediate: true } ) // 监听 highlight 变化:外部传入高亮坐标时绘制 watch( () => props.highlight, (newHighlight) => { if (newHighlight && activeTab.value) { drawHighlight(newHighlight) } }, { immediate: false, deep: true } ) /* ---------------- 关闭弹窗 ---------------- */ function close() { // 中断渲染并重置状态 if (abortController) { abortController.abort() abortController = null } currentScrollContainer = null pdfRealSize.value = { width: 0, height: 0 } emit('update:visible', false) } /* ---------------- 组件卸载清理 ---------------- */ onUnmounted(() => { // 1. 中断所有未完成的渲染 if (abortController) { abortController.abort() abortController = null } // 2. 解绑所有全局事件(防止内存泄漏,确保与绑定函数一致) document.removeEventListener('mousemove', onDrag) document.removeEventListener('mouseup', stopDrag) document.removeEventListener('mousemove', onPdfMouseMove) document.removeEventListener('mouseup', onPdfMouseUp) // 3. 清空所有状态 pdfContainerMap.clear() scrollRefMap.clear() tabScale.clear() tabBaseViewport.clear() currentScrollContainer = null }) /* ---------------- 暴露外部调用的方法 ---------------- */ defineExpose({ openWithTabs, // 打开 PDF 标签页 drawHighlight: drawHighlightByCoords // 绘制高亮区域 }) </script> <style scoped> /* 遮罩层:全屏覆盖 */ .pdf-mask { position: fixed; inset: 0; z-index: 9999; pointer-events: none; /* 仅弹窗区域可点击 */ background: rgba(0, 0, 0, 0.3); /* 增加半透明背景,提升体验 */ } /* 弹窗容器:可拖拽、可缩放 */ .pdf-modal { position: absolute; background: #ffffff; border: 1px solid #e4e7ed; border-radius: 8px; box-shadow: 0 8px 24px rgba(0, 0, 0, 0.15); display: flex; flex-direction: column; pointer-events: auto; min-width: 400px; min-height: 300px; resize: both; /* 允许手动缩放弹窗 */ overflow: hidden; } /* 弹窗头部:标题 + 操作按钮 */ .pdf-header { display: flex; justify-content: space-between; align-items: center; padding: 12px 16px; background: #f5f7fa; cursor: grab; /* 拖拽光标 */ border-bottom: 1px solid #e4e7ed; user-select: none; /* 禁止文本选中 */ } .pdf-header:active { cursor: grabbing; /* 拖拽中光标 */ } .pdf-header .title { font-size: 16px; font-weight: 500; color: #1f2937; } .pdf-header .tools { display: flex; gap: 8px; /* 按钮间距 */ } /* 标签页容器:占满弹窗剩余高度 */ .pdf-tabs { flex: 1; display: flex; flex-direction: column; overflow: hidden; } /* PDF 内容容器:滚动区域 */ .pdf-body { flex: 1; overflow: auto; position: relative; /* 为高亮区域提供定位上下文 */ } /* PDF Canvas:禁止鼠标事件(避免影响拖拽) */ .pdf-body canvas { pointer-events: none; display: block; /* 消除 Canvas 默认空隙 */ } /* 高亮区域样式 */ .pdf-highlight { box-sizing: border-box; /* 边框不影响尺寸计算 */ } /* 适配 Element Plus 标签页样式(避免冲突) */ .pdf-tabs .el-tabs__content { flex: 1; overflow: hidden; padding: 0 !important; /* 清除默认内边距 */ } .pdf-tabs .el-tab-pane { height: 100%; display: flex; flex-direction: column; } </style>这个代码里面有个别是错误的,比如没有PDFLoadingTask,需要帮我重新分析下代码,功能逻辑,把错误的更新掉,并给我重新生成代码
08-28
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值