魔改車鑰匙實現遠端控車:(4)基於compose和經典藍芽編寫一個控制APP

語言: CN / TW / HK

我報名參加金石計劃1期挑戰——瓜分10萬獎池,這是我的第2篇文章,點選檢視活動詳情

前言

這篇文章不出意外的話應該是魔改車鑰匙系列的最後一篇了,自此我們的魔改計劃除了最後的佈線和安裝外已經全部完成了。

不過由於佈線以及安裝不屬於程式設計技術範圍,且我也是第一次做,就不獻醜繼續寫一篇文章了。

在前面的文章中,我們已經完成了 Arduino 控制程式的編寫,接下來就差編寫一個簡單易用的手機端控制 APP 了。

這裡我們依舊選擇使用 compose 作為 UI 框架。

編寫這個控制 APP 會涉及到安卓上的藍芽開發知識,因此我們會先簡要介紹一下如何在安卓上進行藍芽開發。

開始編寫

藍芽基礎

藍芽分為經典藍芽和低功耗藍芽(BLE)這個知識點前面的文章已經介紹過了,在我們當前的需求中,我們只需要使用經典藍芽去與 ESP32 通訊,所以我們也只介紹如何使用經典藍芽。

藍芽許可權

在使用之前,我們需要確保藍芽許可權正確,根據官網教程新增如下許可權:

```xml

```

實際使用時不用新增所有的許可權,只需要根據你的需求新增需要的許可權即可。

詳細可以查閱官網文件:Bluetooth permissions

因為我們在這裡需要連線到 ESP32 所以不要忘記判斷執行時許可權:

kotlin if (ActivityCompat.checkSelfPermission(context, Manifest.permission.BLUETOOTH_CONNECT) != PackageManager.PERMISSION_GRANTED) { // xxxxx // 沒有許可權 }

某些裝置可能不支援經典藍芽或BLE,亦或是兩者均不支援,所以我們需要做一下檢查:

kotlin private fun PackageManager.missingSystemFeature(name: String): Boolean = !hasSystemFeature(name) // ... // Check to see if the Bluetooth classic feature is available. packageManager.takeIf { it.missingSystemFeature(PackageManager.FEATURE_BLUETOOTH) }?.also { Toast.makeText(this, "不支援經典藍芽", Toast.LENGTH_SHORT).show() finish() } // Check to see if the BLE feature is available. packageManager.takeIf { it.missingSystemFeature(PackageManager.FEATURE_BLUETOOTH_LE) }?.also { Toast.makeText(this, "不支援BLE", Toast.LENGTH_SHORT).show() finish() }

以上程式碼來自官網示例

初始化藍芽

在使用藍芽前,我們需要獲取到系統的藍芽介面卡(BluetoothAdapter),後續的大多數操作都將基於這個介面卡展開:

kotlin val bluetoothManager: BluetoothManager = context.getSystemService(BluetoothManager::class.java) val bluetoothAdapter = bluetoothManager.adapter

需要注意的是,獲取到的 bluetoothAdapter 可能為空,需要自己做一下判空處理。

拿到 bluetoothAdapter 後,下一步是判斷是否開啟了藍芽,如果沒有開啟則需要請求開啟:

kotlin if (bluetoothAdapter?.isEnabled == false) { val enableBtIntent = Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE) startActivityForResult(enableBtIntent, REQUEST_ENABLE_BT) }

查詢藍芽裝置

由於在這個專案中, ESP32 沒法實時加密配對,所以我採用的是直接手動配對好我的手機,然後就不再配對新裝置,日後如果有需求,我會研究一下怎麼實時加密配對。

所以我們這裡暫時不需要搜尋新的藍芽裝置,只需要查詢已經連線的裝置即可:

```kotlin fun queryPairDevices(): Set? { if (bluetoothAdapter == null) { Log.e(TAG, "queryPairDevices: bluetoothAdapter is null!") return null }

val pairedDevices: Set<BluetoothDevice>? = bluetoothAdapter!!.bondedDevices

pairedDevices?.forEach { device ->
    val deviceName = device.name
    val deviceHardwareAddress = device.address

    Log.i(TAG, "queryPairDevices: deveice name=$deviceName, mac=$deviceHardwareAddress")
}

return pairedDevices

} ```

連線到指定裝置

連線藍芽裝置有兩種角色:服務端和客戶端,在我們這裡的使用場景中,我們的 APP 是客戶端,而 ESP32 是服務端,所以我們需要實現的是客戶端連線。

因為這裡我們連線的是已配對裝置,所以相對來說簡單的多,不需要做額外的處理,直接連線即可,連線後會拿到一個 BluetoothSocket ,後續的通訊將用到這個 BluetoothSocket

