2023-08-16 18:28:55 -07:00
//
// D e t e c t i o n S e n s o r M o d u l e . s w i f t
// M e s h t a s t i c
//
// C o p y r i g h t ( c ) G a r t h V a n d e r H o u w e n 8 / 1 6 / 2 3 .
//
import SwiftUI
2023-12-01 13:56:29 -08:00
enum DetectionSensorRole : String , CaseIterable , Equatable {
case sensor
case client
var description : String {
switch self {
case . sensor :
return " Sensor "
case . client :
return " Client "
}
}
var localized : String { self . rawValue . localized }
}
2023-08-16 18:28:55 -07:00
struct DetectionSensorConfig : View {
@ Environment ( \ . managedObjectContext ) var context
@ EnvironmentObject var bleManager : BLEManager
@ Environment ( \ . dismiss ) private var goBack
var node : NodeInfoEntity ?
@ State private var isPresentingSaveConfirm : Bool = false
@ State var hasChanges : Bool = false
2023-12-01 13:56:29 -08:00
@ AppStorage ( " detectionSensorRole " ) private var role : DetectionSensorRole = . sensor
@ AppStorage ( " enableDetectionNotifications " ) private var detectionNotificationsEnabled = false
// / M o d u l e C o n f i g S e t t i n g s
2023-08-16 18:28:55 -07:00
@ State var enabled = false
2023-08-17 16:30:48 -05:00
@ State var sendBell : Bool = false
@ State var name : String = " "
@ State var detectionTriggeredHigh : Bool = true
@ State var usePullup : Bool = false
2023-08-17 17:00:20 -07:00
@ State var minimumBroadcastSecs = 0
@ State var stateBroadcastSecs = 0
@ State var monitorPin = 0
2023-08-16 18:28:55 -07:00
var body : some View {
2023-12-06 12:32:17 -08:00
VStack {
Form {
2024-02-17 22:39:22 -07:00
ConfigHeader ( title : " Detection Sensor " , config : \ . detectionSensorConfig , node : node , onAppear : setDetectionSensorValues )
2023-12-06 12:32:17 -08:00
Section ( header : Text ( " options " ) ) {
Toggle ( isOn : $ enabled ) {
Label ( " enabled " , systemImage : " dot.radiowaves.right " )
Text ( " Enables the detection sensor module, it needs to be enabled on both the node with the sensor, and any nodes that you want to receive detection sensor text messages or view the detection sensor log and chart. " )
2023-12-01 13:56:29 -08:00
. font ( . caption )
2023-12-06 12:32:17 -08:00
}
. toggleStyle ( SwitchToggleStyle ( tint : . accentColor ) )
. listRowSeparator ( . visible )
if enabled {
HStack {
Picker ( selection : $ role , label : Text ( " Role " ) ) {
ForEach ( DetectionSensorRole . allCases , id : \ . self ) { r in
Text ( r . description )
. tag ( r )
}
2023-08-17 16:30:48 -05:00
}
2023-12-06 12:32:17 -08:00
. pickerStyle ( SegmentedPickerStyle ( ) )
. padding ( . top , 5 )
. padding ( . bottom , 5 )
2023-08-17 16:30:48 -05:00
}
}
}
2023-12-06 12:32:17 -08:00
if enabled && role = = . client {
Section ( header : Text ( " Client options " ) ) {
Toggle ( isOn : $ detectionNotificationsEnabled ) {
Label ( " Enable Notifications " , systemImage : " bell.badge " )
Text ( " Detection sensor messages are received as text messages. If you enable notifications you will recieve a notification for each detection message received and a corresponding unread message badge. " )
. font ( . caption )
}
. listRowSeparator ( . visible )
2023-12-01 13:56:29 -08:00
}
2023-08-17 16:30:48 -05:00
}
2023-12-06 12:32:17 -08:00
if enabled && role = = . sensor {
Section ( header : Text ( " Sensor options " ) ) {
Toggle ( isOn : $ sendBell ) {
Label ( " Send Bell " , systemImage : " bell " )
Text ( " Send ASCII bell with alert message. Useful for triggering external notification on bell. " )
. font ( . caption )
}
. toggleStyle ( SwitchToggleStyle ( tint : . accentColor ) )
. listRowSeparator ( . visible )
HStack {
Label ( " Name " , systemImage : " signature " )
TextField ( " Friendly name " , text : $ name , axis : . vertical )
. foregroundColor ( . gray )
. autocapitalization ( . none )
. disableAutocorrection ( true )
. onChange ( of : name , perform : { _ in
2023-12-01 13:56:29 -08:00
2023-12-06 12:32:17 -08:00
let totalBytes = name . utf8 . count
// O n l y m e s s w i t h t h e v a l u e i f i t i s t o o b i g
if totalBytes > 20 {
let firstNBytes = Data ( name . utf8 . prefix ( 20 ) )
if let maxBytesString = String ( data : firstNBytes , encoding : String . Encoding . utf8 ) {
// S e t t h e s h o r t N a m e b a c k t o t h e l a s t p l a c e w h e r e i t w a s t h e r i g h t s i z e
name = maxBytesString
}
2023-12-01 13:56:29 -08:00
}
2023-12-06 12:32:17 -08:00
} )
. foregroundColor ( . gray )
}
. listRowSeparator ( . hidden )
Text ( " Friendly name used to format message sent to mesh. Example: A name \" Motion \" would result in a message \" Motion detected \" " )
. font ( . caption )
. foregroundStyle ( . gray )
. listRowSeparator ( . visible )
. offset ( y : - 10 )
Picker ( " GPIO Pin to monitor " , selection : $ monitorPin ) {
2024-01-22 13:21:17 -08:00
ForEach ( 0. . < 49 ) {
2023-12-06 12:32:17 -08:00
if $0 = = 0 {
Text ( " unset " )
} else {
Text ( " Pin \( $0 ) " )
2023-12-01 13:56:29 -08:00
}
}
}
2023-12-06 12:32:17 -08:00
. pickerStyle ( DefaultPickerStyle ( ) )
Toggle ( isOn : $ detectionTriggeredHigh ) {
Label ( " Detection trigger High " , systemImage : " dial.high " )
Text ( " Whether or not the GPIO pin state detection is triggered on HIGH (1) or LOW (0) " )
. font ( . caption )
}
. toggleStyle ( SwitchToggleStyle ( tint : . accentColor ) )
Toggle ( isOn : $ usePullup ) {
Label ( " Uses pullup resistor " , systemImage : " arrow.up.to.line " )
Text ( " Whether or not use INPUT_PULLUP mode for GPIO pin. Only applicable if the board uses pull-up resistors on the pin " )
. font ( . caption )
2023-12-01 13:56:29 -08:00
}
2023-12-06 12:32:17 -08:00
. toggleStyle ( SwitchToggleStyle ( tint : . accentColor ) )
2023-08-17 16:30:48 -05:00
}
2023-12-06 12:32:17 -08:00
Section ( header : Text ( " update.interval " ) ) {
Picker ( " Minimum time between detection broadcasts " , selection : $ minimumBroadcastSecs ) {
ForEach ( UpdateIntervals . allCases ) { ui in
Text ( ui . description ) . tag ( ui . rawValue )
}
}
. pickerStyle ( DefaultPickerStyle ( ) )
. listRowSeparator ( . hidden )
Text ( " Mininum time between detection broadcasts. Default is 45 seconds. " )
. font ( . caption )
. foregroundStyle ( . gray )
. listRowSeparator ( . visible )
Picker ( " State Broadcast Interval " , selection : $ stateBroadcastSecs ) {
Text ( " Never " ) . tag ( 0 )
ForEach ( UpdateIntervals . allCases ) { ui in
Text ( ui . description ) . tag ( ui . rawValue )
}
2023-12-01 13:56:29 -08:00
}
2023-12-06 12:32:17 -08:00
. pickerStyle ( DefaultPickerStyle ( ) )
. listRowSeparator ( . hidden )
Text ( " How often to send detection sensor state to mesh regardless of detection. Default is Never. " )
. font ( . caption )
. foregroundStyle ( . gray )
2023-12-01 13:56:29 -08:00
}
2023-08-17 16:30:48 -05:00
}
2023-08-16 18:28:55 -07:00
}
}
. scrollDismissesKeyboard ( . interactively )
2023-08-17 16:30:48 -05:00
. disabled ( self . bleManager . connectedPeripheral = = nil || node ? . detectionSensorConfig = = nil )
2023-08-16 18:28:55 -07:00
2024-02-18 00:03:34 -07:00
SaveConfigButton ( node : node , hasChanges : $ hasChanges ) {
2023-08-16 18:28:55 -07:00
let connectedNode = getNodeInfo ( id : bleManager . connectedPeripheral ? . num ? ? - 1 , context : context )
if connectedNode != nil {
2024-02-18 00:03:34 -07:00
var dsc = ModuleConfig . DetectionSensorConfig ( )
dsc . enabled = self . enabled
dsc . sendBell = self . sendBell
dsc . name = self . name
dsc . monitorPin = UInt32 ( self . monitorPin )
dsc . detectionTriggeredHigh = self . detectionTriggeredHigh
dsc . usePullup = self . usePullup
dsc . minimumBroadcastSecs = UInt32 ( self . minimumBroadcastSecs )
dsc . stateBroadcastSecs = UInt32 ( self . stateBroadcastSecs )
let adminMessageId = bleManager . saveDetectionSensorModuleConfig ( config : dsc , fromUser : connectedNode ! . user ! , toUser : node ! . user ! , adminIndex : connectedNode ? . myInfo ? . adminIndex ? ? 0 )
if adminMessageId > 0 {
// S h o u l d s h o w a s a v e d s u c c e s s f u l l y a l e r t o n c e I k n o w t h a t t o b e t r u e
// f o r n o w j u s t d i s a b l e t h e b u t t o n a f t e r a s u c c e s s f u l s a v e
hasChanges = false
goBack ( )
2023-08-26 23:17:30 -07:00
}
2023-08-16 18:28:55 -07:00
}
}
2023-08-17 16:30:48 -05:00
. navigationTitle ( " detection.sensor.config " )
2023-08-16 18:28:55 -07:00
. navigationBarItems ( trailing :
ZStack {
2023-09-02 17:37:35 -07:00
ConnectedDevice ( bluetoothOn : bleManager . isSwitchedOn , deviceConnected : bleManager . connectedPeripheral != nil , name : ( bleManager . connectedPeripheral != nil ) ? bleManager . connectedPeripheral . shortName : " ? " )
2023-08-16 18:28:55 -07:00
} )
. onAppear {
2024-01-14 11:25:00 -08:00
if self . bleManager . context = = nil {
self . bleManager . context = context
}
2023-08-16 18:28:55 -07:00
setDetectionSensorValues ( )
2023-08-17 16:30:48 -05:00
// N e e d t o r e q u e s t a D e t e c t i o n S e n s o r M o d u l e C o n f i g f r o m t h e r e m o t e n o d e b e f o r e a l l o w i n g c h a n g e s
if bleManager . connectedPeripheral != nil && node ? . detectionSensorConfig = = nil {
print ( " empty detection sensor module config " )
2023-08-16 18:28:55 -07:00
let connectedNode = getNodeInfo ( id : bleManager . connectedPeripheral . num , context : context )
if node != nil && connectedNode != nil {
2023-08-17 16:30:48 -05:00
_ = bleManager . requestDetectionSensorModuleConfig ( fromUser : connectedNode ! . user ! , toUser : node ! . user ! , adminIndex : connectedNode ? . myInfo ? . adminIndex ? ? 0 )
2023-08-16 18:28:55 -07:00
}
}
}
. onChange ( of : enabled ) { newEnabled in
if node != nil && node ? . detectionSensorConfig != nil {
if newEnabled != node ! . detectionSensorConfig ! . enabled { hasChanges = true }
}
}
2023-08-17 16:30:48 -05:00
. onChange ( of : sendBell ) { newSendBell in
if node != nil && node ? . detectionSensorConfig != nil {
if newSendBell != node ! . detectionSensorConfig ! . sendBell { hasChanges = true }
}
}
. onChange ( of : detectionTriggeredHigh ) { newDetectionTriggeredHigh in
if node != nil && node ? . detectionSensorConfig != nil {
if newDetectionTriggeredHigh != node ! . detectionSensorConfig ! . detectionTriggeredHigh { hasChanges = true }
}
}
. onChange ( of : usePullup ) { newUsePullup in
if node != nil && node ? . detectionSensorConfig != nil {
if newUsePullup != node ! . detectionSensorConfig ! . usePullup { hasChanges = true }
}
}
. onChange ( of : name ) { newName in
if node != nil && node ? . detectionSensorConfig != nil {
if newName != node ! . detectionSensorConfig ! . name { hasChanges = true }
}
}
. onChange ( of : monitorPin ) { newMonitorPin in
if node != nil && node ? . detectionSensorConfig != nil {
if newMonitorPin != node ! . detectionSensorConfig ! . monitorPin { hasChanges = true }
}
}
. onChange ( of : minimumBroadcastSecs ) { newMinimumBroadcastSecs in
if node != nil && node ? . detectionSensorConfig != nil {
if newMinimumBroadcastSecs != node ! . detectionSensorConfig ! . minimumBroadcastSecs { hasChanges = true }
}
}
. onChange ( of : stateBroadcastSecs ) { newStateBroadcastSecs in
if node != nil && node ? . detectionSensorConfig != nil {
if newStateBroadcastSecs != node ! . detectionSensorConfig ! . stateBroadcastSecs { hasChanges = true }
}
}
2023-12-01 13:56:29 -08:00
. onChange ( of : detectionNotificationsEnabled ) { newDetectionNotificationsEnabled in
UserDefaults . enableDetectionNotifications = newDetectionNotificationsEnabled
}
2023-08-16 18:28:55 -07:00
}
func setDetectionSensorValues ( ) {
self . enabled = ( node ? . detectionSensorConfig ? . enabled ? ? false )
2023-08-17 16:30:48 -05:00
self . sendBell = ( node ? . detectionSensorConfig ? . sendBell ? ? false )
self . name = ( node ? . detectionSensorConfig ? . name ? ? " " )
2023-08-17 17:00:20 -07:00
self . monitorPin = Int ( node ? . detectionSensorConfig ? . monitorPin ? ? 0 )
2023-08-17 16:30:48 -05:00
self . usePullup = ( node ? . detectionSensorConfig ? . usePullup ? ? false )
self . detectionTriggeredHigh = ( node ? . detectionSensorConfig ? . detectionTriggeredHigh ? ? true )
2023-08-17 17:00:20 -07:00
self . minimumBroadcastSecs = Int ( node ? . detectionSensorConfig ? . minimumBroadcastSecs ? ? 45 )
self . stateBroadcastSecs = Int ( node ? . detectionSensorConfig ? . stateBroadcastSecs ? ? 0 )
2023-08-17 16:30:48 -05:00
2023-08-16 18:28:55 -07:00
self . hasChanges = false
}
}