Reputation: 2876
I am an android developer and I started to learn Swift. So far I managed to do everything I wanted, but I struggle with listening to updates in SwiftUI
.
I implemented a service that uses Firestore
to read from a database. This service is an ObservableObject
. Inside the service, I have a variable that is annotated as @Published
:
@Published var events = [Event]()
I have implemented Google Maps and I want to fetch these events and do something with them but I don't know how to listen and do something when the call is done.
Google Maps class:
import SwiftUI
import GoogleMaps
struct GoogleMapsView: UIViewRepresentable {
@ObservedObject var databaseService = DatabaseService()
func makeUIView(context: Self.Context) -> GMSMapView {
let camera = GMSCameraPosition.camera(withLatitude: 45.764375, longitude: 24.994272, zoom: 6.0)
let mapView = GMSMapView.map(withFrame: CGRect.zero, camera: camera)
// Here I think I call .getEvents
databaseService.getEvents()
return mapView
}
func updateUIView(_ mapView: GMSMapView, context: Context) {
// Listen to events update and update them like this, but I don't know how...
for event in self.databaseService.events {
let marker = GMSMarker()
marker.position = CLLocationCoordinate2D(latitude: event.lat, longitude: event.long)
marker.title = event.title
marker.snippet = event.title
marker.map = mapView
}
}
}
struct GoogleMapsView_Previews: PreviewProvider {
static var previews: some View {
GoogleMapsView()
}
}
And my SwiftUI
View:
import SwiftUI
struct CommunicationsView: View {
@ObservedObject var databaseService = DatabaseService()
var body: some View {
TabView() {
GoogleMapsView()
.edgesIgnoringSafeArea(.top)
.tabItem {
Image(systemName: "leaf.fill")
Text("Campanii")
}
}.accentColor(.green)
}
}
struct CommunicationsView_Previews: PreviewProvider {
static var previews: some View {
CommunicationsView()
}
}
Also, here is my service:
class DatabaseService : ObservableObject {
let db: Firestore
@Published var events = [Event]()
init() {
db = Firestore.firestore()
}
func getEvents() -> Void {
var tempoEvents = [Event]()
db.collection(DatabaseName.CAMPAIGNS.rawValue).getDocuments { (query: QuerySnapshot?, error: Error?) in
for document in query!.documents {
let data = document.data()
let title = data["title"] as? String ?? ""
let details = data["details"] as? String ?? ""
let lat = data["lat"] as? Double ?? 0.0
let long = data["long"] as? Double ?? 0.0
let date = data["date"] as? Date ?? Date()
let author = data["author"] as? String ?? ""
tempoEvents.append(Event(id: document.documentID, title: title, details: details, lat: lat, long: long, date: date, author: author))
}
self.events.append(contentsOf: tempoEvents)
}
}
}
Upvotes: 2
Views: 209
Reputation: 100523
You need to put
@ObservedObject var databaseService = DatabaseService()
only inside CommunicationsView
with
GoogleMapsView(events:self.databaseService.events)
and add a property var events = [Event]()
inside GoogleMapsView
struct GoogleMapsView: UIViewRepresentable {
var events = [Event]()
......
for event in self.events {
}
}
finally in init
of DatabaseService
init() {
db = Firestore.firestore()
self.getEvents()
}
Upvotes: 1