自动连接到BLE设备

时间:2014-02-20 18:45:00

标签: android-service bluetooth-lowenergy android-bluetooth


我正在开发一个与BLE设备进行通信的应用程序,目前我正在尝试创建一个以应用程序启动并自动连接到TI的CC2541
密钥卡的服务。

问题是gatt服务器似乎每次都失败了.... 我不知道我的代码有什么问题,因为谷歌API和我看到的一些教程
似乎所有的作品都在他们的位置,但仍然没有任何作用...... =(

这是我的服务 -

package com.example.bluetoothgatt;

import java.util.UUID;

import android.app.Service;
import android.bluetooth.BluetoothAdapter;
import android.bluetooth.BluetoothDevice;
import android.bluetooth.BluetoothGatt;
import android.bluetooth.BluetoothGattCharacteristic;
import android.bluetooth.BluetoothGattService;
import android.bluetooth.BluetoothManager;
import android.bluetooth.BluetoothProfile;
import android.content.Intent;
import android.os.Binder;
import android.os.Handler;
import android.os.IBinder;
import android.os.Message;
import android.util.Log;

public class BLE extends Service implements BluetoothAdapter.LeScanCallback {

    private final IBinder mBinder = new BluetoothLeBinder();
    private final static String TAG = "BLE";
    private static final String DEVICE_NAME = "Keyfobdemo";
    private BluetoothManager mBluetoothManager;
    public BluetoothGatt mConnectedGatt;
    private BluetoothAdapter mBluetoothAdapter;
    private BluetoothDevice mDevice;
    private String mDeviceAddress;

    private int mConnectionState = STATE_DISCONNECTED;
    private static final int STATE_DISCONNECTED = 0;
    private static final int STATE_CONNECTING = 1;
    private static final int STATE_CONNECTED = 2;

    /*******************************
     ******************************* 
     ****** Service Inherited ****** Methods **********
     *******************************/

    @Override
    public void onCreate() {
        super.onCreate();
        mBluetoothManager = (BluetoothManager) getSystemService(BLUETOOTH_SERVICE);
        mBluetoothAdapter = mBluetoothManager.getAdapter();
        Thread discoverDevices = new Thread(mStartRunnable);
        discoverDevices.setPriority(discoverDevices.MAX_PRIORITY);
        discoverDevices.start();
    }

    @Override
    public IBinder onBind(Intent intent) {
        return mBinder;
    }

    @Override
    public boolean onUnbind(Intent intent) {
        close();
        return super.onUnbind(intent);
    }

    // Implements callback methods for GATT events that the app cares about.
    // For example, connection change and services discovered.
    private final BluetoothGattExecutor mExecutor = new BluetoothGattExecutor() {

        @Override
        public void onConnectionStateChange(BluetoothGatt gatt, int status,
                int newState) {
            super.onConnectionStateChange(gatt, status, newState);

            if (newState == BluetoothProfile.STATE_CONNECTED) {
                mConnectionState = STATE_CONNECTED;
                mConnectedGatt = gatt;
            } else if (newState == BluetoothProfile.STATE_DISCONNECTED) {
                mConnectionState = STATE_DISCONNECTED;
                Log.i(TAG, "Disconnected from GATT server.");
            }
        }

        @Override
        public void onServicesDiscovered(BluetoothGatt gatt, int status) {
            super.onServicesDiscovered(gatt, status);

            if (status == BluetoothGatt.GATT_SUCCESS) {
            } else {
                Log.w(TAG, "onServicesDiscovered received: " + status);
            }
        }

        @Override
        public void onCharacteristicRead(BluetoothGatt gatt,
                BluetoothGattCharacteristic characteristic, int status) {
            super.onCharacteristicRead(gatt, characteristic, status);

            if (status == BluetoothGatt.GATT_SUCCESS) {
            }
        }

        @Override
        public void onCharacteristicChanged(BluetoothGatt gatt,
                BluetoothGattCharacteristic characteristic) {
            super.onCharacteristicChanged(gatt, characteristic);
        }
    };

    /**
     * Return a reference for the current class
     */
    public class BluetoothLeBinder extends Binder {
        BLE getService() {
            return BLE.this;
        }
    }

    private Runnable mStartRunnable = new Runnable() {
        @Override
        public void run() {
            startScan();
        }
    };

    private void startScan() {
        if (mConnectionState == STATE_DISCONNECTED) {
            mBluetoothAdapter.startLeScan(this);
            mHandler.postDelayed(mStopRunnable, 2500);
        }
    }

    private Runnable mStopRunnable = new Runnable() {
        @Override
        public void run() {
            stopScan();
        }
    };

    private void stopScan() {
        mBluetoothAdapter.stopLeScan(this);
    }

    @Override
    public void onLeScan(BluetoothDevice device, int rssi, byte[] scanRecord) {
        /*
         * We are looking for SensorTag devices only, so validate the name that
         * each device reports before adding it to our collection
         */
        if (DEVICE_NAME.equals(device.getName())) {
            mDevice = device;
            mDeviceAddress = mDevice.getAddress();
            connect(mDeviceAddress);
            mConnectionState = STATE_CONNECTING;
            if(device.getBondState() == BluetoothDevice.BOND_BONDED) {

            } else if (device.getBondState() == BluetoothDevice.BOND_BONDING) {

            } else if(device.getBondState() == BluetoothDevice.BOND_NONE) {
                connect(device.getAddress());
            }
        }
    }

    /**
     * Connects to the GATT server hosted on the Bluetooth LE device.
     * 
     * @param address
     *            The device address of the destination device.
     * 
     * @return Return true if the connection is initiated successfully. The
     *         connection result is reported asynchronously through the
     *         {@code BluetoothGattCallback#onConnectionStateChange(android.bluetooth.BluetoothGatt, int, int)}
     *         callback.
     */
    public boolean connect(final String address) {
        if (mBluetoothAdapter == null || address == null) {
            Log.w(TAG,
                    "BluetoothAdapter not initialized or unspecified address.");
            return false;
        }

        // Previously connected device. Try to reconnect.
        if (mDeviceAddress != null && address.equals(mDeviceAddress)
                && mConnectedGatt != null) {
            Log.d(TAG,
                    "Trying to use an existing BluetoothGatt for connection.");
            if (mConnectedGatt.connect()) {
                mConnectionState = STATE_CONNECTING;
                return true;
            } else {
                return false;
            }
        }

        final BluetoothDevice device = mBluetoothAdapter
                .getRemoteDevice(address);
        if (device == null) {
            Log.w(TAG, "Device not found.  Unable to connect.");
            return false;
        }
        // We want to directly connect to the device, so we are setting the
        // autoConnect
        // parameter to false.
        mConnectedGatt = device.connectGatt(this, false, mExecutor);
        Log.d(TAG, "Trying to create a new connection.");
        mDeviceAddress = address;
        mConnectionState = STATE_CONNECTING;
        return true;
    }

    /**
     * Disconnects an existing connection or cancel a pending connection. The
     * disconnection result is reported asynchronously through the
     * BluetoothGattCallback >>
     * onConnectionStateChange(android.bluetooth.BluetoothGatt, int, int)
     * callback.
     */
    public void disconnect() {
        if (mBluetoothAdapter == null || mConnectedGatt == null) {
            Log.w(TAG, "BluetoothAdapter not initialized");
            return;
        }
        mConnectedGatt.disconnect();
    }

    /**
     * After using a given BLE device, the app must call this method to ensure
     * resources are released properly.
     */
    public void close() {
        if (mConnectedGatt == null) {
            return;
        }
        mConnectedGatt.close();
        mConnectedGatt = null;
    }

    private final UUID IMMEDIATE_ALERT_UUID = UUID
            .fromString("00001802-0000-1000-8000-00805f9b34fb");
    private final UUID ALERT_LEVEL_UUID = UUID
            .fromString("00002a06-0000-1000-8000-00805f9b34fb");

    public void Buzz(BluetoothGatt gatt, int level) {
        BluetoothGattService alertService = gatt
                .getService(IMMEDIATE_ALERT_UUID);
        if (alertService == null) {
            Log.d(TAG, "Immediate Alert service not found!");
            return;
        }
        BluetoothGattCharacteristic alertLevel = alertService
                .getCharacteristic(ALERT_LEVEL_UUID);
        if (alertLevel == null) {
            Log.d(TAG, "Alert Level charateristic not found!");
            return;
        }
        alertLevel.setValue(level, BluetoothGattCharacteristic.FORMAT_UINT8, 0);
        gatt.writeCharacteristic(alertLevel);
        Log.d(TAG, "Alert");
    }

    private final UUID BATTERY_SERVICE_UUID = UUID
            .fromString("0000180F-0000-1000-8000-00805f9b34fb");
    private final UUID BATTERY_LEVEL_UUID = UUID
            .fromString("00002a19-0000-1000-8000-00805f9b34fb");

    public int getbattery(BluetoothGatt mBluetoothGatt) {

        BluetoothGattService batteryService = mConnectedGatt
                .getService(BATTERY_SERVICE_UUID);
        if (batteryService == null) {
            Log.d(TAG, "Battery service not found!");
            return 0;
        }

        BluetoothGattCharacteristic batteryLevel = batteryService
                .getCharacteristic(BATTERY_LEVEL_UUID);
        if (batteryLevel == null) {
            Log.d(TAG, "Battery level not found!");
            return 0;
        }
        mBluetoothGatt.readCharacteristic(batteryLevel);
        return batteryLevel.getIntValue(
                BluetoothGattCharacteristic.FORMAT_SINT8, 0);
    }

    /*
     * We have a Handler to process event results on the main thread
     */
    private static final int MSG_PROGRESS = 201;
    private static final int MSG_DISMISS = 202;
    private static final int MSG_CLEAR = 301;
    private Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            BluetoothGattCharacteristic characteristic;
            switch (msg.what) {
            case MSG_PROGRESS:
                break;
            case MSG_DISMISS:
                break;
            case MSG_CLEAR:
                break;
            }
        }
    };

    public void MakeBuzz() {
        Thread t = new Thread(new Runnable() {

            @Override
            public void run() {
                mConnectedGatt = mDevice.connectGatt(getApplicationContext(),
                        true, mExecutor);
                BluetoothGattService alertService = mConnectedGatt
                        .getService(IMMEDIATE_ALERT_UUID);
                int x = getbattery(mConnectedGatt);
                Buzz(mConnectedGatt, 2);
            }
        });
        t.start();
    }
}

