[英]SwiftUI View doesn't update
当我在ObservableObject
类中更改数组中的属性时,我的视图不会更新。 我什至声明了一个objectWillChange
属性并手动调用它,但视图只是随机更新,或者不是我想要的。 我不明白这个。
import Foundation
import SocketIO
import Combine
class SocketService: ObservableObject {
static let instance = SocketService()
let manager = SocketManager(socketURL: URL(RequestURL.base_url)!)
let socket: SocketIOClient
// let objectWillChange = ObservableObjectPublisher()
@Published var allMessages: [Message] = []
// {
// willSet {
// self.objectWillChange.send()
// }
// }
@Published var writtenUsers: [PreviewMessage] = []
// {
// willSet {
// self.objectWillChange.send()
// }
// }
init() {
socket = manager.defaultSocket
setSocketEvents()
}
// This method is called
func recieve_read_all_messages() {
socket.on("recieve-read-all-messages") { (data, ack) in
guard let arr = data as? [[String: Any]] else { return }
guard let userID = arr[0]["userID"] as? Int else {
print("no userID, \(#function), line: \(#line)"); return
}
// self.objectWillChange.send()
for msg in self.allMessages {
// Here im trying to change the property in the array
msg.content.readStatus = .read
}
}
}
即使我直接更改此属性,视图也不会更新
@EnvironmentObject private var socketService: SocketService
var body: some View {
VStack {
List(filteredMessages, id: \.content.uuid, rowContent: chatSpeechBubbleView)
sendView
}
}
private func chatSpeechBubbleView(forMessage message: Message) -> some View {
ChatSpeechBubble(message: message)
}
private var sendView: some View {
Button(action: sendMessage) {
SFSymbol(.paperplane_fill)
.fontSize(20)
.foregroundColor(.white)
.rotate(.degrees(45))
.padding(10)
.padding(.trailing, 5)
.backgroundColor(.blue)
.clipCircle()
}
.padding(.bottom, 2)
}
func sendMessage() {
for msg in socketService.allMessages {
msg.content.readStatus = .read
}
}
我应该更新的另一个视图:
struct DoubleCheckmark: View {
var messageContent: MessageContent
var body: some View {
HStack(spacing: 0) {
SFSymbol(.checkmark)
.resizable()
.scaledToFit()
SFSymbol(.checkmark)
.resizable()
.scaledToFit()
.padding(.leading, -9)
}
.height(13)
.foregroundColor(self.messageContent.readStatus == .read ? .blue : .gray)
}
}
struct ChatSpeechBubble: View {
// MARK: - init variables
var message: Message
// MARK: - normal variables
var ownSendMessage: Bool {
message.fromUser.id == UDService.shared.user.id
}
// MARK: - Body
var body: some View {
messageContent
}
private var messageContent: some View {
HStack(alignment: .bottom) {
if message.content.text != nil {
Text(message.content.text!)
.foregroundColor(.black)
}
if message.content.imageURL != nil {
Spacer(minLength: 0)
}
Text(message.content.timeHourMinute)
.font(.caption)
.foregroundColor(.gray)
if ownSendMessage {
DoubleCheckmark(messageContent: self.message.content)
}
}
}
}
struct
,而不是class
问题是Message
及其子类型应该声明为struct
,而不是class
。 这是为什么。
这是我用来演示差异的一个基本示例:
看看底部的打印日志。 使用struct
,再次打印“Body”,但为什么呢?
这是因为类是通过引用传递的,而结构是通过值传递的。 这意味着您可以改变类实例的属性,如下所示:
如果SKSpriteNode
本来是一个struct
,并带有let
属性,则尝试更改let
常量时会出现错误:
Cannot assign to property: '*' is a 'let' constant
所以因为Message
是一个class
所以实例不会改变。 但是有了struct
,整个事情都在改变。 您需要更改整个内容(通过使用struct
),以便@Published
& ObservableObject
可以观察到更改。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.