微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

如何在SwiftUI中的Firebase查询期间显示加载动画

如何解决如何在SwiftUI中的Firebase查询期间显示加载动画

我正在使用SwiftUI构建应用程序,并且具有一个ObservableObject来查询我的Firestore数据库。我的文档比较大,经常需要查询很多文档,因此我想在查询下载数据时合并某种加载指示器。

这是我创建的ObservableObject的示例:

import FirebaseFirestore
import SwiftUI

struct Document: Identifiable,Equatable {
    var id: String
    var content: String
}

class Fetch: ObservableObject {
    init(loading: Binding<Bool>) {
        self._loading = loading
        readDocuments()
    }
    
    @Published var documents: [Document] = []
    @Binding var loading: Bool
    
    var collection: CollectionReference = Firestore.firestore().collection("DOCUMENTS")
    
    func newDocument(content: String) {
        let id = self.collection.document().documentID
        self.collection.document(id).setData(["id": id,"content": content]) { (error) in handleError(error: error,message: "\(id) CREATED") }
    }
    
    func deleteDocument(document: Document) {
        if self.documents.contains(document) {
            self.collection.document(document.id).delete() { (error) in handleError(error: error,message: "\(document.id) DELETED") }
        } else { print("\(document.id) NOT FOUND") }
    }
    
    func updateDocument(document: Document,update: [String : Any]) {
        if self.documents.contains(document) {
            self.collection.document(document.id).updateData(update) { (error) in handleError(error: error,message: "\(document.id) UPDATED") }
        } else { print("\(document.id) NOT FOUND") }
    }
    
    func readDocuments() {
        self.collection.addSnapshotListener { (snapshot,error) in
            handleError(error: error,message: "READ DOCUMENTS")
            snapshot?.documentChanges.forEach({ (change) in
                if change.type == .added {
                    self.loading = true
                    self.documents.append(Document(id: change.document.get("id") as? String ?? "Failed TO READ",content: change.document.get("content") as? String ?? "Failed TO READ"))
                    self.loading = false
                }
                if change.type == .modified {
                    self.loading = true
                    self.documents = self.documents.map { (document) -> Document in
                        if document.id == change.document.documentID {
                            let modifiedDocument = Document(id: change.document.get("id") as? String ?? "Failed TO READ",content: change.document.get("content") as? String ?? "Failed TO READ")
                            return modifiedDocument
                        } else {
                            return document
                        }
                    }
                    self.loading = false
                }
                if change.type == .removed {
                    self.loading = true
                    self.documents.removeAll(where: { $0.id == change.document.documentID })
                    self.loading = false
                }
                
            })
        }
    }
    
}

func handleError(error: Error?,message: String) {
    if error != nil { print((error?.localizedDescription)!); return } else { print(message) }
}

这是我创建的示例视图:

struct ContentView: View {
    @State var loading: Bool = false
    var body: some View {
        NavigationView {
            if loading {
                Color.blue.overlay(Text("Loading View"))
            } else {
                Subview(fetch: Fetch(loading: self.$loading))
            }
        }
    }
}

struct Subview: View {
    @Observedobject var fetch: Fetch
    @State var newDocumentContent: String = ""
    
    var body: some View {
        vstack(spacing: 0.0) {
            List {
                ForEach(self.fetch.documents) { document in
                    NavigationLink(destination:
                    UpdateDocument(fetch: self.fetch,documentID: document.id)) {
                        Text(document.content)
                    }
                }.onDelete { indexSet in
                    self.deleteDocument(indexSet: indexSet)
                }
            }
            
            Divider()
            
            NewDocument(fetch: self.fetch,newDocumentContent: self.$newDocumentContent)
        }.navigationBarTitle("CRUD",displayMode: .inline)
    }
    
    func deleteDocument(indexSet: IndexSet) {
        self.fetch.deleteDocument(document: self.fetch.documents[indexSet.first!])
    }
}

请记住,在此示例中,数据并不需要加载视图那么大,这几乎是即时的,但是对于我的应用程序,此代码被拆分为不同文件和场景的加载,因此我认为我将创建这个例子。

我尝试添加一个绑定布尔值,并在readData()函数加载时切换它,但是SwiftUI获取map并出现错误

“在视图更新期间修改状态,这将导致未定义的行为。”

解决方法

您需要在@Published(而非@ObservableObject)中使用@Binding

这是一个可能的演示:

class Fetch: ObservableObject {
    @Published var loading = false

    func longTask() {
        self.loading = true
        // simulates a long asynchronous task (eg. fetching data)
        DispatchQueue.main.asyncAfter(deadline: .now() + 3) {
            self.loading = false
        }
    }
}
struct ContentView: View {
    @ObservedObject private var fetch = Fetch()

    var body: some View {
        ZStack {
            Text("Main view")
            if fetch.loading {
                LoadingView() // can be replaced with any other loading indicator/view
            }
        }
        .onAppear {
            self.fetch.longTask()
        }
    }
}
struct LoadingView: View {
    var body: some View {
        ZStack {
            Color.black
                .opacity(0.5)
                .edgesIgnoringSafeArea(.all)
            Text("Loading")
        }
    }
}

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。