// // LocationsHandler.swift // Meshtastic // // Created by Garth Vander Houwen on 12/4/23. // import SwiftUI import CoreLocation import OSLog // Shared state that manages the `CLLocationManager` and `CLBackgroundActivitySession`. @MainActor class LocationsHandler: ObservableObject { static let shared = LocationsHandler() // Create a single, shared instance of the object. private let manager: CLLocationManager private var background: CLBackgroundActivitySession? var enableSmartPosition: Bool = UserDefaults.enableSmartPosition @Published var locationsArray: [CLLocation] @Published var isStationary = false @Published var count = 0 @Published var isRecording = false @Published var isRecordingPaused = false @Published var recordingStarted: Date? @Published var distanceTraveled = 0.0 @Published var elevationGain = 0.0 @Published var updatesStarted: Bool = UserDefaults.standard.bool(forKey: "liveUpdatesStarted") { didSet { UserDefaults.standard.set(updatesStarted, forKey: "liveUpdatesStarted") } } @Published var backgroundActivity: Bool = UserDefaults.standard.bool(forKey: "BGActivitySessionStarted") { didSet { backgroundActivity ? self.background = CLBackgroundActivitySession() : self.background?.invalidate() UserDefaults.standard.set(backgroundActivity, forKey: "BGActivitySessionStarted") } } private init() { self.manager = CLLocationManager() // Creating a location manager instance is safe to call here in `MainActor`. self.manager.allowsBackgroundLocationUpdates = true locationsArray = [CLLocation]() } func startLocationUpdates() { if self.manager.authorizationStatus == .notDetermined { self.manager.requestWhenInUseAuthorization() } Logger.services.info("📍 [App] Starting location updates") Task { do { self.updatesStarted = true let updates = CLLocationUpdate.liveUpdates() for try await update in updates { if !self.updatesStarted { break } if let loc = update.location { self.isStationary = update.isStationary var locationAdded: Bool locationAdded = addLocation(loc, smartPostion: enableSmartPosition) if !isRecording && locationAdded { self.count = 1 } else if locationAdded && isRecording { self.count += 1 } } } } catch { Logger.services.error("💥 [App] Could not start location updates: \(error.localizedDescription)") } return } } func stopLocationUpdates() { Logger.services.info("🛑 [App] Stopping location updates") self.updatesStarted = false } func addLocation(_ location: CLLocation, smartPostion: Bool) -> Bool { if smartPostion { let age = -location.timestamp.timeIntervalSinceNow if age > 10 { Logger.services.info("📍 [App] Smart Position - Bad Location: Too Old \(age, privacy: .public) seconds ago \(location, privacy: .private)") return false } if location.horizontalAccuracy < 0 { Logger.services.info("📍 [App] Smart Position - Bad Location: Horizontal Accuracy: \(location.horizontalAccuracy) \(location, privacy: .private)") return false } if location.horizontalAccuracy > 5 { Logger.services.info("📍 [App] Smart Position - Bad Location: Horizontal Accuracy: \(location.horizontalAccuracy) \(location, privacy: .private)") return false } } if isRecording { if let lastLocation = locationsArray.last { let distance = location.distance(from: lastLocation) let gain = location.altitude - lastLocation.altitude distanceTraveled += distance if gain > 0 { elevationGain += gain } } locationsArray.append(location) } else { locationsArray = [location] } return true } static let DefaultLocation = CLLocationCoordinate2D(latitude: 37.3346, longitude: -122.0090) static var satsInView: Int { var sats = 0 if let newLocation = shared.locationsArray.last { sats = 1 if newLocation.verticalAccuracy > 0 { sats = 4 if 0...5 ~= newLocation.horizontalAccuracy { sats = 12 } else if 6...15 ~= newLocation.horizontalAccuracy { sats = 10 } else if 16...30 ~= newLocation.horizontalAccuracy { sats = 9 } else if 31...45 ~= newLocation.horizontalAccuracy { sats = 7 } else if 46...60 ~= newLocation.horizontalAccuracy { sats = 5 } } else if newLocation.verticalAccuracy < 0 && 60...300 ~= newLocation.horizontalAccuracy { sats = 3 } else if newLocation.verticalAccuracy < 0 && newLocation.horizontalAccuracy > 300 { sats = 2 } } return sats } }