用SwiftUI给视图添加动画

级别:★☆☆☆☆
标签:「SwiftUI」「常用动画」「转场动画」
作者: WYW
审校: QiShare团队


前言:最近奇舞647写了《用SwiftUI 写了一个简单页面》,笔者也参考官方SwiftUI教程及网络博客,写了一个给视图添加改变位置、透明度、旋转、前景色、背景色及转场动画的Demo。Demo效果如下:

SwiftUIAnimation.gif

简单说一下Demo,首先在设置应用的rootViewController部分,笔者设置了UIHostingController,传入指定的QiRootView,其中QiRootView是用SwiftUI布局的。

window.rootViewController = UIHostingController(rootView: QiRootView())

简单示例如下:在QiRootView 的body里边写上我们要显示的视图。

import SwiftUI

struct QiRootView: View {
    var body: some View {
        VStack {
            HStack {
                Button(action: {
                    withAnimation (.easeInOut(duration: 2.0)){
                        self.showDetail.toggle()
                    }
                }) {
                    Text("常用动画")
                }
            }
        }
    }
}

Demo 里边,笔者在body里边,上边的HStack默认情况显示了2个按钮,其中左边的按钮给按钮添加了普通的位置、透明度、旋转、圆角、前景色、背景色的动画。右边的按钮会触发state属性showDetail,转场显示出奇舞647之前写的简单页面。
下边的HStack显示了5个按钮,第一个按钮显示还没绘制完的徽章。后4个按钮分别以不同的转场动画显示奇舞647写的简单页面。

下边笔者分2部分介绍给视图添加动画的内容:

  1. 给视图添加常用动画
  2. 给视图添加转场动画

使用SwiftUI给视图添加动画的方式比较简单。分2种方式可以给视图添加动画。

  1. 直接在 View 上使用 .animation 类型添加动画
  2. 在按钮的action里边用withAnimation { } 来控制某个 State 属性,进而触发动画。

给视图添加常用动画使用的是直接在 View 上使用 .animation 类型的方式给视图添加动画。

给视图添加转场动画部分使用的是在按钮的action里边用withAnimation { } 来控制某个 State 属性,进而触发的转场动画。

给视图添加常用动画

下边笔者介绍下上述Demo 动画中用到的相应API。

  • 改变位置
.offset(x: changeAnimation ? offsetX2 : offsetX1, y: changeAnimation ? offsetY2 : offsetY1)
  • 改变尺寸的话,也可以改变后边的width height的值。
.frame(width: 100.0, height: 100.0, alignment: .center)
  • 改变视图圆角
.cornerRadius(changeAnimation ? cornerR2 : cornerR1)
  • 改变背景色
.background(changeAnimation ? backColor2 : backColor1)
  • 改变前景色
.foregroundColor(changeAnimation ? foreColor2 : foreColor1)
  • 改变透明度
.opacity(changeAnimation ? 0.5 : 1.0)
  • 改变scale
.scaleEffect(changeAnimation ? 1.5 : 1.0)
  • 旋转视图
.rotationEffect(.degrees(changeAnimation ? 90 : 0))

呈现动画的方式可以使用如下API 控制动画类型

.animation(.easeInOut(duration: 2.0))
.animation(.spring())

综上用到如下代码:

import SwiftUI

struct QiRootView: View {
    
    @State private var changeAnimation = false
    private var offsetX1: CGFloat = 0.0
    private var offsetY1: CGFloat = 0.0
    private var offsetX2: CGFloat = 20.0
    private var offsetY2: CGFloat = 20.0
    
    private var cornerR1: CGFloat = 0.0
    private var cornerR2: CGFloat = 50.0
    
    private var foreColor1 = Color.blue
    private var foreColor2 = Color.white
    
    private var backColor1 = Color.gray
    private var backColor2 = Color.black
    
    var body: some View {
        VStack {
           
            HStack {
                
                Button() {
                                Text("位置")
                                    .offset(x: changeAnimation ? offsetX2 : offsetX1, y: changeAnimation ? offsetY2 : offsetY1)
                                    .frame(width: 100.0, height: 100.0, alignment: .center)
                                    .foregroundColor(changeAnimation ? foreColor2 : foreColor1)
                                    .background(changeAnimation ? backColor2 : backColor1)
                                    .opacity(changeAnimation ? 0.5 : 1.0)
                                    .cornerRadius(changeAnimation ? cornerR2 : cornerR1)
                                    .padding()
                                    .rotationEffect(.degrees(changeAnimation ? 90 : 0))
                                    .animation(.easeInOut(duration: 2.0))
                              }
            }
        }
    }
}

给视图添加转场动画

转场动画部分,笔者参照官方教程写了四种转场动画。
分别从左边、右边、上边、下边呈现出来奇舞647写的简单页面。
代码层面的传入的转场类型为给AnyTransition添加了extention的转场类型。

import SwiftUI

extension AnyTransition {
    static var moveAndFade: AnyTransition {
        AnyTransition.move(edge: .trailing)
    }
    static var moveAndFadeLeading: AnyTransition {
        AnyTransition.move(edge: .leading)
    }
    static var moveAndFadeUp: AnyTransition {
        AnyTransition.move(edge: .top)
    }
    static var moveAndFadeBottom: AnyTransition {
        AnyTransition.move(edge: .bottom)
    }
}

struct QiRootView: View {
    
    @State private var showDetail = false
    
    var body: some View {
        VStack {
           
            HStack {
                Button(action: {
                               withAnimation (.easeInOut(duration: 2.0)){
                                   self.showDetail.toggle()
                               }
                }) {
                             Image(systemName: "chevron.right.circle")
                                .imageScale(.large)
                                .rotationEffect(.degrees(showDetail ? 90 : 0))
                                .opacity(showDetail ? 0.5 : 1.0)
                                .scaleEffect(showDetail3 ? 1.5 : 1.0)
                                .padding()
                          }
            }
            
            if showDetail {
                ContentView()
                    .transition(.moveAndFade)
                    // .transition(.moveAndFadeBottom)
                    // .transition(.moveAndFadeLeading)
                    // .transition(.moveAndFadeUp)
            }
        }
    }
}

Demo

Demo 下载地址:QiSwiftUIAnimation

参考学习网址


推荐文章:
用SwiftUI写一个简单页面
iOS 控制日志的开关
iOS App中可拆卸一个framework的两种方式
自定义WKWebView显示内容(一)
Swift 5.1 (6) - 函数
Swift 5.1 (5) - 控制流
Xcode11 新建工程中的SceneDelegate
iOS App启动优化(二)—— 使用“Time Profiler”工具监控App的启动耗时
iOS App启动优化(一)—— 了解App的启动流程
iOS WKWebView的基本使用

最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容