0

我现在正在学习 SwiftUI,遇到了这个我似乎无法解决的问题。可能有一个非常简单的解决方案。本质上,我有一个包含 NSManagedObjects(房间)列表的视图模型:

class RoomViewModel: ObservableObject, ViewModelProtocol {
    
    enum ViewState {
        case Loading
        case Normal
        case NoData
    }
    
    @Published var viewState: ViewState = .Loading
    @Published var rooms: [Room] = []
    
    // Get all the rooms
    func getRooms() {
        viewState = .Loading
        rooms = CoreDataManager.shared.getRooms()
        updateViewState()
    }
    
    // MARK: Update the view state
    func updateViewState() {
        viewState = rooms.count == 0 ? .NoData : .Normal
    }
}

从视图模型中的房间列表中删除最后一个元素时,foreach 崩溃:

if viewModel.rooms.count > 0 {
    List {
        ForEach(Array(viewModel.rooms.enumerated()), id: \.self.element.id) { index, room in
            RoomRow(room: self.viewModel.rooms[index], viewModel: viewModel, index: index)
        }.listRowBackground(Color.backgroundColor)
    }
} else {
    NoData()
}

当我删除一个房间时,这是运行的代码:

PersistenceController.shared.container.viewContext.delete(self.viewModel.rooms[selectedIndex])
self.viewModel.rooms.remove(at: selectedIndex)
PersistenceController.shared.saveContext()

然后应用程序在视图中的 ForEach 循环上崩溃,指出索引超出范围。这是一些异步问题,在从视图模型的列表中删除元素之前,视图状态会更新吗?

4

1 回答 1

0

我找到了解决我的问题的方法。问题似乎是 ForEach 试图查看以前数据和新数据之间的差异,并试图抓取已删除的核心数据对象。我通过更改 ForEach 循环来检查我们正在查看的对象是否已被删除来解决问题

ForEach(Array(viewModel.rooms.enumerated()), id: \.self.element.id) { index, room in
    if room.isFault {
        EmptyView()
    } else {
        RoomRow(room: self.viewModel.rooms[index], viewModel: viewModel, index: index)
    }
}
于 2020-12-23T18:34:08.100 回答