```kotlin suspend fun connectDevice(device: BluetoothDevice, onConnected : (socket: Result) -> Unit) { val mmSocket: BluetoothSocket? by lazy(LazyThreadSafetyMode.NONE) { device.createRfcommSocketToServiceRecord(UUID.fromString("00001101-0000-1000-8000-00805f9b34fb")) }

    withContext(Dispatchers.IO) {

        kotlin.runCatching {
            // 開始連線前應該關閉掃描,否則會減慢連線速度
            bluetoothAdapter?.cancelDiscovery()

            mmSocket?.connect()
        }.fold({
            withContext(Dispatchers.Main) {
                socket = mmSocket
                onConnected(Result.success(mmSocket!!))
            }
        }, {
            withContext(Dispatchers.Main) {
                onConnected(Result.failure(it))
            }
            Log.e(TAG, "connectDevice: connect fail!", it)
        })
    }
}

```

需要注意的一點是,UUID需要和 ESP32 設定的 UUID 一致,這裡我的 ESP32 並沒有設定什麼特殊的 UUID, 所以我們在 APP 中使用的是常用的 UUID:

Hint: If you are connecting to a Bluetooth serial board then try using the well-known SPP UUID 00001101-0000-1000-8000-00805F9B34FB. However if you are connecting to an Android peer then please generate your own unique UUID.

另外,其實從名字 Socket 就能看出,這是個耗時操作,所以我們將其放到協程中,並使用工作執行緒執行 withContext(Dispatchers.IO)

對了,上面的程式碼中,我加了一句 bluetoothAdapter?.cancelDiscovery() 其實這行程式碼在這裡純屬多餘,因為我壓根沒有搜尋裝置的操作,但是為了避免我以後新增搜尋裝置後忘記加上,所以我沒有給它刪掉。

最後,我這裡使用了一個匿名函式回撥連線結果 onConnected : (socket: Result<BluetoothSocket>) -> Unit

資料通訊

資料通訊需要使用上一節拿到的 BluetoothSocket ,通過 read BluetoothSocketInputStream 從服務端讀取資料;write BluetoothSocketOutputStream 往服務端寫入資料:

```kotlin suspend fun startBtReceiveServer(mmSocket: BluetoothSocket, onReceive: (numBytes: Int, byteBufferArray: ByteArray) -> Unit) { keepReceive = true val mmInStream: InputStream = mmSocket.inputStream val mmBuffer = ByteArray(1024) // 緩衝區大小

withContext(Dispatchers.IO) {
    var numBytes = 0 // 實際讀取的資料大小
    while (true) {

        kotlin.runCatching {
            mmInStream.read(mmBuffer)
        }.fold(
            {
                numBytes = it
            },
            {
                Log.e(TAG, "Input stream was disconnected", it)
                return@withContext
            }
        )

        withContext(Dispatchers.Main) {
            onReceive(numBytes, mmBuffer)
        }
    }
}

}

suspend fun sendByteToDevice(mmSocket: BluetoothSocket, bytes: ByteArray, onSend: (result: Result) -> Unit) { val mmOutStream: OutputStream = mmSocket.outputStream

withContext(Dispatchers.IO) {
    val result = kotlin.runCatching {
        mmOutStream.write(bytes)
    }

    if (result.isFailure) {
        Log.e(TAG, "Error occurred when sending data", result.exceptionOrNull())
        onSend(Result.failure(result.exceptionOrNull() ?: Exception("not found exception")))
    }
    else {
        onSend(Result.success(bytes))
    }
}

} ```

同樣的,這裡的讀取和寫入都是耗時操作,所以我都聲明瞭是掛起函式 suspend

另外,接收伺服器的資料時,需要一直迴圈讀取 inputStream 直至 socket 丟擲異常(連線被斷開)。

這裡我們在接收到新資料時,依然使用一個匿名函式回撥接收到的資料 onReceive: (numBytes: Int, byteBufferArray: ByteArray) -> Unit

其中 numBytes 是本次接收到的資料大小, byteBufferArray 是完整的緩衝陣列,實際資料可能沒有這麼多。

完整的幫助類

結合我們的需求,我寫了一個藍芽連線和通訊的幫助類 BtHelper :

