【发布时间】:2021-12-06 06:14:22
【问题描述】:
从 ViewModel 更新 cartArray 不会附加到当前元素,而是每次都新添加对象。我需要将 cartArray 维护为全局数组,以便可以从项目的任何视图访问它。我正在从 ViewModel 向 cartArray 添加元素。我采用了一个单独的类 DataStorage,其中包含可以通过项目访问的对象
Example_AppApp.swift
import SwiftUI
@main
struct Example_AppApp: App {
var body: some Scene {
WindowGroup {
ContentView().environmentObject(DataStorage())
}
}
}
DataStorage.swift
import Foundation
class DataStorage: ObservableObject {
@Published var cartArray = [Book]()
}
ContentView.swift
import SwiftUI
struct ContentView: View {
@State var showSheetView = false
var body: some View {
NavigationView{
ListViewDisplay()
.navigationBarItems(trailing:
Button(action: {
self.showSheetView.toggle()
}) {
Image(systemName: "cart.circle.fill")
.font(Font.system(.title))
}
)
}.sheet(isPresented: $showSheetView) {
View3()
}
}
}
struct ListViewDisplay: View{
var book = [
Book(bookId: 1 ,bookName: "Catch-22"),
Book(bookId: 2 ,bookName: "Just-Shocking" ),
Book(bookId: 3 ,bookName: "Stephen King" ),
Book(bookId: 4,bookName: "A Gentleman in Moscow"),
]
var body: some View {
List(book, id: \.id) { book in
Text(book.bookName)
NavigationLink(destination: View1(book: book)) {
}
}
}
}
View1Modal.swift
import Foundation
struct Book: Codable, Identifiable {
var id:String{bookName}
var bookId : Int
var bookName: String
}
struct BookOption: Codable{
var name: String
var price: Int
}
View1ViewModel.swift
import Foundation
import Combine
class View1ViewModel : ObservableObject{
var dataStorage = DataStorage()
func addBook (bookId:Int ,bookName : String){
dataStorage.cartArray.append(Book(bookId:bookId, bookName: bookName)) // Adding to global array
print(dataStorage.cartArray)
}
}
View1.swift
import SwiftUI
struct View1: View {
@ObservedObject var vwModel = View1ViewModel()
@EnvironmentObject var datastrg: DataStorage
var book:Book
var body: some View {
Text(book.bookName).font(.title)
Spacer()
Button(action: {
vwModel.addBook(bookId: book.bookId, bookName: book.bookName)
}, label: {
Text("Add Book to Cart")
.frame(maxWidth: .infinity, minHeight: 60)
.background(Color.red)
.foregroundColor(Color.white)
.font(.custom("OpenSans-Bold", size: 24))
})
}
}
View3.swift
import SwiftUI
struct View3: View {
@EnvironmentObject var datastorage : DataStorage
var body: some View {
NavigationView {
List(datastorage.cartArray,id:\.id){book in
VStack{
Text(book.bookName)
.font(.custom("OpenSans-Bold", size: 20))
}
}
.navigationBarTitle(Text("Cart"), displayMode: .inline)
}
}
}
第一次调用 addBook 函数时,它会打印为
[Example_App.Book(bookId: 1, bookName: "Catch-22")]
当我返回此 View1 并通过调用 addBook func 添加另一本书时,它将作为新对象添加到 cartArray
[Example_App.Book(bookId: 3, bookName: "Stephen King")]
打印cartArray 中的元素数量为1 element 而不是2 elements。当我转到 View3 并显示 Books in list 时,cartArray 显示为 empty(0 elements)
我认为 ViewModel 类中的 var dataStorage = DataStorage() 有问题。每次都是新创建的,因此不会存储先前的值。但我不明白如何保持它的状态 如何在 View3 中显示列表?任何想法/建议都会有所帮助
【问题讨论】:
-
请包含一个可以编译的minimal reproducible example。没有它,就不清楚这里到底发生了什么。
-
@jnpdx 请立即检查。我还举例说明了它是如何打印 cartArray 的
-
请包含可编译代码。如果你将你拥有的东西复制并粘贴到 Xcode 中,会有很多东西因为缺少代码或结构不清晰而无法编译。那我就可以给出答案了。您在寻找单例解决方案还是
EnvironmentObject解决方案?在您的代码中,您似乎正在尝试使用EnvironmentObject,但您引用了其他 cmets 中的单例(而不是您评论的我的答案)。 -
另外,同时还有一个指针:任何时候你做
DataStorage()(带括号),你正在创建一个DataStorage的新实例——你需要通过 EnvironmentObject 或单例共享 一个实例。 -
另外值得注意的是:EnvironmentObject 解决方案可能比单例解决方案更具可测试性。
标签: swift iphone xcode swiftui