Vue.Draggable 踩坑:add 事件与 change 事件中 newIndex 字段不同之谜

news2025/1/14 1:19:36

背景

  最近在弄自定义表单,需要拖动组件进行表单设计,所以用到了 Vue.Draggable(中文文档)。Vue.Draggable 是一款基于 Sortable.js 实现的 vue 拖拽插件,文档挺简单的,用起来也方便,但没想到接下来给我遇到了灵异事件…

坑的表现

  当我写完了由配置对象到组件的渲染逻辑之后,便开始了阶段性测试。我先是拖入了一个输入框,它正常的渲染了出来,并且各项功能都很正常。
在这里插入图片描述

  然后我又拖了个文本域进去,随手把它放在输入框的下面。

在这里插入图片描述

  结果意想不到的事发生了,文本域居然跑到了输入框的上面去了,我惊呆了…

在这里插入图片描述
  印象中拖入放置时元素在列表中的位置是 Vue.Draggable 自己维护的啊,我没做什么控制,怎么可能出问题呢?满脑子疑惑的我又拖了个文本域放在输入框下面,结果它有一次惊呆了我,它正常了,没有跑到输入框上面去…

  我刷新页面打算重新试一下。

    ● 第一步,拖入一个输入框,正常。
    ● 第二步,拖入一个文本域放在输入框下面,不正常,跑上面去了。
    ● 第三步,再次拖入一个文本域放在输入框下面,正常。

  好家伙,看来按这个步骤是百分百重现了。老老实实去检查代码,确认没有手动维护过 Vue.Draggable 中的 list。在 add 事件中打印 event.newIndex (以下称 addEvent.newIndex),发现 addEvent.newIndex 的值是正常的,但是却与新增元素在 list 中的下标不一致,又在 change 事件中打印 newIndex (以下称 changeEvent.newIndex),发现 changeEvent.newIndex 却是指向新元素在 list 中的位置。

  但是 changeEvent.newIndex 的值不对啊!它应该跟 addEvent.newIndex 一样才对啊!文本域应该在输入框的下面才对啊!啊啊啊!!!难道我发现了 Vue.Draggable 的 BUG?

填坑

  结论直达

  两个事件中的 newIndex 完全是由 Vue.Draggable 自身维护的,要想找到导致它俩不同的原因,只能去看看 Vue.Draggable 的源码了。于是我拉取了 Vue.Draggable 的源码,打算来研究一下。值得庆幸的是 Vue.Draggable 的源码很少,只有 400 多行,读起来比较简单。

  我很快找到了下面处理 add 事件的代码。

// Vue.Draggable 源码

// onDragAdd 方法是 Draggable 组件内部方法,它调用之后才会 emit Draggable 组件的 add 事件
// 可以在源码中搜索 delegateAndEmit 查找绑定事件的位置
// vue 组件 methods 选项中的方法
onDragAdd(evt) {
    const element = evt.item._underlying_vm_;
    if (element === undefined) {
        return;
    }
    removeNode(evt.item);
    // evt.newIndex 是 add 事件中的 newIndex
    const newIndex = this.getVmIndex(evt.newIndex);
    this.spliceList(newIndex, 0, element);
    this.computeIndexes();
	
	// added.newIndex 是 change 事件中的 newIndex
    const added = { element, newIndex };
    this.emitChanges({ added });
},

  从上面的代码中可以看出,change 事件中的 newIndexadd 事件中的 newIndex 经由 this.getVmIndex() 方法加工而来的。那么我们看一下 this.getVmIndex() 方法做了什么加工导致了它们的不一样。

// Vue.Draggable 源码

// added.newIndex 依赖于 this.visibleIndexes (一个数组),当新元素的下标小于 this.visibleIndexes 的长度减一时,返回 this.visibleIndexes 的长度,否则返回 this.visibleIndexes 中下标为 domIndex 的值
/**
 * vue 组件 methods 选项中的方法,计算并返回 change 事件中的 newIndex
 * @param {number} domIndex - add 事件中的 newIndex
 * @returns {number}
 */
