SwiftUI之Transitions过渡动画

Transition是什么?

在SwiftUI中,transition决定了某个View如何插入到视图栈中,或者如何在视图栈中移除。transition自身并没有任何效果, 需要配合动画一起使用,举个例子:

struct Example1: View {@State private var show = falsevar body: some View {VStack {Spacer()if show {LabelView().transition(.opacity)}Spacer()Button("点击") {self.show.toggle()}.padding(20)}}
}

可以看出,并没有什么动画效果,其实,这也很好理解,transition只是告诉系统试图如何过渡,系统并不知道过渡的动画函数是什么,也就无法做动画。

注意,即使使用隐式动画,也就是.animation()modifier也不起作用。代码如下:

struct Example1: View {@State private var show = falsevar body: some View {VStack {Spacer()if show {LabelView().animation(.easeInOut).transition(.opacity)}...}
}

要想让transition有动画,有两种方法:

第一种是给出一个显式动画,代码如下:

struct Example1: View {@State private var show = falsevar body: some View {VStack {...Button("点击") {withAnimation(.easeInOut(duration: 1.0)) {self.show.toggle()}}.padding(20)}}
}

另一种方法是为transition关联一个动画,这里值得注意的是,我们下边代码中与transition关联的动画作用于transition,并不是作用于view的。

struct Example2: View {@State private var show = falsevar body: some View {VStack {Spacer()if show {LabelView().transition(AnyTransition.opacity.animation(.easeInOut(duration: 1.0)))}Spacer()Button("点击") {self.show.toggle()}.padding(20)}}
}

添加了动画的效果如下图所示:

非对称的Transitions

在了解什么叫非对称之前,我们先了解一下对称,对于transition来说,当view出现的时候,会执行某个过渡效果,在默认情况下,当该view消失的时候,会执行与出现相反的过渡效果,这就是transiton的对称性。

可以看到,绿色文本从左边滑入,然后从右边滑出,是一个对称的过渡效果。

我们可以使用.asymmetric来实现非对称的过渡效果,代码如下:

.transition(.asymmetric(insertion: .opacity, removal: .scale))

可以看出,出现和消失使用了不同的过渡效果。

组合Transitions

我们还想更进一步,我们可以使用组合来为某个过渡效果实现多个动画过程,在SwiftUI中的实现代码也超级简单:

.transition(AnyTransition.opacity.combined(with: .slide))

可以看到,绿色文本的过渡动画,通知执行了opacityslide两种效果,当然我们也可以在asymmetric中使用:

.transition(.asymmetric(insertion: AnyTransition.opacity.combined(with: .slide), removal: AnyTransition.scale.combined(with: .slide)))

效果如下:

带有参数的Transitions

我们在上边的代码中,只使用了类似.slide这样的参数,其实这些参数还可以接受一些额外的参数,例如下边这些:

.scale(scale: 0.0, anchor: UnitPoint(x: 1, y: 0))
.scale(scale: 2.0)
.move(edge: .leading)
.offset(x: 30)
.offset(y: 50)
.offset(x: 100, y: 10)

自定义Transitions

本篇文章的核心内容来了,上边介绍的各种效果基本上能够满足我们大部分的开发需求,但是,总有例外,当我们需要复杂的过渡效果的时候,这一小节的内容能够给你提供更多的思路

比如, 在App中的各种样式的弹屏,翻页等等,你能想到的过渡都属于Transitions的范畴。当然我们这里只是演示了自定义这些过渡效果的核心思想。

我们先做一个简单的例子,我们自定义一个过渡效果,类似与上边用到的opacity效果。代码如下:

extension AnyTransition {static var myCustomOpacity: AnyTransition {AnyTransition.modifier(active: MyOpacityModifier(opacity: 0), identity: MyOpacityModifier(opacity: 1))}
}struct MyOpacityModifier: ViewModifier {let opacity: Doublefunc body(content: Content) -> some View {content.opacity(opacity)}
}
  • 写一个AnyTransition的扩展
  • 实现一个myCustomOpacity的静态类型
  • 返回值为AnyTransition.modifier,它接受两个参数,activeidentity,分别表示开始和结束
  • activeidentity是个ViewModifier类型

基本上就这几步,然后我们这么使用:

.transition(.myCustomOpacity)

大家仔细看上边的代码,由于本质上是个ViewModifier,相当于修改了view的opacity,这也就是我们上边说过的,不加显式动画,不会产生过渡效果的原因。

有很多动画效果,比如.rotationEffect() 和.transformEffect(),用transition都可以实现,我们在最后,使用GeometryEffect来实现一个下边这样的效果:

动图

我们先讲一下该动画的实现思路:

