Skip to content

Commit 48f7279

Browse files
committed
ios: add a connectionState enum for the frontend
We are either scanning, connecting/pairing, or connected. This should help the frontend display everything appropriately.
1 parent 14fab83 commit 48f7279

File tree

4 files changed

+49
-21
lines changed

4 files changed

+49
-21
lines changed

backend/devices/bluetooth/bluetooth.go

+16-4
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,18 @@ import (
2121
"github.com/sirupsen/logrus"
2222
)
2323

24+
// ConnectionState represents the current Bluetooth connection state.
25+
type ConnectionState string
26+
27+
const (
28+
// ConnectionStateScanning means actively searching for devices.
29+
ConnectionStateScanning ConnectionState = "scanning"
30+
// ConnectionStateConnecting means a connection & pairing attempt is in progress.
31+
ConnectionStateConnecting ConnectionState = "connecting"
32+
// ConnectionStateConnected means successfully connected and paired.
33+
ConnectionStateConnected ConnectionState = "connected"
34+
)
35+
2436
// Peripheral is a bluetooth peripheral.
2537
type Peripheral struct {
2638
Identifier string `json:"identifier"`
@@ -33,9 +45,8 @@ type State struct {
3345
// BluetoothAvailable is false if bluetooth is powered off or otherwise unavailable.
3446
BluetoothAvailable bool `json:"bluetoothAvailable"`
3547
Peripherals []*Peripheral `json:"peripherals"`
36-
// Connecting is true from the moment we try to connect until after we are paired (or until
37-
// either step fails).
38-
Connecting bool `json:"connecting"`
48+
// ConnectionState shows the current bluetooth connection phase
49+
ConnectionState ConnectionState `json:"connectionState"`
3950
}
4051

4152
// Bluetooth manages a list of peripherals.
@@ -52,7 +63,8 @@ type Bluetooth struct {
5263
func New(log *logrus.Entry) *Bluetooth {
5364
b := &Bluetooth{
5465
state: &State{
55-
Peripherals: []*Peripheral{},
66+
Peripherals: []*Peripheral{},
67+
ConnectionState: ConnectionStateScanning,
5668
},
5769
log: log,
5870
}

frontends/ios/BitBoxApp/BitBoxApp/Bluetooth.swift

+31-15
Original file line numberDiff line numberDiff line change
@@ -19,11 +19,18 @@ struct ProductInfo: Codable {
1919
}
2020
}
2121

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+
2230
struct State {
2331
var bluetoothAvailable: Bool
2432
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
2734
}
2835

2936
struct PeripheralMetadata {
@@ -43,14 +50,18 @@ var pairedDeviceIdentifiers: Set<String> {
4350
}
4451

4552
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+
)
4758

4859
var centralManager: CBCentralManager!
4960
var connectedPeripheral: CBPeripheral?
5061
var pWriter: CBCharacteristic?
5162
var pReader: CBCharacteristic?
5263
var pProduct: CBCharacteristic?
53-
64+
5465
private var isPaired: Bool = false
5566

5667
// Peripherals in this set will not be auto-connected even if previously paired.
@@ -76,7 +87,7 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
7687
guard let metadata = state.discoveredPeripherals[peripheralID] else { return }
7788
centralManager.stopScan()
7889
state.discoveredPeripherals[peripheralID]?.connectionError = nil
79-
state.connecting = true
90+
state.connectionState = .connecting
8091
updateBackendState()
8192
centralManager.connect(metadata.peripheral, options: nil)
8293
}
@@ -86,6 +97,7 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
8697
!centralManager.isScanning,
8798
connectedPeripheral == nil else { return }
8899
state.discoveredPeripherals.removeAll()
100+
state.connectionState = .scanning
89101
updateBackendState()
90102
centralManager.scanForPeripherals(
91103
withServices: [CBUUID(string: "e1511a45-f3db-44c0-82b8-6c880790d1f1")],
@@ -150,10 +162,10 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
150162
func centralManager(_ central: CBCentralManager, didFailToConnect peripheral: CBPeripheral, error: Error?) {
151163
let errorMessage = error?.localizedDescription ?? "unknown error"
152164
state.discoveredPeripherals[peripheral.identifier]?.connectionError = errorMessage
153-
state.connecting = false
154-
dontAutoConnectSet.insert(peripheral.identifier)
155165
updateBackendState()
166+
dontAutoConnectSet.insert(peripheral.identifier)
156167
print("BLE: connection failed to \(peripheral.name ?? "unknown device"): \(errorMessage)")
168+
restartScan()
157169
}
158170

159171
func peripheral(_ peripheral: CBPeripheral, didDiscoverServices error: Error?) {
@@ -240,29 +252,29 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
240252
// Add to paired devices
241253
pairedDeviceIdentifiers.insert(peripheral.identifier.uuidString)
242254
}
243-
state.connecting = false
255+
state.connectionState = .connected
244256
updateBackendState()
245257
// Invoke device manager to scan now, which will make it detect the device being connected
246258
// (or disconnected, in case the product string indicates that) now instead of waiting for
247259
// the next scan.
248260
MobileserverUsbUpdate()
249261
}
250262
}
251-
263+
252264
func handleDisconnect() {
253265
connectedPeripheral = nil
254266
pReader = nil
255267
pWriter = nil
256268
pProduct = nil
257269
state.discoveredPeripherals.removeAll()
258-
state.connecting = false
270+
state.connectionState = .scanning
259271
isPaired = false
260272
updateBackendState()
261-
273+
262274
// Have the backend scan right away, which will make it detect that we disconnected.
263275
// Otherwise there would be up to a second of delay (the backend device manager scan interval).
264276
MobileserverUsbUpdate()
265-
277+
266278
restartScan()
267279
}
268280

@@ -299,7 +311,7 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
299311
let value = pProduct.value else {
300312
return nil
301313
}
302-
314+
303315
if value.isEmpty {
304316
return nil
305317
}
@@ -326,7 +338,7 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
326338
struct StateJSON: Codable {
327339
let bluetoothAvailable: Bool
328340
let peripherals: [PeripheralJSON]
329-
let connecting: Bool
341+
let connectionState: ConnectionState
330342
}
331343

332344
// Convert discoveredPeripherals to the JSON structure
@@ -338,7 +350,11 @@ class BluetoothManager: NSObject, ObservableObject, CBCentralManagerDelegate, CB
338350
)
339351
}
340352

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+
)
342358

343359
do {
344360
let encoder = JSONEncoder()

frontends/web/src/api/bluetooth.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@ export type TPeripheral = {
2626
export type TState = {
2727
bluetoothAvailable: boolean;
2828
peripherals: TPeripheral[];
29-
connecting: boolean;
29+
connectionState: 'scanning' | 'connecting' | 'connected';
3030
};
3131

3232
export const getState = (): Promise<TState> => {

frontends/web/src/components/bluetooth/bluetooth.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,7 @@ const _Bluetooth = () => {
3737
{t('bluetooth.select')}
3838
</div>
3939
<div className={styles.container}>
40-
{ state.connecting ? <p>connecting</p> : null }
40+
State: { state.connectionState }
4141
{state.peripherals.map(peripheral => {
4242
return (
4343
<ActionableItem

0 commit comments

Comments
 (0)