Merge "[Audiosharing] Show/hide audio sharing settings based on BT state." into main

This commit is contained in:
Yiyi Shen
2023-11-29 02:32:42 +00:00
committed by Android (Google) Code Review
8 changed files with 205 additions and 63 deletions

View File

@@ -16,8 +16,12 @@
package com.android.settings.connecteddevice.audiosharing; package com.android.settings.connecteddevice.audiosharing;
import android.bluetooth.BluetoothAdapter;
import android.content.Context; import android.content.Context;
import androidx.annotation.NonNull;
import androidx.lifecycle.DefaultLifecycleObserver;
import androidx.lifecycle.LifecycleOwner;
import androidx.preference.Preference; import androidx.preference.Preference;
import androidx.preference.PreferenceScreen; import androidx.preference.PreferenceScreen;
@@ -26,14 +30,18 @@ import com.android.settings.core.BasePreferenceController;
import com.android.settings.flags.Flags; import com.android.settings.flags.Flags;
import com.android.settingslib.bluetooth.LocalBluetoothLeBroadcast; import com.android.settingslib.bluetooth.LocalBluetoothLeBroadcast;
import com.android.settingslib.bluetooth.LocalBluetoothManager; import com.android.settingslib.bluetooth.LocalBluetoothManager;
import com.android.settingslib.utils.ThreadUtils;
public abstract class AudioSharingBasePreferenceController extends BasePreferenceController { public abstract class AudioSharingBasePreferenceController extends BasePreferenceController
implements DefaultLifecycleObserver {
private final BluetoothAdapter mBluetoothAdapter;
private final LocalBluetoothManager mBtManager; private final LocalBluetoothManager mBtManager;
protected final LocalBluetoothLeBroadcast mBroadcast; protected final LocalBluetoothLeBroadcast mBroadcast;
protected Preference mPreference; protected Preference mPreference;
public AudioSharingBasePreferenceController(Context context, String preferenceKey) { public AudioSharingBasePreferenceController(Context context, String preferenceKey) {
super(context, preferenceKey); super(context, preferenceKey);
mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
mBtManager = Utils.getLocalBtManager(context); mBtManager = Utils.getLocalBtManager(context);
mBroadcast = mBroadcast =
mBtManager == null mBtManager == null
@@ -43,28 +51,40 @@ public abstract class AudioSharingBasePreferenceController extends BasePreferenc
@Override @Override
public int getAvailabilityStatus() { public int getAvailabilityStatus() {
return mBtManager != null && Flags.enableLeAudioSharing() return Flags.enableLeAudioSharing() ? AVAILABLE : UNSUPPORTED_ON_DEVICE;
? AVAILABLE
: UNSUPPORTED_ON_DEVICE;
} }
@Override @Override
public void displayPreference(PreferenceScreen screen) { public void displayPreference(PreferenceScreen screen) {
super.displayPreference(screen); super.displayPreference(screen);
mPreference = screen.findPreference(getPreferenceKey()); mPreference = screen.findPreference(getPreferenceKey());
updateVisibility(isBroadcasting());
} }
/** @Override
* Update the visibility of the preference. public void onStart(@NonNull LifecycleOwner owner) {
* if (isAvailable()) {
* @param isVisible the latest visibility state for the preference. updateVisibility();
*/ }
public void updateVisibility(boolean isVisible) { }
mPreference.setVisible(isVisible);
/** Update the visibility of the preference. */
protected void updateVisibility() {
if (mPreference != null) {
var unused =
ThreadUtils.postOnBackgroundThread(
() -> {
boolean isVisible = isBroadcasting() && isBluetoothStateOn();
ThreadUtils.postOnMainThread(
() -> mPreference.setVisible(isVisible));
});
}
} }
protected boolean isBroadcasting() { protected boolean isBroadcasting() {
return mBroadcast != null && mBroadcast.isEnabled(null); return mBroadcast != null && mBroadcast.isEnabled(null);
} }
protected boolean isBluetoothStateOn() {
return mBluetoothAdapter != null && mBluetoothAdapter.isEnabled();
}
} }

