mirror of
https://github.com/meshtastic/Meshtastic-Apple.git
synced 2026-04-20 22:13:56 +00:00
* Initial implementation of transports * Initial LogRadio implementation * Fixes for Settings view (caused by debug commenting) * Refinement of the object and actor model * Connect view text and tab updates * Fix mac catalyst and tests * Warning and logging clean-up * In progress commit * Serial Transport and Reconnect draft work * Serial transport and reconnection draft work * Quick fix for BLE - still more work to do * interim commit * More in progress changes * Minor improvements * Pretty good initial implementation * Bump version beyond the app store * Fix for disconnection swipeAction * Tweaks to TCPConnection implementation * Retry for NONCE_ONLY_DB * Revert json string change * Simplified some of the API + "Anti-discovery" * Tweaks for devices leaving the discovery process * Bump version * iOS26 Tweaks * Tweaks and bug fixes * Add link with slash sf symbol * update symbol image on connect view * BLE disconnect handling * Log privacy attributes * Onboarding and minor fixes. * change database to nodes, add emoji to tcp logs * Error handling improvements * More logging emojis * Suppressed unnecessary errors on disconnect * Heartbeat emoji * Add bluetooth symbol * add privacy attributes to [TCP] logs, add custom bluetooth logo * Improve routing logs * Emoji for connect logs * Heartbeat emoji * Add CBCentralManagerScanOptionAllowDuplicatesKey options to central for bluetooth * fix nav errors by switching from observableobject to state * Update connection indicator icon * fix for BLE disconnects * Connection process fixes * More fixes/tweaks to connection process * Strict concurrency * Fix some warnings, remove wifi warning * delete stale keys * interim commit * Update privacy for log, fix wrong space * fix a couple of linting items * Switch to targeted * interim commit * BLE Signal strenth on connect view * Remove BLE RSSI from long press menu * Modem lights * minor spacing tweak * Additional BLE logging and a scanning fix. * Discovery and BLE RSSI improvements * Background suspension * Update isConnected to enable UI during db load * update protobufs * Replace config if statements with switches, Fix unknown module config logging, make dark mode modem circle stroke color white so they are visible * Additional logging cleanup * hast * Set unmessagable to true if the longname has the unmessagable emoji * Connect error handling improvements * Admin popup list icon and activity lights updates * Revert use of .toolbar back to .navigationBarItems * More public logging * Better BLE error handling * Node DB progress meter * minor tweak to activity light interaction timing * Fix comment linting, remove stale keys * Remove stale keys * Easy linting fixes * Two more simple linting fixes * clean up meshtasticapp * More public logging * Replay config * Logging * Fix for unselected node on Settings * Tweak to progress meter based on device idiom * Update protos * Session replay redaction of messages * Serial fix for old devices, and a let statement * Mask text too * Fix typo * BLE poweredOff is now an auto-reconnectable error * Update logging * Fix for peerRemovedPairingInformation * Logging for BLE peripheral:didUpdateValueFor errors. * Fix for inconsistent swipe disconnect behavior * periperal:didUpdateValueFor error handling * Fix for BLEConnection continuation guarding * BLEConnection actor deadlock on disconnect * Heartbeat nonce * Fix for swipe disconnect and task cancellation * Fix for swipe actions not honoring .disabled() * Tell BLETransport when BLEConnection is cancelled * Update navigation logging * Logging updates * Bump version to 2.7.0 * Organize into folders and heartbeat stuff * Minor improvements to manual TCP connection * Auto-connect toggle * Possible BLE bug, still waiting to see in logs * Concurrency tweaks * Concurrency improvements * requestDeviceMetadata fix. fixes remote admin * Minor typo fixes * "All" button for log filters: category and level * More robust continuation handling for BLE * @FetchRequest based ChannelMessageList * Update info.plist and device hardware file * Move auto connect toggle to app settings and debug mode, tint properly with the accent color * Add label to auto connect toggle * Update log for node info received from ourselves over the mesh * Remove unused scrollViewProxy * Update Meshtastic/Views/Onboarding/DeviceOnboarding.swift Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Update target for connect view * Properly Set datadog environment * Comment out ble manager * Adjust cyclomatic complexity thresholds in .swiftlint.yml * Linting fixes, delete ble manager * Make session replay debug only --------- Co-authored-by: jake-b <jake-b@users.noreply.github.com> Co-authored-by: jake <jake@jakes-Mac-mini.local> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
565 lines
20 KiB
Swift
565 lines
20 KiB
Swift
//
|
|
// PositionConfig.swift
|
|
// Meshtastic Apple
|
|
//
|
|
// Copyright (c) Garth Vander Houwen 6/11/22.
|
|
//
|
|
|
|
import SwiftUI
|
|
import MeshtasticProtobufs
|
|
import OSLog
|
|
|
|
struct PositionFlags: OptionSet {
|
|
let rawValue: Int
|
|
static let Altitude = PositionFlags(rawValue: 1)
|
|
static let AltitudeMsl = PositionFlags(rawValue: 2)
|
|
static let GeoidalSeparation = PositionFlags(rawValue: 4)
|
|
static let Dop = PositionFlags(rawValue: 8)
|
|
static let Hvdop = PositionFlags(rawValue: 16)
|
|
static let Satsinview = PositionFlags(rawValue: 32)
|
|
static let SeqNo = PositionFlags(rawValue: 64)
|
|
static let Timestamp = PositionFlags(rawValue: 128)
|
|
static let Speed = PositionFlags(rawValue: 256)
|
|
static let Heading = PositionFlags(rawValue: 512)
|
|
}
|
|
|
|
struct PositionConfig: View {
|
|
|
|
@Environment(\.managedObjectContext) var context
|
|
@EnvironmentObject var accessoryManager: AccessoryManager
|
|
@Environment(\.dismiss) private var goBack
|
|
var node: NodeInfoEntity?
|
|
@State var hasChanges = false
|
|
@State var hasFlagChanges = false
|
|
@State var smartPositionEnabled = true
|
|
@State var deviceGpsEnabled = true
|
|
@State var gpsMode = 0
|
|
@State var rxGpio = 0
|
|
@State var txGpio = 0
|
|
@State var gpsEnGpio = 0
|
|
@State var fixedPosition = false
|
|
@State var gpsUpdateInterval = 0
|
|
@State var positionBroadcastSeconds = 0
|
|
@State var broadcastSmartMinimumDistance = 0
|
|
@State var broadcastSmartMinimumIntervalSecs = 0
|
|
@State var positionFlags = 811
|
|
|
|
/// Position Flags
|
|
/// Altitude value - 1
|
|
@State var includeAltitude = false
|
|
/// Altitude value is MSL - 2
|
|
@State var includeAltitudeMsl = false
|
|
/// Include geoidal separation - 4
|
|
@State var includeGeoidalSeparation = false
|
|
/// Include the DOP value ; PDOP used by default, see below - 8
|
|
@State var includeDop = false
|
|
/// If POS_DOP set, send separate HDOP / VDOP values instead of PDOP - 16
|
|
@State var includeHvdop = false
|
|
/// Include number of "satellites in view" - 32
|
|
@State var includeSatsinview = false
|
|
/// Include a sequence number incremented per packet - 64
|
|
@State var includeSeqNo = false
|
|
/// Include positional timestamp (from GPS solution) - 128
|
|
@State var includeTimestamp = false
|
|
/// Include positional heading - 256
|
|
/// Intended for use with vehicle not walking speeds
|
|
/// walking speeds are likely to be error prone like the compass
|
|
@State var includeSpeed = false
|
|
/// Include positional speed - 512
|
|
/// Intended for use with vehicle not walking speeds
|
|
/// walking speeds are likely to be error prone like the compass
|
|
@State var includeHeading = false
|
|
/// Minimum Version for fixed postion admin messages
|
|
@State var minimumVersion = "2.3.3"
|
|
@State private var supportedVersion = true
|
|
@State private var showingSetFixedAlert = false
|
|
// @State private var showingRemoveFixedAlert = false
|
|
|
|
@ViewBuilder
|
|
var positionPacketSection: some View {
|
|
Section(header: Text("Position Packet")) {
|
|
|
|
VStack(alignment: .leading) {
|
|
Picker("Broadcast Interval", selection: $positionBroadcastSeconds) {
|
|
ForEach(UpdateIntervals.allCases) { at in
|
|
if at.rawValue >= 300 {
|
|
Text(at.description)
|
|
}
|
|
}
|
|
}
|
|
.pickerStyle(DefaultPickerStyle())
|
|
Text("The maximum interval that can elapse without a node broadcasting a position")
|
|
.foregroundColor(.gray)
|
|
.font(.callout)
|
|
}
|
|
|
|
Toggle(isOn: $smartPositionEnabled) {
|
|
Label("Smart Position", systemImage: "brain")
|
|
}
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
|
|
if smartPositionEnabled {
|
|
VStack(alignment: .leading) {
|
|
Picker("Minimum Interval", selection: $broadcastSmartMinimumIntervalSecs) {
|
|
ForEach(UpdateIntervals.allCases) { at in
|
|
Text(at.description)
|
|
}
|
|
}
|
|
.pickerStyle(DefaultPickerStyle())
|
|
Text("The fastest that position updates will be sent if the minimum distance has been satisfied")
|
|
.foregroundColor(.gray)
|
|
.font(.callout)
|
|
}
|
|
VStack(alignment: .leading) {
|
|
Picker("Minimum Distance", selection: $broadcastSmartMinimumDistance) {
|
|
ForEach(10..<151) {
|
|
if $0 == 0 {
|
|
Text("Unset")
|
|
} else {
|
|
if $0.isMultiple(of: 5) {
|
|
Text("\($0)")
|
|
.tag($0)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
.pickerStyle(DefaultPickerStyle())
|
|
Text("The minimum distance change in meters to be considered for a smart position broadcast.")
|
|
.foregroundColor(.gray)
|
|
.font(.callout)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
@ViewBuilder
|
|
var deviceGPSSection: some View {
|
|
Section(header: Text("Device GPS")) {
|
|
Picker("", selection: $gpsMode) {
|
|
ForEach(GpsMode.allCases, id: \.self) { at in
|
|
Text(at.description)
|
|
.tag(at.id)
|
|
}
|
|
}
|
|
.pickerStyle(SegmentedPickerStyle())
|
|
.padding(.top, 5)
|
|
.padding(.bottom, 5)
|
|
.disabled(fixedPosition && !(gpsMode == 1))
|
|
if gpsMode == 1 {
|
|
Text("Positions will be provided by your device GPS, if you select disabled or not present you can set a fixed position.")
|
|
.foregroundColor(.gray)
|
|
.font(.callout)
|
|
VStack(alignment: .leading) {
|
|
Picker("Update Interval", selection: $gpsUpdateInterval) {
|
|
ForEach(GpsUpdateIntervals.allCases) { ui in
|
|
Text(ui.description)
|
|
}
|
|
}
|
|
Text("How often should we try to get a GPS position.")
|
|
.foregroundColor(.gray)
|
|
.font(.callout)
|
|
}
|
|
}
|
|
if (gpsMode != 1 && node?.num ?? 0 == accessoryManager.activeDeviceNum ?? -1) || fixedPosition {
|
|
VStack(alignment: .leading) {
|
|
Toggle(isOn: $fixedPosition) {
|
|
Label("Fixed Position", systemImage: "location.square.fill")
|
|
if !(node?.positionConfig?.fixedPosition ?? false) {
|
|
Text("Your current location will be set as the fixed position and broadcast over the mesh on the position interval.")
|
|
} else {
|
|
|
|
}
|
|
}
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
@ViewBuilder
|
|
var positionFlagsSection: some View {
|
|
Section(header: Text("Position Flags")) {
|
|
|
|
Text("Optional fields to include when assembling position messages. the more fields are included, the larger the message will be - leading to longer airtime and a higher risk of packet loss")
|
|
.foregroundColor(.gray)
|
|
.font(.callout)
|
|
|
|
Toggle(isOn: $includeAltitude) {
|
|
Label("Altitude", systemImage: "arrow.up")
|
|
}
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
.onChange(of: includeAltitude) { _, newIncludeAltitude in
|
|
if newIncludeAltitude != PositionFlags(rawValue: self.positionFlags).contains(.Altitude) { hasChanges = true }
|
|
}
|
|
|
|
Toggle(isOn: $includeSatsinview) {
|
|
Label("Number of satellites", systemImage: "skew")
|
|
}
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
.onChange(of: includeSatsinview) { _, newIncludeSatsinview in
|
|
if newIncludeSatsinview != PositionFlags(rawValue: self.positionFlags).contains(.Satsinview) { hasChanges = true }
|
|
}
|
|
|
|
Toggle(isOn: $includeSeqNo) { // 64
|
|
Label("Sequence number", systemImage: "number")
|
|
}
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
.onChange(of: includeSeqNo) { _, newIncludeSeqNo in
|
|
if newIncludeSeqNo != PositionFlags(rawValue: self.positionFlags).contains(.SeqNo) { hasChanges = true }
|
|
}
|
|
|
|
Toggle(isOn: $includeTimestamp) { // 128
|
|
Label("Timestamp", systemImage: "clock")
|
|
}
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
.onChange(of: includeTimestamp) { _, newIncludeTimestamp in
|
|
if newIncludeTimestamp != PositionFlags(rawValue: self.positionFlags).contains(.Timestamp) { hasChanges = true }
|
|
}
|
|
|
|
Toggle(isOn: $includeHeading) { // 128
|
|
Label("Vehicle heading", systemImage: "location.circle")
|
|
}
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
.onChange(of: includeHeading) { _, newIncludeHeading in
|
|
if newIncludeHeading != PositionFlags(rawValue: self.positionFlags).contains(.Heading) { hasChanges = true }
|
|
}
|
|
|
|
Toggle(isOn: $includeSpeed) { // 128
|
|
Label("Vehicle speed", systemImage: "speedometer")
|
|
}
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
.onChange(of: includeSpeed) { _, newIncludeSpeed in
|
|
if newIncludeSpeed != PositionFlags(rawValue: self.positionFlags).contains(.Speed) { hasChanges = true }
|
|
}
|
|
}
|
|
}
|
|
|
|
@ViewBuilder
|
|
var advancedPositionFlagsSection: some View {
|
|
Section(header: Text("Advanced Position Flags")) {
|
|
|
|
if includeAltitude {
|
|
Toggle(isOn: $includeAltitudeMsl) {
|
|
Label("Altitude is Mean Sea Level", systemImage: "arrow.up.to.line.compact")
|
|
}
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
.onChange(of: includeAltitudeMsl) { _, newIncludeAltitudeMsl in
|
|
if newIncludeAltitudeMsl != PositionFlags(rawValue: self.positionFlags).contains(.AltitudeMsl) { hasChanges = true }
|
|
}
|
|
|
|
Toggle(isOn: $includeGeoidalSeparation) {
|
|
Label("Altitude Geoidal Separation", systemImage: "globe.americas")
|
|
}
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
.onChange(of: includeGeoidalSeparation) { _, newIncludeGeoidalSeparation in
|
|
if newIncludeGeoidalSeparation != PositionFlags(rawValue: self.positionFlags).contains(.GeoidalSeparation) { hasChanges = true }
|
|
}
|
|
}
|
|
|
|
Toggle(isOn: $includeDop) {
|
|
Text("Dilution of precision (DOP) PDOP used by default")
|
|
}
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
.onChange(of: includeDop) { _, newIncludeDop in
|
|
if newIncludeDop != PositionFlags(rawValue: self.positionFlags).contains(.Dop) { hasChanges = true }
|
|
}
|
|
|
|
if includeDop {
|
|
Toggle(isOn: $includeHvdop) {
|
|
Text("If DOP is set, use HDOP / VDOP values instead of PDOP")
|
|
}
|
|
.toggleStyle(SwitchToggleStyle(tint: .accentColor))
|
|
.onChange(of: includeHvdop) { _, newIncludeHvdop in
|
|
if newIncludeHvdop != PositionFlags(rawValue: self.positionFlags).contains(.Hvdop) { hasChanges = true }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
@ViewBuilder
|
|
var advancedDeviceGPSSection: some View {
|
|
Section(header: Text("Advanced Device GPS")) {
|
|
Picker("GPS Receive GPIO", selection: $rxGpio) {
|
|
ForEach(0..<49) {
|
|
if $0 == 0 {
|
|
Text("Unset")
|
|
} else {
|
|
Text("Pin \($0)")
|
|
}
|
|
}
|
|
}
|
|
.pickerStyle(DefaultPickerStyle())
|
|
Picker("GPS Transmit GPIO", selection: $txGpio) {
|
|
ForEach(0..<49) {
|
|
if $0 == 0 {
|
|
Text("Unset")
|
|
} else {
|
|
Text("Pin \($0)")
|
|
}
|
|
}
|
|
}
|
|
.pickerStyle(DefaultPickerStyle())
|
|
Picker("GPS EN GPIO", selection: $gpsEnGpio) {
|
|
ForEach(0..<49) {
|
|
if $0 == 0 {
|
|
Text("Unset")
|
|
} else {
|
|
Text("Pin \($0)")
|
|
}
|
|
}
|
|
}
|
|
.pickerStyle(DefaultPickerStyle())
|
|
Text("(Re)define PIN_GPS_EN for your board.")
|
|
.font(.caption)
|
|
}
|
|
}
|
|
var saveButton: some View {
|
|
SaveConfigButton(node: node, hasChanges: $hasChanges) {
|
|
if fixedPosition && !supportedVersion {
|
|
Task {
|
|
try await accessoryManager.sendPosition(channel: 0, destNum: node?.num ?? 0, wantResponse: true)
|
|
}
|
|
}
|
|
if let deviceNum = accessoryManager.activeDeviceNum, let connectedNode = getNodeInfo(id: deviceNum, context: context) {
|
|
var pc = Config.PositionConfig()
|
|
pc.positionBroadcastSmartEnabled = smartPositionEnabled
|
|
pc.gpsEnabled = gpsMode == 1
|
|
pc.gpsMode = Config.PositionConfig.GpsMode(rawValue: gpsMode) ?? Config.PositionConfig.GpsMode.notPresent
|
|
pc.fixedPosition = fixedPosition
|
|
pc.gpsUpdateInterval = UInt32(gpsUpdateInterval)
|
|
pc.positionBroadcastSecs = UInt32(positionBroadcastSeconds)
|
|
pc.broadcastSmartMinimumIntervalSecs = UInt32(broadcastSmartMinimumIntervalSecs)
|
|
pc.broadcastSmartMinimumDistance = UInt32(broadcastSmartMinimumDistance)
|
|
pc.rxGpio = UInt32(rxGpio)
|
|
pc.txGpio = UInt32(txGpio)
|
|
pc.gpsEnGpio = UInt32(gpsEnGpio)
|
|
var pf: PositionFlags = []
|
|
if includeAltitude { pf.insert(.Altitude) }
|
|
if includeAltitudeMsl { pf.insert(.AltitudeMsl) }
|
|
if includeGeoidalSeparation { pf.insert(.GeoidalSeparation) }
|
|
if includeDop { pf.insert(.Dop) }
|
|
if includeHvdop { pf.insert(.Hvdop) }
|
|
if includeSatsinview { pf.insert(.Satsinview) }
|
|
if includeSeqNo { pf.insert(.SeqNo) }
|
|
if includeTimestamp { pf.insert(.Timestamp) }
|
|
if includeSpeed { pf.insert(.Speed) }
|
|
if includeHeading { pf.insert(.Heading) }
|
|
pc.positionFlags = UInt32(pf.rawValue)
|
|
Task {
|
|
_ = try await accessoryManager.savePositionConfig(config: pc, fromUser: connectedNode.user!, toUser: node!.user!)
|
|
Task { @MainActor in
|
|
// Disable the button after a successful save
|
|
hasChanges = false
|
|
goBack()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
var setFixedAlertTitle: String {
|
|
if node?.positionConfig?.fixedPosition == true {
|
|
return "Remove Fixed Position"
|
|
} else {
|
|
return "Set Fixed Position"
|
|
}
|
|
}
|
|
|
|
var body: some View {
|
|
VStack {
|
|
Form {
|
|
ConfigHeader(title: "Position", config: \.positionConfig, node: node, onAppear: setPositionValues)
|
|
positionPacketSection
|
|
deviceGPSSection
|
|
positionFlagsSection
|
|
advancedPositionFlagsSection
|
|
if gpsMode == 1 {
|
|
advancedDeviceGPSSection
|
|
}
|
|
}
|
|
.disabled(!accessoryManager.isConnected || node?.positionConfig == nil)
|
|
.alert(setFixedAlertTitle, isPresented: $showingSetFixedAlert) {
|
|
Button("Cancel", role: .cancel) {
|
|
fixedPosition = !fixedPosition
|
|
}
|
|
if node?.positionConfig?.fixedPosition ?? false {
|
|
Button("Remove", role: .destructive) {
|
|
removeFixedPosition()
|
|
}
|
|
} else {
|
|
Button("Set") {
|
|
setFixedPosition()
|
|
}
|
|
}
|
|
} message: {
|
|
Text(node?.positionConfig?.fixedPosition ?? false ? "This will disable fixed position and remove the currently set position." : "This will send a current position from your phone and enable fixed position.")
|
|
}
|
|
saveButton
|
|
}
|
|
.navigationTitle("Position Config")
|
|
.navigationBarItems(
|
|
trailing: ZStack {
|
|
ConnectedDevice(deviceConnected: accessoryManager.isConnected, name: accessoryManager.activeConnection?.device.shortName ?? "?")
|
|
}
|
|
)
|
|
.onFirstAppear {
|
|
supportedVersion = accessoryManager.checkIsVersionSupported(forVersion: minimumVersion)
|
|
// Need to request a NetworkConfig from the remote node before allowing changes
|
|
if let deviceNum = accessoryManager.activeDeviceNum, let node {
|
|
let connectedNode = getNodeInfo(id: deviceNum, context: context)
|
|
if let connectedNode {
|
|
if node.num != deviceNum {
|
|
if UserDefaults.enableAdministration {
|
|
/// 2.5 Administration with session passkey
|
|
let expiration = node.sessionExpiration ?? Date()
|
|
if expiration < Date() || node.positionConfig == nil {
|
|
Task {
|
|
do {
|
|
Logger.mesh.info("⚙️ Empty or expired position config requesting via PKI admin")
|
|
try await accessoryManager.requestPositionConfig(fromUser: connectedNode.user!, toUser: node.user!)
|
|
} catch {
|
|
Logger.mesh.info("🚨 Position config request failed")
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
/// Legacy Administration
|
|
Logger.mesh.info("☠️ Using insecure legacy admin that is no longer supported, please upgrade your firmware.")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
.onChange(of: fixedPosition) { _, newFixed in
|
|
if supportedVersion {
|
|
if let positionConfig = node?.positionConfig {
|
|
/// Fixed Position is off to start
|
|
if !positionConfig.fixedPosition && newFixed {
|
|
showingSetFixedAlert = true
|
|
} else if positionConfig.fixedPosition && !newFixed {
|
|
/// Fixed Position is on to start
|
|
showingSetFixedAlert = true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
.onChange(of: gpsMode) { _, newGpsMode in
|
|
if newGpsMode != node?.positionConfig?.gpsMode ?? 0 { hasChanges = true }
|
|
}
|
|
.onChange(of: rxGpio) { _, newRxGpio in
|
|
if newRxGpio != node?.positionConfig?.rxGpio ?? 0 { hasChanges = true }
|
|
}
|
|
.onChange(of: txGpio) { _, newTxGpio in
|
|
if newTxGpio != node?.positionConfig?.txGpio ?? 0 { hasChanges = true }
|
|
}
|
|
.onChange(of: gpsEnGpio) { _, newGpsEnGpio in
|
|
if newGpsEnGpio != node?.positionConfig?.gpsEnGpio ?? 0 { hasChanges = true }
|
|
}
|
|
.onChange(of: smartPositionEnabled) { _, newSmartPositionEnabled in
|
|
if newSmartPositionEnabled != node?.positionConfig?.smartPositionEnabled { hasChanges = true }
|
|
}
|
|
.onChange(of: positionBroadcastSeconds) { _, newPositionBroadcastSeconds in
|
|
if newPositionBroadcastSeconds != node?.positionConfig?.positionBroadcastSeconds ?? 0 { hasChanges = true }
|
|
}
|
|
.onChange(of: broadcastSmartMinimumIntervalSecs) { _, newBroadcastSmartMinimumIntervalSecs in
|
|
if newBroadcastSmartMinimumIntervalSecs != node?.positionConfig?.broadcastSmartMinimumIntervalSecs ?? 0 { hasChanges = true }
|
|
}
|
|
.onChange(of: broadcastSmartMinimumDistance) { _, newBroadcastSmartMinimumDistance in
|
|
if newBroadcastSmartMinimumDistance != node?.positionConfig?.broadcastSmartMinimumDistance ?? 0 { hasChanges = true }
|
|
}
|
|
.onChange(of: gpsUpdateInterval) { _, newGpsUpdateInterval in
|
|
if newGpsUpdateInterval != node?.positionConfig?.gpsUpdateInterval ?? 0 { hasChanges = true }
|
|
}
|
|
}
|
|
|
|
func handlePositionFlagtChanges() {
|
|
guard (node?.positionConfig) != nil else { return }
|
|
let pf = PositionFlags(rawValue: self.positionFlags)
|
|
hasChanges =
|
|
pf.contains(.Altitude) ||
|
|
pf.contains(.AltitudeMsl) ||
|
|
pf.contains(.Satsinview) ||
|
|
pf.contains(.SeqNo) ||
|
|
pf.contains(.Timestamp) ||
|
|
pf.contains(.Speed) ||
|
|
pf.contains(.Heading) ||
|
|
pf.contains(.GeoidalSeparation) ||
|
|
pf.contains(.Dop) ||
|
|
pf.contains(.Hvdop)
|
|
}
|
|
|
|
func setPositionValues() {
|
|
self.smartPositionEnabled = node?.positionConfig?.smartPositionEnabled ?? true
|
|
self.deviceGpsEnabled = node?.positionConfig?.deviceGpsEnabled ?? false
|
|
self.gpsMode = Int(node?.positionConfig?.gpsMode ?? 0)
|
|
if node?.positionConfig?.deviceGpsEnabled ?? false && gpsMode != 1 {
|
|
self.gpsMode = 1
|
|
}
|
|
self.rxGpio = Int(node?.positionConfig?.rxGpio ?? 0)
|
|
self.txGpio = Int(node?.positionConfig?.txGpio ?? 0)
|
|
self.gpsEnGpio = Int(node?.positionConfig?.gpsEnGpio ?? 0)
|
|
self.fixedPosition = node?.positionConfig?.fixedPosition ?? false
|
|
self.gpsUpdateInterval = Int(node?.positionConfig?.gpsUpdateInterval ?? 30)
|
|
self.positionBroadcastSeconds = Int(node?.positionConfig?.positionBroadcastSeconds ?? 900)
|
|
self.broadcastSmartMinimumIntervalSecs = Int(node?.positionConfig?.broadcastSmartMinimumIntervalSecs ?? 30)
|
|
self.broadcastSmartMinimumDistance = Int(node?.positionConfig?.broadcastSmartMinimumDistance ?? 50)
|
|
self.positionFlags = Int(node?.positionConfig?.positionFlags ?? 3)
|
|
let pf = PositionFlags(rawValue: self.positionFlags)
|
|
self.includeAltitude = pf.contains(.Altitude)
|
|
self.includeAltitudeMsl = pf.contains(.AltitudeMsl)
|
|
self.includeGeoidalSeparation = pf.contains(.GeoidalSeparation)
|
|
self.includeDop = pf.contains(.Dop)
|
|
self.includeHvdop = pf.contains(.Hvdop)
|
|
self.includeSatsinview = pf.contains(.Satsinview)
|
|
self.includeSeqNo = pf.contains(.SeqNo)
|
|
self.includeTimestamp = pf.contains(.Timestamp)
|
|
self.includeSpeed = pf.contains(.Speed)
|
|
self.includeHeading = pf.contains(.Heading)
|
|
self.hasChanges = false
|
|
}
|
|
|
|
private func setFixedPosition() {
|
|
guard let nodeNum = accessoryManager.activeDeviceNum,
|
|
nodeNum > 0 else { return }
|
|
Task {
|
|
do {
|
|
try await accessoryManager.setFixedPosition(fromUser: node!.user!, channel: 0)
|
|
} catch {
|
|
Logger.mesh.error("Set Position Failed")
|
|
}
|
|
}
|
|
node?.positionConfig?.fixedPosition = true
|
|
do {
|
|
try context.save()
|
|
Logger.data.info("💾 Updated Position Config with Fixed Position = true")
|
|
} catch {
|
|
context.rollback()
|
|
let nsError = error as NSError
|
|
Logger.data.error("Error Saving Position Config Entity \(nsError, privacy: .public)")
|
|
}
|
|
}
|
|
|
|
private func removeFixedPosition() {
|
|
guard let nodeNum = accessoryManager.activeDeviceNum,
|
|
nodeNum > 0 else { return }
|
|
Task {
|
|
do {
|
|
try await accessoryManager.removeFixedPosition(fromUser: node!.user!, channel: 0)
|
|
} catch {
|
|
Logger.mesh.error("Remove Fixed Position Failed")
|
|
}
|
|
}
|
|
let mutablePositions = node?.positions?.mutableCopy() as? NSMutableOrderedSet
|
|
mutablePositions?.removeAllObjects()
|
|
node?.positions = mutablePositions
|
|
node?.positionConfig?.fixedPosition = false
|
|
do {
|
|
try context.save()
|
|
Logger.data.info("💾 Updated Position Config with Fixed Position = false")
|
|
} catch {
|
|
context.rollback()
|
|
let nsError = error as NSError
|
|
Logger.data.error("Error Saving Position Config Entity \(nsError, privacy: .public)")
|
|
}
|
|
}
|
|
}
|