用SwiftUI给视图添加动画

avatar
奇舞团移动端团队 @奇舞团

级别:★☆☆☆☆
标签:「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的基本使用
奇舞周刊