admin管理员组文章数量:1344955
tl;dr:
In SwiftUI, I want proper animations when a list of rows changes order, at the same time as row content changes values. Works with List
, does not work with VStack
/HStack
. Probably something to do with view identity, transitions, animations and transactions.
Animation glitch: Works, but uses List
:
I have a common setup but I am not able to achieve perfect animations.
- A list of items.
- One of the labels in each row reflects a property.
- The list is sorted by that property.
- On a change to the property - the row label needs to animate, but the overall rows order itself might also change and need animating.
In the Minimal reproducible example I am providing, specifically we have:
- A list of comments.
- One of the labels in each row reflects the 'upvote' count for a comment.
- The list is sorted by 'most upvoted'.
The issue:
When both the content AND order changes - the label 'gets detached' and 'jumps' to the final position after the row animation.
The issue is the same regardless of whether ScrollView
is used (wraps everything), or just directly a VStack
/LazyVStack
.
The issue is the same regardless of whether LazyVStack
/ VStack
is used.
Here is a code you can paste in a new project's ContentView and observe the issue.
Run in Simulator and use Debug > Slow animations
.
struct ContentView: View {
@State private var comments: [Comment] = []
var body: some View {
HStack {
Text("Sate 1").onTapGesture { comments = [.init("Well done", 6), .init("Interesting", 5), .init("Nice", 4)] }
Text("Sate 2").onTapGesture { comments = [.init("Interesting", 7), .init("Well done", 6), .init("Nice", 4)] }
}
// Option 1: Animation glitch
VStack { ForEach(comments, content: CommentRowView.init(comment:)) }
.padding()
.animation(.default, value: comments)
// Option 2: Works, but uses `List`
// List { ForEach(comments, content: CommentRowView.init(comment:)) }
// .animation(.default, value: comments)
}
}
struct CommentRowView: View {
let comment: Comment
var body: some View {
HStack {
Text(comment.text).font(.largeTitle)
Spacer()
VStack {
Text(comment.upvotes.formatted())
.contentTransition(.numericText())
.animation(.default, value: comment.upvotes)
}
.font(.title)
}.padding().border(.gray)
}
}
struct Comment: Identifiable, Equatable {
var id: String { text }; let text: String; let upvotes: Int
init(_ text: String, _ upvotes: Int) { self.text = text; self.upvotes = upvotes }
}
tl;dr:
In SwiftUI, I want proper animations when a list of rows changes order, at the same time as row content changes values. Works with List
, does not work with VStack
/HStack
. Probably something to do with view identity, transitions, animations and transactions.
Animation glitch: Works, but uses List
:
I have a common setup but I am not able to achieve perfect animations.
- A list of items.
- One of the labels in each row reflects a property.
- The list is sorted by that property.
- On a change to the property - the row label needs to animate, but the overall rows order itself might also change and need animating.
In the Minimal reproducible example I am providing, specifically we have:
- A list of comments.
- One of the labels in each row reflects the 'upvote' count for a comment.
- The list is sorted by 'most upvoted'.
The issue:
When both the content AND order changes - the label 'gets detached' and 'jumps' to the final position after the row animation.
The issue is the same regardless of whether ScrollView
is used (wraps everything), or just directly a VStack
/LazyVStack
.
The issue is the same regardless of whether LazyVStack
/ VStack
is used.
Here is a code you can paste in a new project's ContentView and observe the issue.
Run in Simulator and use Debug > Slow animations
.
struct ContentView: View {
@State private var comments: [Comment] = []
var body: some View {
HStack {
Text("Sate 1").onTapGesture { comments = [.init("Well done", 6), .init("Interesting", 5), .init("Nice", 4)] }
Text("Sate 2").onTapGesture { comments = [.init("Interesting", 7), .init("Well done", 6), .init("Nice", 4)] }
}
// Option 1: Animation glitch
VStack { ForEach(comments, content: CommentRowView.init(comment:)) }
.padding()
.animation(.default, value: comments)
// Option 2: Works, but uses `List`
// List { ForEach(comments, content: CommentRowView.init(comment:)) }
// .animation(.default, value: comments)
}
}
struct CommentRowView: View {
let comment: Comment
var body: some View {
HStack {
Text(comment.text).font(.largeTitle)
Spacer()
VStack {
Text(comment.upvotes.formatted())
.contentTransition(.numericText())
.animation(.default, value: comment.upvotes)
}
.font(.title)
}.padding().border(.gray)
}
}
struct Comment: Identifiable, Equatable {
var id: String { text }; let text: String; let upvotes: Int
init(_ text: String, _ upvotes: Int) { self.text = text; self.upvotes = upvotes }
}
Share
Improve this question
asked yesterday
Nikolay SuvandzhievNikolay Suvandzhiev
9,0956 gold badges46 silver badges53 bronze badges
1
|
1 Answer
Reset to default 3struct CommentRowView: View {
let comment: Comment
var body: some View {
HStack {
//content
}
.padding().border(.gray)
.background(Color.black) //THIS - solid background
.drawingGroup() // THIS - upvotes number animation fix
}
}
本文标签: iosSwiftUIanimating VStack order and content at the same time without glitchesStack Overflow
版权声明:本文标题:ios - SwiftUI - animating VStack order and content at the same time without glitches - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743796306a2540495.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
CommentRowView
, try adding.drawingGroup()
to the nestedText
showing the formattedupvotes
(after.contentTransition
). See also How to avoid contentTransition apply black in new text - SwiftUI – Benzy Neez Commented yesterday