2017-03-04 15:48:10 -08:00
|
|
|
package org.thoughtcrime.securesms.webrtc.audio;
|
|
|
|
|
|
|
|
import android.bluetooth.BluetoothAdapter;
|
|
|
|
import android.bluetooth.BluetoothClass;
|
|
|
|
import android.bluetooth.BluetoothDevice;
|
|
|
|
import android.bluetooth.BluetoothHeadset;
|
|
|
|
import android.bluetooth.BluetoothProfile;
|
|
|
|
import android.content.BroadcastReceiver;
|
|
|
|
import android.content.Context;
|
|
|
|
import android.content.Intent;
|
|
|
|
import android.content.IntentFilter;
|
|
|
|
import android.media.AudioManager;
|
|
|
|
import android.os.Build;
|
|
|
|
import android.support.annotation.NonNull;
|
|
|
|
import android.support.annotation.Nullable;
|
|
|
|
import android.support.annotation.RequiresApi;
|
2018-08-01 11:09:24 -04:00
|
|
|
import org.thoughtcrime.securesms.logging.Log;
|
2017-03-04 15:48:10 -08:00
|
|
|
|
|
|
|
import org.thoughtcrime.securesms.util.ServiceUtil;
|
|
|
|
|
|
|
|
import java.util.List;
|
|
|
|
|
|
|
|
public class BluetoothStateManager {
|
|
|
|
|
|
|
|
private static final String TAG = BluetoothStateManager.class.getSimpleName();
|
|
|
|
|
|
|
|
private enum ScoConnection {
|
|
|
|
DISCONNECTED,
|
|
|
|
IN_PROGRESS,
|
|
|
|
CONNECTED
|
|
|
|
}
|
|
|
|
|
|
|
|
private final Object LOCK = new Object();
|
|
|
|
|
|
|
|
private final Context context;
|
|
|
|
private final BluetoothAdapter bluetoothAdapter;
|
2017-12-11 09:52:03 -08:00
|
|
|
private BluetoothScoReceiver bluetoothScoReceiver;
|
|
|
|
private BluetoothConnectionReceiver bluetoothConnectionReceiver;
|
2017-03-04 15:48:10 -08:00
|
|
|
private final BluetoothStateListener listener;
|
|
|
|
|
|
|
|
private BluetoothHeadset bluetoothHeadset = null;
|
|
|
|
private ScoConnection scoConnection = ScoConnection.DISCONNECTED;
|
|
|
|
private boolean wantsConnection = false;
|
|
|
|
|
|
|
|
public BluetoothStateManager(@NonNull Context context, @Nullable BluetoothStateListener listener) {
|
|
|
|
this.context = context.getApplicationContext();
|
|
|
|
this.bluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
|
|
|
|
this.bluetoothScoReceiver = new BluetoothScoReceiver();
|
|
|
|
this.bluetoothConnectionReceiver = new BluetoothConnectionReceiver();
|
|
|
|
this.listener = listener;
|
|
|
|
|
2017-03-12 02:23:40 +01:00
|
|
|
if (this.bluetoothAdapter == null)
|
|
|
|
return;
|
|
|
|
|
2017-03-04 15:48:10 -08:00
|
|
|
requestHeadsetProxyProfile();
|
|
|
|
|
2017-12-12 11:16:40 -08:00
|
|
|
this.context.registerReceiver(bluetoothConnectionReceiver, new IntentFilter(BluetoothHeadset.ACTION_CONNECTION_STATE_CHANGED));
|
2017-03-04 15:48:10 -08:00
|
|
|
|
2017-12-12 11:16:40 -08:00
|
|
|
Intent sticky = this.context.registerReceiver(bluetoothScoReceiver, new IntentFilter(getScoChangeIntent()));
|
2017-03-04 15:48:10 -08:00
|
|
|
|
|
|
|
if (sticky != null) {
|
|
|
|
bluetoothScoReceiver.onReceive(context, sticky);
|
|
|
|
}
|
|
|
|
|
|
|
|
handleBluetoothStateChange();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void onDestroy() {
|
2017-12-11 09:52:03 -08:00
|
|
|
if (bluetoothHeadset != null && bluetoothAdapter != null) {
|
2017-03-04 15:48:10 -08:00
|
|
|
this.bluetoothAdapter.closeProfileProxy(BluetoothProfile.HEADSET, bluetoothHeadset);
|
|
|
|
}
|
|
|
|
|
2017-12-11 09:52:03 -08:00
|
|
|
if (bluetoothConnectionReceiver != null) {
|
2017-03-04 15:48:10 -08:00
|
|
|
context.unregisterReceiver(bluetoothConnectionReceiver);
|
2017-12-11 09:52:03 -08:00
|
|
|
bluetoothConnectionReceiver = null;
|
2017-03-04 15:48:10 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (bluetoothScoReceiver != null) {
|
|
|
|
context.unregisterReceiver(bluetoothScoReceiver);
|
2017-12-11 09:52:03 -08:00
|
|
|
bluetoothScoReceiver = null;
|
2017-03-04 15:48:10 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
this.bluetoothHeadset = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setWantsConnection(boolean enabled) {
|
|
|
|
synchronized (LOCK) {
|
|
|
|
AudioManager audioManager = ServiceUtil.getAudioManager(context);
|
|
|
|
|
|
|
|
this.wantsConnection = enabled;
|
|
|
|
|
|
|
|
if (wantsConnection && isBluetoothAvailable() && scoConnection == ScoConnection.DISCONNECTED) {
|
|
|
|
audioManager.startBluetoothSco();
|
|
|
|
scoConnection = ScoConnection.IN_PROGRESS;
|
|
|
|
} else if (!wantsConnection && scoConnection == ScoConnection.CONNECTED) {
|
|
|
|
audioManager.stopBluetoothSco();
|
|
|
|
audioManager.setBluetoothScoOn(false);
|
|
|
|
scoConnection = ScoConnection.DISCONNECTED;
|
|
|
|
} else if (!wantsConnection && scoConnection == ScoConnection.IN_PROGRESS) {
|
|
|
|
audioManager.stopBluetoothSco();
|
|
|
|
scoConnection = ScoConnection.DISCONNECTED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void handleBluetoothStateChange() {
|
|
|
|
if (listener != null) listener.onBluetoothStateChanged(isBluetoothAvailable());
|
|
|
|
}
|
|
|
|
|
|
|
|
private boolean isBluetoothAvailable() {
|
|
|
|
try {
|
|
|
|
synchronized (LOCK) {
|
|
|
|
AudioManager audioManager = ServiceUtil.getAudioManager(context);
|
|
|
|
|
|
|
|
if (bluetoothAdapter == null || !bluetoothAdapter.isEnabled()) return false;
|
|
|
|
if (!audioManager.isBluetoothScoAvailableOffCall()) return false;
|
|
|
|
|
2017-12-11 09:52:03 -08:00
|
|
|
return bluetoothHeadset != null && !bluetoothHeadset.getConnectedDevices().isEmpty();
|
2017-03-04 15:48:10 -08:00
|
|
|
}
|
|
|
|
} catch (Exception e) {
|
|
|
|
Log.w(TAG, e);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private String getScoChangeIntent() {
|
|
|
|
if (Build.VERSION.SDK_INT >= 14) {
|
|
|
|
return AudioManager.ACTION_SCO_AUDIO_STATE_UPDATED;
|
|
|
|
} else {
|
|
|
|
return AudioManager.ACTION_SCO_AUDIO_STATE_CHANGED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private void requestHeadsetProxyProfile() {
|
2017-12-11 09:52:03 -08:00
|
|
|
this.bluetoothAdapter.getProfileProxy(context, new BluetoothProfile.ServiceListener() {
|
|
|
|
@RequiresApi(api = Build.VERSION_CODES.HONEYCOMB)
|
|
|
|
@Override
|
|
|
|
public void onServiceConnected(int profile, BluetoothProfile proxy) {
|
|
|
|
if (profile == BluetoothProfile.HEADSET) {
|
|
|
|
synchronized (LOCK) {
|
|
|
|
bluetoothHeadset = (BluetoothHeadset) proxy;
|
|
|
|
}
|
2017-03-04 15:48:10 -08:00
|
|
|
|
2017-12-11 09:52:03 -08:00
|
|
|
Intent sticky = context.registerReceiver(null, new IntentFilter(getScoChangeIntent()));
|
|
|
|
bluetoothScoReceiver.onReceive(context, sticky);
|
2017-03-04 15:48:10 -08:00
|
|
|
|
2017-12-11 09:52:03 -08:00
|
|
|
synchronized (LOCK) {
|
|
|
|
if (wantsConnection && isBluetoothAvailable() && scoConnection == ScoConnection.DISCONNECTED) {
|
|
|
|
AudioManager audioManager = ServiceUtil.getAudioManager(context);
|
|
|
|
audioManager.startBluetoothSco();
|
|
|
|
scoConnection = ScoConnection.IN_PROGRESS;
|
2017-03-04 15:48:10 -08:00
|
|
|
}
|
|
|
|
}
|
2017-12-11 09:52:03 -08:00
|
|
|
|
|
|
|
handleBluetoothStateChange();
|
2017-03-04 15:48:10 -08:00
|
|
|
}
|
2017-12-11 09:52:03 -08:00
|
|
|
}
|
2017-03-04 15:48:10 -08:00
|
|
|
|
2017-12-11 09:52:03 -08:00
|
|
|
@Override
|
|
|
|
public void onServiceDisconnected(int profile) {
|
2018-08-02 09:25:33 -04:00
|
|
|
Log.i(TAG, "onServiceDisconnected");
|
2017-12-11 09:52:03 -08:00
|
|
|
if (profile == BluetoothProfile.HEADSET) {
|
|
|
|
bluetoothHeadset = null;
|
|
|
|
handleBluetoothStateChange();
|
2017-03-04 15:48:10 -08:00
|
|
|
}
|
2017-12-11 09:52:03 -08:00
|
|
|
}
|
|
|
|
}, BluetoothProfile.HEADSET);
|
2017-03-04 15:48:10 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
private class BluetoothScoReceiver extends BroadcastReceiver {
|
|
|
|
@Override
|
|
|
|
public void onReceive(Context context, Intent intent) {
|
|
|
|
if (intent == null) return;
|
2018-08-02 09:25:33 -04:00
|
|
|
Log.i(TAG, "onReceive");
|
2017-03-04 15:48:10 -08:00
|
|
|
|
|
|
|
synchronized (LOCK) {
|
|
|
|
if (getScoChangeIntent().equals(intent.getAction())) {
|
|
|
|
int status = intent.getIntExtra(AudioManager.EXTRA_SCO_AUDIO_STATE, AudioManager.SCO_AUDIO_STATE_ERROR);
|
|
|
|
|
|
|
|
if (status == AudioManager.SCO_AUDIO_STATE_CONNECTED) {
|
2017-12-11 09:52:03 -08:00
|
|
|
if (bluetoothHeadset != null) {
|
2017-03-04 15:48:10 -08:00
|
|
|
List<BluetoothDevice> devices = bluetoothHeadset.getConnectedDevices();
|
|
|
|
|
|
|
|
for (BluetoothDevice device : devices) {
|
|
|
|
if (bluetoothHeadset.isAudioConnected(device)) {
|
|
|
|
int deviceClass = device.getBluetoothClass().getDeviceClass();
|
|
|
|
|
|
|
|
if (deviceClass == BluetoothClass.Device.AUDIO_VIDEO_HANDSFREE ||
|
|
|
|
deviceClass == BluetoothClass.Device.AUDIO_VIDEO_CAR_AUDIO ||
|
|
|
|
deviceClass == BluetoothClass.Device.AUDIO_VIDEO_WEARABLE_HEADSET)
|
|
|
|
{
|
|
|
|
scoConnection = ScoConnection.CONNECTED;
|
|
|
|
|
|
|
|
if (wantsConnection) {
|
|
|
|
AudioManager audioManager = ServiceUtil.getAudioManager(context);
|
|
|
|
audioManager.setBluetoothScoOn(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
handleBluetoothStateChange();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private class BluetoothConnectionReceiver extends BroadcastReceiver {
|
|
|
|
@Override
|
|
|
|
public void onReceive(Context context, Intent intent) {
|
2018-08-02 09:25:33 -04:00
|
|
|
Log.i(TAG, "onReceive");
|
2017-03-04 15:48:10 -08:00
|
|
|
handleBluetoothStateChange();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public interface BluetoothStateListener {
|
|
|
|
public void onBluetoothStateChanged(boolean isAvailable);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|