```kotlin class BtHelper { private var bluetoothAdapter: BluetoothAdapter? = null private var keepReceive: Boolean = true

companion object {
    private const val TAG = "BtHelper"

    val instance by lazy(mode = LazyThreadSafetyMode.SYNCHRONIZED) {
        BtHelper()
    }
}

fun init(bluetoothAdapter: BluetoothAdapter) {
    this.bluetoothAdapter = bluetoothAdapter
}

fun init(context: Context): Boolean {
    val bluetoothManager: BluetoothManager = context.getSystemService(BluetoothManager::class.java)
    this.bluetoothAdapter = bluetoothManager.adapter
    return if (bluetoothAdapter == null) {
        Log.e(TAG, "init: bluetoothAdapter is null, may this device not support bluetooth!")
        false
    } else {
        true
    }
}

fun checkBluetooth(context: Context): Boolean {
    return ActivityCompat.checkSelfPermission(context, Manifest.permission.BLUETOOTH_CONNECT) == PackageManager.PERMISSION_GRANTED
            && bluetoothAdapter?.isEnabled == true
}

@SuppressLint("MissingPermission")
fun queryPairDevices(): Set<BluetoothDevice>? {
    if (bluetoothAdapter == null) {
        Log.e(TAG, "queryPairDevices: bluetoothAdapter is null!")
        return null
    }

    val pairedDevices: Set<BluetoothDevice>? = bluetoothAdapter!!.bondedDevices

    pairedDevices?.forEach { device ->
        val deviceName = device.name
        val deviceHardwareAddress = device.address

        Log.i(TAG, "queryPairDevices: deveice name=$deviceName, mac=$deviceHardwareAddress")
    }

    return pairedDevices
}

@SuppressLint("MissingPermission")
suspend fun connectDevice(device: BluetoothDevice, onConnected : (socket: Result<BluetoothSocket>) -> Unit) {
    val mmSocket: BluetoothSocket? by lazy(LazyThreadSafetyMode.NONE) {
        device.createRfcommSocketToServiceRecord(UUID.fromString("00001101-0000-1000-8000-00805f9b34fb"))
    }

    withContext(Dispatchers.IO) {

        kotlin.runCatching {
            // 開始連線前應該關閉掃描,否則會減慢連線速度
            bluetoothAdapter?.cancelDiscovery()

            mmSocket?.connect()
        }.fold({
            withContext(Dispatchers.Main) {
                onConnected(Result.success(mmSocket!!))
            }
        }, {
            withContext(Dispatchers.Main) {
                onConnected(Result.failure(it))
            }
            Log.e(TAG, "connectDevice: connect fail!", it)
        })
    }
}

fun cancelConnect(mmSocket: BluetoothSocket?) {
    try {
        mmSocket?.close()
    } catch (e: IOException) {
        Log.e(TAG, "Could not close the client socket", e)
    }
}

suspend fun startBtReceiveServer(mmSocket: BluetoothSocket, onReceive: (numBytes: Int, byteBufferArray: ByteArray) -> Unit) {
    keepReceive = true
    val mmInStream: InputStream = mmSocket.inputStream
    val mmBuffer = ByteArray(1024) // mmBuffer store for the stream

    withContext(Dispatchers.IO) {
        var numBytes = 0 // bytes returned from read()
        while (true) {

            kotlin.runCatching {
                mmInStream.read(mmBuffer)
            }.fold(
                {
                    numBytes = it
                },
                {
                    Log.e(TAG, "Input stream was disconnected", it)
                    return@withContext
                }
            )

            withContext(Dispatchers.Main) {
                onReceive(numBytes, mmBuffer)
            }
        }
    }
}

fun stopBtReceiveServer() {
    keepReceive = false
}

suspend fun sendByteToDevice(mmSocket: BluetoothSocket, bytes: ByteArray, onSend: (result: Result<ByteArray>) -> Unit) {
    val mmOutStream: OutputStream = mmSocket.outputStream

    withContext(Dispatchers.IO) {
        val result = kotlin.runCatching {
            mmOutStream.write(bytes)
        }

        if (result.isFailure) {
            Log.e(TAG, "Error occurred when sending data", result.exceptionOrNull())
            onSend(Result.failure(result.exceptionOrNull() ?: Exception("not found exception")))
        }
        else {
            onSend(Result.success(bytes))
        }
    }
}

} ```

通訊協議與需求

在上一篇文章寫完之後,其實我又加了許多功能。

但是我們的需求實際上總結來說就兩個:

  1. 能夠直接在手機 APP 上模擬觸發遙控器按鍵
  2. 能夠設定 ESP32 的某些引數

結合這個需求,我們制定瞭如下通訊協議(這裡只寫了重要的):

單指令:

