Split BluetoothDevice into BluetoothDevice and BluetoothAdapter. BluetoothAdapter: Represents the local BT adapter. Operations on the local adapter (start a scan, etc). BluetoothDevice: Represents a remote BT device. Operations on remote devices (pair, connect, etc).
295 lines
10 KiB
Java
295 lines
10 KiB
Java
/*
|
|
* Copyright (C) 2008 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.
|
|
*/
|
|
|
|
package com.android.settings.bluetooth;
|
|
|
|
import android.bluetooth.BluetoothA2dp;
|
|
import android.bluetooth.BluetoothDevice;
|
|
import android.bluetooth.BluetoothError;
|
|
import android.bluetooth.BluetoothHeadset;
|
|
import android.os.Handler;
|
|
import android.text.TextUtils;
|
|
|
|
import com.android.settings.R;
|
|
|
|
import java.util.HashMap;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import java.util.Set;
|
|
|
|
/**
|
|
* LocalBluetoothProfileManager is an abstract class defining the basic
|
|
* functionality related to a profile.
|
|
*/
|
|
public abstract class LocalBluetoothProfileManager {
|
|
|
|
// TODO: close profiles when we're shutting down
|
|
private static Map<Profile, LocalBluetoothProfileManager> sProfileMap =
|
|
new HashMap<Profile, LocalBluetoothProfileManager>();
|
|
|
|
protected LocalBluetoothManager mLocalManager;
|
|
|
|
public static LocalBluetoothProfileManager getProfileManager(LocalBluetoothManager localManager,
|
|
Profile profile) {
|
|
|
|
LocalBluetoothProfileManager profileManager;
|
|
|
|
synchronized (sProfileMap) {
|
|
profileManager = sProfileMap.get(profile);
|
|
|
|
if (profileManager == null) {
|
|
switch (profile) {
|
|
case A2DP:
|
|
profileManager = new A2dpProfileManager(localManager);
|
|
break;
|
|
|
|
case HEADSET:
|
|
profileManager = new HeadsetProfileManager(localManager);
|
|
break;
|
|
}
|
|
|
|
sProfileMap.put(profile, profileManager);
|
|
}
|
|
}
|
|
|
|
return profileManager;
|
|
}
|
|
|
|
/**
|
|
* Temporary method to fill profiles based on a device's class.
|
|
*
|
|
* NOTE: This list happens to define the connection order. We should put this logic in a more
|
|
* well known place when this method is no longer temporary.
|
|
*
|
|
* @param btClass The class
|
|
* @param profiles The list of profiles to fill
|
|
*/
|
|
public static void fill(int btClass, List<Profile> profiles) {
|
|
profiles.clear();
|
|
|
|
if (BluetoothHeadset.doesClassMatch(btClass)) {
|
|
profiles.add(Profile.HEADSET);
|
|
}
|
|
|
|
if (BluetoothA2dp.doesClassMatchSink(btClass)) {
|
|
profiles.add(Profile.A2DP);
|
|
}
|
|
}
|
|
|
|
protected LocalBluetoothProfileManager(LocalBluetoothManager localManager) {
|
|
mLocalManager = localManager;
|
|
}
|
|
|
|
public abstract int connect(BluetoothDevice device);
|
|
|
|
public abstract int disconnect(BluetoothDevice device);
|
|
|
|
public abstract int getConnectionStatus(BluetoothDevice device);
|
|
|
|
public abstract int getSummary(BluetoothDevice device);
|
|
|
|
public abstract int convertState(int a2dpState);
|
|
|
|
public abstract boolean isPreferred(BluetoothDevice device);
|
|
|
|
public abstract void setPreferred(BluetoothDevice device, boolean preferred);
|
|
|
|
public boolean isConnected(BluetoothDevice device) {
|
|
return SettingsBtStatus.isConnectionStatusConnected(getConnectionStatus(device));
|
|
}
|
|
|
|
// TODO: int instead of enum
|
|
public enum Profile {
|
|
HEADSET(R.string.bluetooth_profile_headset),
|
|
A2DP(R.string.bluetooth_profile_a2dp);
|
|
|
|
public final int localizedString;
|
|
|
|
private Profile(int localizedString) {
|
|
this.localizedString = localizedString;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* A2dpProfileManager is an abstraction for the {@link BluetoothA2dp} service.
|
|
*/
|
|
private static class A2dpProfileManager extends LocalBluetoothProfileManager {
|
|
private BluetoothA2dp mService;
|
|
|
|
public A2dpProfileManager(LocalBluetoothManager localManager) {
|
|
super(localManager);
|
|
mService = new BluetoothA2dp(localManager.getContext());
|
|
}
|
|
|
|
@Override
|
|
public int connect(BluetoothDevice device) {
|
|
Set<BluetoothDevice> sinks = mService.getConnectedSinks();
|
|
if (sinks != null) {
|
|
for (BluetoothDevice sink : sinks) {
|
|
mService.disconnectSink(sink);
|
|
}
|
|
}
|
|
return mService.connectSink(device);
|
|
}
|
|
|
|
@Override
|
|
public int disconnect(BluetoothDevice device) {
|
|
return mService.disconnectSink(device);
|
|
}
|
|
|
|
@Override
|
|
public int getConnectionStatus(BluetoothDevice device) {
|
|
return convertState(mService.getSinkState(device));
|
|
}
|
|
|
|
@Override
|
|
public int getSummary(BluetoothDevice device) {
|
|
int connectionStatus = getConnectionStatus(device);
|
|
|
|
if (SettingsBtStatus.isConnectionStatusConnected(connectionStatus)) {
|
|
return R.string.bluetooth_a2dp_profile_summary_connected;
|
|
} else {
|
|
return SettingsBtStatus.getConnectionStatusSummary(connectionStatus);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean isPreferred(BluetoothDevice device) {
|
|
return mService.getSinkPriority(device) > BluetoothA2dp.PRIORITY_OFF;
|
|
}
|
|
|
|
@Override
|
|
public void setPreferred(BluetoothDevice device, boolean preferred) {
|
|
mService.setSinkPriority(device,
|
|
preferred ? BluetoothA2dp.PRIORITY_AUTO : BluetoothA2dp.PRIORITY_OFF);
|
|
}
|
|
|
|
@Override
|
|
public int convertState(int a2dpState) {
|
|
switch (a2dpState) {
|
|
case BluetoothA2dp.STATE_CONNECTED:
|
|
return SettingsBtStatus.CONNECTION_STATUS_CONNECTED;
|
|
case BluetoothA2dp.STATE_CONNECTING:
|
|
return SettingsBtStatus.CONNECTION_STATUS_CONNECTING;
|
|
case BluetoothA2dp.STATE_DISCONNECTED:
|
|
return SettingsBtStatus.CONNECTION_STATUS_DISCONNECTED;
|
|
case BluetoothA2dp.STATE_DISCONNECTING:
|
|
return SettingsBtStatus.CONNECTION_STATUS_DISCONNECTING;
|
|
case BluetoothA2dp.STATE_PLAYING:
|
|
return SettingsBtStatus.CONNECTION_STATUS_ACTIVE;
|
|
default:
|
|
return SettingsBtStatus.CONNECTION_STATUS_UNKNOWN;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* HeadsetProfileManager is an abstraction for the {@link BluetoothHeadset} service.
|
|
*/
|
|
private static class HeadsetProfileManager extends LocalBluetoothProfileManager
|
|
implements BluetoothHeadset.ServiceListener {
|
|
private BluetoothHeadset mService;
|
|
private Handler mUiHandler = new Handler();
|
|
|
|
public HeadsetProfileManager(LocalBluetoothManager localManager) {
|
|
super(localManager);
|
|
mService = new BluetoothHeadset(localManager.getContext(), this);
|
|
}
|
|
|
|
public void onServiceConnected() {
|
|
// This could be called on a non-UI thread, funnel to UI thread.
|
|
mUiHandler.post(new Runnable() {
|
|
public void run() {
|
|
/*
|
|
* We just bound to the service, so refresh the UI of the
|
|
* headset device.
|
|
*/
|
|
BluetoothDevice device = mService.getCurrentHeadset();
|
|
if (device == null) return;
|
|
mLocalManager.getCachedDeviceManager()
|
|
.onProfileStateChanged(device, Profile.HEADSET,
|
|
BluetoothHeadset.STATE_CONNECTED);
|
|
}
|
|
});
|
|
}
|
|
|
|
public void onServiceDisconnected() {
|
|
}
|
|
|
|
@Override
|
|
public int connect(BluetoothDevice device) {
|
|
// Since connectHeadset fails if already connected to a headset, we
|
|
// disconnect from any headset first
|
|
mService.disconnectHeadset();
|
|
return mService.connectHeadset(device)
|
|
? BluetoothError.SUCCESS : BluetoothError.ERROR;
|
|
}
|
|
|
|
@Override
|
|
public int disconnect(BluetoothDevice device) {
|
|
if (mService.getCurrentHeadset().equals(device)) {
|
|
return mService.disconnectHeadset() ? BluetoothError.SUCCESS : BluetoothError.ERROR;
|
|
} else {
|
|
return BluetoothError.SUCCESS;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public int getConnectionStatus(BluetoothDevice device) {
|
|
BluetoothDevice currentDevice = mService.getCurrentHeadset();
|
|
return currentDevice != null && currentDevice.equals(device)
|
|
? convertState(mService.getState())
|
|
: SettingsBtStatus.CONNECTION_STATUS_DISCONNECTED;
|
|
}
|
|
|
|
@Override
|
|
public int getSummary(BluetoothDevice device) {
|
|
int connectionStatus = getConnectionStatus(device);
|
|
|
|
if (SettingsBtStatus.isConnectionStatusConnected(connectionStatus)) {
|
|
return R.string.bluetooth_headset_profile_summary_connected;
|
|
} else {
|
|
return SettingsBtStatus.getConnectionStatusSummary(connectionStatus);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean isPreferred(BluetoothDevice device) {
|
|
return mService.getPriority(device) > BluetoothHeadset.PRIORITY_OFF;
|
|
}
|
|
|
|
@Override
|
|
public void setPreferred(BluetoothDevice device, boolean preferred) {
|
|
mService.setPriority(device,
|
|
preferred ? BluetoothHeadset.PRIORITY_AUTO : BluetoothHeadset.PRIORITY_OFF);
|
|
}
|
|
|
|
@Override
|
|
public int convertState(int headsetState) {
|
|
switch (headsetState) {
|
|
case BluetoothHeadset.STATE_CONNECTED:
|
|
return SettingsBtStatus.CONNECTION_STATUS_CONNECTED;
|
|
case BluetoothHeadset.STATE_CONNECTING:
|
|
return SettingsBtStatus.CONNECTION_STATUS_CONNECTING;
|
|
case BluetoothHeadset.STATE_DISCONNECTED:
|
|
return SettingsBtStatus.CONNECTION_STATUS_DISCONNECTED;
|
|
default:
|
|
return SettingsBtStatus.CONNECTION_STATUS_UNKNOWN;
|
|
}
|
|
}
|
|
}
|
|
}
|