  • 出现的时候,一边缩放,一边旋转
  • 仔细观察,缩放动画在整个动画时间的一半的时候,就已经缩放完毕
  • 旋转沿着x轴

有了上边的思路后,我们再看下边的代码:

struct GeometryEffectTransitionsDemo: View {@State private var show = falsevar body: some View {return ZStack {Button("Open Booking") {withAnimation(.easeInOut(duration: 0.8)) {self.show.toggle()}}.position(x: 100, y: 20)if show {RoundedRectangle(cornerRadius: 15).fill(Color.green).frame(width: 300, height: 400).shadow(color: .black, radius: 3).transition(.fly).zIndex(1)}}}
}extension AnyTransition {static var fly: AnyTransition {AnyTransition.modifier(active: FlyModifier(pct: 0), identity: FlyModifier(pct: 1))}
}struct FlyModifier: GeometryEffect {var pct: Doublevar animatableData: Double {get {pct}set {pct = newValue}}func effectValue(size: CGSize) -> ProjectionTransform {let a = CGFloat(Angle(degrees: 90 * (1 - pct)).radians)var transform3d = CATransform3DIdentitytransform3d.m34 = -1 / max(size.width, size.height)transform3d = CATransform3DRotate(transform3d, a, 1, 0, 0)transform3d = CATransform3DTranslate(transform3d, -size.width / 2.0, -size.width / 2.0, 0)let afffineTransform1 = ProjectionTransform(CGAffineTransform(translationX: size.width / 2.0, y: size.width / 2.0))let afffineTransform2 = ProjectionTransform(CGAffineTransform(scaleX: CGFloat(pct * 2), y: CGFloat(pct * 2)))if pct <= 0.5 {return ProjectionTransform(transform3d).concatenating(afffineTransform2).concatenating(afffineTransform1)} else {return ProjectionTransform(transform3d).concatenating(afffineTransform1)}}
}

GeometryEffect本身即实现了ViewModifier协议,又实现了Animatable协议,因此它可以作为activeidentity的参数,也可以通过animatableData获取动画状态。

整个过渡效果的核心代码如下:

func effectValue(size: CGSize) -> ProjectionTransform {...if pct <= 0.5 {return ProjectionTransform(transform3d).concatenating(afffineTransform2).concatenating(afffineTransform1)} else {return ProjectionTransform(transform3d).concatenating(afffineTransform1)}}

我们用pct跟0.5做判断,返回不同的形变值,就实现了上边的效果。

总结

当我们考虑为某个View使用过渡动画的时候,我们就可以考虑Transitions了,Transitions强大的自定义功能能够让我们实现很多复杂的UI效果。

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

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

相关文章

个人百度百科怎么创建

编辑百度词条是一个相对简单的流程&#xff0c;但需要注意的是&#xff0c;并不是所有的词条都可以编辑&#xff0c;部分锁定的词条是无法编辑的&#xff0c;但可以通过官方平台申请解封。以下百科优化网yajje分享是详细的步骤&#xff1a; 注册百度账号 首先&#xff0c;用户…

基于Docker+Jenkins实现自动部署SpringBoot+Maven项目

安装Docker随便根据其他教程安装即可&#xff0c;本文着重讲jenkins的安装与环境配置。 一、安装jenkins 1.运行命令搜索Jenkins docker search jenkins deprecated 是弃用的意思&#xff0c;第一条搜索记录就是告诉我们 jenkins 镜像已经弃用&#xff0c;让我们使用 jenkins…

实战:Zig 编写高性能 Web 服务(1)

1.1 认识 std.http std.http 是 Zig 标准库中用于处理 HTTP 相关操作的类库。以我学习新的编程语言的经历来看&#xff0c;编写web程序是最常见的技术场景&#xff0c;所以熟练掌握 HTTP server/client 服务相关的编程知识是比较重要的。 std.http 主要包含以下API: Client…

计算机类专业应该怎么选学校和方向?优先选这些!

&#x1f446;点击关注 获取更多编程干货&#x1f446; 高考季临近&#xff0c;不少有意向报考计算机专业的同学在为院校和细分专业的选择而苦恼&#xff0c;以下是一些建议&#xff0c;希望能帮到大家&#xff01; 01 选校建议 在选择计算机科学&#xff08;CS&#xff09…

震撼揭秘:超53%热效率的潍柴柴油机如何成为中国动力心?

世界内燃机领域迎来历史性突破! 2024年4月20日&#xff0c;2024世界内燃机大会在天津隆重开幕&#xff0c;潍柴动力震撼发布了全球首款本体热效率高达53.09%的柴油机。这一技术奇迹标志着中国内燃机行业再一次登顶全球科技巅峰&#xff0c;彰显了中国在高端装备制造业中的领先…

我的app开始养活我了

大家在日常使用各类 app 时应该会发现&#xff0c;进入 app 会有个开屏广告&#xff0c;在使用 app 中&#xff0c;时不时的也会有广告被我们刷到。 这时候如果我们看完了这个广告&#xff0c;或者点击了这个广告的话&#xff0c;app商家就会获得这个广告的佣金。 这个佣金就是…

uniapp开发环境及项目创建

uniapp官网&#xff1a;https://uniapp.dcloud.net.cn/tutorial/ 使用HBuilderX编辑器 点击 文件 -> 新建 - > 项目 创建好之后的目录结构 uniapp项目目录详细结构 ┌─uniCloud 云空间目录&#xff0c;支付宝小程序云为uniCloud-alipay&#xff0c;阿里…

社交媒体数据恢复:Weico

一、从备份中恢复数据 云备份 希望这篇教程能帮助你恢复Weico中的聊天记录和文件。如有其他问题&#xff0c;请随时联系我们。 三、注意事项 在尝试恢复数据的过程中&#xff0c;请避免执行任何可能导致数据进一步丢失的操作。 数据恢复的效果取决于多种因素&#xff0c;包…

OWASP API Security Top 10解读

数字经济时代&#xff0c;无论是互联网商业创新还是传统企业数字化转型&#xff0c;都在推动API数量与应用范围的爆发式增长。从只用于企业内部服务调用的1.0时代&#xff0c;到面向服务架构的2.0时代&#xff0c;再到如今成为开放平台和云原生微服务的3.0时代&#xff0c;API正…

百度百科人物百科修改注意事项

在修改百度百科人物百科时&#xff0c;百科优化网yajje分享需要注意以下几个方面&#xff1a; 参考资料的选择和使用 选择权威参考资料&#xff1a;在修改过程中&#xff0c;参考资料的选择至关重要。应尽可能从靠谱的来源寻找资料&#xff0c;如教科书、国家标准或公文、学术…

深度学习入门-第3章-神经网络

前面的待补充 3.6 手写数字识别 3.6.1 MNIST 数据集 本书提供了便利的 Python 脚本 mnist.py &#xff0c;该脚本支持从下载 MNIST 数据集到将这些数据转换成 NumPy 数组等处理&#xff08;mnist.py 在 dataset 目录下&#xff09;。 使用 mnist.py 时&#xff0c;当前目录必须…

Day13 : 队列 LeedCode:239. 滑动窗口最大值 347.前 K 个高频元素

详细讲解:点击跳转 239. 滑动窗口最大值 给你一个整数数组 nums&#xff0c;有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧。你只可以看到在滑动窗口内的 k 个数字。滑动窗口每次只向右移动一位。 返回 滑动窗口中的最大值 。 示例 1&#xff1a; 输入&#x…

【C语言】字符串左旋(三种方法)

&#xff08;方法3只给出思路参考&#xff09; 问题 描述&#xff1a; 实现一个函数&#xff0c;可以左旋字符串中的k个字符。 例如&#xff1a; ABCD左旋一个字符得到BCDA ABCD左旋两个字符得到CDAB 分析 我们先来理解一下&#xff0c;什么叫“左旋”&#xff1f;其实是这…

解决odbc 数据源创建之后删除失败问题

问题描述 该问题一般存在于用32位odbc管理程序删除了64位管理程序创建的数据源&#xff0c;被提示失败&#xff0c;然后再用64位odbc管理程序删除时发现删除不掉。怎么都删除不掉&#xff0c;不会影响体验&#xff0c;但是很膈应 问题解决 将该路径下的对应的ODBC数据源&…

使用Minikube+docker+harbor+k8s自动化部署 @by_TWJ

目录 1. 开始1.1. 环境1.2. 测试的git仓库1.3. 离线文件1.4. 安装docker1.5. 安装docker-compose&#xff08;非必要&#xff09;1.6. 安装Jenkins1.7. 安装harbor1.8. 允许docker通过http访问私有仓库1.9. 修改/etc/hosts&#xff0c;追加自定义域名1.10. 安装Minikube 2. min…

【Javascript系列】Terser通过调用API来实现代码的压缩和优化功能

Terser通过调用API来实现代码的压缩和优化功能 起源通过API来调用API调用过程中的一个隐含的技术点 - 异步调用和Promise对象官方文档中的一个有点容易忽略和混淆的地方关于Promise 起源 书接 上回&#xff0c;对Terser的功能做了一个初步的探索。在官方的主页上&#xff0c;有…

Windows系统电脑本地部署AI音乐创作工具并实现无公网IP远程使用

文章目录 前言1. 本地部署2. 使用方法介绍3. 内网穿透工具下载安装4. 配置公网地址5. 配置固定公网地址 前言 本文主要介绍如何在Windows系统电脑上快速本地部署一个文字生成音乐的AI创作工具MusicGPT&#xff0c;并结合cpolar内网穿透工具实现随时随地远程访问使用。 MusicG…

22、matlab锯齿波、三角波、方波:rectpuls()函数/sawtooth()函数/square()函数

1、采样的非周期性矩形 语法 语法1&#xff1a;y rectpuls(t) 返回一个以数组 t 中指示的采样时间采样的连续非周期性单位高度矩形脉冲&#xff0c;该矩形脉冲以 t 0 为中心。 语法2&#xff1a;y rectpuls(t,w) 生成一个宽度为 w 的矩形 参数 t:采样时间 w:矩形宽度…

两步教你学会内网穿透

文章目录 1.下载安装ngrok安装包2.生成固定的域名3.访问 1.下载安装ngrok安装包 1.注册登录ngrok网站ngrok官网 页面下滑点击下载下载&#xff1a; 2.解压&#xff08;双击ngrok.exe进入终端&#xff09; 3.终端绑定账户信息&#xff08;复制如下图信息到终端&#xff09; …

Win11下只支持IE浏览器的老网站顺畅运行的方法

在Windows 11操作系统中&#xff0c;由于Internet Explorer&#xff08;IE&#xff09;浏览器的逐步淘汰&#xff0c;微软官方已不再直接支持IE浏览器。然而&#xff0c;当您遇到必须访问仅支持IE的老旧网站时&#xff0c;Windows 11仍然提供了一些实用的替代方案来应对这一挑战…