欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

SwiftUI 中的 Animation

程序员文章站 2022-03-16 15:10:23
...

原文:Animations in SwiftUI

26 Jun 2019

SwiftUI 创建了一种声明式的和简单明了的 UI 构建方式。我们介绍了 List、Form 组件和绑定。它们是的 SwiftUI 使用起来更简单和强大。今天,我们将介绍另一种 SwiftUI 特性:Animations。

Animation

在 SwiftUI 中,你可以将任意的改变过程封装进一个 withAnimation 块中。默认,SwiftUI 会对这种改变采用 fade in/out 的方式进行动画。来看个例子。

struct ContentView : View {
    @State private var isButtonVisible = true

    var body: some View {
        VStack {
            Button(action: {
                withAnimation {
                    self.isButtonVisible.toggle()
                }
            }) {
                Text("Press me")
            }

            if isButtonVisible {
                Button(action: {}) {
                    Text("Hidden Button")
                }
            }
        }
    }
}

在这个例子中,我们将状态改变动作封装到了 withAnimation 块中,以产生一个漂亮的 fade in 动画。通过指定 timing 和 spring 参数,你还可以修改动画的效果。也可以在要动画的 view 后面添加一个 animation 修饰符。

struct ContentView : View {
    @State private var isButtonVisible = true

    var body: some View {
        VStack {
            Button(action: {
                self.isButtonVisible.toggle()
            }) {
                Text("Press me")
            }

            if isButtonVisible {
                Button(action: {}) {
                    Text("Hidden Button")
                }.animation(.easeInOut)
            }
        }
    }
}

在上面的例子中,通过直接添加一个 animation 修饰符达到了同样的效果。这里我们指定动画方式为 easeInOut,但你可以自己指定这个 animation 属性。

是有时候,存在多个视图依赖了同一个状态的情况,我们想对所有存在这种依赖的视图进行动画。在这种情况下,我们可以用 animatable 绑定。

struct ContentView : View {
    @State private var isButtonVisible = true

    var body: some View {
        VStack {
            Toggle(isOn: $isButtonVisible.animation()) {
                Text("Show/Hide button")
            }

            if isButtonVisible {
                Button(action: {}) {
                    Text("Hidden Button")
                }
            }
        }
    }
}

如上所示,我们可以通过 animation 方法将我们的绑定变成 animatable 绑定。改方法可以将绑定值的每个变化放进一个 animation 块中。这个方法中,你可以指定动画参数。关于绑定,请阅读上一篇文章。

Transitions

我前面已经说过,SwiftUI 默认使用 fade in out 动画,但我们可以使用任意类型的动画。例如将 fade 动画替换成 moving 动画。

struct ContentView : View {
    @State private var isButtonVisible = true

    var body: some View {
        VStack {
            Toggle(isOn: $isButtonVisible.animation()) {
                Text("Show/Hide button")
            }

            if isButtonVisible {
                Button(action: {}) {
                    Text("Hidden Button")
                }.transition(.move(edge: .trailing))
            }
        }
    }
}

在上面的例子中,我们向视图传递了一个 transition 修饰符。SwiftUI 有许多现成的转场动画比如 move、slide、scale、offset、opacity 等等。可以将它们组合成单一动画。例如:

extension AnyTransition {
    static func moveAndScale(edge: Edge) -> AnyTransition {
        AnyTransition.move(edge: edge).combined(with: .scale())
    }
}

struct ContentView : View {
    @State private var isButtonVisible = true

    var body: some View {
        VStack {
            Toggle(isOn: $isButtonVisible.animation()) {
                Text("Show/Hide button")
            }

            if isButtonVisible {
                Button(action: {}) {
                    Text("Hidden Button")
                }.transition(.moveAndScale(edge: .trailing))
            }
        }
    }
}

我们创建了一个 moveAndScale 动画,它只是将 move 和 scale 动画捏在了一起。SwiftUI 会将你在动画方法中指定的时间值和 spring 参数均匀地应用到当前动画。

SwiftUI 也提供了构建非对称动画的方法。假设你想在插入时应用 move 动画,而在删除时应用 fade 动画。这时,可以使用 AnyTransition 结构中的 asymmetric 方法来构建非对称动画。

extension AnyTransition {
    static func moveOrFade(edge: Edge) -> AnyTransition {
        AnyTransition.asymmetric(
            insertion: .move(edge: edge),
            removal: .opacity
        )
    }
}

struct ContentView : View {
    @State private var isButtonVisible = true

    var body: some View {
        VStack {
            Toggle(isOn: $isButtonVisible.animation()) {
                Text("Show/Hide button")
            }

            if isButtonVisible {
                Button(action: {}) {
                    Text("Hidden Button")
                }.transition(.moveOrFade(edge: .trailing))
            }
        }
    }
}

看到了吗?我们向 asymmetric 方法传入了两个动画,一个用于插入操作时,一个用于删除操作时。

注意:我们也可以传入我们之前创建的组合动画。

结论

今天,我们讨论了 SwiftUI 中的许多动画方法。你可以根据需要来选用这些方法。随着对 SwiftUI 学习的不断深入,我觉得它确实是一个迷人的框架。请关注我的 Twitter,对本文有任何问题请问我。感谢您的阅读。下周再见。

相关标签: SwiftUI