forked from dotnet/android-samples
-
Notifications
You must be signed in to change notification settings - Fork 0
/
BluetoothLeService.cs
341 lines (300 loc) · 11.3 KB
/
BluetoothLeService.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
/*
* Copyright (C) 2013 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using Android.App;
using Android.Content;
using Android.OS;
using Android.Runtime;
using Android.Views;
using Android.Widget;
using Android.Bluetooth;
using Java.Util;
using Android.Util;
namespace BluetoothLeGatt
{
/**
* Service for managing connection and data communication with a GATT server hosted on a
* given Bluetooth LE device.
*/
[Service (Name = "bluetoothlegatt.BluetoothLeService", Enabled = true )]
public class BluetoothLeService : Service
{
public readonly static String TAG = typeof (BluetoothLeService).Name;
public static BluetoothManager mBluetoothManager;
public static BluetoothAdapter mBluetoothAdapter;
public static String mBluetoothDeviceAddress;
public static BluetoothGatt mBluetoothGatt;
public static State mConnectionState = State.Disconnected;
public readonly static String ACTION_GATT_CONNECTED =
"com.xamarin.bluetooth.le.ACTION_GATT_CONNECTED";
public readonly static String ACTION_GATT_DISCONNECTED =
"com.xamarin.bluetooth.le.ACTION_GATT_DISCONNECTED";
public readonly static String ACTION_GATT_SERVICES_DISCOVERED =
"com.xamarin.bluetooth.le.ACTION_GATT_SERVICES_DISCOVERED";
public readonly static String ACTION_DATA_AVAILABLE =
"com.xamarin.bluetooth.le.ACTION_DATA_AVAILABLE";
public readonly static String EXTRA_DATA =
"com.xamarin.bluetooth.le.EXTRA_DATA";
public readonly static UUID UUID_HEART_RATE_MEASUREMENT =
UUID.FromString (SampleGattAttributes.HEART_RATE_MEASUREMENT);
IBinder mBinder;
public void BroadcastUpdate (String action)
{
Intent intent = new Intent (action);
SendBroadcast (intent);
}
public void BroadcastUpdate (String action, BluetoothGattCharacteristic characteristic)
{
Intent intent = new Intent (action);
// This is special handling for the Heart Rate Measurement profile. Data parsing is
// carried out as per profile specifications:
// http://developer.bluetooth.org/gatt/characteristics/Pages/CharacteristicViewer.aspx?u=org.bluetooth.characteristic.heart_rate_measurement.xml
if (UUID_HEART_RATE_MEASUREMENT == (characteristic.Uuid)) {
GattProperty flag = characteristic.Properties;
GattFormat format = (GattFormat) (-1);
if (((int) flag & 0x01) != 0) {
format = GattFormat.Uint16;
Log.Debug (TAG, "Heart rate format UINT16.");
} else {
format = GattFormat.Uint8;
Log.Debug (TAG, "Heart rate format UINT8.");
}
var heartRate = characteristic.GetIntValue (format, 1);
Log.Debug (TAG, String.Format ("Received heart rate: {0}", heartRate));
intent.PutExtra (EXTRA_DATA, heartRate);
} else {
// For all other profiles, writes the data formatted in HEX.
byte[] data = characteristic.GetValue ();
if (data != null && data.Length > 0) {
StringBuilder stringBuilder = new StringBuilder (data.Length);
foreach (byte byteChar in data)
stringBuilder.Append (String.Format ("{0}02X ", byteChar));
intent.PutExtra (EXTRA_DATA, Convert.ToBase64String (data) + "\n" + stringBuilder.ToString());
}
}
SendBroadcast (intent);
}
public class LocalBinder : Binder
{
BluetoothLeService service;
public LocalBinder (BluetoothLeService service)
{
this.service = service;
}
public BluetoothLeService GetService ()
{
return service;
}
}
public override IBinder OnBind (Intent intent)
{
mBinder = new LocalBinder (this);
return mBinder;
}
public override bool OnUnbind (Intent intent)
{
// After using a given device, you should make sure that BluetoothGatt.close() is called
// such that resources are cleaned up properly. In this particular example, close() is
// invoked when the UI is disconnected from the Service.
Close ();
return base.OnUnbind (intent);
}
/**
* Initializes a reference to the local Bluetooth adapter.
*
* @return Return true if the initialization is successful.
*/
public bool Initialize()
{
// For API level 18 and above, get a reference to BluetoothAdapter through
// BluetoothManager.
if (mBluetoothManager == null) {
mBluetoothManager = (BluetoothManager) GetSystemService (Context.BluetoothService);
if (mBluetoothManager == null) {
Log.Error (TAG, "Unable to initialize BluetoothManager.");
return false;
}
}
mBluetoothAdapter = mBluetoothManager.Adapter;
if (mBluetoothAdapter == null) {
Log.Error (TAG, "Unable to obtain a BluetoothAdapter.");
return false;
}
return true;
}
/**
* 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 bool Connect (String address)
{
if (mBluetoothAdapter == null || address == null) {
Log.Warn (TAG, "BluetoothAdapter not initialized or unspecified address.");
return false;
}
// Previously connected device. Try to reconnect.
if (mBluetoothDeviceAddress != null && address == mBluetoothDeviceAddress && mBluetoothGatt != null) {
Log.Debug (TAG, "Trying to use an existing mBluetoothGatt for connection.");
if (mBluetoothGatt.Connect ()) {
mConnectionState = State.Connecting;
return true;
} else {
return false;
}
}
BluetoothDevice device = mBluetoothAdapter.GetRemoteDevice (address);
if (device == null) {
Log.Warn (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.
mBluetoothGatt = device.ConnectGatt (this, false, new BGattCallback (this));
Log.Debug (TAG, "Trying to create a new connection.");
mBluetoothDeviceAddress = address;
mConnectionState = State.Connecting;
return true;
}
/**
* Disconnects an existing connection or cancel a pending connection. The disconnection result
* is reported asynchronously through the
* {@code BluetoothGattCallback#onConnectionStateChange(android.bluetooth.BluetoothGatt, int, int)}
* callback.
*/
public void Disconnect ()
{
if (mBluetoothAdapter == null || mBluetoothGatt == null) {
Log.Warn (TAG, "BluetoothAdapter not initialized");
return;
}
mBluetoothGatt.Disconnect ();
}
/**
* After using a given BLE device, the app must call this method to ensure resources are
* released properly.
*/
public void Close ()
{
if (mBluetoothGatt == null) {
return;
}
mBluetoothGatt.Close ();
mBluetoothGatt = null;
}
/**
* Request a read on a given {@code BluetoothGattCharacteristic}. The read result is reported
* asynchronously through the {@code BluetoothGattCallback#onCharacteristicRead(android.bluetooth.BluetoothGatt, android.bluetooth.BluetoothGattCharacteristic, int)}
* callback.
*
* @param characteristic The characteristic to read from.
*/
public void ReadCharacteristic (BluetoothGattCharacteristic characteristic)
{
if (mBluetoothAdapter == null || mBluetoothGatt == null) {
Log.Warn (TAG, "BluetoothAdapter not initialized");
return;
}
mBluetoothGatt.ReadCharacteristic (characteristic);
}
/**
* Enables or disables notification on a give characteristic.
*
* @param characteristic Characteristic to act on.
* @param enabled If true, enable notification. False otherwise.
*/
public void SetCharacteristicNotification (BluetoothGattCharacteristic characteristic, bool enabled)
{
if (mBluetoothAdapter == null || mBluetoothGatt == null) {
Log.Warn (TAG, "BluetoothAdapter not initialized");
return;
}
mBluetoothGatt.SetCharacteristicNotification (characteristic, enabled);
// This is specific to Heart Rate Measurement.
if (UUID_HEART_RATE_MEASUREMENT == characteristic.Uuid) {
BluetoothGattDescriptor descriptor = characteristic.GetDescriptor (
UUID.FromString (SampleGattAttributes.CLIENT_CHARACTERISTIC_CONFIG));
descriptor.SetValue (BluetoothGattDescriptor.EnableNotificationValue.ToArray ());
mBluetoothGatt.WriteDescriptor (descriptor);
}
}
/**
* Retrieves a list of supported GATT services on the connected device. This should be
* invoked only after {@code BluetoothGatt#discoverServices()} completes successfully.
*
* @return A {@code List} of supported services.
*/
public IList <BluetoothGattService> GetSupportedGattServices ()
{
if (mBluetoothGatt == null) return null;
return mBluetoothGatt.Services;
}
}
// Implements callback methods for GATT events that the app cares about. For example,
// connection change and services discovered.
class BGattCallback : BluetoothGattCallback
{
BluetoothLeService service;
public BGattCallback (BluetoothLeService s)
{
service = s;
}
public override void OnConnectionStateChange (BluetoothGatt gatt, GattStatus status, ProfileState newState)
{
String intentAction;
if (newState == ProfileState.Connected) {
intentAction = BluetoothLeService.ACTION_GATT_CONNECTED;
BluetoothLeService.mConnectionState = State.Connected;
service.BroadcastUpdate (intentAction);
Log.Info (BluetoothLeService.TAG, "Connected to GATT server.");
// Attempts to discover services after successful connection.
Log.Info (BluetoothLeService.TAG, "Attempting to start service discovery:" +
BluetoothLeService.mBluetoothGatt.DiscoverServices ());
} else if (newState == ProfileState.Disconnected) {
intentAction = BluetoothLeService.ACTION_GATT_DISCONNECTED;
BluetoothLeService.mConnectionState = State.Disconnected;
Log.Info (BluetoothLeService.TAG, "Disconnected from GATT server.");
service.BroadcastUpdate (intentAction);
}
}
public override void OnServicesDiscovered (BluetoothGatt gatt, GattStatus status)
{
if (status == GattStatus.Success) {
service.BroadcastUpdate (BluetoothLeService.ACTION_GATT_SERVICES_DISCOVERED);
} else {
Log.Warn (BluetoothLeService.TAG, "onServicesDiscovered received: " + status);
}
}
public override void OnCharacteristicRead (BluetoothGatt gatt, BluetoothGattCharacteristic characteristic, GattStatus status)
{
if (status == GattStatus.Success) {
service.BroadcastUpdate (BluetoothLeService.ACTION_DATA_AVAILABLE, characteristic);
}
}
public override void OnCharacteristicChanged (BluetoothGatt gatt, BluetoothGattCharacteristic characteristic)
{
service.BroadcastUpdate (BluetoothLeService.ACTION_DATA_AVAILABLE, characteristic);
}
}
}