PopupInner源码分析 -- ant-design-vue系列
PopupInner源码分析 – ant-design-vue系列
1 综述
上一篇讲解了vc-align
的工作原理,也就是对齐是如何完成的。这一篇主要讲述包裹 Align
的组件:PopupInner
组件是如何工作的。
PopupInner
主要是对动画状态的管理,比如打开弹窗的时候,弹出的动画是从上到下的,关闭的时候正好相反。我们把动画时长改成两秒,进行观察。
**动画的管理,更深层次来说是对元素类名的管理。**接下来需要搞清楚在动画的过程中,状态是如何发生变化的。
2 极简代码
使用vue3
提供的动画组件Transition
来包裹Align
组件,但是没有对Transition
组件设置任何属性,所以暂时还没有动画效果。
return () => (<Transition>{props.visible ? (<Align align={props.align} target={props.target}>{slots.default?.()}</Align>) : null}</Transition>
);
3 源码实现
3.1 前置知识
组件Transition
:https://cn.vuejs.org/guide/built-ins/transition.html#transition-on-appear
3.2 PopupInner
组件定义的动画状态
源码地址:https://github.com/vueComponent/ant-design-vue/blob/main/components/vc-trigger/Popup/useVisibleStatus.ts
在源码中有这样一个hook
,监控visible
的变化,并且提供了当前状态和修改状态的函数。doMeasure
是另一个hook
提供的函数,目的是得到source
节点的宽/高度。
// ======================== Status ========================const [status, goNextStatus] = useVisibleStatus(visible, doMeasure);
在useVisibleStatus
函数中,注释如下:
/*** 每个组件正确工作的步骤如下,弹出窗口应遵循这个流程。* measure - 检查拉伸尺寸值* align - 让组件对齐位置* aligned - 再次重新对齐,以防止其他className更改了大小(这一步后续再解释)* afterAlign - 选择下一步是触发运动还是结束* beforeMotion - 应将motion重置为invisible,以便CSSMotion可以进行正常运动* motion - 执行动作* stable 一切结束*/
对源码进行拆解:
- 整体框架:
type PopupStatus = null | 'measure' | 'align' | 'aligned' | 'motion' | 'stable';
type Func = () => void;
const StatusQueue: PopupStatus[] = ['measure', 'align', null, 'motion'];export default (visible: Ref<boolean>,doMeasure: Func,
): [Ref<PopupStatus>, (callback?: () => void) => void] => {// 弹出状态const status = ref<PopupStatus>(null);// raf = (callback: FrameRequestCallback) => window.requestAnimationFrame(callback);// rafRef 就是requestAnimationFrame执行器的引用,这个hook不做拆解了。const rafRef = ref<number>();// 标记组件的销毁状态const destroyRef = ref(false);const goNextStatus = () => {// ......}onMounted(() => {// ......})return [status, goNextStatus];
}
🎯 状态驱动如图所示:
- 监控visible的变化
当visible
变化的时候,重新从measure
状态开始流转。
watch(visible,() => {setStatus('measure');},{ immediate: true, flush: 'post' },
);
- 监听status的变化
在组件挂载时,监控了status
的变化。
如果当前是measure
,调用传入的doMeasure
,测量target
的大小。接下来会设置新状态:
- 如果
status === 'measure'
,则执行setStatus('align')
- 如果
status === 'align'
,这里因为要多次定位,所以nextStatus
为空,也就是无法自动进入下个状态 - 如果
status === 'motion'
,nextStatus
同样为空,无法自动进入下个状态
所以,当popup
处于align
和motion
状态时,都需要等待动作完成,由外部调用goNextStatus
来进入下个状态。
onMounted(() => {// Go next statuswatch(status,() => {switch (status.value) {case 'measure':doMeasure();break;default:}if (status.value) {rafRef.value = raf(async () => {const index = StatusQueue.indexOf(status.value);const nextStatus = StatusQueue[index + 1];if (nextStatus && index !== -1) {setStatus(nextStatus);}});}},{ immediate: true, flush: 'post' },);
});function setStatus(nextStatus: PopupStatus) {if (!destroyRef.value) {status.value = nextStatus;}
}
- goNextStatus 函数
可以看到,正好弥补了上一个watch
不能自动触发的状态变更。
function goNextStatus(callback?: () => void) {// 取消原先的动作,注册新的。目的是为了动画的流程。cancelRaf();rafRef.value = raf(() => {// Only align should be manually triggerlet newStatus = status.value;switch (status.value) {case 'align':newStatus = 'motion';break;case 'motion':newStatus = 'stable';break;default:}setStatus(newStatus);callback?.();});
}
- 其他部分
都是一些清理函数。
onBeforeUnmount(() => {destroyRef.value = true;cancelRaf();
});function cancelRaf() {raf.cancel(rafRef.value);
}
3.3 核心变量
先看一下渲染函数,最重要的就是transitionProps
和mergedStyle
,其他变量暂时不管。
return (<Transition// ......{...transitionProps}v-slots={{default: () => {return !destroyPopupOnHide || props.visible ? (<Align// ......v-slots={{default: () => (<div// ......style={mergedStyle}>{childNode}</div>),}}></Align>) : null;},}}></Transition>
);
对代码进行debug
,观察这两个变量在动画状态过程中的变化。可以看到,在第一次measure
过程结束后:
const transitionProps = {name: 'ant-slide-up',appear: true,enterFromClass: 'ant-slide-up-enter ant-slide-up-enter-prepare',enterActiveClass: 'ant-slide-up-enter ant-slide-up-enter-prepare',enterToClass: 'ant-slide-up-enter ant-slide-up-enter-active',leaveFromClass: ' ant-slide-up-leave',leaveActiveClass: 'ant-slide-up-leave ant-slide-up-leave-active',leaveToClass: 'ant-slide-up-leave ant-slide-up-leave-active'
};const mergedStyle = [{ minWidth: '76px', opacity: 0, pointerEvents: 'none' }, null];// 后续opacity会变成null,弹窗逐步显现。从源码中可以清晰的看到opacity: statusValue === 'motion' || statusValue === 'stable' || !visible.value ? null : 0,pointerEvents: !visible.value && statusValue !== 'stable' ? 'none' : null,
这个就是动画的核心了,其他代码都是为这个服务的。比如对齐popup
的位置;控制内部status
状态,进而让动画在对齐完成后再开始。
动画的源码:https://github.com/vueComponent/ant-design-vue/blob/main/components/style/core/motion/slide.less
3.4 代码详解
由于代码逻辑都穿插在一起,所以先把所有逻辑分开看一下,最后在用一个流程进行串联。
遵照源码对代码的分块。
3.4.1 Measure
// ======================= Measure ========================
/**
* 如果 stretch 是width,那么stretchStyle返回元素的 width;如果stretch 是minWidth,那么stretchStyle返回元素的 minWidth
* height也是一样的。
*
* 开始时 width = height = 0px
* 只有调用measureStretchStyle才能获得目标元素的属性
*/
const [stretchStyle, measureStretchStyle] = useStretchStyle(toRef(props, 'stretch'));const doMeasure = () => {if (props.stretch) {measureStretchStyle(props.getRootDomNode());}
};const visible = ref(false);
let timeoutId: any;
watch(() => props.visible,val => {clearTimeout(timeoutId);if (val) {/*** 如果visible变成true,那么延迟变更*/timeoutId = setTimeout(() => {visible.value = props.visible;});} else {visible.value = false;}},{ immediate: true },
);
3.4.2 Aligns
// ======================== Aligns ========================
/**
* 解释见下文
*/
const prepareResolveRef = ref<(value?: unknown) => void>();/**
* Align组件的target可以接受函数或者对象,获取的时候也要区分。
*/
const getAlignTarget = () => {if (props.point) {return props.point;}return props.getRootDomNode;
};/**
* 调用Align组件的对齐方法
*/
const forceAlign = () => {alignRef.value?.forceAlign();
};/**
* popupDomNode是弹窗节点,matchAlign是对齐的结果reuslt
* 使用align-dom对齐,会有自适应视口的位置调整,因此result并不一定是是设置的位置
*/
const onInternalAlign = (popupDomNode: HTMLElement, matchAlign: AlignType) => {/*** 根据对齐结果获取类名,比如默认的左上角对齐,类名是 ant-dropdown-placement-bottomLeft*/const nextAlignedClassName = props.getClassNameFromAlign(matchAlign);const preAlignedClassName = alignedClassName.value;if (alignedClassName.value !== nextAlignedClassName) {alignedClassName.value = nextAlignedClassName;}/*** 如果当前状态是align,并且新的类名和旧的不同(也就是对齐方式不同),就重新执行对齐方法,拿到新的对齐结果再比较* 直到对齐无误后,status进入下个阶段motion,同时放开动画,进入动画的下个阶段*/if (status.value === 'align') {// Repeat until not more align neededif (preAlignedClassName !== nextAlignedClassName) {Promise.resolve().then(() => {forceAlign();});} else {goNextStatus(() => {prepareResolveRef.value?.();});}props.onAlign?.(popupDomNode, matchAlign);}
};
prepareResolveRef
是一个Promise
的resolve
方法,执行这个方法,可以让Promise
结束。这个promise
的定义在Motion
模块。
/**
* 从名字可以看出,这个是动画开始前的一个阶段。
*/
const onShowPrepare = () => {return new Promise(resolve => {prepareResolveRef.value = resolve;});
};
使用的地方在Transition
组件,也就是说,onBeforeEnter
要想结束,需要等到prepareResolveRef.value?.()
的执行。
<Transition// ......onBeforeEnter={onShowPrepare}
</Transition>
onInternalAlign
在Align
组件中,有以下这句代码。执行的就是onInternalAlign
这个函数,传入source
和result
if (latestOnAlign && result) {latestOnAlign(source, result);
}
3.4.3 Motion
// ======================== Motion ========================
const motion = computed(() => {/*** 如果设置了动画,就使用设置好的,否则使用默认的。也就是3.3 展示的变量*/const m = typeof props.animation === 'object' ? props.animation : getMotion(props as any);['onAfterEnter', 'onAfterLeave'].forEach(eventName => {/*** 拦截了两个状态* 如果动画状态是onAfterEnter或者onAfterLeave,也就是打开和关闭的结束时间点。* 就手动把status状态设置为stable,同时执行原动画逻辑。*/const originFn = m[eventName];m[eventName] = node => {goNextStatus();// 结束后,强制 stablestatus.value = 'stable';originFn?.(node);};});return m;
});const onShowPrepare = () => {return new Promise(resolve => {prepareResolveRef.value = resolve;});
};/**
* 如果不需要动画,且status是动画状态,则直接进入下个阶段stable
*/
watch([motion, status],() => {if (!motion.value && status.value === 'motion') {goNextStatus();}},{ immediate: true },
);expose({forceAlign,getElement: () => {return (elementRef.value as any).$el || elementRef.value;},
});const alignDisabled = computed(() => {if ((props.align as any)?.points && (status.value === 'align' || status.value === 'stable')) {return false;}return true;
});
3.5 主流程讲解
Align
组件使用了v-show
属性,可以应用Transition
的动画效果。由visible
变量控制。- 当
visible
变成true
时,触发了useVisibleStatus
这个hook
,status
状态开始流转。 - 在
useVisibleStatus
这个hook
中,status被设置成了measure
- 还是在这个hook中,因为
status
被watch
监听,因此触发了回调,执行了doMeasure
方法,算出popup
的样式,同时status
被推到align
- 此时再次触发
useVisibleStatus
这个hook
中status
的监听,状态被推到了null
,也就是这个hook
暂时不能控制状态了。 - 因为
doMeasure
方法的执行,source
的大小发生了变化,对齐方法forceAlign
开始执行(详情请看上一篇)。 - 每次
forceAlign
执行的末尾,都会执行latestOnAlign(source, result);
对应的就是PopupInner
中的onInternalAlign
方法。 - 这个方法中,会多次调用
forceAlign()
,直到source
被准确定位(排除各种视口变化对位置的影响),以便动画的位置没有错误。 - 当位置稳定后,
onInternalAlign
会手动将状态推进到下一步motion
。同时执行prepareResolveRef.value?.()
,让css
动画也进入下一个阶段。 - 如果动画执行完成,通过劫持动画的
onAfterEnter
阶段,推进状态到stable
,到此一次动画结束。
4 总结
本篇介绍了PopupInner
的源码实现,由于状态变化的复杂,所以只要理解流程即可,在实际开发中,我们的对齐定位大多不会如此复杂。如果只允许超着一个方向对齐,那么只要为Transtion
设置类名即可。
相关文章:

PopupInner源码分析 -- ant-design-vue系列
PopupInner源码分析 – ant-design-vue系列 1 综述 上一篇讲解了vc-align的工作原理,也就是对齐是如何完成的。这一篇主要讲述包裹 Align的组件:PopupInner组件是如何工作的。 PopupInner主要是对动画状态的管理,比如打开弹窗的时候&#…...
Maven 的 pom.xml 文件中<dependency> 元素及其各个参数的解释
在 Maven 的 pom.xml 文件中,<dependency> 标签用于定义项目依赖的外部库。每个 <dependency> 元素包含了一系列的子元素,这些子元素定义了依赖库的各种属性。下面是一个典型的 <dependency> 元素及其各个参数的解释: <…...

【信创】Linux终端禁用USB存储 _ 统信 _ 麒麟 _ 方德
原文链接:【信创】Linux终端禁用USB存储 | 统信 | 麒麟 | 方德 Hello,大家好啊!今天给大家带来一篇关于在Linux终端下禁用USB存储设备的文章。禁用USB存储设备可以提高系统的安全性,防止未经授权的人员将数据拷贝到外部存储设备或…...
开放API接口时要注意的安全处理总结
开发API接口:开放给别人调用的接口。未经过安全处理的开发API接口安全弱点:数据窃取(密码等信息被窃取,盗刷,敏感信息的等)——RSA/DES加密: 签名机制在API接口中的应用:签名用于验证…...

FastGPT自定义插件的icon
最近研究FastGPT的自定义插件,经过好几天的折磨,终于实现了一个简单的发送邮件功能,但是呢在使用的时候发现插件的icon是默认的fastgpt的logo,那肯定得自定义一个啊。直接说方法: 1、自定义插件下面的template.json文件…...

SprinBoot+Vue旅游网站的设计与实现
目录 1 项目介绍2 项目截图3 核心代码3.1 Controller3.2 Service3.3 Dao3.4 application.yml3.5 SpringbootApplication3.5 Vue 4 数据库表设计5 文档参考6 计算机毕设选题推荐7 源码获取 1 项目介绍 博主个人介绍:CSDN认证博客专家,CSDN平台Java领域优质…...

代码随想录刷题day27丨455.分发饼干 ,376. 摆动序列 ,53. 最大子序和
代码随想录刷题day27丨455.分发饼干 ,376. 摆动序列 ,53. 最大子序和 1.贪心算法理论基础 贪心的本质是选择每一阶段的局部最优,从而达到全局最优。 这么说有点抽象,来举一个例子: 例如,有一堆钞票,你可以拿走十张&a…...

Detect It Easy
Detect It Easy(简称 DIE)项目的网址为 https://github.com/horsicq/Detect-It-Easy 下载完安装包后,直接双击die.exe即可进入到操作界面 工具介绍: 它可以用来检测程序架构和文件类型。如图所示。其中,「模式」说明程…...
c++开关灯
题目描述 现有 𝑛n 盏灯排成一排,从左到右依次编号为:11,22,……,𝑛n。然后依次执行 𝑚m 项操作。 操作分为两种: 指定一个区间 [𝑎,𝑏][a,b]&…...

DevOps实现CI/CD实战(六)- Jenkins集成k8s
十、 Jenkins集成k8s Jenkins在集成K8s之前,需要搭建k8s集群,具体搭建步骤,完整笔记 https://github.com/ITenderL/ITenderL.github.io/tree/main/docs/DevOps, 包括完整的DevOps的笔记。 1. 准备部署的yml文件 pipeline.yml …...

张雪峰:物联网行业迎高光时刻!如何选择?我们诚聘销售工程师!
作为一间10多年的物联网公司,各位求职人士可以看看我们其中一个招聘要求,和自己需求结合分析分析,希望对你们有所帮助。 【公司实力底蕴】 盈电智控物联网科技(广东)有限公司,2024年7月成立,是…...
利用多文件编程实现顺序表的创建,判满,插入,输出
文章目录 🍊自我介绍🍊利用多文件编程实现顺序表的创建,判满,插入,输出seqlist.cseqlist.hmain.c 你的点赞评论就是对博主最大的鼓励 当然喜欢的小伙伴可以:点赞关注评论收藏(一键四连ÿ…...

百度快照劫持之JS劫持诊断与恢复一例
劫持现象: 百度搜索结果中,被劫持网站出现在搜索结果中, 点击进入网站,网站显示正常,数秒后网站自动跳转到彩票网站f51688.com/ff6/。但是第二次点击搜索结果,正常进入网站缺不会跳转到彩票网站。 初步认…...
深入探讨Go语言中的切片与数组操作
在编程世界中,数组一直是非常流行的数据结构,主要有两个原因:其一是简单易懂,其二是非常灵活,可以存储多种不同类型的数据。在Go语言中,数组的用法有其独特的特点,但与此同时,Go语言…...

【WPS Excel】复制表格时,提示“图片太大,超过部份将被截去“ 问题
WPS表格 2019版本 升级到 WPS最新版 WPS-支持多人在线协作编辑Word、Excel和PPT文档_WPS官方网站 使用最新版就能够解决这个问题,如果仍旧无法解决可以勾选如下配置 重启Excel解决。 请勾选:文件 - 选项 - 编辑 - 不提示且不压缩文件中的图像...

驱动(RK3588S)第九课时:多节点驱动与函数接口
目录 一、多节点概念1、所用到的结构体说明2、函数接口主要是read和write函数2.1、把应用层的数据拷贝给底层2.2、把应用层的数据拷贝给底层 3、应用层的read和write函数4、底层的read和write函数二、ioctl控制命令接口1、概念2、函数介绍应用层和驱动层 三、代码与现象1.编写L…...
Linux系统下配置MySQL
1. 寻找MySQL的配置文件 MySQL的配置文件通常位于以下位置: 在大多数Linux系统上,主配置文件通常位于/etc/mysql/my.cnf或/etc/my.cnf。在macOS上,如果你使用Homebrew安装MySQL,配置文件通常位于/usr/local/etc/my.cnf。在Window…...

信捷 XD PLC POU编程之FB
在使用信捷的POU方式编程,可以建立两种POU:FB和FC。 FB和FC这两种POU又各自可以建立梯形图语言POU和C语言POU。 函数块(FB)是把反复使用的部分程序块转换成一种通用部件,他可以在程序中反复被调用,不仅 提高了程序的开…...

终于有人把云计算、大数据和人工智能讲明白了!
引言 在当今数字化时代,云计算、大数据和人工智能成为了全球科技界的热门话题。这些技术的迅猛发展以及应用范围的不断扩大,正深刻地改变着我们的生活和工作方式。云计算为我们提供了有效的计算和存储能力,大数据则以海量的信息资源为基础&a…...
【编程底层思考】详解Java内存模型(JMM)原理及其作用
Java内存模型(Java Memory Model, JMM)是Java虚拟机(JVM)的一个核心概念,它定义了Java程序中各种变量(线程共享变量)的访问规则,以及在并发环境下,为了确保数据的可见性、…...

安全突围:重塑内生安全体系:齐向东在2025年BCS大会的演讲
文章目录 前言第一部分:体系力量是突围之钥第一重困境是体系思想落地不畅。第二重困境是大小体系融合瓶颈。第三重困境是“小体系”运营梗阻。 第二部分:体系矛盾是突围之障一是数据孤岛的障碍。二是投入不足的障碍。三是新旧兼容难的障碍。 第三部分&am…...
【SSH疑难排查】轻松解决新版OpenSSH连接旧服务器的“no matching...“系列算法协商失败问题
【SSH疑难排查】轻松解决新版OpenSSH连接旧服务器的"no matching..."系列算法协商失败问题 摘要: 近期,在使用较新版本的OpenSSH客户端连接老旧SSH服务器时,会遇到 "no matching key exchange method found", "n…...

Razor编程中@Html的方法使用大全
文章目录 1. 基础HTML辅助方法1.1 Html.ActionLink()1.2 Html.RouteLink()1.3 Html.Display() / Html.DisplayFor()1.4 Html.Editor() / Html.EditorFor()1.5 Html.Label() / Html.LabelFor()1.6 Html.TextBox() / Html.TextBoxFor() 2. 表单相关辅助方法2.1 Html.BeginForm() …...

【p2p、分布式,区块链笔记 MESH】Bluetooth蓝牙通信 BLE Mesh协议的拓扑结构 定向转发机制
目录 节点的功能承载层(GATT/Adv)局限性: 拓扑关系定向转发机制定向转发意义 CG 节点的功能 节点的功能由节点支持的特性和功能决定。所有节点都能够发送和接收网格消息。节点还可以选择支持一个或多个附加功能,如 Configuration …...
在树莓派上添加音频输入设备的几种方法
在树莓派上添加音频输入设备可以通过以下步骤完成,具体方法取决于设备类型(如USB麦克风、3.5mm接口麦克风或HDMI音频输入)。以下是详细指南: 1. 连接音频输入设备 USB麦克风/声卡:直接插入树莓派的USB接口。3.5mm麦克…...
Java求职者面试指南:Spring、Spring Boot、Spring MVC与MyBatis技术解析
Java求职者面试指南:Spring、Spring Boot、Spring MVC与MyBatis技术解析 一、第一轮基础概念问题 1. Spring框架的核心容器是什么?它的作用是什么? Spring框架的核心容器是IoC(控制反转)容器。它的主要作用是管理对…...

Linux中《基础IO》详细介绍
目录 理解"文件"狭义理解广义理解文件操作的归类认知系统角度文件类别 回顾C文件接口打开文件写文件读文件稍作修改,实现简单cat命令 输出信息到显示器,你有哪些方法stdin & stdout & stderr打开文件的方式 系统⽂件I/O⼀种传递标志位…...
第22节 Node.js JXcore 打包
Node.js是一个开放源代码、跨平台的、用于服务器端和网络应用的运行环境。 JXcore是一个支持多线程的 Node.js 发行版本,基本不需要对你现有的代码做任何改动就可以直接线程安全地以多线程运行。 本文主要介绍JXcore的打包功能。 JXcore 安装 下载JXcore安装包&a…...

运动控制--BLDC电机
一、电机的分类 按照供电电源 1.直流电机 1.1 有刷直流电机(BDC) 通过电刷与换向器实现电流方向切换,典型应用于电动工具、玩具等 1.2 无刷直流电机(BLDC) 电子换向替代机械电刷,具有高可靠性,常用于无人机、高端家电…...

暴雨新专利解决服务器噪音与性能悖论
6月1日,我国首部数据中心绿色化评价方面国家标准《绿色数据中心评价》正式实施,为我国数据中心的绿色低碳建设提供了明确指引。《评价》首次将噪音控制纳入国家级绿色评价体系,要求从设计隔声结构到运维定期监测实现闭环管控,加速…...