2021-08-20 07:56:05 -07:00
|
|
|
//
|
2021-11-29 21:35:23 -08:00
|
|
|
// NodeMap.swift
|
2022-06-09 22:11:54 -07:00
|
|
|
// MeshtasticApple
|
2021-08-20 07:56:05 -07:00
|
|
|
//
|
|
|
|
|
// Created by Garth Vander Houwen on 8/7/21.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
import SwiftUI
|
|
|
|
|
import MapKit
|
|
|
|
|
import CoreLocation
|
2021-12-20 22:29:28 -08:00
|
|
|
import CoreData
|
2021-08-20 07:56:05 -07:00
|
|
|
|
2021-09-18 15:33:35 -07:00
|
|
|
struct NodeMap: View {
|
2021-12-15 23:53:45 -08:00
|
|
|
@Environment(\.managedObjectContext) var context
|
2021-11-29 15:59:06 -08:00
|
|
|
@EnvironmentObject var bleManager: BLEManager
|
2023-05-08 07:35:31 -07:00
|
|
|
@ObservedObject var tileManager = OfflineTileManager.shared
|
2023-08-22 11:18:10 -05:00
|
|
|
@StateObject var appState = AppState.shared
|
2023-05-06 16:15:12 -07:00
|
|
|
@State var selectedMapLayer: MapLayer = UserDefaults.mapLayer
|
2023-04-25 17:56:57 -07:00
|
|
|
@State var enableMapRecentering: Bool = UserDefaults.enableMapRecentering
|
|
|
|
|
@State var enableMapRouteLines: Bool = UserDefaults.enableMapRouteLines
|
|
|
|
|
@State var enableMapNodeHistoryPins: Bool = UserDefaults.enableMapNodeHistoryPins
|
|
|
|
|
@State var enableOfflineMaps: Bool = UserDefaults.enableOfflineMaps
|
2023-05-14 00:16:55 -07:00
|
|
|
@State var selectedTileServer: MapTileServer = UserDefaults.mapTileServer
|
2023-05-06 18:40:11 -07:00
|
|
|
@State var enableOfflineMapsMBTiles: Bool = UserDefaults.enableOfflineMapsMBTiles
|
2023-05-14 00:16:55 -07:00
|
|
|
@State var enableOverlayServer: Bool = UserDefaults.enableOverlayServer
|
|
|
|
|
@State var selectedOverlayServer: MapOverlayServer = UserDefaults.mapOverlayServer
|
2023-05-07 08:01:16 -07:00
|
|
|
@State var mapTilesAboveLabels: Bool = UserDefaults.mapTilesAboveLabels
|
2023-06-28 22:16:38 -07:00
|
|
|
let fromDate: NSDate = Calendar.current.date(byAdding: .month, value: -1, to: Date())! as NSDate
|
2023-09-10 14:59:51 -07:00
|
|
|
// @FetchRequest(sortDescriptors: [NSSortDescriptor(key: "time", ascending: true)],
|
|
|
|
|
// predicate: NSPredicate(format: "time >= %@ && nodePosition != nil", Calendar.current.date(byAdding: .day, value: -7, to: Date())! as NSDate), animation: .none)
|
2023-02-10 20:50:25 -08:00
|
|
|
@FetchRequest(sortDescriptors: [NSSortDescriptor(key: "time", ascending: true)],
|
2023-09-10 14:59:51 -07:00
|
|
|
predicate: NSPredicate(format: "nodePosition != nil", Calendar.current.date(byAdding: .day, value: -7, to: Date())! as NSDate), animation: .none)
|
2023-01-13 15:49:25 -08:00
|
|
|
private var positions: FetchedResults<PositionEntity>
|
2023-01-15 19:15:00 -08:00
|
|
|
@FetchRequest(sortDescriptors: [NSSortDescriptor(key: "name", ascending: false)],
|
|
|
|
|
predicate: NSPredicate(
|
|
|
|
|
format: "expire == nil || expire >= %@", Date() as NSDate
|
2023-02-06 18:45:03 -08:00
|
|
|
), animation: .none)
|
2023-01-13 22:30:10 -08:00
|
|
|
private var waypoints: FetchedResults<WaypointEntity>
|
2023-05-06 16:15:12 -07:00
|
|
|
@State var waypointCoordinate: WaypointCoordinate?
|
2023-04-23 22:38:02 -07:00
|
|
|
@State var selectedTracking: UserTrackingModes = .none
|
|
|
|
|
@State var isPresentingInfoSheet: Bool = false
|
2023-01-13 15:49:25 -08:00
|
|
|
@State private var customMapOverlay: MapViewSwiftUI.CustomMapOverlay? = MapViewSwiftUI.CustomMapOverlay(
|
2023-05-14 00:16:55 -07:00
|
|
|
mapName: "offlinemap",
|
|
|
|
|
tileType: "png",
|
|
|
|
|
canReplaceMapContent: true
|
|
|
|
|
)
|
2023-02-27 11:16:46 -08:00
|
|
|
var body: some View {
|
|
|
|
|
NavigationStack {
|
2023-01-13 15:49:25 -08:00
|
|
|
ZStack {
|
2023-04-22 12:31:06 -07:00
|
|
|
MapViewSwiftUI(
|
2023-05-14 00:16:55 -07:00
|
|
|
onLongPress: { coord in
|
2023-04-22 12:31:06 -07:00
|
|
|
waypointCoordinate = WaypointCoordinate(id: .init(), coordinate: coord, waypointId: 0)
|
|
|
|
|
}, onWaypointEdit: { wpId in
|
|
|
|
|
if wpId > 0 {
|
|
|
|
|
waypointCoordinate = WaypointCoordinate(id: .init(), coordinate: nil, waypointId: Int64(wpId))
|
|
|
|
|
}
|
|
|
|
|
},
|
2023-05-14 00:16:55 -07:00
|
|
|
selectedMapLayer: selectedMapLayer,
|
|
|
|
|
positions: Array(positions),
|
|
|
|
|
waypoints: Array(waypoints),
|
|
|
|
|
userTrackingMode: selectedTracking.MKUserTrackingModeValue(),
|
|
|
|
|
showNodeHistory: enableMapNodeHistoryPins,
|
|
|
|
|
showRouteLines: enableMapRouteLines,
|
|
|
|
|
customMapOverlay: self.customMapOverlay
|
2022-01-10 06:09:31 +13:00
|
|
|
)
|
2023-04-23 22:38:02 -07:00
|
|
|
VStack(alignment: .trailing) {
|
|
|
|
|
HStack(alignment: .top) {
|
|
|
|
|
Spacer()
|
|
|
|
|
MapButtons(tracking: $selectedTracking, isPresentingInfoSheet: $isPresentingInfoSheet)
|
|
|
|
|
.padding(.trailing, 8)
|
|
|
|
|
.padding(.top, 16)
|
|
|
|
|
}
|
|
|
|
|
Spacer()
|
|
|
|
|
}
|
2022-12-30 13:18:02 -08:00
|
|
|
}
|
2023-01-13 15:49:25 -08:00
|
|
|
.ignoresSafeArea(.all, edges: [.top, .leading, .trailing])
|
|
|
|
|
.frame(maxHeight: .infinity)
|
2023-04-22 12:31:06 -07:00
|
|
|
.sheet(item: $waypointCoordinate, content: { wpc in
|
2023-11-09 17:43:38 -08:00
|
|
|
WaypointFormMapKit(coordinate: wpc)
|
2023-05-14 00:16:55 -07:00
|
|
|
.presentationDetents([.medium, .large])
|
|
|
|
|
.presentationDragIndicator(.automatic)
|
2023-04-22 12:31:06 -07:00
|
|
|
})
|
2023-04-25 17:56:57 -07:00
|
|
|
.sheet(isPresented: $isPresentingInfoSheet) {
|
|
|
|
|
VStack {
|
|
|
|
|
Form {
|
|
|
|
|
Section(header: Text("Map Options")) {
|
2023-05-06 16:15:12 -07:00
|
|
|
Picker(selection: $selectedMapLayer, label: Text("")) {
|
|
|
|
|
ForEach(MapLayer.allCases, id: \.self) { layer in
|
|
|
|
|
if layer == MapLayer.offline && UserDefaults.enableOfflineMaps {
|
|
|
|
|
Text(layer.localized)
|
|
|
|
|
} else if layer != MapLayer.offline {
|
|
|
|
|
Text(layer.localized)
|
|
|
|
|
}
|
2023-04-25 17:56:57 -07:00
|
|
|
}
|
|
|
|
|
}
|
2023-05-06 16:15:12 -07:00
|
|
|
.pickerStyle(SegmentedPickerStyle())
|
|
|
|
|
.onChange(of: (selectedMapLayer)) { newMapLayer in
|
|
|
|
|
UserDefaults.mapLayer = newMapLayer
|
2023-04-25 17:56:57 -07:00
|
|
|
}
|
2023-05-06 16:15:12 -07:00
|
|
|
.padding(.top, 5)
|
|
|
|
|
.padding(.bottom, 5)
|
2023-04-25 17:56:57 -07:00
|
|
|
Toggle(isOn: $enableMapRecentering) {
|
|
|
|
|
Label("map.recentering", systemImage: "camera.metering.center.weighted")
|
|
|
|
|
}
|
|
|
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
2023-04-25 21:44:34 -07:00
|
|
|
.onTapGesture {
|
|
|
|
|
self.enableMapRecentering.toggle()
|
|
|
|
|
UserDefaults.enableMapRecentering = self.enableMapRecentering
|
|
|
|
|
}
|
2023-04-25 17:56:57 -07:00
|
|
|
Toggle(isOn: $enableMapNodeHistoryPins) {
|
|
|
|
|
Label("Show Node History", systemImage: "building.columns.fill")
|
|
|
|
|
}
|
|
|
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
2023-04-25 21:44:34 -07:00
|
|
|
.onTapGesture {
|
|
|
|
|
self.enableMapNodeHistoryPins.toggle()
|
|
|
|
|
UserDefaults.enableMapNodeHistoryPins = self.enableMapNodeHistoryPins
|
|
|
|
|
}
|
2023-04-25 17:56:57 -07:00
|
|
|
Toggle(isOn: $enableMapRouteLines) {
|
|
|
|
|
Label("Show Route Lines", systemImage: "road.lanes")
|
|
|
|
|
}
|
|
|
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
2023-04-25 21:44:34 -07:00
|
|
|
.onTapGesture {
|
|
|
|
|
self.enableMapRouteLines.toggle()
|
|
|
|
|
UserDefaults.enableMapRouteLines = self.enableMapRouteLines
|
2023-04-25 17:56:57 -07:00
|
|
|
}
|
2023-05-14 00:16:55 -07:00
|
|
|
let locale = Locale.current
|
|
|
|
|
if locale.region?.identifier ?? "no locale" == "US" {
|
|
|
|
|
Toggle(isOn: $enableOverlayServer) {
|
|
|
|
|
Label("Show Weather", systemImage: "cloud.fill")
|
|
|
|
|
}
|
|
|
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
|
|
|
.onTapGesture {
|
|
|
|
|
self.enableOverlayServer.toggle()
|
|
|
|
|
UserDefaults.enableOverlayServer = self.enableOverlayServer
|
|
|
|
|
}
|
|
|
|
|
if enableOverlayServer {
|
|
|
|
|
Picker(selection: $selectedOverlayServer,
|
|
|
|
|
label: Text("Radar")) {
|
|
|
|
|
ForEach(MapOverlayServer.allCases, id: \.self) { mos in
|
|
|
|
|
Text(mos.description)
|
|
|
|
|
.font(.footnote)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
.pickerStyle(DefaultPickerStyle())
|
|
|
|
|
.onChange(of: (selectedOverlayServer)) { newSelectedOverlayServer in
|
|
|
|
|
UserDefaults.mapOverlayServer = newSelectedOverlayServer
|
|
|
|
|
}
|
|
|
|
|
Text(LocalizedStringKey(selectedOverlayServer.attribution))
|
|
|
|
|
.font(.footnote)
|
|
|
|
|
.foregroundColor(.gray)
|
|
|
|
|
.padding(0)
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-04-25 17:56:57 -07:00
|
|
|
}
|
|
|
|
|
Section(header: Text("Offline Maps")) {
|
|
|
|
|
Toggle(isOn: $enableOfflineMaps) {
|
|
|
|
|
Text("Enable Offline Maps")
|
|
|
|
|
}
|
|
|
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
2023-08-26 23:17:30 -07:00
|
|
|
.onChange(of: enableOfflineMaps) { newEnableOfflineMaps in
|
|
|
|
|
UserDefaults.enableOfflineMaps = newEnableOfflineMaps
|
2023-05-16 05:54:12 -07:00
|
|
|
if !enableOfflineMaps {
|
2023-05-07 08:01:16 -07:00
|
|
|
if self.selectedMapLayer == .offline {
|
|
|
|
|
self.selectedMapLayer = .standard
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-04-25 17:56:57 -07:00
|
|
|
}
|
2023-05-16 05:54:12 -07:00
|
|
|
if enableOfflineMaps {
|
2023-08-26 23:17:30 -07:00
|
|
|
VStack(alignment: .leading) {
|
2023-05-06 18:40:11 -07:00
|
|
|
if !enableOfflineMapsMBTiles {
|
2023-05-08 07:35:31 -07:00
|
|
|
Picker(selection: $selectedTileServer,
|
|
|
|
|
label: Text("Tile Server")) {
|
2023-05-14 00:16:55 -07:00
|
|
|
ForEach(MapTileServer.allCases, id: \.self) { tsl in
|
2023-05-08 07:35:31 -07:00
|
|
|
Text(tsl.description)
|
2023-05-06 18:40:11 -07:00
|
|
|
}
|
2023-05-05 17:13:35 -07:00
|
|
|
}
|
2023-05-14 00:16:55 -07:00
|
|
|
.pickerStyle(DefaultPickerStyle())
|
|
|
|
|
.onChange(of: (selectedTileServer)) { newSelectedTileServer in
|
|
|
|
|
UserDefaults.mapTileServer = newSelectedTileServer
|
|
|
|
|
}
|
2023-05-09 19:31:25 -07:00
|
|
|
Text("Attribution:")
|
|
|
|
|
.fontWeight(.semibold)
|
|
|
|
|
.font(.footnote)
|
2023-05-08 07:35:31 -07:00
|
|
|
Text(LocalizedStringKey(selectedTileServer.attribution))
|
2023-05-09 19:31:25 -07:00
|
|
|
.font(.footnote)
|
2023-05-07 08:01:16 -07:00
|
|
|
.foregroundColor(.gray)
|
2023-05-09 19:31:25 -07:00
|
|
|
.padding(0)
|
2023-05-07 08:01:16 -07:00
|
|
|
Divider()
|
|
|
|
|
Toggle(isOn: $mapTilesAboveLabels) {
|
|
|
|
|
Text("Tiles above Labels")
|
|
|
|
|
}
|
|
|
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
|
|
|
.onTapGesture {
|
|
|
|
|
self.mapTilesAboveLabels.toggle()
|
|
|
|
|
UserDefaults.mapTilesAboveLabels = self.mapTilesAboveLabels
|
|
|
|
|
}
|
2023-04-25 17:56:57 -07:00
|
|
|
}
|
2023-05-08 07:35:31 -07:00
|
|
|
Divider()
|
2023-05-06 18:40:11 -07:00
|
|
|
Toggle(isOn: $enableOfflineMapsMBTiles) {
|
|
|
|
|
Text("Enable MB Tiles")
|
|
|
|
|
}
|
|
|
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
|
|
|
.onTapGesture {
|
|
|
|
|
self.enableOfflineMapsMBTiles.toggle()
|
|
|
|
|
UserDefaults.enableOfflineMapsMBTiles = self.enableOfflineMapsMBTiles
|
|
|
|
|
}
|
2023-05-07 08:01:16 -07:00
|
|
|
Text("The latest MBTiles file shared with meshtastic will be loaded into the map.")
|
2023-05-09 19:31:25 -07:00
|
|
|
.font(.footnote)
|
2023-05-07 08:01:16 -07:00
|
|
|
.foregroundColor(.gray)
|
2023-04-25 17:56:57 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#if targetEnvironment(macCatalyst)
|
|
|
|
|
Button {
|
|
|
|
|
isPresentingInfoSheet = false
|
|
|
|
|
} label: {
|
|
|
|
|
Label("close", systemImage: "xmark")
|
|
|
|
|
}
|
|
|
|
|
.buttonStyle(.bordered)
|
|
|
|
|
.buttonBorderShape(.capsule)
|
|
|
|
|
.controlSize(.large)
|
|
|
|
|
.padding(.bottom)
|
|
|
|
|
#endif
|
|
|
|
|
}
|
2023-05-14 00:16:55 -07:00
|
|
|
.presentationDetents([UserDefaults.enableOfflineMaps || UserDefaults.enableOverlayServer ? .large : .medium])
|
2023-04-25 17:56:57 -07:00
|
|
|
.presentationDragIndicator(.visible)
|
|
|
|
|
}
|
2023-02-27 11:16:46 -08:00
|
|
|
}
|
2022-10-17 21:18:57 -07:00
|
|
|
.navigationBarItems(leading:
|
2023-02-27 11:16:46 -08:00
|
|
|
MeshtasticLogo(), trailing:
|
|
|
|
|
ZStack {
|
2022-10-17 21:18:57 -07:00
|
|
|
ConnectedDevice(
|
|
|
|
|
bluetoothOn: bleManager.isSwitchedOn,
|
|
|
|
|
deviceConnected: bleManager.connectedPeripheral != nil,
|
|
|
|
|
name: (bleManager.connectedPeripheral != nil) ? bleManager.connectedPeripheral.shortName :
|
2023-09-02 17:37:35 -07:00
|
|
|
"?")
|
2022-10-17 21:18:57 -07:00
|
|
|
})
|
|
|
|
|
.onAppear(perform: {
|
2023-02-26 08:57:13 -08:00
|
|
|
UIApplication.shared.isIdleTimerDisabled = true
|
2023-09-10 18:50:11 -07:00
|
|
|
if self.bleManager.context == nil {
|
|
|
|
|
self.bleManager.context = context
|
|
|
|
|
}
|
2022-10-17 21:18:57 -07:00
|
|
|
})
|
2023-03-06 10:33:18 -08:00
|
|
|
.onDisappear(perform: {
|
2023-02-27 11:16:46 -08:00
|
|
|
UIApplication.shared.isIdleTimerDisabled = false
|
|
|
|
|
})
|
2023-04-18 00:09:13 -07:00
|
|
|
}
|
2021-08-20 07:56:05 -07:00
|
|
|
}
|