这是Application类 -

package com.example.bluetoothgatt;


import android.app.Application;
import android.content.Intent;

public class ApplicationBleTest extends Application {
    // Application variables
    public final String SMOKE_TALK_PACKAGE_NAME = "com.smoketalk";
    private BluetoothLEService mBleService;
    private static int MODE_PRIVATE;

    /**
     * Application OnCreate event initiate the class parameters
     */
    public void onCreate() {
        super.onCreate();
        getApplicationContext().startService(new Intent(this, BLE.class));
    }
}

这是主要活动(我试图通过点击按钮使密钥嗡嗡声响起)

package com.example.bluetoothgatt;

import com.example.bluetoothgatt.BluetoothLowEnergyService.BluetoothLeBinder;

import android.app.Activity;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;

/**
 * Created by Dave Smith Double Encore, Inc. MainActivity
 */
public class MainActivity extends Activity {

    BluetoothLowEnergyService mBluetoothService;
    boolean isBound = false;
    Button buzz;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Intent intent = new Intent(this, BluetoothLowEnergyService.class);
        bindService(intent, mBleServiceConnection, Context.BIND_AUTO_CREATE);
        buzz = (Button) findViewById(R.id.btn1);
        buzz.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View v) {
                mBluetoothService.MakeBuzz();
            }
        });
    }

    private ServiceConnection mBleServiceConnection = new ServiceConnection() {

        public void onServiceConnected(ComponentName className, IBinder service) {
            BluetoothLeBinder binder = (BluetoothLeBinder) service;
            mBluetoothService = binder.getService();
            isBound = true;
        }

        public void onServiceDisconnected(ComponentName arg0) {
            isBound = false;
        }

    };
}