getVmIndex(domIndex) {
    const indexes = this.visibleIndexes;
    const numberIndexes = indexes.length;
    return domIndex > numberIndexes - 1 ? numberIndexes : indexes[domIndex];
},

  getVmIndex() 方法用于计算 changeEvent.newIndex。它的参数 domIndex 即为 addEvent.newIndex

  从上面的代码可以看出 changeEvent.newIndex 还依赖于 this.visibleIndexes(一个数组),当 domIndex(新元素的下标)大于 this.visibleIndexes 的长度 - 1(最后一个元素的下标)时,changeEvent.newIndexthis.visibleIndexes的长度(最后一个元素的下标 + 1),否则为 this.visibleIndexes 中下标为 domIndex 的值。

  看来还要弄明白 this.visibleIndexes 是什么,下面的代码说明了 this.visibleIndexes 的由来。

// vue 组件 methods 选项中的方法,
computeIndexes() {
	this.$nextTick(() => {
		// 这个 computeIndexes 并不是在 methods 中声明的,因此调用时没有使用 this
		this.visibleIndexes = computeIndexes(
			this.getChildrenNodes(),
			this.rootContainer.children,
			this.transitionMode,
			this.footerOffset
		);
	});
},

/**
 * 计算 this.visibleIndexes 列表
 * @param {Array<VNode>} slots - isTransition 为 true 时,表示 TransitionGroup 的默认插槽,否则表示 draggable 组件的默认插槽
 * @param {Array<Node>} children - isTransition 为 true 时,表示 TransitionGroup 子元素列表,否则表示 draggable 组件子元素列表
 * @param {boolean} isTransition - 是否使用了 TransitionGroup 组件
 * @param {number} footerOffset - footer 插槽根元素的个数,没有使用 footer 插槽时为 0
 * @returns
 */
function computeIndexes(slots, children, isTransition, footerOffset) {
    if (!slots) {
        return [];
    }

    const elmFromNodes = slots.map(elt => elt.elm);
    const footerIndex = children.length - footerOffset;

	// rawIndexes 列表表示显示的节点,其虚拟节点在 slots 中的位置
    const rawIndexes = [...children].map((elt, idx) => {
        return idx >= footerIndex ? elmFromNodes.length : elmFromNodes.indexOf(elt);
    });
	
	// 如果使用了 TransitionGroup 组件,则将 children 中有而 slots 中没有的过滤掉
    return isTransition ? rawIndexes.filter(ind => ind !== -1) : rawIndexes;
}

  由上面的代码可以看出 rawIndexes 数组表示:显示的节点,其虚拟节点在 slots 中的位置。rawIndexes 元素的下标表示节点在 children 中的下标,元素的值表示节点在 slots 中的下标(如果节点在 footer 之后,则值为 slots 的长度)。

  现在我们再来看 getVmIndex() 方法,this.visibleIndexes 是由 slotschildren 维护的,而它决定了 changeEvent.newIndex 的值,所以影响 changeEvent.newIndex 的根本因素就是 slotschildren

  找到了根本因素接下来就简单了。我重复执行出现问题的操作步骤,然后在这个过程中打印 slotschildren,我惊讶的发现当我向 Vue.Draggable 第一次拖入输入框组件时,slotschildren 的长度居然不一样!children 是空的, 而 slots 的长度虽然正常,但其中的虚拟节点的 elm 属性却是 undefined

  看到这里我恍然大悟,正是 slotschildren 异常的值导致了 changeEvent.newIndex 的计算错误,那么是什么导致了它们值的异常呢?也许你有注意到 slots 虽然长度正常,但其中的虚拟节点的 elm 属性却是 undefined

  是的,没错,正是因为输入框组件采用了懒加载的方式进行引入,而导致的这个诡异的问题!

  万万没想到,组件的引入方式居然还会导致奇怪的问题出现!

总结

  所以,如果想在 Vue.Draggable 中使用自定义组件,那么千万不要使用懒加载的方式引入这些组件!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1192412.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Python数据容器(元组)