View File

@@ -93,14 +93,14 @@ public class AudioSharingDashboardFragment extends DashboardFragment
} }
@Override @Override
public void onSwitchBarChanged(boolean newState) { public void onSwitchBarChanged() {
updateVisibilityForAttachedPreferences(newState); updateVisibilityForAttachedPreferences();
} }
private void updateVisibilityForAttachedPreferences(boolean isVisible) { private void updateVisibilityForAttachedPreferences() {
mAudioSharingDeviceVolumeGroupController.updateVisibility(isVisible); mAudioSharingDeviceVolumeGroupController.updateVisibility();
mCallsAndAlarmsPreferenceController.updateVisibility(isVisible); mCallsAndAlarmsPreferenceController.updateVisibility();
mAudioSharingNamePreferenceController.updateVisibility(isVisible); mAudioSharingNamePreferenceController.updateVisibility();
mAudioStreamsCategoryController.updateVisibility(isVisible); mAudioStreamsCategoryController.updateVisibility();
} }
} }

View File

@@ -27,7 +27,6 @@ import android.util.Log;
import androidx.annotation.NonNull; import androidx.annotation.NonNull;
import androidx.fragment.app.FragmentManager; import androidx.fragment.app.FragmentManager;
import androidx.lifecycle.DefaultLifecycleObserver;
import androidx.lifecycle.LifecycleOwner; import androidx.lifecycle.LifecycleOwner;
import androidx.preference.Preference; import androidx.preference.Preference;
import androidx.preference.PreferenceGroup; import androidx.preference.PreferenceGroup;
@@ -48,7 +47,7 @@ import java.util.concurrent.Executor;
import java.util.concurrent.Executors; import java.util.concurrent.Executors;
public class AudioSharingDeviceVolumeGroupController extends AudioSharingBasePreferenceController public class AudioSharingDeviceVolumeGroupController extends AudioSharingBasePreferenceController
implements DefaultLifecycleObserver, DevicePreferenceCallback { implements DevicePreferenceCallback {
private static final String TAG = "AudioSharingDeviceVolumeGroupController"; private static final String TAG = "AudioSharingDeviceVolumeGroupController";
private static final String KEY = "audio_sharing_device_volume_group"; private static final String KEY = "audio_sharing_device_volume_group";
@@ -162,6 +161,7 @@ public class AudioSharingDeviceVolumeGroupController extends AudioSharingBasePre
@Override @Override
public void onStart(@NonNull LifecycleOwner owner) { public void onStart(@NonNull LifecycleOwner owner) {
super.onStart(owner);
if (mAssistant == null) { if (mAssistant == null) {
Log.d(TAG, "onStart() Broadcast or assistant is not supported on this device"); Log.d(TAG, "onStart() Broadcast or assistant is not supported on this device");
return; return;
@@ -176,6 +176,7 @@ public class AudioSharingDeviceVolumeGroupController extends AudioSharingBasePre
@Override @Override
public void onStop(@NonNull LifecycleOwner owner) { public void onStop(@NonNull LifecycleOwner owner) {
super.onStop(owner);
if (mAssistant == null) { if (mAssistant == null) {
Log.d(TAG, "onStop() Broadcast or assistant is not supported on this device"); Log.d(TAG, "onStop() Broadcast or assistant is not supported on this device");
return; return;
@@ -233,10 +234,12 @@ public class AudioSharingDeviceVolumeGroupController extends AudioSharingBasePre
} }
@Override @Override
public void updateVisibility(boolean isVisible) { public void updateVisibility() {
super.updateVisibility(isVisible);
if (mPreferenceGroup != null) { if (mPreferenceGroup != null) {
mPreferenceGroup.setVisible(mPreferenceGroup.getPreferenceCount() > 0); mPreferenceGroup.setVisible(false);
if (mPreferenceGroup.getPreferenceCount() > 0) {
super.updateVisibility();
}
} }
} }

View File

@@ -19,16 +19,13 @@ package com.android.settings.connecteddevice.audiosharing;
import android.content.Context; import android.content.Context;
import androidx.annotation.NonNull; import androidx.annotation.NonNull;
import androidx.lifecycle.DefaultLifecycleObserver;
import androidx.lifecycle.LifecycleOwner; import androidx.lifecycle.LifecycleOwner;
import androidx.preference.Preference; import androidx.preference.Preference;
import com.android.settings.widget.ValidatedEditTextPreference; import com.android.settings.widget.ValidatedEditTextPreference;
public class AudioSharingNamePreferenceController extends AudioSharingBasePreferenceController public class AudioSharingNamePreferenceController extends AudioSharingBasePreferenceController
implements ValidatedEditTextPreference.Validator, implements ValidatedEditTextPreference.Validator, Preference.OnPreferenceChangeListener {
Preference.OnPreferenceChangeListener,
DefaultLifecycleObserver {
private static final String TAG = "AudioSharingNamePreferenceController"; private static final String TAG = "AudioSharingNamePreferenceController";
@@ -59,11 +56,13 @@ public class AudioSharingNamePreferenceController extends AudioSharingBasePrefer
@Override @Override
public void onStart(@NonNull LifecycleOwner owner) { public void onStart(@NonNull LifecycleOwner owner) {
super.onStart(owner);
// TODO // TODO
} }
@Override @Override
public void onStop(@NonNull LifecycleOwner owner) { public void onStop(@NonNull LifecycleOwner owner) {
super.onStop(owner);
// TODO // TODO
} }
} }

View File

@@ -31,6 +31,7 @@ import android.widget.CompoundButton;
import android.widget.CompoundButton.OnCheckedChangeListener; import android.widget.CompoundButton.OnCheckedChangeListener;
import androidx.annotation.NonNull; import androidx.annotation.NonNull;
import androidx.annotation.VisibleForTesting;
import androidx.lifecycle.DefaultLifecycleObserver; import androidx.lifecycle.DefaultLifecycleObserver;
import androidx.lifecycle.LifecycleOwner; import androidx.lifecycle.LifecycleOwner;
@@ -58,19 +59,20 @@ public class AudioSharingSwitchBarController extends BasePreferenceController
private static final String PREF_KEY = "audio_sharing_main_switch"; private static final String PREF_KEY = "audio_sharing_main_switch";
interface OnSwitchBarChangedListener { interface OnSwitchBarChangedListener {
void onSwitchBarChanged(boolean newState); void onSwitchBarChanged();
} }
private final SettingsMainSwitchBar mSwitchBar; private final SettingsMainSwitchBar mSwitchBar;
private final BluetoothAdapter mBluetoothAdapter; private final BluetoothAdapter mBluetoothAdapter;
private final IntentFilter mIntentFilter;
private final LocalBluetoothManager mBtManager; private final LocalBluetoothManager mBtManager;
private final LocalBluetoothLeBroadcast mBroadcast; private final LocalBluetoothLeBroadcast mBroadcast;
private final LocalBluetoothLeBroadcastAssistant mAssistant; private final LocalBluetoothLeBroadcastAssistant mAssistant;
private final Executor mExecutor; private final Executor mExecutor;
private final OnSwitchBarChangedListener mListener; private final OnSwitchBarChangedListener mListener;
private DashboardFragment mFragment; private DashboardFragment mFragment;
@VisibleForTesting IntentFilter mIntentFilter;
@VisibleForTesting
BroadcastReceiver mReceiver = BroadcastReceiver mReceiver =
new BroadcastReceiver() { new BroadcastReceiver() {
@Override @Override
@@ -80,6 +82,7 @@ public class AudioSharingSwitchBarController extends BasePreferenceController
intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, BluetoothDevice.ERROR); intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, BluetoothDevice.ERROR);
mSwitchBar.setChecked(isBroadcasting()); mSwitchBar.setChecked(isBroadcasting());
mSwitchBar.setEnabled(adapterState == BluetoothAdapter.STATE_ON); mSwitchBar.setEnabled(adapterState == BluetoothAdapter.STATE_ON);
mListener.onSwitchBarChanged();
} }
}; };
@@ -346,14 +349,18 @@ public class AudioSharingSwitchBarController extends BasePreferenceController
} }
private void updateSwitch() { private void updateSwitch() {
ThreadUtils.postOnMainThread( var unused =
ThreadUtils.postOnBackgroundThread(
() -> { () -> {
boolean isBroadcasting = isBroadcasting(); boolean isBroadcasting = isBroadcasting();
ThreadUtils.postOnMainThread(
() -> {
if (mSwitchBar.isChecked() != isBroadcasting) { if (mSwitchBar.isChecked() != isBroadcasting) {
mSwitchBar.setChecked(isBroadcasting); mSwitchBar.setChecked(isBroadcasting);
} }
mSwitchBar.setEnabled(true); mSwitchBar.setEnabled(true);
mListener.onSwitchBarChanged(isBroadcasting); mListener.onSwitchBarChanged();
});
}); });
} }

View File

@@ -22,7 +22,6 @@ import android.content.Context;
import android.util.Log; import android.util.Log;
import androidx.annotation.NonNull; import androidx.annotation.NonNull;
import androidx.lifecycle.DefaultLifecycleObserver;
import androidx.lifecycle.LifecycleOwner; import androidx.lifecycle.LifecycleOwner;
import androidx.preference.PreferenceScreen; import androidx.preference.PreferenceScreen;
@@ -31,6 +30,7 @@ import com.android.settings.dashboard.DashboardFragment;
import com.android.settingslib.bluetooth.BluetoothCallback; import com.android.settingslib.bluetooth.BluetoothCallback;
import com.android.settingslib.bluetooth.CachedBluetoothDevice; import com.android.settingslib.bluetooth.CachedBluetoothDevice;
import com.android.settingslib.bluetooth.LocalBluetoothManager; import com.android.settingslib.bluetooth.LocalBluetoothManager;
import com.android.settingslib.utils.ThreadUtils;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.HashMap; import java.util.HashMap;
@@ -39,7 +39,7 @@ import java.util.Map;
/** PreferenceController to control the dialog to choose the active device for calls and alarms */ /** PreferenceController to control the dialog to choose the active device for calls and alarms */
public class CallsAndAlarmsPreferenceController extends AudioSharingBasePreferenceController public class CallsAndAlarmsPreferenceController extends AudioSharingBasePreferenceController
implements BluetoothCallback, DefaultLifecycleObserver { implements BluetoothCallback {
private static final String TAG = "CallsAndAlarmsPreferenceController"; private static final String TAG = "CallsAndAlarmsPreferenceController";
private static final String PREF_KEY = "calls_and_alarms"; private static final String PREF_KEY = "calls_and_alarms";
@@ -86,6 +86,7 @@ public class CallsAndAlarmsPreferenceController extends AudioSharingBasePreferen
@Override @Override
public void onStart(@NonNull LifecycleOwner owner) { public void onStart(@NonNull LifecycleOwner owner) {
super.onStart(owner);
if (mLocalBtManager != null) { if (mLocalBtManager != null) {
mLocalBtManager.getEventManager().registerCallback(this); mLocalBtManager.getEventManager().registerCallback(this);
} }
@@ -93,25 +94,46 @@ public class CallsAndAlarmsPreferenceController extends AudioSharingBasePreferen
@Override @Override
public void onStop(@NonNull LifecycleOwner owner) { public void onStop(@NonNull LifecycleOwner owner) {
super.onStop(owner);
if (mLocalBtManager != null) { if (mLocalBtManager != null) {
mLocalBtManager.getEventManager().unregisterCallback(this); mLocalBtManager.getEventManager().unregisterCallback(this);
} }
} }
@Override @Override
public void updateVisibility(boolean isVisible) { public void updateVisibility() {
super.updateVisibility(isVisible); if (mPreference == null) return;
if (isVisible && mPreference != null) { var unused =
ThreadUtils.postOnBackgroundThread(
() -> {
boolean isVisible = isBroadcasting() && isBluetoothStateOn();
if (!isVisible) {
ThreadUtils.postOnMainThread(() -> mPreference.setVisible(false));
} else {
updateDeviceItemsInSharingSession(); updateDeviceItemsInSharingSession();
// mDeviceItemsInSharingSession is ordered. The active device is the first place if // mDeviceItemsInSharingSession is ordered. The active device is the
// exits. // first
// place if exits.
if (!mDeviceItemsInSharingSession.isEmpty() if (!mDeviceItemsInSharingSession.isEmpty()
&& mDeviceItemsInSharingSession.get(0).isActive()) { && mDeviceItemsInSharingSession.get(0).isActive()) {
mPreference.setSummary(mDeviceItemsInSharingSession.get(0).getName()); ThreadUtils.postOnMainThread(
() -> {
mPreference.setVisible(true);
mPreference.setSummary(
mDeviceItemsInSharingSession
.get(0)
.getName());
});
} else { } else {
mPreference.setSummary(""); ThreadUtils.postOnMainThread(
() -> {
mPreference.setVisible(true);
mPreference.setSummary(
"No active device in sharing");
});
} }
} }
});
} }
@Override @Override

View File

@@ -22,7 +22,6 @@ import android.content.Context;
import android.util.Log; import android.util.Log;
import androidx.annotation.NonNull; import androidx.annotation.NonNull;
import androidx.lifecycle.DefaultLifecycleObserver;
import androidx.lifecycle.LifecycleOwner; import androidx.lifecycle.LifecycleOwner;
import com.android.settings.bluetooth.Utils; import com.android.settings.bluetooth.Utils;
@@ -38,8 +37,7 @@ import com.android.settingslib.utils.ThreadUtils;
import java.util.concurrent.Executor; import java.util.concurrent.Executor;
import java.util.concurrent.Executors; import java.util.concurrent.Executors;
public class AudioStreamsCategoryController extends AudioSharingBasePreferenceController public class AudioStreamsCategoryController extends AudioSharingBasePreferenceController {
implements DefaultLifecycleObserver {
private static final String TAG = "AudioStreamsCategoryController"; private static final String TAG = "AudioStreamsCategoryController";
private static final boolean DEBUG = BluetoothUtils.D; private static final boolean DEBUG = BluetoothUtils.D;
private final LocalBluetoothManager mLocalBtManager; private final LocalBluetoothManager mLocalBtManager;
@@ -50,7 +48,7 @@ public class AudioStreamsCategoryController extends AudioSharingBasePreferenceCo
public void onActiveDeviceChanged( public void onActiveDeviceChanged(
@Nullable CachedBluetoothDevice activeDevice, int bluetoothProfile) { @Nullable CachedBluetoothDevice activeDevice, int bluetoothProfile) {
if (bluetoothProfile == BluetoothProfile.LE_AUDIO) { if (bluetoothProfile == BluetoothProfile.LE_AUDIO) {
updateVisibility(isBroadcasting()); updateVisibility();
} }
} }
}; };
@@ -63,14 +61,15 @@ public class AudioStreamsCategoryController extends AudioSharingBasePreferenceCo
@Override @Override
public void onStart(@NonNull LifecycleOwner owner) { public void onStart(@NonNull LifecycleOwner owner) {
super.onStart(owner);
if (mLocalBtManager != null) { if (mLocalBtManager != null) {
mLocalBtManager.getEventManager().registerCallback(mBluetoothCallback); mLocalBtManager.getEventManager().registerCallback(mBluetoothCallback);
} }
updateVisibility(isBroadcasting());
} }
@Override @Override
public void onStop(@NonNull LifecycleOwner owner) { public void onStop(@NonNull LifecycleOwner owner) {
super.onStop(owner);
if (mLocalBtManager != null) { if (mLocalBtManager != null) {
mLocalBtManager.getEventManager().unregisterCallback(mBluetoothCallback); mLocalBtManager.getEventManager().unregisterCallback(mBluetoothCallback);
} }
@@ -84,21 +83,28 @@ public class AudioStreamsCategoryController extends AudioSharingBasePreferenceCo
} }
@Override @Override
public void updateVisibility(boolean isBroadcasting) { public void updateVisibility() {
if (mPreference == null) return;
mExecutor.execute( mExecutor.execute(
() -> { () -> {
boolean hasActiveLe = boolean hasActiveLe =
AudioSharingUtils.getActiveSinkOnAssistant(mLocalBtManager).isPresent(); AudioSharingUtils.getActiveSinkOnAssistant(mLocalBtManager).isPresent();
boolean isBroadcasting = isBroadcasting();
boolean isBluetoothOn = isBluetoothStateOn();
if (DEBUG) { if (DEBUG) {
Log.d( Log.d(
TAG, TAG,
"updateVisibility() isBroadcasting : " "updateVisibility() isBroadcasting : "
+ isBroadcasting + isBroadcasting
+ " hasActiveLe : " + " hasActiveLe : "
+ hasActiveLe); + hasActiveLe
+ " is BT on : "
+ isBluetoothOn);
} }
ThreadUtils.postOnMainThread( ThreadUtils.postOnMainThread(
() -> super.updateVisibility(hasActiveLe && !isBroadcasting)); () ->
mPreference.setVisible(
isBluetoothOn && hasActiveLe && !isBroadcasting));
}); });
} }
} }

View File

@@ -0,0 +1,85 @@
/*
* Copyright (C) 2023 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.connecteddevice.audiosharing;
import static com.google.common.truth.Truth.assertThat;
import static org.mockito.Mockito.verify;
import static org.robolectric.Shadows.shadowOf;
import android.bluetooth.BluetoothAdapter;
import android.content.Context;
import android.content.Intent;
import android.os.Looper;
import android.platform.test.annotations.RequiresFlagsEnabled;
import android.platform.test.flag.junit.CheckFlagsRule;
import android.platform.test.flag.junit.DeviceFlagsValueProvider;
import android.widget.Switch;
import androidx.test.core.app.ApplicationProvider;
import com.android.settings.flags.Flags;
import com.android.settings.widget.SettingsMainSwitchBar;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.Mock;
import org.mockito.Spy;
import org.mockito.junit.MockitoJUnit;
import org.mockito.junit.MockitoRule;
import org.robolectric.RobolectricTestRunner;
@RunWith(RobolectricTestRunner.class)
public class AudioSharingSwitchBarControllerTest {
@Rule public final MockitoRule mMockitoRule = MockitoJUnit.rule();
@Rule
public final CheckFlagsRule mCheckFlagsRule = DeviceFlagsValueProvider.createCheckFlagsRule();
@Spy Context mContext = ApplicationProvider.getApplicationContext();
@Mock private Switch mSwitch;
private SettingsMainSwitchBar mSwitchBar;
private AudioSharingSwitchBarController mController;
private AudioSharingSwitchBarController.OnSwitchBarChangedListener mListener;
private boolean mOnSwitchBarChanged;
@Before
public void setUp() {
mSwitchBar = new SettingsMainSwitchBar(mContext);
mOnSwitchBarChanged = false;
mListener = () -> mOnSwitchBarChanged = true;
mController = new AudioSharingSwitchBarController(mContext, mSwitchBar, mListener);
}
@Test
@RequiresFlagsEnabled(Flags.FLAG_ENABLE_LE_AUDIO_SHARING)
public void bluetoothOff_switchDisabled() {
mContext.registerReceiver(
mController.mReceiver,
mController.mIntentFilter,
Context.RECEIVER_EXPORTED_UNAUDITED);
Intent intent = new Intent(BluetoothAdapter.ACTION_STATE_CHANGED);
intent.putExtra(BluetoothAdapter.EXTRA_STATE, BluetoothAdapter.STATE_OFF);
mContext.sendBroadcast(intent);
shadowOf(Looper.getMainLooper()).idle();
verify(mSwitch).setEnabled(false);
assertThat(mOnSwitchBarChanged).isTrue();
}
}