Android 藍(lán)牙自動匹配PIN碼跳過用戶交互

近期項目中需要連接藍(lán)牙設(shè)備,起初只是設(shè)置藍(lán)牙列表界面讓用戶點擊然后輸入默認(rèn)PIN碼,后來改需求了 = = ,要求自動連接指定設(shè)備并不需要用戶手動輸入PIN碼,作為Android 小白的我是拒絕的,但是拒絕有什么用~

首先說一下之后會用到的關(guān)于藍(lán)牙方面的東西:

  • 斷開藍(lán)牙已配對的設(shè)備
  • 搜索附近藍(lán)牙設(shè)備
  • 攔截用戶交互頁面,使用代碼輸入
  • 由于在最后連接的時候使用的是設(shè)備的SDK所以在這里就不介紹了

1.斷開已配對設(shè)備

最后在項目中發(fā)現(xiàn)沒有用。這里就先記錄一下。

    //得到配對的設(shè)備列表,清除已配對的設(shè)備
    public void removePairDevice() {
        if (mBluetoothAdapter != null) {
            //mBluetoothAdapter初始化方式 mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
            //這個就是獲取已配對藍(lán)牙列表的方法
            Set<BluetoothDevice> bondedDevices = mBluetoothAdapter.getBondedDevices();
            for (BluetoothDevice device : bondedDevices) {
                //這里可以通過device.getName()  device.getAddress()來判斷是否是自己需要斷開的設(shè)備
                unpairDevice(device);
            }
        }
    }

    //反射來調(diào)用BluetoothDevice.removeBond取消設(shè)備的配對
    private void unpairDevice(BluetoothDevice device) {
        try {
            Method m = device.getClass().getMethod("removeBond", (Class[]) null);
            m.invoke(device, (Object[]) null);
        } catch (Exception e) {
            Log.e("mate", e.getMessage());
        }
    }

2.搜索附近藍(lán)牙

首先我們需要注冊兩個廣播,第一個為正在搜索時的,第二個為搜索完成的。

        // Register for broadcasts when a device is discovered
        IntentFilter filter = new IntentFilter(BluetoothDevice.ACTION_FOUND);
        this.registerReceiver(mFindBlueToothReceiver, filter);
        // Register for broadcasts when discovery has finished
        filter = new IntentFilter(BluetoothAdapter.ACTION_DISCOVERY_FINISHED);
        this.registerReceiver(mFindBlueToothReceiver, filter);
        
        //需要時開始搜索
        if (mBluetoothAdapter.isDiscovering()) {
                    mBluetoothAdapter.cancelDiscovery();
         }
        mBluetoothAdapter.startDiscovery();

然后對廣播進(jìn)行處理。這里要說明一下ClsUtils.createBond()這個方法如果連接設(shè)備SDK中有配對的方法,建議把這個方法去掉,我這里是去掉的,加上的話偶爾會Toast出無法配對。

 private final BroadcastReceiver mFindBlueToothReceiver = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {
            String action = intent.getAction();
            // When discovery finds a device
            if (BluetoothDevice.ACTION_FOUND.equals(action)) {
                //TODO 開始搜索
                BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
             //   if (device.getBondState() != BluetoothDevice.BOND_BONDED) {//判斷藍(lán)牙狀態(tài),是否是已配對
                    //TODO 可以在這判斷名字 如果搜索結(jié)束后沒有,再到已配對中尋找
                    String BTName[] = device.getName().split("-");
                    if (BTName[0].equals("xxx")) {
                          //在這連接設(shè)備 一般需要藍(lán)牙地址 device.getAddress();
                          try {
                              ClsUtils.createBond(device.getClass(), device);
                          } catch (Exception e) {
                              // TODO Auto-generated catch block
                              e.printStackTrace();
                          }
                    }
              //  }
            } else if (BluetoothAdapter.ACTION_DISCOVERY_FINISHED.equals(action)) {
                //TODO 搜索結(jié)束
                Toast.makeText(context, "搜索結(jié)束",Toast.LENGTH_SHORT).show();
            }
        }
    };

這里在Activity結(jié)束時記得取消注冊和取消搜索

unregisterReceiver(mFindBlueToothReceiver);
 if (mBluetoothAdapter != null) {
         mBluetoothAdapter.cancelDiscovery();
 }

3.攔截用戶交互頁面

通過廣播可以監(jiān)聽到輸入PIN碼的那個頁面將要彈出

       <receiver android:name=".BluetoothConnectActivityReceiver" >
            <intent-filter android:priority="1000">
                <action android:name="android.bluetooth.device.action.PAIRING_REQUEST" />
            </intent-filter>
        </receiver>

廣播中需要做的事情,注意一定要調(diào)用abortBroadcast(),不然交互頁面還是會出現(xiàn)一下然后消失。就是這個方法找了一天(╯‵□′)╯︵┴─┴。。。