清单文件 -

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.bluetoothgatt"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-feature
        android:name="android.hardware.bluetooth_le"
        android:required="true" />

    <uses-sdk
        android:minSdkVersion="18"
        android:targetSdkVersion="18" />

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

    <application
        android:name="com.example.bluetoothgatt.ApplicationBleTest"
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name=".MainActivity"
            android:label="SensorTag Weather" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

        <service android:name="com.example.bluetoothgatt.BLE" />
    </application>

</manifest>

和最后一个主要活动的布局 -

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin" >

    <TextView
        android:id="@+id/textView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:gravity="center_horizontal"
        android:padding="@dimen/activity_horizontal_margin"
        android:text="Android BLE Test"
        android:textSize="42sp" />

    <Button
        android:id="@+id/btn1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_below="@+id/textView1"
        android:layout_centerHorizontal="true"
        android:layout_marginTop="56dp"
        android:text="Buzz" />

</RelativeLayout>

任何帮助都将受到赞赏,因为我重新定位不知道出了什么问题... =(

4 个答案:

答案 0 :(得分:3)

对于初学者,我建议注释掉粘合代码(在ifLeScan方法中,if之后的所有东西(device.getBondState()..)整个粘合过程在4.3(至少是Nexus设备)上不稳定,并且变得更加稳定4.4。 您应该能够发现设备,并且用户选择的BluetoothDevice应该在停止发现后调用ConnectGatt。这将尝试连接到设备上的Gatt服务器。如果连接成功,您应该在connectionStateChange上收到回调,表明连接成功。

绑定背后的概念与设备配对和交换密钥有关,如果您的特性是加密的。通常,您应该能够连接到Gatt服务器而无需绑定,但是一旦连接,如果您尝试读取加密特性,它将失败。

答案 1 :(得分:2)

我尝试了你的代码并且它有效。你需要遵循这个过程:

  1. BluetoothAdapter.startLeScan(leCallback)

  2. 在leCallback中的onLeScan(最终的BluetoothDevice设备,int rssi,byte [] scanRecord)中,调用btDevice.connectGatt(上下文上下文,布尔autoConnect,BluetoothGattCallback gattCallback);

  3. 在gattCallBack中的onConnectionStateChange(BluetoothGatt gatt,int status,int newState)中,检查newState是否为BluetoothProfile.STATE_CONNECTED,如果是,则调用gatt.discoverServices();

  4. 在gattCallBack中的onServicesDiscovered(BluetoothGatt gatt,int status)中,检查状态是否为BluetoothGatt.GATT_SUCCESS,如果是,则按UUID获取服务:BluetoothGattService service = gatt.getService(YOUR_SERVICE_UUID);

  5. 如果服务为空,则表示尚未发现服务,您需要在发现下一个服务时再次检查,并再次调用onServicesDiscovered。

  6. 当发现所有服务时,您应该已经获得了服务,除非设备不支持它。

  7. 现在,您可以在Buzz方法中使用您的服务。

答案 2 :(得分:1)

另外值得注意的是,BLE动作必须全部由您序列化。例如,如果您对某个特性进行了读/写操作,则需要在执行另一个操作之前等待回调。如果不是,这将导致错误。

答案 3 :(得分:0)

由于您从服务运行,您可以尝试在主线程上运行connect,如下所示:

public void connectToDevice( String deviceAddress) {
mDeviceAddress = deviceAddress;
final BluetoothDevice device = mBluetoothAdapter.getRemoteDevice(mDeviceAddress);

Handler handler = new Handler(Looper.getMainLooper());
handler.post(new Runnable() {
    @Override
    public void run() {


        if (device != null) {

              mGatt = device.connectGatt(getApplicationContext(), true, mGattCallback);
              scanLeDevice(false);// will stop after first device detection
          }
       }
   });
}

希望它有所帮助。