1 /* 2 * Copyright (C) 2011 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.settingslib.bluetooth; 18 19 import android.bluetooth.BluetoothAdapter; 20 import android.bluetooth.BluetoothClass; 21 import android.bluetooth.BluetoothDevice; 22 import android.content.BroadcastReceiver; 23 import android.content.Context; 24 import android.content.Intent; 25 import android.content.IntentFilter; 26 import android.util.Log; 27 28 import com.android.settingslib.R; 29 30 import java.util.ArrayList; 31 import java.util.Collection; 32 import java.util.HashMap; 33 import java.util.Map; 34 import java.util.Set; 35 36 /** 37 * BluetoothEventManager receives broadcasts and callbacks from the Bluetooth 38 * API and dispatches the event on the UI thread to the right class in the 39 * Settings. 40 */ 41 public final class BluetoothEventManager { 42 private static final String TAG = "BluetoothEventManager"; 43 44 private final LocalBluetoothAdapter mLocalAdapter; 45 private final CachedBluetoothDeviceManager mDeviceManager; 46 private LocalBluetoothProfileManager mProfileManager; 47 private final IntentFilter mAdapterIntentFilter, mProfileIntentFilter; 48 private final Map<String, Handler> mHandlerMap; 49 private Context mContext; 50 51 private final Collection<BluetoothCallback> mCallbacks = 52 new ArrayList<BluetoothCallback>(); 53 54 private android.os.Handler mReceiverHandler; 55 56 interface Handler { 57 void onReceive(Context context, Intent intent, BluetoothDevice device); 58 } 59 60 private void addHandler(String action, Handler handler) { 61 mHandlerMap.put(action, handler); 62 mAdapterIntentFilter.addAction(action); 63 } 64 65 void addProfileHandler(String action, Handler handler) { 66 mHandlerMap.put(action, handler); 67 mProfileIntentFilter.addAction(action); 68 } 69 70 // Set profile manager after construction due to circular dependency 71 void setProfileManager(LocalBluetoothProfileManager manager) { 72 mProfileManager = manager; 73 } 74 75 BluetoothEventManager(LocalBluetoothAdapter adapter, 76 CachedBluetoothDeviceManager deviceManager, Context context) { 77 mLocalAdapter = adapter; 78 mDeviceManager = deviceManager; 79 mAdapterIntentFilter = new IntentFilter(); 80 mProfileIntentFilter = new IntentFilter(); 81 mHandlerMap = new HashMap<String, Handler>(); 82 mContext = context; 83 84 // Bluetooth on/off broadcasts 85 addHandler(BluetoothAdapter.ACTION_STATE_CHANGED, new AdapterStateChangedHandler()); 86 // Generic connected/not broadcast 87 addHandler(BluetoothAdapter.ACTION_CONNECTION_STATE_CHANGED, 88 new ConnectionStateChangedHandler()); 89 90 // Discovery broadcasts 91 addHandler(BluetoothAdapter.ACTION_DISCOVERY_STARTED, new ScanningStateChangedHandler(true)); 92 addHandler(BluetoothAdapter.ACTION_DISCOVERY_FINISHED, new ScanningStateChangedHandler(false)); 93 addHandler(BluetoothDevice.ACTION_FOUND, new DeviceFoundHandler()); 94 addHandler(BluetoothDevice.ACTION_DISAPPEARED, new DeviceDisappearedHandler()); 95 addHandler(BluetoothDevice.ACTION_NAME_CHANGED, new NameChangedHandler()); 96 addHandler(BluetoothDevice.ACTION_ALIAS_CHANGED, new NameChangedHandler()); 97 98 // Pairing broadcasts 99 addHandler(BluetoothDevice.ACTION_BOND_STATE_CHANGED, new BondStateChangedHandler()); 100 addHandler(BluetoothDevice.ACTION_PAIRING_CANCEL, new PairingCancelHandler()); 101 102 // Fine-grained state broadcasts 103 addHandler(BluetoothDevice.ACTION_CLASS_CHANGED, new ClassChangedHandler()); 104 addHandler(BluetoothDevice.ACTION_UUID, new UuidChangedHandler()); 105 106 // Dock event broadcasts 107 addHandler(Intent.ACTION_DOCK_EVENT, new DockEventHandler()); 108 109 mContext.registerReceiver(mBroadcastReceiver, mAdapterIntentFilter, null, mReceiverHandler); 110 } 111 112 void registerProfileIntentReceiver() { 113 mContext.registerReceiver(mBroadcastReceiver, mProfileIntentFilter, null, mReceiverHandler); 114 } 115 116 public void setReceiverHandler(android.os.Handler handler) { 117 mContext.unregisterReceiver(mBroadcastReceiver); 118 mReceiverHandler = handler; 119 mContext.registerReceiver(mBroadcastReceiver, mAdapterIntentFilter, null, mReceiverHandler); 120 registerProfileIntentReceiver(); 121 } 122 123 /** Register to start receiving callbacks for Bluetooth events. */ 124 public void registerCallback(BluetoothCallback callback) { 125 synchronized (mCallbacks) { 126 mCallbacks.add(callback); 127 } 128 } 129 130 /** Unregister to stop receiving callbacks for Bluetooth events. */ 131 public void unregisterCallback(BluetoothCallback callback) { 132 synchronized (mCallbacks) { 133 mCallbacks.remove(callback); 134 } 135 } 136 137 private final BroadcastReceiver mBroadcastReceiver = new BroadcastReceiver() { 138 @Override 139 public void onReceive(Context context, Intent intent) { 140 String action = intent.getAction(); 141 BluetoothDevice device = intent 142 .getParcelableExtra(BluetoothDevice.EXTRA_DEVICE); 143 144 Handler handler = mHandlerMap.get(action); 145 if (handler != null) { 146 handler.onReceive(context, intent, device); 147 } 148 } 149 }; 150 151 private class AdapterStateChangedHandler implements Handler { 152 public void onReceive(Context context, Intent intent, 153 BluetoothDevice device) { 154 int state = intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, 155 BluetoothAdapter.ERROR); 156 // update local profiles and get paired devices 157 mLocalAdapter.setBluetoothStateInt(state); 158 // send callback to update UI and possibly start scanning 159 synchronized (mCallbacks) { 160 for (BluetoothCallback callback : mCallbacks) { 161 callback.onBluetoothStateChanged(state); 162 } 163 } 164 // Inform CachedDeviceManager that the adapter state has changed 165 mDeviceManager.onBluetoothStateChanged(state); 166 } 167 } 168 169 private class ScanningStateChangedHandler implements Handler { 170 private final boolean mStarted; 171 172 ScanningStateChangedHandler(boolean started) { 173 mStarted = started; 174 } 175 public void onReceive(Context context, Intent intent, 176 BluetoothDevice device) { 177 synchronized (mCallbacks) { 178 for (BluetoothCallback callback : mCallbacks) { 179 callback.onScanningStateChanged(mStarted); 180 } 181 } 182 mDeviceManager.onScanningStateChanged(mStarted); 183 } 184 } 185 186 private class DeviceFoundHandler implements Handler { 187 public void onReceive(Context context, Intent intent, 188 BluetoothDevice device) { 189 short rssi = intent.getShortExtra(BluetoothDevice.EXTRA_RSSI, Short.MIN_VALUE); 190 BluetoothClass btClass = intent.getParcelableExtra(BluetoothDevice.EXTRA_CLASS); 191 String name = intent.getStringExtra(BluetoothDevice.EXTRA_NAME); 192 // TODO Pick up UUID. They should be available for 2.1 devices. 193 // Skip for now, there's a bluez problem and we are not getting uuids even for 2.1. 194 CachedBluetoothDevice cachedDevice = mDeviceManager.findDevice(device); 195 if (cachedDevice == null) { 196 cachedDevice = mDeviceManager.addDevice(mLocalAdapter, mProfileManager, device); 197 Log.d(TAG, "DeviceFoundHandler created new CachedBluetoothDevice: " 198 + cachedDevice); 199 } 200 cachedDevice.setRssi(rssi); 201 cachedDevice.setBtClass(btClass); 202 cachedDevice.setNewName(name); 203 cachedDevice.setVisible(true); 204 } 205 } 206 207 private class ConnectionStateChangedHandler implements Handler { 208 @Override 209 public void onReceive(Context context, Intent intent, BluetoothDevice device) { 210 CachedBluetoothDevice cachedDevice = mDeviceManager.findDevice(device); 211 int state = intent.getIntExtra(BluetoothAdapter.EXTRA_CONNECTION_STATE, 212 BluetoothAdapter.ERROR); 213 dispatchConnectionStateChanged(cachedDevice, state); 214 } 215 } 216 217 private void dispatchConnectionStateChanged(CachedBluetoothDevice cachedDevice, int state) { 218 synchronized (mCallbacks) { 219 for (BluetoothCallback callback : mCallbacks) { 220 callback.onConnectionStateChanged(cachedDevice, state); 221 } 222 } 223 } 224 225 void dispatchDeviceAdded(CachedBluetoothDevice cachedDevice) { 226 synchronized (mCallbacks) { 227 for (BluetoothCallback callback : mCallbacks) { 228 callback.onDeviceAdded(cachedDevice); 229 } 230 } 231 } 232 233 private class DeviceDisappearedHandler implements Handler { 234 public void onReceive(Context context, Intent intent, 235 BluetoothDevice device) { 236 CachedBluetoothDevice cachedDevice = mDeviceManager.findDevice(device); 237 if (cachedDevice == null) { 238 Log.w(TAG, "received ACTION_DISAPPEARED for an unknown device: " + device); 239 return; 240 } 241 if (CachedBluetoothDeviceManager.onDeviceDisappeared(cachedDevice)) { 242 synchronized (mCallbacks) { 243 for (BluetoothCallback callback : mCallbacks) { 244 callback.onDeviceDeleted(cachedDevice); 245 } 246 } 247 } 248 } 249 } 250 251 private class NameChangedHandler implements Handler { 252 public void onReceive(Context context, Intent intent, 253 BluetoothDevice device) { 254 mDeviceManager.onDeviceNameUpdated(device); 255 } 256 } 257 258 private class BondStateChangedHandler implements Handler { 259 public void onReceive(Context context, Intent intent, 260 BluetoothDevice device) { 261 if (device == null) { 262 Log.e(TAG, "ACTION_BOND_STATE_CHANGED with no EXTRA_DEVICE"); 263 return; 264 } 265 int bondState = intent.getIntExtra(BluetoothDevice.EXTRA_BOND_STATE, 266 BluetoothDevice.ERROR); 267 CachedBluetoothDevice cachedDevice = mDeviceManager.findDevice(device); 268 if (cachedDevice == null) { 269 Log.w(TAG, "CachedBluetoothDevice for device " + device + 270 " not found, calling readPairedDevices()."); 271 if (!readPairedDevices()) { 272 Log.e(TAG, "Got bonding state changed for " + device + 273 ", but we have no record of that device."); 274 return; 275 } 276 cachedDevice = mDeviceManager.findDevice(device); 277 if (cachedDevice == null) { 278 Log.e(TAG, "Got bonding state changed for " + device + 279 ", but device not added in cache."); 280 return; 281 } 282 } 283 284 synchronized (mCallbacks) { 285 for (BluetoothCallback callback : mCallbacks) { 286 callback.onDeviceBondStateChanged(cachedDevice, bondState); 287 } 288 } 289 cachedDevice.onBondingStateChanged(bondState); 290 291 if (bondState == BluetoothDevice.BOND_NONE) { 292 int reason = intent.getIntExtra(BluetoothDevice.EXTRA_REASON, 293 BluetoothDevice.ERROR); 294 295 showUnbondMessage(context, cachedDevice.getName(), reason); 296 } 297 } 298 299 /** 300 * Called when we have reached the unbonded state. 301 * 302 * @param reason one of the error reasons from 303 * BluetoothDevice.UNBOND_REASON_* 304 */ 305 private void showUnbondMessage(Context context, String name, int reason) { 306 int errorMsg; 307 308 switch(reason) { 309 case BluetoothDevice.UNBOND_REASON_AUTH_FAILED: 310 errorMsg = R.string.bluetooth_pairing_pin_error_message; 311 break; 312 case BluetoothDevice.UNBOND_REASON_AUTH_REJECTED: 313 errorMsg = R.string.bluetooth_pairing_rejected_error_message; 314 break; 315 case BluetoothDevice.UNBOND_REASON_REMOTE_DEVICE_DOWN: 316 errorMsg = R.string.bluetooth_pairing_device_down_error_message; 317 break; 318 case BluetoothDevice.UNBOND_REASON_DISCOVERY_IN_PROGRESS: 319 case BluetoothDevice.UNBOND_REASON_AUTH_TIMEOUT: 320 case BluetoothDevice.UNBOND_REASON_REPEATED_ATTEMPTS: 321 case BluetoothDevice.UNBOND_REASON_REMOTE_AUTH_CANCELED: 322 errorMsg = R.string.bluetooth_pairing_error_message; 323 break; 324 default: 325 Log.w(TAG, "showUnbondMessage: Not displaying any message for reason: " + reason); 326 return; 327 } 328 Utils.showError(context, name, errorMsg); 329 } 330 } 331 332 private class ClassChangedHandler implements Handler { 333 public void onReceive(Context context, Intent intent, 334 BluetoothDevice device) { 335 mDeviceManager.onBtClassChanged(device); 336 } 337 } 338 339 private class UuidChangedHandler implements Handler { 340 public void onReceive(Context context, Intent intent, 341 BluetoothDevice device) { 342 mDeviceManager.onUuidChanged(device); 343 } 344 } 345 346 private class PairingCancelHandler implements Handler { 347 public void onReceive(Context context, Intent intent, BluetoothDevice device) { 348 if (device == null) { 349 Log.e(TAG, "ACTION_PAIRING_CANCEL with no EXTRA_DEVICE"); 350 return; 351 } 352 CachedBluetoothDevice cachedDevice = mDeviceManager.findDevice(device); 353 if (cachedDevice == null) { 354 Log.e(TAG, "ACTION_PAIRING_CANCEL with no cached device"); 355 return; 356 } 357 int errorMsg = R.string.bluetooth_pairing_error_message; 358 if (context != null && cachedDevice != null) { 359 Utils.showError(context, cachedDevice.getName(), errorMsg); 360 } 361 } 362 } 363 364 private class DockEventHandler implements Handler { 365 public void onReceive(Context context, Intent intent, BluetoothDevice device) { 366 // Remove if unpair device upon undocking 367 int anythingButUnDocked = Intent.EXTRA_DOCK_STATE_UNDOCKED + 1; 368 int state = intent.getIntExtra(Intent.EXTRA_DOCK_STATE, anythingButUnDocked); 369 if (state == Intent.EXTRA_DOCK_STATE_UNDOCKED) { 370 if (device != null && device.getBondState() == BluetoothDevice.BOND_NONE) { 371 CachedBluetoothDevice cachedDevice = mDeviceManager.findDevice(device); 372 if (cachedDevice != null) { 373 cachedDevice.setVisible(false); 374 } 375 } 376 } 377 } 378 } 379 boolean readPairedDevices() { 380 Set<BluetoothDevice> bondedDevices = mLocalAdapter.getBondedDevices(); 381 if (bondedDevices == null) { 382 return false; 383 } 384 385 boolean deviceAdded = false; 386 for (BluetoothDevice device : bondedDevices) { 387 CachedBluetoothDevice cachedDevice = mDeviceManager.findDevice(device); 388 if (cachedDevice == null) { 389 cachedDevice = mDeviceManager.addDevice(mLocalAdapter, mProfileManager, device); 390 dispatchDeviceAdded(cachedDevice); 391 deviceAdded = true; 392 } 393 } 394 395 return deviceAdded; 396 } 397 } 398