我最近做了一個(gè)軟硬件結(jié)合的項(xiàng)目诫肠,通過小程序藍(lán)牙去連接設(shè)備司澎,控制鎖的開門,總結(jié)了一下知識(shí)和坑點(diǎn)栋豫,希望對(duì)大家有點(diǎn)幫助挤安。
首先說我做的這個(gè)項(xiàng)目的作用吧
我的項(xiàng)目是一個(gè)醫(yī)療陪護(hù)床,他做初衷是丧鸯,給長(zhǎng)期在醫(yī)院陪病人家屬的人準(zhǔn)備的蛤铜,當(dāng)病人家屬累了,或者困了的時(shí)候骡送,就可以關(guān)注我的小程序昂羡,進(jìn)入小程序,點(diǎn)擊掃碼交押金摔踱,然后開鎖使用虐先,把陪護(hù)床拉開,即可使用派敷。
開始介紹我的藍(lán)牙代碼和坑點(diǎn)
開鎖使用流程代碼
1蛹批,進(jìn)入小程序點(diǎn)擊開鎖,判斷手機(jī)藍(lán)牙是否開啟篮愉,這一步最開始的就是判斷小程序?qū)κ謾C(jī)藍(lán)牙的狀態(tài)查看腐芍。
wx.closeBluetoothAdapter({
? ? success: res => {
? ? ? wx.openBluetoothAdapter({ //初始化 藍(lán)牙模塊? 成功 和 失敗的回調(diào)
? ? ? ? success: res => {
? ? ? ? ? console.log('初始化藍(lán)牙成功' + res)
? ? ? ? ? this.context.setData({ bluetoothState: true })
? ? ? ? ? this.getBluetoothAdapterState()? //藍(lán)牙開啟成功,執(zhí)行去查看藍(lán)牙適配情況
? ? ? ? },
? ? ? ? fail: err => {
? ? ? ? ? console.log(err);
? ? ? ? ? wx.hideLoading();
? ? ? ? ? this.context.setData({ bluetoothState: false })
? ? ? ? }
? ? ? })
? ? }
? })
2试躏,手機(jī)藍(lán)牙開啟成功猪勇,進(jìn)一步判斷藍(lán)牙適配是否可用
wx.getBluetoothAdapterState({ //獲取本機(jī)藍(lán)牙適配器狀態(tài) 判斷用戶是否開啟藍(lán)牙
? ? ? success: res => {
? ? ? ? console.log('藍(lán)牙狀態(tài)', res)
? ? ? ? //discovering 是否正在搜索設(shè)備? available 藍(lán)牙適配器是否可用
? ? ? ? if (res.available == false) {?
? ? ? ? ? wx.showToast({
? ? ? ? ? ? title: '設(shè)備無法開啟藍(lán)牙連接',
? ? ? ? ? ? icon: 'none'
? ? ? ? ? })
? ? ? ? } else if (res.discovering == false){
? ? ? ? ? ? this.startBluetoothDevicesDiscovery()
? ? ? ? } else if (res.available){
? ? ? ? ? this.startBluetoothDevicesDiscovery()? //藍(lán)牙適配器正常,去執(zhí)行搜索外圍設(shè)備
? ? ? ? }
? ? ? }
? ? })
3颠蕴,開始搜索外圍設(shè)備泣刹,(注意:執(zhí)行該操作比較浪費(fèi)資源,在搜索到mac地址成功后停止該操作)
wx.startBluetoothDevicesDiscovery({ //開始搜尋附近的藍(lán)牙外圍設(shè)備
? ? services: [this.serviceId], //搜索對(duì)應(yīng)設(shè)備的id? 以微信硬件平臺(tái)的藍(lán)牙智能燈為例犀被,主服務(wù)的 UUID 是 FEE7椅您。傳入這個(gè)參數(shù),只搜索主服務(wù) UUID 為 FEE7 的設(shè)備
我這邊 因?yàn)楣碳讓哟a放設(shè)定了? serviceId = '0000FFF0-0000-1000-8000-00805F9B34FB'? 寡键,所以只會(huì)所搜0000FFF0-0000-1000-8000-00805F9B34FB 的住服務(wù)掀泳,這邊的話類似一個(gè)過濾器
? ? allowDuplicatesKey:false,
? ? success:res =>{
? ? ? ? console.log(res);
? ? ? if (!res.isDiscovering) { //是否在搜索到了設(shè)備
? ? ? ? this.getBluetoothAdapterState()
? ? ? }else{
? ? ? ? console.log('1');
????????/ /當(dāng)時(shí)成功時(shí)候,就去執(zhí)行左邊監(jiān)聽附近新設(shè)備的api
? ? ? ? this.onBluetoothDeviceFound()
? ? ? }
? ? },
? ? fail: err => {
? ? ? console.log(err)
? ? ? this.stopBluetoothDevicesDiscovery()
? ? }
? })
4西轩,這一步是去執(zhí)行監(jiān)聽附近設(shè)備员舵,獲取附近設(shè)備的devices 結(jié)構(gòu)
? ?這邊是有兩個(gè)場(chǎng)景,一個(gè)位ios藕畔,一個(gè)安卓固灵,安卓其實(shí)通過搜索到devices 里面的mac地直接配對(duì)通過,而ios所搜不到mac劫流,只能得到devices 里面uuid,(uuid 可以通過小程序官方提供的一個(gè)方法去轉(zhuǎn)成十六進(jìn)制的數(shù)據(jù)的mac地址),我這邊安卓和ios全部是通過uuid去轉(zhuǎn)為六十機(jī)制mac地址祠汇,去做配對(duì)的仍秤,這樣一來就不用區(qū)分ios和安卓?jī)蓚€(gè)場(chǎng)景。
注意可很,這里設(shè)備的mac搜索到之后诗力,一定要執(zhí)行wx.stopBluetoothDevicesDiscovery() 函數(shù),暫定搜索我抠,不然后臺(tái)資源很消耗性能的苇本。
functionab2hex(buffer){consthexArr =Array.prototype.map.call(newUint8Array(buffer),function(bit){return('00'+ bit.toString(16)).slice(-2) } )returnhexArr.join('')}
wx.onBluetoothDeviceFound((res) => {
? ? console.log('搜索到的設(shè)備沒配對(duì)成功有', res)
? ? res.devices.forEach(device => {
? ? ? console.log(device);
? ? ? let _advertisData = this.ab2hex(device.advertisData) //得到設(shè)備的mac地址
? ? ? if (_advertisData == that.advertisData){)
? ? ? ? this.context.connectCallBack('1'); 執(zhí)行成功對(duì)應(yīng)的回調(diào)函數(shù)
? ? ? ? wx.setStorageSync('device', device.deviceId)//第一搜索到設(shè)備 保存在本地,給關(guān)鎖的時(shí)候使用
? ? ? ? that.deviceId = device.deviceId;
? ? ? ? that.stopBluetoothDevicesDiscovery() //設(shè)備已經(jīng)搜索到菜拓,停止搜索
? ? ? ? console.log('設(shè)備已經(jīng)搜索到瓣窄,停止搜索')
? ? ? ? that.createBLEConnection()
? ? ? }
? ? })
? })
4,獲取onBluetoothDeviceFound 通過的?devices 里面 deviceId去連接設(shè)備纳鼎。
連接這邊安卓和ios 比較坑的俺夕,一次連接失敗的概率性特別大,所以為了用戶體驗(yàn)上贱鄙,必須的通過多次循環(huán)返回連接劝贸,直到連接成功之后,才跳出循環(huán)逗宁。
我這邊的小程序循環(huán)的
全局的openFlag=false 在點(diǎn)擊開鎖的第一個(gè)函數(shù)里面映九,執(zhí)行到createBLEConnection函數(shù)的時(shí)候,調(diào)用openlock函數(shù) 里面判斷條件不成立就是一直循環(huán)值瞎颗,直到? if (count == 6 && this.openFlag ==false) 時(shí)候件甥,這個(gè)時(shí)候說明6次多連接失敗,那么全局openFlag=ftrue 就會(huì)?if (count == 6 && this.openFlag == true) 這個(gè)終止的條件,跳出界面言缤,這是現(xiàn)階段最好解決方法嚼蚀,歡迎大家指點(diǎn)。
連接成功之后
createBLEConnection = function () {
? ? this.openlock(0);
}
openlock=function(count){
? ? //連接藍(lán)牙之后管挟,就一直是true·
? if (count == 6 && this.openFlag ==false){
? ? //五次沒有連接上轿曙,重新卸載藍(lán)牙模塊,重新連接
? ? this.startConnect(this.context,true);
? ? return false;
? }
? //成立
? if (count == 6 && this.openFlag == true){
? ? //執(zhí)行到這一步 一定是deviceId值錯(cuò)誤
? ? ? wx.showToast({
? ? ? ? title: '連接中斷,請(qǐng)重試',
? ? ? ? icon:'none'
? ? ? })
? ? ? setTimeout(()=>{
? ? ? ? wx.navigateBack({
? ? ? ? ? data: 1,
? ? ? ? })
? ? ? },2000)
? ? ? ? return false;? ? ? ?
? ? }
? const self = this;
? //500毫秒連接一次僻孝,連接一次失敗导帝,繼續(xù)連接,直到10次連接失敗穿铆,就讓他跳轉(zhuǎn)首頁(yè)
? setTimeout(() => {
? ? wx.createBLEConnection({
? ? ? deviceId: this.deviceId,
? ? //? deviceId:"a434f199651s",
? ? ? success: res => {
? ? ? ? console.log('連接', res)
? ? ? ? if (res.errCode == 0) {
? ? ? ? ? wx.hideLoading() //連接成功? 關(guān)閉提示狂
? ? ? ? ? self.funk = true //防止 連接點(diǎn)擊開鎖按鈕處理
? ? ? ? ? wx.showToast({
? ? ? ? ? ? title: '藍(lán)牙連接設(shè)備成功',
? ? ? ? ? ? icon: 'none'
? ? ? ? ? })
? ? ? ? ? this.getBLEDeviceServices(this.deviceId)
? ? ? ? }
? ? ? },
? ? ? fail: err => {
? ? ? ? console.log(err)
? ? ? ? console.log(count)
? ? ? ? self.openlock(count+1);
? ? ? }
? ? })
? }, 500)
}
5您单,連接藍(lán)牙成功之后,這里就是得獲取藍(lán)牙所有的服務(wù)荞雏,這一步為了與建立通訊而準(zhǔn)備的
/注意:這里的 deviceId 需要已經(jīng)通過 createBLEConnection 與對(duì)應(yīng)設(shè)備建立鏈接
wx.getBLEDeviceServices({
? ? deviceId,
? ? success: (res) => {
? ? ? console.log('服務(wù)', res)
? ? ? this.getBLEDeviceCharacteristics(this.deviceId, this.serviceId)
? ? }
? })
6,獲取設(shè)備的特征值虐秦,這一步很重要
?判斷是否 write? boolean? 該特征值是否支持 write 操作?write?該操作是藍(lán)牙寫人二進(jìn)制數(shù)據(jù)
??判斷是否 notifyboolean該特征值是否支持 notify 操作
??判斷是否? indicateboolean該特征值是否支持 indicate 操作
??判斷是否??readboolean該特征值是否支持 read 操作
上述判斷?notifyboolean?indicateboolean?readboolean 至少這三種通過平酿,不然寫二進(jìn)制數(shù)據(jù)的時(shí)候,回調(diào)失敗悦陋。
wx.getBLEDeviceCharacteristics({
? ? deviceId,
? ? serviceId,
? ? success: (res) => {
? ? ? console.log('特征值', res.characteristics)
? ? ? for (let i = 0; i < res.characteristics.length; i++) {
? ? ? ? let item = res.characteristics[i]
? ? ? ? if (item.uuid == this.characteristicId) {
? ? ? ? ? this.notifyBLECharacteristicValueChange(this.deviceId, this.serviceId, this.characteristicId)
? ? ? ? ? this.context.connectCallBack()
? ? ? ? }
? ? ? }
? ? },
? ? fail(res) {
? ? ? console.error('getBLEDeviceCharacteristics', res)
? ? }
? })
7.這個(gè)一步是比較重要蜈彼,向設(shè)備藍(lán)牙進(jìn)入十六進(jìn)制數(shù)據(jù),列表你想要設(shè)備做一些動(dòng)作俺驶,比如開鎖關(guān)鎖幸逆,一些指令。
wx.writeBLECharacteristicValue({
? ? ? deviceId,//設(shè)備的 id
? ? ? serviceId, //0000FFF0-0000-1000-8000-00805F9B34FB
? ? ? characteristicId: this.characteristicId,//0000FFF6-0000-1000-8000-00805F9B34FB
? ? ? value: buffer,//藍(lán)牙設(shè)備特征值對(duì)應(yīng)的二進(jìn)制值
? ? ? success: res => { resolve(res) },
? ? ? fail: err => { reject(err) }
? ? })
? })
8暮现,啟用低功耗藍(lán)牙設(shè)備特征值變化時(shí)的 notify 功能还绘,訂閱特征值。注意:必須設(shè)備的特征值支持 notify 或者 indicate 才可以成功調(diào)用栖袋。
另外拍顷,必須先啟用?notifyBLECharacteristicValueChange?才能監(jiān)聽到設(shè)備?characteristicValueChange?事件
wx.notifyBLECharacteristicValueChange({
? ? state: true,
? ? deviceId,
? ? serviceId,
? ? characteristicId,
? ? success: res => {
? ? ? console.log('監(jiān)聽成功:', res)
? ? ? this.onBLECharacteristicValueChange()
? ? }
? })
9.監(jiān)聽低功耗藍(lán)牙設(shè)備的特征值變化事件。必須先啟用?notifyBLECharacteristicValueChange?接口才能接收到設(shè)備推送的 notification栋荸。
wx.onBLECharacteristicValueChange(res => {
? ? console.log('123')
? ? console.log('監(jiān)聽值', this.ab2hex(res.value))
? ? this.context.callBack(this.ab2hex(res.value))
? })菇怀、
10,當(dāng)你所有功能pai執(zhí)行完成之后晌块,就去寶珠
wx.stopBluetoothDevicesDiscovery()