元组 1.定义元组 定义元组使用小括号&#xff0c;且使用逗号隔开各个数据&#xff0c;数据是不同的数据类型 # 定义元组字面量 (元素,元素,...,元素) # 定义元组变量 变量名称 (元素,元素,...,元素) # 定义空元组 变量名称 () 变量名称 tuple()2.元组的相关操作 编号方法…

JavaScript_Node节点属性_nodeName

nodeName属性&#xff1a;返回节点的名称 节点的类型有七种 Document&#xff1a;整个文档树的顶层节点 DocumentType&#xff1a;doctype标签 Element&#xff1a;网页的各种HTML标签 Attribute&#xff1a;网页元素的属性 Text&#xff1a;标签之间或标签包含的文本 C…

【kylin】使用nmtui软件配置网桥

文章目录 一、什么是网桥二、如何配置网桥域名解析失败 一、什么是网桥 网桥也叫桥接器&#xff0c;是连接两个局域网的一种存储/转发设备&#xff0c;它能将一个大的LAN分割为多个网段&#xff0c;或将两个以上的LAN互联为一个逻辑LAN&#xff0c;使LAN上的所有用户都可访问服…

go 引入包报错“构建约束排除‘D/...vendor/pkg包’”中所有的GO文件

解决方案&#xff1a; 方案一&#xff1a;没生效 go - 构建约束排除所有 Go 文件 - IT工具网 go modules - build constraints exclude all Go files in - Stack Overflow 方案二&#xff1a;生效&#xff0c;手动初始化创建一个目录 后续再研究原因&#xff0c;有明白的大…

第18章Swing程序设计

Swing程序设计 Swing用于开发桌面窗体程序用于JDK的第二代GUI框架&#xff0c;其功能比JDK第一代GUI框架AWT更为强大&#xff0c;性能更加优良。但因为Swing技术推出时间太早&#xff0c;七性能&#xff0c;开发效率等不及一些其他的留下技术&#xff0c;所以目前市场大多数桌面…

ISP算法——UVNR

ISP算法——UVNR 概念简介 UVNR也就是经过CSC只有在YUV域对UV两个色域进行降噪&#xff0c;在有些方案里也叫CNR&#xff08;chroma noise reduction&#xff09;。主要就是在YUV域针对彩燥进行特殊处理的一系列算法。 关于噪声产生的原因在前面关于降噪的文章和视频中已经做…

latex cite命令、款式

UTS SEDE 的 latex 模板 [1,2] 用 biblatex&#xff0c;默认用的引用格式是 ieee。然而 Research Foundation 的 literature review 这个作业要用 APA 7&#xff0c;想在保留 biblatex 的情况下区分有括号和无括号两种引用格式&#xff0c;即 [3] 中 \citet、\citep 的分别。 …

OSG交互:选中场景模型并高亮显示

1、目的 可以在osg视图中选中指定模型实体,并高亮显示。共分为两种,一种鼠标点选,一种框选。 2、鼠标点选 2.1 功能说明 生成两组对象,一组cow对象可以被选中,另一组robot不能被选中;点击cow对象被选中高亮,点击robot被选中不高亮;点击空白处,弹出“select nothing!…

rocketMq消息堆积处理方式

消息堆积常见于以下几种情况&#xff1a; &#xff08;1&#xff09;新上线的消费者功能有BUG&#xff0c;消息无法被消费。 &#xff08;2&#xff09;消费者实例宕机或因网络问题暂时无法同Broker建立连接。 &#xff08;3&#xff09;生产者短时间内推送大量消息至Broker…

PTA_乙级_1096

Q1&#xff1a;因数 在数学中&#xff0c;一个数的因数是能够整除该数的整数。换句话说&#xff0c;如果我们将一个数 a 除以另一个整数 b 而得到整数商&#xff0c;那么 b 就是 a 的因数。以下是一些例子&#xff1a; 1.因数的定义&#xff1a; 如果整数 b 可以被整数 a 整除&…

IP行业API助力于网络分析和数据挖掘

引言 在当今数字化时代&#xff0c;数据成为了企业、科研机构和政府决策者的重要资源&#xff0c;而IP行业API则成为了数据分析及挖掘的工具之一。IP行业API是一种能够查询IP地址所属的行业分类信息的应用程序接口&#xff0c;它能够提供在网络分析、用户行为分析及大数据挖掘…

