SwiftUI:调用带有锚点.center的scrollTo方法时崩溃

时间:2020-10-11 21:47:12

标签: swiftui scrollview

不确定100%是否是由anchor引起的,因为我刚刚将其移除并且到目前为止还没有发生过崩溃(如果发生更多崩溃,将会报告)-有人可以检查并帮助查看是否可能错误在哪里?也要提交Apple Bug。

我的视图如下:

var body: some View {
    ScrollViewReader { scrollProxy in
        ScrollView {
            if isShowingContent {
                LazyVStack {
                    ForEach(mo.msgMos.indices, id: \.self) { i in
                        ...
                    }
                }
            }
        }
        .onChange(of: mo.scrollToIdx) { idx in
            withAnimation {
                scrollProxy.scrollTo(targetIdx, anchor: .center)
            }
        }
    }
}

崩溃堆栈如下所示:

Crashed: com.apple.main-thread
0  AttributeGraph                 0x1cedce974 AGGraphGetValue + 284
1  SwiftUI                        0x1addcbe5c _ViewCache.layout.getter + 52
2  SwiftUI                        0x1addcc2f4 _ViewCache.withPlacementData<A>(_:) + 160
3  SwiftUI                        0x1addd02d0 closure #1 in IncrementalScrollable.makeTarget(at:anchor:) + 336
4  SwiftUI                        0x1addd6a9c partial apply for closure #1 in IncrementalScrollable.makeTarget(at:anchor:) + 56
5  SwiftUI                        0x1ae2241fc HostingScrollView.updateAnimationTarget(_:) + 216
6  SwiftUI                        0x1ae2245c4 HostingScrollView.bounds.didset + 188
7  SwiftUI                        0x1ae2244e8 HostingScrollView.bounds.setter + 128
8  SwiftUI                        0x1ae22444c @objc HostingScrollView.bounds.setter + 44
9  UIKitCore                      0x1aa494e4c -[UIScrollView setContentOffset:] + 804
10 UIKitCore                      0x1aa48f1e0 -[UIScrollViewScrollAnimation setProgress:] + 320
11 UIKitCore                      0x1a95094b8 -[UIAnimator _advanceAnimationsOfType:withTimestamp:] + 276
12 QuartzCore                     0x1aa8e46a4 CA::Display::DisplayLink::dispatch_items(unsigned long long, unsigned long long, unsigned long long) + 664
13 QuartzCore                     0x1aa9bb1a0 display_timer_callback(__CFMachPort*, void*, long, void*) + 280
14 CoreFoundation                 0x1a7633ce4 __CFMachPortPerform + 176
15 CoreFoundation                 0x1a7658098 __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE1_PERFORM_FUNCTION__ + 60
16 CoreFoundation                 0x1a7657440 __CFRunLoopDoSource1 + 596
17 CoreFoundation                 0x1a7651320 __CFRunLoopRun + 2360
18 CoreFoundation                 0x1a76504bc CFRunLoopRunSpecific + 600
19 GraphicsServices               0x1be0d5820 GSEventRunModal + 164
20 UIKitCore                      0x1a9ff4734 -[UIApplication _run] + 1072
21 UIKitCore                      0x1a9ff9e10 UIApplicationMain + 168
22 Fablur                         0x102453f8c main + 12 (AppDelegate.swift:12)
23 libdyld.dylib                  0x1a7317e60 start + 4

4 个答案:

答案 0 :(得分:1)

我发现,如果您使用 scrollTo 和锚点 并且它将滚动视图的内容定位到通常无法通过正常滚动到达的位置,那么它会在以下情况下崩溃视图被移除。

例如。滚动到滚动视图中的最后一个视图并使用 .leading 或 .center 位置。

我通过在滚动视图的末尾添加一个空白视图来解决这个问题。不理想但有效。

答案 1 :(得分:0)

我认为应该将其连接在其他位置

var body: some View {
    ScrollViewReader { scrollProxy in
        ScrollView {
            if isShowingContent {
                LazyVStack {
                    ForEach(mo.msgMos.indices, id: \.self) { i in
                        ...
                    }
                }
                .onChange(of: mo.scrollToIdx) { idx in
                   withAnimation {
                      scrollProxy.scrollTo(targetIdx, anchor: .center)
                   }
                }
            }
        }
    }
}

答案 2 :(得分:0)

我遇到了非常相似的问题,并得出了(潜在的)结论,即ScrollViewReader不能正确计算,因为您的内容位于LazyVStack中,并且由于Lazy项没有预加载...阅读器没有知道滚动视图实际上有多长时间。至少这是我的猜测...

答案 3 :(得分:0)

去掉 withAnimation 块,你会没事的

.onChange(of: mo.scrollToIdx) { idx in
    scrollProxy.scrollTo(targetIdx, anchor: .center)
}