@@ -19,11 +19,18 @@ struct ProductInfo: Codable {
19
19
}
20
20
}
21
21
22
+ enum ConnectionState : String , Codable {
23
+ // we are scanning for peripherals by default
24
+ case scanning
25
+ // from the moment we try to connect until after we are paired (or until either step fails).
26
+ case connecting
27
+ case connected
28
+ }
29
+
22
30
struct State {
23
31
var bluetoothAvailable : Bool
24
32
var discoveredPeripherals : [ UUID : PeripheralMetadata ]
25
- // true from the moment we try to connect until after we are paired (or until either step fails).
26
- var connecting : Bool
33
+ var connectionState : ConnectionState
27
34
}
28
35
29
36
struct PeripheralMetadata {
@@ -43,14 +50,18 @@ var pairedDeviceIdentifiers: Set<String> {
43
50
}
44
51
45
52
class BluetoothManager : NSObject , ObservableObject , CBCentralManagerDelegate , CBPeripheralDelegate {
46
- private var state : State = State ( bluetoothAvailable: false , discoveredPeripherals: [ : ] , connecting: false )
53
+ private var state : State = State (
54
+ bluetoothAvailable: false ,
55
+ discoveredPeripherals: [ : ] ,
56
+ connectionState: . scanning // Initial state
57
+ )
47
58
48
59
var centralManager : CBCentralManager !
49
60
var connectedPeripheral : CBPeripheral ?
50
61
var pWriter : CBCharacteristic ?
51
62
var pReader : CBCharacteristic ?
52
63
var pProduct : CBCharacteristic ?
53
-
64
+
54
65
private var isPaired : Bool = false
55
66
56
67
// Peripherals in this set will not be auto-connected even if previously paired.
@@ -76,7 +87,7 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
76
87
guard let metadata = state. discoveredPeripherals [ peripheralID] else { return }
77
88
centralManager. stopScan ( )
78
89
state. discoveredPeripherals [ peripheralID] ? . connectionError = nil
79
- state. connecting = true
90
+ state. connectionState = . connecting
80
91
updateBackendState ( )
81
92
centralManager. connect ( metadata. peripheral, options: nil )
82
93
}
@@ -86,6 +97,7 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
86
97
!centralManager. isScanning,
87
98
connectedPeripheral == nil else { return }
88
99
state. discoveredPeripherals. removeAll ( )
100
+ state. connectionState = . scanning
89
101
updateBackendState ( )
90
102
centralManager. scanForPeripherals (
91
103
withServices: [ CBUUID ( string: " e1511a45-f3db-44c0-82b8-6c880790d1f1 " ) ] ,
@@ -150,10 +162,10 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
150
162
func centralManager( _ central: CBCentralManager , didFailToConnect peripheral: CBPeripheral , error: Error ? ) {
151
163
let errorMessage = error? . localizedDescription ?? " unknown error "
152
164
state. discoveredPeripherals [ peripheral. identifier] ? . connectionError = errorMessage
153
- state. connecting = false
154
- dontAutoConnectSet. insert ( peripheral. identifier)
155
165
updateBackendState ( )
166
+ dontAutoConnectSet. insert ( peripheral. identifier)
156
167
print ( " BLE: connection failed to \( peripheral. name ?? " unknown device " ) : \( errorMessage) " )
168
+ restartScan ( )
157
169
}
158
170
159
171
func peripheral( _ peripheral: CBPeripheral , didDiscoverServices error: Error ? ) {
@@ -240,29 +252,29 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
240
252
// Add to paired devices
241
253
pairedDeviceIdentifiers. insert ( peripheral. identifier. uuidString)
242
254
}
243
- state. connecting = false
255
+ state. connectionState = . connected
244
256
updateBackendState ( )
245
257
// Invoke device manager to scan now, which will make it detect the device being connected
246
258
// (or disconnected, in case the product string indicates that) now instead of waiting for
247
259
// the next scan.
248
260
MobileserverUsbUpdate ( )
249
261
}
250
262
}
251
-
263
+
252
264
func handleDisconnect( ) {
253
265
connectedPeripheral = nil
254
266
pReader = nil
255
267
pWriter = nil
256
268
pProduct = nil
257
269
state. discoveredPeripherals. removeAll ( )
258
- state. connecting = false
270
+ state. connectionState = . scanning
259
271
isPaired = false
260
272
updateBackendState ( )
261
-
273
+
262
274
// Have the backend scan right away, which will make it detect that we disconnected.
263
275
// Otherwise there would be up to a second of delay (the backend device manager scan interval).
264
276
MobileserverUsbUpdate ( )
265
-
277
+
266
278
restartScan ( )
267
279
}
268
280
@@ -299,7 +311,7 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
299
311
let value = pProduct. value else {
300
312
return nil
301
313
}
302
-
314
+
303
315
if value. isEmpty {
304
316
return nil
305
317
}
@@ -326,7 +338,7 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
326
338
struct StateJSON : Codable {
327
339
let bluetoothAvailable : Bool
328
340
let peripherals : [ PeripheralJSON ]
329
- let connecting : Bool
341
+ let connectionState : ConnectionState
330
342
}
331
343
332
344
// Convert discoveredPeripherals to the JSON structure
@@ -338,7 +350,11 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
338
350
)
339
351
}
340
352
341
- let state = StateJSON ( bluetoothAvailable: state. bluetoothAvailable, peripherals: peripherals, connecting: state. connecting)
353
+ let state = StateJSON (
354
+ bluetoothAvailable: state. bluetoothAvailable,
355
+ peripherals: peripherals,
356
+ connectionState: state. connectionState
357
+ )
342
358
343
359
do {
344
360
let encoder = JSONEncoder ( )
0 commit comments