有啥方法可以判断哪个状态更改导致在 SwiftUI 中重建视图?
Posted
技术标签:
【中文标题】有啥方法可以判断哪个状态更改导致在 SwiftUI 中重建视图?【英文标题】:Is there any way to tell which state change caused the view to be rebuilt in SwiftUI?有什么方法可以判断哪个状态更改导致在 SwiftUI 中重建视图? 【发布时间】:2020-07-24 17:54:30 【问题描述】:我有一个视图需要像手风琴一样工作,用户可以拖动它来打开/折叠它。而且我希望它在用户拖动时没有任何动画,但在他离开时动画。所以,我想在我的视图中有两个状态变量(例如,高度和 tempHeight),但我只会在高度变量触发视图重建过程时制作动画。
对 height 和 tempHeight 变化的动画在拖动时看起来很奇怪,感觉有延迟。
描述我的问题的示例代码:
struct ContentView: View
@State var tempHeight: CGFloat = 200
@State var height: CGFloat = 200
var dragGesture: some Gesture
DragGesture(),
.onChanged value in
tempHeight = min(400, max(200, height + value.translation.height))
.onEnded value in
height = tempHeight > 300 ? 400 : 200
tempHeight = height
var body: some View
VStack
Spacer()
.frame(height: 100)
Color.blue.frame(height: tempHeight)
.animation(.easeIn)
.gesture(dragGesture)
Spacer()
我试图解决这个问题的方法是使用一个静态变量来跟踪最后发生的状态变化,并且它似乎有效。 我目前的解决方法:
enum StateChange
case height
case tempHeight
static var lastChangeBy: StateChange = .height
struct ContentView: View
@State var tempHeight: CGFloat = 200
@State var height: CGFloat = 200
var dragGesture: some Gesture
DragGesture()
.onChanged value in
tempHeight = min(400, max(200, height + value.translation.height))
StateChange.lastChangeBy = .tempHeight
.onEnded value in
height = tempHeight > 300 ? 400 : 200
tempHeight = height
StateChange.lastChangeBy = .height
var body: some View
VStack
Spacer()
.frame(height: 100)
Color.blue.frame(height: tempHeight)
.gesture(dragGesture)
.animation(StateChange.lastChangeBy == .height ? .easeInOut : .none)
.animation(.easeInOut)
Spacer()
这对我来说就像一个黑客。如果我有很多 @State 包装的变量,它很快就会变得一团糟。是否可以知道哪个状态更改触发了在 body 内构建视图,以便我可以有条件地将动画应用于特定触发器或更好的编写方式。
【问题讨论】:
【参考方案1】:您可以明确指定哪些值更改在修改器中激活动画,例如
Color.blue.frame(height: tempHeight)
.gesture(dragGesture)
.animation(.easeInOut, value: height) // << here
【讨论】:
以上是关于有啥方法可以判断哪个状态更改导致在 SwiftUI 中重建视图?的主要内容,如果未能解决你的问题,请参考以下文章
SwiftUI 如何快速识别视图(View)界面的刷新是由哪个状态的改变导致的?
SwiftUI视图状态更改使用withAnimation()方法无动画效果的解决
SwiftUI视图状态更改使用withAnimation()方法无动画效果的解决