public class BluetoothConnectActivityReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        if (intent.getAction().equals("android.bluetooth.device.action.PAIRING_REQUEST")) {
            BluetoothDevice mBluetoothDevice = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
            try {
                //(三星)4.3版本測試手機(jī)還是會彈出用戶交互頁面(閃一下),如果不注釋掉下面這句頁面不會取消但可以配對成功。(中興,魅族4(Flyme 6))5.1版本手機(jī)兩中情況下都正常
                //ClsUtils.setPairingConfirmation(mBluetoothDevice.getClass(), mBluetoothDevice, true);
                abortBroadcast();//如果沒有將廣播終止,則會出現(xiàn)一個一閃而過的配對框。
                //3.調(diào)用setPin方法進(jìn)行配對...
                boolean ret = ClsUtils.setPin(mBluetoothDevice.getClass(), mBluetoothDevice, "你需要設(shè)置的PIN碼");
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
}

然后只要在搜索到自己需要的設(shè)備后連接進(jìn)行操作就可以了!!!一定要記得加權(quán)限~

 <uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
 <uses-permission android:name="android.permission.BLUETOOTH" />

在Android6.0之后還需要一個模糊定位的權(quán)限

<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />

最后把ClsUtils類奉上,網(wǎng)上有很多的。

/**************** 藍(lán)牙配對函數(shù) ***************/

import java.lang.reflect.Field;
import java.lang.reflect.Method;

import android.bluetooth.BluetoothDevice;
import android.util.Log;

public class ClsUtils {
    /**
     * 與設(shè)備配對 參考源碼:platform/packages/apps/Settings.git
     * /Settings/src/com/android/settings/bluetooth/CachedBluetoothDevice.java
     */
    static public boolean createBond(Class btClass, BluetoothDevice btDevice) throws Exception {
        Method createBondMethod = btClass.getMethod("createBond");
        Boolean returnValue = (Boolean) createBondMethod.invoke(btDevice);
        return returnValue.booleanValue();
    }

    /**
     * 與設(shè)備解除配對 參考源碼:platform/packages/apps/Settings.git
     * /Settings/src/com/android/settings/bluetooth/CachedBluetoothDevice.java
     */
    static public boolean removeBond(Class<?> btClass, BluetoothDevice btDevice) throws Exception {
        Method removeBondMethod = btClass.getMethod("removeBond");
        Boolean returnValue = (Boolean) removeBondMethod.invoke(btDevice);
        return returnValue.booleanValue();
    }

    static public boolean setPin(Class<? extends BluetoothDevice> btClass, BluetoothDevice btDevice, String str) throws Exception {
        try {
            Method removeBondMethod = btClass.getDeclaredMethod("setPin", new Class[]{byte[].class});
            Boolean returnValue = (Boolean) removeBondMethod.invoke(btDevice,
                    new Object[]
                            {str.getBytes()});
            Log.e("returnValue", "" + returnValue);
        } catch (SecurityException e) {
            // throw new RuntimeException(e.getMessage());
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            // throw new RuntimeException(e.getMessage());
            e.printStackTrace();
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        return true;

    }

    // 取消用戶輸入
    static public boolean cancelPairingUserInput(Class<?> btClass, BluetoothDevice device) throws Exception {
        Method createBondMethod = btClass.getMethod("cancelPairingUserInput");
//        cancelBondProcess(btClass, device);
        Boolean returnValue = (Boolean) createBondMethod.invoke(device);
        return returnValue.booleanValue();
    }

    // 取消配對
    static public boolean cancelBondProcess(Class<?> btClass, BluetoothDevice device) throws Exception {
        Method createBondMethod = btClass.getMethod("cancelBondProcess");
        Boolean returnValue = (Boolean) createBondMethod.invoke(device);
        return returnValue.booleanValue();
    }

    //確認(rèn)配對

    static public void setPairingConfirmation(Class<?> btClass, BluetoothDevice device, boolean isConfirm) throws Exception {
        Method setPairingConfirmation = btClass.getDeclaredMethod("setPairingConfirmation", boolean.class);
        setPairingConfirmation.invoke(device, isConfirm);
    }


    /**
     *
     * @param clsShow
     */
    static public void printAllInform(Class clsShow) {
        try {
            // 取得所有方法
            Method[] hideMethod = clsShow.getMethods();
            int i = 0;
            for (; i < hideMethod.length; i++) {
                Log.e("method name", hideMethod[i].getName() + ";and the i is:"+ i);
            }
            // 取得所有常量
            Field[] allFields = clsShow.getFields();
            for (i = 0; i < allFields.length; i++) {
                Log.e("Field name", allFields[i].getName());
            }
        } catch (SecurityException e) {
            // throw new RuntimeException(e.getMessage());
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            // throw new RuntimeException(e.getMessage());
            e.printStackTrace();
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
}

文章最后提供兩篇促使我寫這篇文章的背后資源╮( ̄▽ ̄)╭。
Markdown新手指南
Android藍(lán)牙自動配對Demo,親測好使!!!

第一次寫文章,不知道會是什么效果,我的第一次就這么出去了o(〃'▽'〃)o。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

推薦閱讀更多精彩內(nèi)容