| 指令 | 功能 | 說明 | | :----: | :----: | :----: | | 1 | 開啟電源 | 給遙控器供電 | | 2 | 關閉電源 | 斷開遙控器供電 | | 8 | 讀取當前主機板狀態(友好文字) | 讀取當前主機板的狀態資訊,以友好文字形式返回 | | 9 | 讀取主機板設定引數(格式化文字) | 讀取當前主機板儲存的設定引數,以格式化文字返回 | | 101 | 觸發上鎖按鍵 | 無 | | 102 | 斷開上鎖按鍵 | 無 | | 103 | 觸發解鎖按鍵 | 無 | | 104 | 斷開解鎖按鍵 | 無 | | 105 | 觸發多功能按鍵 | 無 | | 106 | 斷開多功能按鍵 | 無 |

設定引數指令:

設定引數內容格式依舊如同上篇文章所述,這裡並沒有做更改。

| 引數碼 | 功能 | 說明 | | :----: | :----: | :----: | | 1 | 設定間隔時間 | 設定 BLE 掃描一次的時間 | | 2 | 設定 RSSI 閾值 | 設定識別 RSSI 的閾值 | | 3 | 設定是否觸發解鎖按鍵 | 設定掃描到手環且RSSI閾值符合後,是否觸發解鎖按鍵,不開啟該項則只會給遙控起上電,不會自動解鎖 | | 4 | 設定是否啟用感應解鎖 | 設定是否啟用感應解鎖,不開啟則不會掃描手環,只能手動連線主機板並給遙控器上電解鎖 | | 5 | 設定掃描失敗多少次後觸發上鎖 | 設定掃描裝置失敗多少次後才會觸發上鎖並斷電,有時掃描藍芽會間歇性的掃描失敗,增加該選項是為了避免正常使用時被錯誤的上鎖 |

編寫 APP

介面設計

由於本文的重點不在於如何設計介面,所以這裡不再贅述怎麼實現介面,我直接就上最終實現效果即可。

對了,由於現在還在測試,所以最終介面肯定不會這麼簡陋的(也許吧)。

主頁(等待連線):

s1.jpg

控制頁:

s2.jpg

控制頁(開啟設定):

s3.jpg

邏輯實現

其實,這個程式碼邏輯也很簡單,這裡就挑幾個說說,其他的大夥可以直接看原始碼。

何時初始化

上面說到,我為藍芽通訊編寫了一個簡單的幫助類,並且實現了一個單例模式:

```kotlin companion object {

val instance by lazy(mode = LazyThreadSafetyMode.SYNCHRONIZED) {
    BtHelper()
}

} ```

我最開始是在 Application 中呼叫 BtHelper.instance.init(this) 初始化,但是我後來發現,這樣初始化的話,在實際使用中時,bluetoothAdapter 始終為 null 。

沒辦法,我把初始化放到了頂級 composable 中:

```kotlin @Composable fun HomeView(viewModel: HomeViewModel) { val context = LocalContext.current DisposableEffect(Unit) { viewModel.dispatch(HomeAction.InitBt(context))

    onDispose { }
}
// .....

} ```

InitBt 這個 Action 中,我呼叫了 BtHelper.instance.init(context) 重新初始化。

這下基本沒問題了。

傳送模擬按鍵資料

因為遙控器的按鍵涉及到短按和長按的邏輯操作,所以這裡我不能直接使用 Button 的點選回撥,而是要自己處理按下和擡起手指事件。

並且在按下 Button 時傳送觸發按鍵指令,鬆開 Button 時觸發斷開按鍵命令。

以上鎖這個 Button 為例:

kotlin Button( onClick = { }, modifier = Modifier.presBtn { viewModel.dispatch(HomeAction.OnClickButton(ButtonIndex.Lock, it)) } ) { Text(text = "上鎖") }

其中,presBtn 是我自己定義的一個擴充套件函式:

```kotlin @SuppressLint("UnnecessaryComposedModifier") @OptIn(ExperimentalComposeUiApi::class) inline fun Modifier.presBtn(crossinline onPress: (btnAction: ButtonAction)->Unit): Modifier = composed {

pointerInteropFilter {
    when (it.action) {
        MotionEvent.ACTION_DOWN -> {
            onPress(ButtonAction.Down)
        }
        MotionEvent.ACTION_UP -> {
            onPress(ButtonAction.Up)
        }
    }
    true
}

} ```

我在這個擴充套件函式中通過 pointerInteropFilter 獲取原始觸控事件,並回調其中的 ACTION_DOWNACTION_UP 事件。

然後在 OnClickButton 這個 Action 中做如下處理:

```kotlin private fun onClickButton(index: ButtonIndex, action: ButtonAction) { val sendValue: Byte = when (index) { ButtonIndex.Lock -> { if (action == ButtonAction.Down) 101 else 102 } }

viewModelScope.launch {
    BtHelper.instance.sendByteToDevice(socket!!, byteArrayOf(sendValue)) {
        it.fold(
            {
                Log.i(TAG, "seed successful: byte= ${it.toHexStr()}")
            },
            {
                Log.e(TAG, "seed fail", it)
            }
        )
    }
}

} ```

為了避免讀者看起來太混亂,這裡刪除了其他按鍵的判斷,只保留了上鎖按鍵。

通過判斷是 按下事件 還是 擡起事件 來決定傳送給 ESP32 的指令是 101 還是 102

讀取資料

在這個專案中,我們涉及到讀取資料的地方其實就兩個:讀取狀態(友好文字和格式化文字)。

其中返回的資料格式,在上面介面設計一節中的最後兩張截圖已經有所體現,上面返回的是友好文字,下面是格式化文字。

其中格式化文字我需要解析出來並更新到 UI 上(設定介面):

```kotlin BtHelper.instance.startBtReceiveServer(socket!!, onReceive = { numBytes, byteBufferArray -> if (numBytes > 0) { val contentArray = byteBufferArray.sliceArray(0..numBytes) val contentText = contentArray.toText()

    Log.i(TAG, "connectDevice: rev:numBytes=$numBytes, " +
            "\nbyteBuffer(hex)=${contentArray.toHexStr()}, " +
            "\nbyteBuffer(ascii)=$contentText"
    )

    viewStates = viewStates.copy(logText = "${viewStates.logText}\n$contentText")

    if (contentText.length > 6 && contentText.slice(0..2) == "Set") {
        Log.i(TAG, "connectDevice: READ from setting")
        val setList = contentText.split(",")
        viewStates = viewStates.copy(
            availableInduction = setList[1] != "0",
            triggerUnlock = setList[2] != "0",
            scanningTime = setList[3],
            rssiThreshold = setList[4],
            shutdownThreshold = setList[5],
            isReadSettingState = false
        )
    }
}

}) ```

對了,我還寫了一個轉換類(FormatUtils),用於處理返回資料:

```kotlin object FormatUtils {

/**
 * 將十六進位制字串轉成 ByteArray
 * */
fun hexStrToBytes(hexString: String): ByteArray {
    check(hexString.length % 2 == 0) { return ByteArray(0) }

    return hexString.chunked(2)
        .map { it.toInt(16).toByte() }
        .toByteArray()
}

/**
 * 將十六進位制字串轉成 ByteArray
 * */
fun String.toBytes(): ByteArray {
    return hexStrToBytes(this)
}

/**
 * 將 ByteArray 轉成 十六進位制字串
 * */
fun bytesToHexStr(byteArray: ByteArray) =
    with(StringBuilder()) {
        byteArray.forEach {
            val hex = it.toInt() and (0xFF)
            val hexStr = Integer.toHexString(hex)
            if (hexStr.length == 1) append("0").append(hexStr)
            else append(hexStr)
        }
        toString().uppercase(Locale.CHINA)
    }

/**
 * 將位元組陣列轉成十六進位制字串
 * */
fun ByteArray.toHexStr(): String {
    return bytesToHexStr(this)
}

/**
 * 將位元組陣列解析成文字(ASCII)
 * */
fun ByteArray.toText(): String {
    return String(this)
}

/**
 * 將 ByteArray 轉為 bit 字串
 * */
fun ByteArray.toBitsStr(): String {
    if (this.isEmpty()) return ""
    val sb = java.lang.StringBuilder()
    for (aByte in this) {
        for (j in 7 downTo 0) {
            sb.append(if (aByte.toInt() shr j and 0x01 == 0) '0' else '1')
        }
    }
    return sb.toString()
}

/**
 *
 * 將十六進位制字串轉成 ASCII 文字
 *
 * */
fun String.toText(): String {
    val output = java.lang.StringBuilder()
    var i = 0
    while (i < this.length) {
        val str = this.substring(i, i + 2)
        output.append(str.toInt(16).toChar())
        i += 2
    }
    return output.toString()
}

/**
 * 將十六進位制字串轉為帶符號的 Int
 * */
fun String.toNumber(): Int {
    return this.toInt(16).toShort().toInt()
}

/**
 * 將整數轉成有符號十六進位制字串
 *
 * @param length 返回的十六進位制總長度,不足會在前面補 0 ,超出會將前面多餘的去除
 * */
fun Int.toHex(length: Int = 4): String {
    val hex = Integer.toHexString(this).uppercase(Locale.CHINA)
    return hex.padStart(length, '0').drop((hex.length-length).coerceAtLeast(0))
}

} ```

專案地址

auto_controller

歡迎 star!