- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我从 Firestore 表中获取一个数组。显然是实时的,因此任何新添加的文档或对现有文档的修改都会反射(reflect)在 View 中。这适用于字符串和数字等简单属性,但我也需要显示和刷新其他数据。
我还存储坐标,如果坐标发生变化,我需要调用 geocoder.reverseGeocodeLocation()
以便从该位置获取地标并将地址显示为字符串。为此,我创建了一个名为 PlacemarkService 的服务(代码如下)。
问题是这样的。每当模型被修改时,数据都会通过 PackagesViewModel
来,它有一个 @Published var results = [Package]()
这是由 Firestore 动态修改的(可编码)。然后,对数组造成的任何更改都将导致 PackagesView
内的 List
被刷新,这将导致 PackageView(packageService: PackageService(package: package))
被创建,并且由于 PackageService
作为参数传递,因此将创建 PackageService
的新实例。最后这将触发该服务内的 init() 被调用。
@Published var results = [Package]() -> List -> PackageView(packageService: PackageService(package: package)) -> init()
现在,如果我遵循此流程并在 init 内触发 getPlacemarks()
,一切都会正常工作(大多数情况下),并且如果包模型发生更改,该函数将在 init 上调用,并将反转 GeocodeLocation修改后的位置。但这里有一个大问题,这就是原因。我无法将此逻辑放在 init 上,因为我无法对 for 循环内的数百个位置进行地理编码。我在列表中显示了多个包,而 MapKit 不允许这样做。
很明显,这个逻辑需要在显示 PackageView
之后触发,因此当选择这些包之一时。
如果我在 PackageView 出现时调用 packageService.getPlacemarks()
,那么第一次工作正常,但是...当 @Published var results = [Package]( 时,不会触发 onAppear )
已更新。
最后的问题是:
在哪里调用 packageService.getPlacemarks()
,以便每当 @Published 内的
已更新Package
时都会调用它 var results = [Package]
但由于我上面解释的原因,不在 PlacemarkService 的初始化中。
抱歉这么长的解释,我只是想弄清楚。
class PackageService: ObservableObject {
var package: Package
private var cancellables = Set<AnyCancellable>()
@Published var sourcePlacemarkService = PlacemarkService()
@Published var destinationPlacemarkService = PlacemarkService()
var cancellable: AnyCancellable? = nil
init(package: Package) {
self.package = package
startListening()
}
func startListening() {
// Listen to placemark changes.
cancellable = Publishers.CombineLatest(sourcePlacemarkService.$placemark, destinationPlacemarkService.$placemark).sink(receiveValue: {_ in
// Publish changes manually to the view.
self.objectWillChange.send()
})
}
// Get placemarks from locations
func getPlacemarks() {
sourcePlacemarkService.reverseGeocodeLocation(location: package.source.toCLLocation)
destinationPlacemarkService.reverseGeocodeLocation(location: package.destination.toCLLocation)
}
}
class PackagesViewModel: ObservableObject {
@Published var results = [Package]()
func load() {
query.addSnapshotListener { (querySnapshot, error) in
// updates results array when a document is modified.
}
}
}
struct PackagesView: View {
@StateObject var packagessViewModel = PackagesViewModel()
var body: some View {
List(packagessViewModel.results, id: \.self) { package in
NavigationLink(destination: PackageView(packageService: PackageService(package: package))) {
Text(package.title)
}
}
}
}
struct PackageView: View {
@ObservedObject var packageService: PackageService
func onAppear() {
packageService.getPlacemarks()
}
var body: some View {
// show the address from placemark after it is geocoded.
VStack {
Text(packageService.sourcePlacemarkService.placemark.title)
Text(packageService.destinationPlacemarkService.placemark.title)
}
.onAppear(perform: onAppear)
}
}
class PlacemarkService: ObservableObject {
@Published var placemark: CLPlacemark?
init(placemark: CLPlacemark? = nil) {
self.placemark = placemark
}
func reverseGeocodeLocation(location: CLLocation?) {
if let location = location {
geocoder.reverseGeocodeLocation(location, completionHandler: { (placemark, error) in
// some code here
self.placemark = placemark
})
}
}
}
struct Package: Identifiable, Codable {
@DocumentID var id: String?
var documentReference: DocumentReference
var uid: String
var title: String
var description: String
var source, destination: GeoPoint
var amount: Double
var createdAt: Timestamp = Timestamp()
var paid: Bool = false
}
最佳答案
首先 - 简化这段代码。大部分代码对于重现问题来说是不必要的,甚至无法编译。下面的代码不是工作代码,而是我们稍后必须开始和更改的代码:
struct ContentView: View {
var body: some View {
NavigationView {
PackagesView()
}
}
}
class PackageService: ObservableObject {
let package: Package
init(package: Package) {
self.package = package
}
// Get placemarks from locations
func getPlacemarks() {
print("getPlacements called")
}
}
class PackagesViewModel: ObservableObject {
@Published var results = [Package]()
}
struct PackagesView: View {
@StateObject var packagesViewModel = PackagesViewModel()
var body: some View {
VStack {
Button("Add new package") {
let number = packagesViewModel.results.count + 1
let new = Package(title: "title \(number)", description: "description \(number)")
packagesViewModel.results.append(new)
}
Button("Change random title") {
guard let randomIndex = packagesViewModel.results.indices.randomElement() else {
return
}
packagesViewModel.results[randomIndex].title = "new title (\(Int.random(in: 1 ... 100)))"
}
List(packagesViewModel.results, id: \.self) { package in
NavigationLink(destination: PackageView(packageService: PackageService(package: package))) {
Text(package.title)
}
}
}
}
}
struct PackageView: View {
@ObservedObject var packageService: PackageService
var body: some View {
VStack {
Text("Title: \(packageService.package.title)")
Text("Description: \(packageService.package.description)")
}
}
}
struct Package: Identifiable, Hashable {
let id = UUID()
var title: String
let description: String
}
现在,解决问题。我通过检测 结果
随 onChange(of:perform:)
的变化解决了这个问题。但是,从这里无法访问 View 主体中使用的 PackageService
。
为了防止出现此问题,PackageService
实际上存储在 PackagesViewModel
中,这在逻辑上对数据流更有意义。现在,由于 PackageService
也是一个 struct
,因此 @Published
在数组上工作以获取 结果
,现在可以使用了。
请参阅下面的代码:
struct PackageService: Hashable {
var package: Package
init(package: Package) {
self.package = package
}
// Get placemarks from locations
mutating func getPlacemarks() {
print("getPlacements called")
// This function is mutating, feel free to set any properties in here
}
}
class PackagesViewModel: ObservableObject {
@Published var results = [PackageService]()
}
struct PackagesView: View {
@StateObject var packagesViewModel = PackagesViewModel()
var body: some View {
VStack {
Button("Add new package") {
let number = packagesViewModel.results.count + 1
let new = Package(title: "title \(number)", description: "description \(number)")
packagesViewModel.results.append(PackageService(package: new))
}
Button("Change random title") {
guard let randomIndex = packagesViewModel.results.indices.randomElement() else {
return
}
let newTitle = "new title (\(Int.random(in: 1 ... 100)))"
packagesViewModel.results[randomIndex].package.title = newTitle
}
List($packagesViewModel.results, id: \.self) { $packageService in
NavigationLink(destination: PackageView(packageService: $packageService)) {
Text(packageService.package.title)
}
}
}
.onChange(of: packagesViewModel.results) { _ in
for packageService in $packagesViewModel.results {
packageService.wrappedValue.getPlacemarks()
}
}
}
}
struct PackageView: View {
@Binding var packageService: PackageService
var body: some View {
VStack {
Text("Title: \(packageService.package.title)")
Text("Description: \(packageService.package.description)")
}
}
}
关于ios - 当模型在 @Published 属性内更新时,SwiftUI 触发函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/69982541/
可不可以命名为MVVM模型?因为View通过查看模型数据。 View 是否应该只与 ViewModelData 交互?我确实在某处读到正确的 MVVM 模型应该在 ViewModel 而不是 Mode
我正在阅读有关设计模式的文章,虽然作者们都认为观察者模式很酷,但在设计方面,每个人都在谈论 MVC。 我有点困惑,MVC 图不是循环的,代码流具有闭合拓扑不是很自然吗?为什么没有人谈论这种模式: mo
我正在开发一个 Sticky Notes 项目并在 WPF 中做 UI,显然将 MVVM 作为我的架构设计选择。我正在重新考虑我的模型、 View 和 View 模型应该是什么。 我有一个名为 Not
不要混淆:How can I convert List to Hashtable in C#? 我有一个模型列表,我想将它们组织成一个哈希表,以枚举作为键,模型列表(具有枚举的值)作为值。 publi
我只是花了一些时间阅读这些术语(我不经常使用它们,因为我们没有任何 MVC 应用程序,我通常只说“模型”),但我觉得根据上下文,这些意味着不同的东西: 实体 这很简单,它是数据库中的一行: 2) In
我想知道你们中是否有人知道一些很好的教程来解释大型应用程序的 MVVM。我发现关于 MVVM 的每个教程都只是基础知识解释(如何实现模型、 View 模型和 View ),但我对在应用程序页面之间传递
我想realm.delete() 我的 Realm 中除了一个模型之外的所有模型。有什么办法可以不列出所有这些吗? 也许是一种遍历 Realm 中当前存在的所有类型的方法? 最佳答案 您可以从您的 R
我正在尝试使用 alias 指令模拟一个 Eloquent 模型,如下所示: $transporter = \Mockery::mock('alias:' . Transporter::class)
我正在使用 stargazer 创建我的 plm 汇总表。 library(plm) library(pglm) data("Unions", package = "pglm") anb1 <- pl
我读了几篇与 ASP.NET 分层架构相关的文章和问题,但是读得太多后我有点困惑。 UI 层是在 ASP.NET MVC 中开发的,对于数据访问,我在项目中使用 EF。 我想通过一个例子来描述我的问题
我收到此消息错误: Inceptionv3.mlmodel: unable to read document 我下载了最新版本的 xcode。 9.4 版测试版 (9Q1004a) 最佳答案 您没有
(同样,一个 MVC 验证问题。我知道,我知道......) 我想使用 AutoMapper ( http://automapper.codeplex.com/ ) 来验证我的创建 View 中不在我
需要澄清一件事,现在我正在处理一个流程,其中我有两个 View 模型,一个依赖于另一个 View 模型,为了处理这件事,我尝试在我的基本 Activity 中注入(inject)两个 View 模型,
如果 WPF MVVM 应该没有代码,为什么在使用 ICommand 时,是否需要在 Window.xaml.cs 代码中实例化 DataContext 属性?我已经并排观看并关注了 YouTube
当我第一次听说 ASP.NET MVC 时,我认为这意味着应用程序由三个部分组成:模型、 View 和 Controller 。 然后我读到 NerdDinner并学习了存储库和 View 模型的方法
Platform : ubuntu 16.04 Python version: 3.5.2 mmdnn version : 0.2.5 Source framework with version :
我正在学习本教程:https://www.raywenderlich.com/160728/object-oriented-programming-swift ...并尝试对代码进行一些个人调整,看看
我正试图围绕 AngularJS。我很喜欢它,但一个核心概念似乎在逃避我——模型在哪里? 例如,如果我有一个显示多个交易列表的应用程序。一个列表向服务器查询匹配某些条件的分页事务集,另一个列表使用不同
我在为某个应用程序找出最佳方法时遇到了麻烦。我不太习惯取代旧 TLA(三层架构)的新架构,所以这就是我的来源。 在为我的应用程序(POCO 类,对吧??)设计模型和 DAL 时,我有以下疑问: 我的模
我有两个模型:Person 和 Department。每个人可以在一个部门工作。部门可以由多人管理。我不确定如何在 Django 模型中构建这种关系。 这是我不成功的尝试之一 [models.py]:
我是一名优秀的程序员,十分优秀!