Meshtastic-Apple/Meshtastic/Views/Nodes/NodeMap.swift

273 lines
9.1 KiB
Swift
Raw Normal View History

2021-08-20 07:56:05 -07:00
//
2021-11-29 21:35:23 -08:00
// NodeMap.swift
// 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
struct NodeMap: View {
2023-05-14 00:16:55 -07:00
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-05-14 00:16:55 -07:00
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
@State var mapTilesAboveLabels: Bool = UserDefaults.mapTilesAboveLabels
2023-05-14 00:16:55 -07:00
let fromDate: NSDate = Calendar.current.date(byAdding: .month, value: -1, to: Date())! as NSDate
@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)
private var positions: FetchedResults<PositionEntity>
2023-05-14 00:16:55 -07:00
2023-01-15 19:15:00 -08:00
@FetchRequest(sortDescriptors: [NSSortDescriptor(key: "name", ascending: false)],
predicate: NSPredicate(
format: "expire == nil || expire >= %@", Date() as NSDate
), 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-05-14 00:16:55 -07:00
@State var selectedTracking: UserTrackingModes = .none
2023-05-08 07:35:31 -07:00
@State var isPresentingInfoSheet: Bool = false
@State private var customMapOverlay: MapViewSwiftUI.CustomMapOverlay? = MapViewSwiftUI.CustomMapOverlay(
2023-05-14 00:16:55 -07:00
mapName: "offlinemap",
tileType: "png",
canReplaceMapContent: true
)
var body: some View {
2023-05-14 00:16:55 -07:00
NavigationStack {
ZStack {
2023-05-14 00:16:55 -07:00
MapViewSwiftUI(
2023-05-14 00:16:55 -07:00
onLongPress: { coord in
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
)
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
}
.ignoresSafeArea(.all, edges: [.top, .leading, .trailing])
.frame(maxHeight: .infinity)
.sheet(item: $waypointCoordinate, content: { wpc in
2023-05-14 00:16:55 -07:00
WaypointFormView(coordinate: wpc)
.presentationDetents([.medium, .large])
.presentationDragIndicator(.automatic)
})
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))
.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))
.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))
.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-05-13 15:46:18 -07:00
.onChange(of: (enableOfflineMaps)) { newEnableOfflineMaps in
2023-05-16 05:54:12 -07:00
UserDefaults.enableOfflineMaps = enableOfflineMaps
if !enableOfflineMaps {
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 {
VStack (alignment: .leading) {
2023-04-25 17:56:57 -07:00
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
}
Text("Attribution:")
.fontWeight(.semibold)
.font(.footnote)
2023-05-08 07:35:31 -07:00
Text(LocalizedStringKey(selectedTileServer.attribution))
.font(.footnote)
.foregroundColor(.gray)
.padding(0)
Divider()
Toggle(isOn: $mapTilesAboveLabels) {
Text("Tiles above Labels")
}
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
.onTapGesture {
self.mapTilesAboveLabels.toggle()
UserDefaults.mapTilesAboveLabels = self.mapTilesAboveLabels
}
2023-05-08 07:35:31 -07:00
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
}
Text("The latest MBTiles file shared with meshtastic will be loaded into the map.")
.font(.footnote)
.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)
}
}
2022-10-17 21:18:57 -07:00
.navigationBarItems(leading:
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 :
"????")
})
.onAppear(perform: {
2023-02-26 08:57:13 -08:00
UIApplication.shared.isIdleTimerDisabled = true
2022-10-17 21:18:57 -07:00
self.bleManager.context = context
})
2023-03-06 10:33:18 -08:00
.onDisappear(perform: {
UIApplication.shared.isIdleTimerDisabled = false
})
2023-04-18 00:09:13 -07:00
}
2021-08-20 07:56:05 -07:00
}