奇异矩阵、非奇异矩阵

对于一个方阵A&#xff1a; 如果A的行列式等于0&#xff0c;称矩阵A为奇异矩阵如果A的行列式不等于0&#xff0c;称A 非奇异矩阵 也就是说&#xff0c;对于方阵A&#xff0c;如果它是满秩的&#xff0c;即它的秩等于矩阵的阶数&#xff0c;就是非奇异矩阵&#xff1b;如果秩小…

MS2358:96KHz、24bit 音频 ADC

MS2358 是带有采样速率 8kHz-96kHz 的立体声音频模数 转换器&#xff0c;适合于面向消费者的专业音频系统。 MS2358 通过使用增强型双位 Δ - ∑ 技术来实现其高精度 的特点。 MS2358 支持单端的模拟输入&#xff0c;所以不需要外部器 件&#xff0c;非常适合用于像 …

【无标题(PC+WAP)花卉租赁盆栽绿植类pbootcms站模板

(PCWAP)花卉租赁盆栽绿植类pbootcms网站模板 PbootCMS内核开发的网站模板&#xff0c;该模板适用于盆栽绿植网站等企业&#xff0c;当然其他行业也可以做&#xff0c;只需要把文字图片换成其他行业的即可&#xff1b; PCWAP&#xff0c;同一个后台&#xff0c;数据即时同步&…

【iOS开发】iOS App的加固保护原理:使用ipaguard混淆加固

​ 摘要 在开发iOS应用时&#xff0c;保护应用程序的安全是非常重要的。本文将介绍一种使用ipaguard混淆加固的方法来保护iOS应用的安全。通过字符串混淆、类名和方法名混淆、程序结构混淆加密以及反调试、反注入等主动保护策略&#xff0c;可以有效地保护应用程序的安全性。 …

远程运维如何更高效的远程管理?向日葵的这几项功能会帮到你

具备一定规模的企业&#xff0c;其IT运维需求普遍会面临设备数量众多、难以统一高效管理、始终存在安全敞口等问题&#xff0c;尤其是针对分部广泛的无人值守设备时&#xff0c;更是如此。 举一个简单的例子&#xff0c;一台位于商圈的无人值守可互动广告机设备&#xff0c;所…

怎么设置代理IP进行网络爬取呢?代理访问网络如何设置?

在如今网络爬虫广泛应用的年代&#xff0c;很多时候我们都会遇到需要使用代理IP进行网络爬取的情况。代理IP可以帮助我们隐藏真实的IP地址&#xff0c;从而保护我们的隐私和安全。那么&#xff0c;怎么设置代理IP进行网络爬取呢&#xff1f;代理访问网络如何设置&#xff1f;下…

腾讯云3年540,买其他服务器的都是韭菜!

你是否曾经为选择一款合适的云服务器而烦恼&#xff1f;市场上的云服务器品牌繁多&#xff0c;价格各异&#xff0c;如何才能找到一款性价比高&#xff0c;又适合自己的服务器呢&#xff1f;今天&#xff0c;我要给大家介绍一款腾讯云服务器&#xff0c;3年只需540元&#xff0…

viple进阶4:打印空心三角形

题目&#xff1a;根据用户输入的行数n打印空心三角形&#xff0c;下图分别为n3、n4、n5和n10的效果图 第一步&#xff1a;观察效果图 输入的行数为3&#xff0c;打印结果就有3行&#xff1b;输入的行数为4&#xff0c;则打印结果就有4行&#xff1b;以此类推&#xff0c;输入的…

LED显示屏像素技术

LED显示屏的像素技术是LED显示屏的核心技术之一&#xff0c;它决定了显示屏的清晰度、亮度和色彩表现。以下是一些常见的LED显示屏像素技术&#xff1a; 直插式LED显示屏像素技术&#xff1a;该技术采用LED灯珠直接插入到电路板上的方式&#xff0c;通过电路板上的电路连接实现…