【发布时间】:2020-08-15 12:26:14
【问题描述】:
我有一个可以保存图像的游戏对象。每当找到游戏的图像 URL 时,都应该创建一个新的 GameImage-object 实例。然后它将获取图像并填充 UIImage 属性。发生这种情况时,应更新 UI 以显示图像。
class Game: ObservableObject {
@Published var image: GameImage?
}
class GameImage: ObservableObject {
let url: URL
@Published var image: UIImage?
private var cancellable: AnyCancellable?
init(url: URL) {
self.url = url
}
func fetch() {
self.cancellable = URLSession.shared.dataTaskPublisher(for: self.url)
.map { UIImage(data: $0.data) }
.replaceError(with: nil)
.receive(on: DispatchQueue.main)
.sink(receiveValue: { [weak self] (image) in
guard let self = self else { return }
self.image = image
print(self.url)
print(self.image)
})
}
func cancel() {
cancellable?.cancel()
}
deinit {
cancel()
}
}
struct ContentView: View {
@StateObject var game = Game()
var body: some View {
VStack {
if let image = game.image?.image {
Image(uiImage: image)
} else {
Text("No image.")
}
}
.onAppear(perform: {
guard let gameImageURL = URL(string: "https://cf.geekdo-images.com/itemrep/img/oVEpcbtyWkJjIjk1peTJo6hI1yk=/fit-in/246x300/pic4884996.jpg") else { return }
game.image = GameImage(url: gameImageURL)
game.image!.fetch()
})
}
}
问题是。获取完成后,调试控制台将显示该图像包含一个 UIImage。但是 UI 不会更新以显示图像。我在这里错过了什么?
【问题讨论】:
-
你会显示依赖于这个视图模型的 UI 吗?
-
我不确定你的意思。游戏实例
game应该包含单一的事实来源。游戏对象还可以包含其他属性,例如我从该示例中删除的名称,以免使事情变得过于复杂。每当图像可用时,它就应该呈现它,以这种方式它是依赖的。
标签: asynchronous swiftui combine