gpt4 book ai didi

swiftui - watchOS 上使用 SwiftUI 的意外(自动)导航

转载 作者:行者123 更新时间:2023-12-05 07:04:15 26 4
gpt4 key购买 nike

我有一个简单的 watchOS 6.2.8 SwiftUI 应用程序,我在其中向用户显示消息列表。

这些消息被建模为类并具有标题、正文和类别名称。我还有一个类别对象,它是这些消息的 View ,仅显示特定的类别名称。

I specifically mention watchOS 6.2.8 because it seems SwiftUI behaves a bit different there than on other platforms.

class Message: Identifiable {
let identifier: String
let date: Date
let title: String
let body: String
let category: String

var id: String {
identifier
}

init(identifier: String, date: Date, title: String, body: String, category: String) {
self.identifier = identifier
self.date = date
self.title = title
self.body = body
self.category = category
}
}

class Category: ObservableObject, Identifiable {
let name: String
@Published var messages: [Message] = []

var id: String {
name
}

init(name: String, messages: [Message] = []) {
self.name = name
self.messages = messages
}
}

Category 本身是一个 @ObservableObject 并发布 messages,这样当一个类别被更新时(比如在后台),显示类别消息列表的 View 也会更新。 (这很好用)

为了存储这些消息,我有一个简单的 MessageStore,它是一个 @ObservableObject,如下所示:

class MessageStore: ObservableObject {
@Published var messages: [Message] = []
@Published var categories: [Category] = []

static let sharedInstance = MessageStore()

func insert(message: Message) throws { ... mutage messages and categories ... }
func delete(message: Message) throws { ... mutage messages and categories ... }
}

(为简单起见,我使用单例,因为环境对象在 watchOS 上无法正确传递存在问题)

故事使消息类别保持同步。添加具有类别名称集的新 Message 时,它还会在 categories 列表中创建或更新 Category 对象。

在我的主要观点中,我提出了两件事:

  • All Messages NavigationLink 转到 View 以显示所有消息
  • 对于每个类别,我都会显示一个 NavigationLink,它会转到一个 View 以仅显示该特定类别中的消息。

enter image description here

这一切都有效,令人惊讶。但是发生了一件我不明白的非常奇怪的事情。 (第一个 SwiftUI 项目)

当我进入所有消息列表并删除所有包含特定类别的消息时,当我导航回主视图时发生了意想不到的事情。

首先我观察到该类别已从列表中正确删除。

但随后,主视图自动快速导航到“所有消息”列表,然后返回。

最后一部分快把我逼疯了..我不明白为什么会这样。从数据的角度来看,一切看起来都很好——消息和类别都被删除了。在自动导航之后,最终的 UI 状态看起来也不错——所有消息的消息计数是正确的,现在消息为零的类别不再显示在列表中。

这是主要 ContentViewAllMessagesView 的代码。如果有帮助,我当然可以在这里发布完整的代码。

struct AllMessagesView: View {
@ObservedObject var messageStore = MessageStore.sharedInstance

@ViewBuilder
var body: some View {
if messageStore.messages.count == 0 {
Text("No messages").multilineTextAlignment(.center)
.navigationBarTitle("All Messages")
} else {
List {
ForEach(messageStore.messages) { message in
MessageCellView(message: message)
}.onDelete(perform: deleteMessages)
}
.navigationBarTitle("All Messages")
}
}

func deleteMessages(at offsets: IndexSet) {
for index in offsets {
do {
try messageStore.delete(message: messageStore.messages[index])
} catch {
NSLog("Failed to delete message: \(error.localizedDescription)")
}
}
}
}

//

struct CategoryMessagesView: View {
@ObservedObject var messageStore = MessageStore.sharedInstance

@ObservedObject var category: Category

var body: some View {
Group {
if category.messages.count == 0 {
Text("No messages in category “\(category.name)”").multilineTextAlignment(.center)
} else {
List {
ForEach(category.messages) { message in
MessageCellView(message: message)
}.onDelete(perform: deleteMessages)
}
}
}.navigationBarTitle(category.name)
}

func deleteMessages(at offsets: IndexSet) {
for index in offsets {
do {
try messageStore.delete(message: category.messages[index])
} catch {
NSLog("Cannot delete message: \(error.localizedDescription)")
}
}
}
}

struct ContentView: View {
@ObservedObject var messageStore = MessageStore.sharedInstance

var body: some View {
List {
Section {
NavigationLink(destination: AllMessagesView()) {
HStack {
Image(systemName: "tray.2")
Text("All Messages")
Spacer()
Text("\(messageStore.messages.count)")
.font(messageCountFont())
.bold()
.layoutPriority(1)
.foregroundColor(.green)
}
}
}

Section {
Group {
if messageStore.categories.count > 0 {
Section {
ForEach(messageStore.categories) { category in
NavigationLink(destination: CategoryMessagesView(category: category)) {
HStack {
Image(systemName: "tray") // .foregroundColor(.green)
Text("\(category.name)").lineLimit(1).truncationMode(.tail)
Spacer()
Text("\(category.messages.count)")
.font(self.messageCountFont())
.bold()
.layoutPriority(1)
.foregroundColor(.green)
}
}
}
}
} else {
EmptyView()
}
}
}
}
}

// TODO This is pretty inefficient
func messageCountFont() -> Font {
let font = UIFont.preferredFont(forTextStyle: .caption1)
return Font(font.withSize(font.pointSize * 0.75))
}
}

抱歉,我知道这是很多代码,但我觉得我需要提供足够的上下文和可见性来展示这里发生的事情。

完整项目位于 https://github.com/st3fan/LearningSwiftUI/tree/master/MasterDetail - 我不认为更多的代码是相关的,但如果是,请告诉我,我会把它移到这里的问题中。

最佳答案

问题出在更新的 ForEach 中,这会导致重新创建 List 并因此断开链接。这看起来像是 SwiftUI 缺陷,因此值得向 Apple 提交反馈。

经过测试的解决方法是将“所有消息”导航链接移出列表(看起来有点不同,但可能是合适的)。使用 Xcode 12/watchOS 7.0 测试

struct ContentView: View {
@ObservedObject var messageStore = MessageStore.sharedInstance

var body: some View {
VStack {
NavigationLink(destination: AllMessagesView()) {
HStack {
Image(systemName: "tray.2")
Text("All Messages")
Spacer()
Text("\(messageStore.messages.count)")
.font(messageCountFont())
.bold()
.layoutPriority(1)
.foregroundColor(.green)
}
}
List {
ForEach(messageStore.categories) { category in
NavigationLink(destination: CategoryMessagesView(category: category)) {
HStack {
Image(systemName: "tray") // .foregroundColor(.green)
Text("\(category.name)").lineLimit(1).truncationMode(.tail)
Spacer()
Text("\(category.messages.count)")
.font(self.messageCountFont())
.bold()
.layoutPriority(1)
.foregroundColor(.green)
}
}
}
}
}
}

// ... other code

关于swiftui - watchOS 上使用 SwiftUI 的意外(自动)导航,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/62971563/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com