ios - 列出重新加载动画故障

标签 ios swift swiftui combine

Animation Glitch

所以我有一个列表,当用户填写搜索关键字时会发生变化,当没有结果时,所有单元格都会崩溃,并且它们会以某种方式飞到看起来很难看的第一部分。我的代码中是否存在错误或者这是预期的 SwiftUI 行为?谢谢。

import SwiftUI

struct ContentView: View {

    @ObservedObject var viewModel = ViewModel(photoLibraryService: PhotoLibraryService.shared)

    var body: some View {
        NavigationView {
            List {
                Section {
                    TextField("Enter Album Name", text: $viewModel.searchText)
                }
                Section {
                    if viewModel.libraryAlbums.count > 0 {
                        ForEach(viewModel.libraryAlbums) { libraryAlbum -> Text in
                            let title = libraryAlbum.assetCollection.localizedTitle ?? "Album"
                            return Text(title)
                        }
                    }
                }
            }.listStyle(GroupedListStyle())
                .navigationBarTitle(
                    Text("Albums")
                ).navigationBarItems(trailing: Button("Add Album", action: {
                        PhotoLibraryService.shared.createAlbum(withTitle: "New Album \(Int.random(in: 1...100))")
                }))
        }.animation(.default)
    }
}

最佳答案

1) 你必须使用一些去抖动来减少在搜索字段中键入时刷新列表的需要

2) 禁用行动画

第二个是最难的部分。诀窍是通过设置它的 id 来强制重新创建一些 View 。

这是一个简单的应用程序代码(能够测试这个想法)

import SwiftUI
import Combine

class Model: ObservableObject {
    @Published var text: String = ""
    @Published var debouncedText: String = ""
    @Published var data = ["art", "audience", "association", "attitude", "ambition", "assistance", "awareness", "apartment", "artisan", "airport", "atmosphere", "actor", "army", "attention", "agreement", "application", "agency", "article", "affair", "apple", "argument", "analysis", "appearance", "assumption", "arrival", "assistant", "addition", "accident", "appointment", "advice", "ability", "alcohol", "anxiety", "ad", "activity"].map(DataRow.init)
    var filtered: [DataRow] {
        data.filter { (row) -> Bool in
            row.txt.lowercased().hasPrefix(debouncedText.lowercased())
        }
    }

    var id: UUID {
        UUID()
    }

    private var store = Set<AnyCancellable>()
    init(delay: Double) {
        $text
            .debounce(for: .seconds(delay), scheduler: RunLoop.main)
            .sink { [weak self] (s) in
            self?.debouncedText = s
        }.store(in: &store)
    }
}

struct DataRow: Identifiable {
    let id = UUID()
    let txt: String
    init(_ txt: String) {
        self.txt = txt
    }
}

struct ContentView: View {

    @ObservedObject var search = Model(delay: 0.5)

    var body: some View {

        NavigationView {
            VStack(alignment: .leading) {

                TextField("filter", text: $search.text)
                    .padding(.vertical)
                    .padding(.horizontal)

                List(search.filtered) { (e) in
                    Text(e.txt)
                }.id(search.id)
            }.navigationBarTitle("Navigation")
        }
    }
}


struct ContentView_Previews: PreviewProvider {
    static var previews: some View {
        ContentView()
    }
}

我对结果很满意

enter image description here

关于ios - 列出重新加载动画故障,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59602045/

相关文章:

ios - 我想将每个单元格链接到不同的 URL

ios - 全屏显示时的透明 View Controller 背景

arrays - Swift 4 替换字符串中的单词

SwiftUI 链接长文本对齐多行

iphone - 如何自定义 TTPostController 操作按钮

ios - 具有 MVVM 架构的 Swiftui 在复杂应用程序中导航屏幕

ios - Multipeer Connectivity 如何获取广告对等点的名称?

ios - 具有媒体和自定义属性的 Twilio 可编程聊天消息

ios - SwiftUI:如何 "swipe delete"嵌套 ForEach 中的元素(分组)

swift - 如何在 SwiftUI/Mapkit 中正确显示自定义引脚?