| /* |
| * Copyright (C) 2022 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 static android.media.Spatializer.SPATIALIZER_IMMERSIVE_LEVEL_NONE; |
| |
| import android.content.Context; |
| import android.media.AudioDeviceAttributes; |
| import android.media.AudioDeviceInfo; |
| import android.media.Spatializer; |
| import android.text.TextUtils; |
| import android.util.Log; |
| |
| import androidx.annotation.VisibleForTesting; |
| import androidx.preference.Preference; |
| import androidx.preference.PreferenceCategory; |
| import androidx.preference.PreferenceFragmentCompat; |
| import androidx.preference.PreferenceScreen; |
| import androidx.preference.SwitchPreferenceCompat; |
| import androidx.preference.TwoStatePreference; |
| |
| import com.android.settings.R; |
| import com.android.settings.overlay.FeatureFactory; |
| import com.android.settingslib.bluetooth.CachedBluetoothDevice; |
| import com.android.settingslib.core.lifecycle.Lifecycle; |
| import com.android.settingslib.utils.ThreadUtils; |
| |
| import java.util.concurrent.atomic.AtomicBoolean; |
| |
| /** |
| * The controller of the Spatial audio setting in the bluetooth detail settings. |
| */ |
| public class BluetoothDetailsSpatialAudioController extends BluetoothDetailsController |
| implements Preference.OnPreferenceClickListener { |
| |
| private static final String TAG = "BluetoothSpatialAudioController"; |
| private static final String KEY_SPATIAL_AUDIO_GROUP = "spatial_audio_group"; |
| private static final String KEY_SPATIAL_AUDIO = "spatial_audio"; |
| private static final String KEY_HEAD_TRACKING = "head_tracking"; |
| |
| private final Spatializer mSpatializer; |
| |
| @VisibleForTesting |
| PreferenceCategory mProfilesContainer; |
| @VisibleForTesting |
| AudioDeviceAttributes mAudioDevice = null; |
| |
| AtomicBoolean mHasHeadTracker = new AtomicBoolean(false); |
| |
| public BluetoothDetailsSpatialAudioController( |
| Context context, |
| PreferenceFragmentCompat fragment, |
| CachedBluetoothDevice device, |
| Lifecycle lifecycle) { |
| super(context, fragment, device, lifecycle); |
| mSpatializer = FeatureFactory.getFeatureFactory().getBluetoothFeatureProvider() |
| .getSpatializer(context); |
| } |
| |
| @Override |
| public boolean isAvailable() { |
| return mSpatializer.getImmersiveAudioLevel() != SPATIALIZER_IMMERSIVE_LEVEL_NONE; |
| } |
| |
| @Override |
| public boolean onPreferenceClick(Preference preference) { |
| TwoStatePreference switchPreference = (TwoStatePreference) preference; |
| String key = switchPreference.getKey(); |
| if (TextUtils.equals(key, KEY_SPATIAL_AUDIO)) { |
| updateSpatializerEnabled(switchPreference.isChecked()); |
| ThreadUtils.postOnBackgroundThread( |
| () -> { |
| mHasHeadTracker.set( |
| mAudioDevice != null && mSpatializer.hasHeadTracker(mAudioDevice)); |
| mContext.getMainExecutor() |
| .execute(() -> refreshSpatialAudioEnabled(switchPreference)); |
| }); |
| return true; |
| } else if (TextUtils.equals(key, KEY_HEAD_TRACKING)) { |
| updateSpatializerHeadTracking(switchPreference.isChecked()); |
| return true; |
| } else { |
| Log.w(TAG, "invalid key name."); |
| return false; |
| } |
| } |
| |
| private void updateSpatializerEnabled(boolean enabled) { |
| if (mAudioDevice == null) { |
| Log.w(TAG, "cannot update spatializer enabled for null audio device."); |
| return; |
| } |
| if (enabled) { |
| mSpatializer.addCompatibleAudioDevice(mAudioDevice); |
| } else { |
| mSpatializer.removeCompatibleAudioDevice(mAudioDevice); |
| } |
| } |
| |
| private void updateSpatializerHeadTracking(boolean enabled) { |
| if (mAudioDevice == null) { |
| Log.w(TAG, "cannot update spatializer head tracking for null audio device."); |
| return; |
| } |
| mSpatializer.setHeadTrackerEnabled(enabled, mAudioDevice); |
| } |
| |
| @Override |
| public String getPreferenceKey() { |
| return KEY_SPATIAL_AUDIO_GROUP; |
| } |
| |
| @Override |
| protected void init(PreferenceScreen screen) { |
| mProfilesContainer = screen.findPreference(getPreferenceKey()); |
| refresh(); |
| } |
| |
| @Override |
| protected void refresh() { |
| if (mAudioDevice == null) { |
| getAvailableDevice(); |
| } |
| ThreadUtils.postOnBackgroundThread( |
| () -> { |
| mHasHeadTracker.set( |
| mAudioDevice != null && mSpatializer.hasHeadTracker(mAudioDevice)); |
| mContext.getMainExecutor().execute(this::refreshUi); |
| }); |
| } |
| |
| private void refreshUi() { |
| TwoStatePreference spatialAudioPref = mProfilesContainer.findPreference(KEY_SPATIAL_AUDIO); |
| if (spatialAudioPref == null && mAudioDevice != null) { |
| spatialAudioPref = createSpatialAudioPreference(mProfilesContainer.getContext()); |
| mProfilesContainer.addPreference(spatialAudioPref); |
| } else if (mAudioDevice == null || !mSpatializer.isAvailableForDevice(mAudioDevice)) { |
| if (spatialAudioPref != null) { |
| mProfilesContainer.removePreference(spatialAudioPref); |
| } |
| final TwoStatePreference headTrackingPref = |
| mProfilesContainer.findPreference(KEY_HEAD_TRACKING); |
| if (headTrackingPref != null) { |
| mProfilesContainer.removePreference(headTrackingPref); |
| } |
| mAudioDevice = null; |
| return; |
| } |
| |
| refreshSpatialAudioEnabled(spatialAudioPref); |
| } |
| |
| private void refreshSpatialAudioEnabled( |
| TwoStatePreference spatialAudioPref) { |
| boolean isSpatialAudioOn = mSpatializer.getCompatibleAudioDevices().contains(mAudioDevice); |
| Log.d(TAG, "refresh() isSpatialAudioOn : " + isSpatialAudioOn); |
| spatialAudioPref.setChecked(isSpatialAudioOn); |
| |
| TwoStatePreference headTrackingPref = mProfilesContainer.findPreference(KEY_HEAD_TRACKING); |
| if (headTrackingPref == null) { |
| headTrackingPref = createHeadTrackingPreference(mProfilesContainer.getContext()); |
| mProfilesContainer.addPreference(headTrackingPref); |
| } |
| refreshHeadTracking(spatialAudioPref, headTrackingPref); |
| } |
| |
| private void refreshHeadTracking(TwoStatePreference spatialAudioPref, |
| TwoStatePreference headTrackingPref) { |
| boolean isHeadTrackingAvailable = spatialAudioPref.isChecked() && mHasHeadTracker.get(); |
| Log.d(TAG, "refresh() has head tracker : " + mHasHeadTracker.get()); |
| headTrackingPref.setVisible(isHeadTrackingAvailable); |
| if (isHeadTrackingAvailable) { |
| headTrackingPref.setChecked(mSpatializer.isHeadTrackerEnabled(mAudioDevice)); |
| } |
| } |
| |
| @VisibleForTesting |
| TwoStatePreference createSpatialAudioPreference(Context context) { |
| TwoStatePreference pref = new SwitchPreferenceCompat(context); |
| pref.setKey(KEY_SPATIAL_AUDIO); |
| pref.setTitle(context.getString(R.string.bluetooth_details_spatial_audio_title)); |
| pref.setSummary(context.getString(R.string.bluetooth_details_spatial_audio_summary)); |
| pref.setOnPreferenceClickListener(this); |
| return pref; |
| } |
| |
| @VisibleForTesting |
| TwoStatePreference createHeadTrackingPreference(Context context) { |
| TwoStatePreference pref = new SwitchPreferenceCompat(context); |
| pref.setKey(KEY_HEAD_TRACKING); |
| pref.setTitle(context.getString(R.string.bluetooth_details_head_tracking_title)); |
| pref.setSummary(context.getString(R.string.bluetooth_details_head_tracking_summary)); |
| pref.setOnPreferenceClickListener(this); |
| return pref; |
| } |
| |
| private void getAvailableDevice() { |
| AudioDeviceAttributes a2dpDevice = new AudioDeviceAttributes( |
| AudioDeviceAttributes.ROLE_OUTPUT, |
| AudioDeviceInfo.TYPE_BLUETOOTH_A2DP, |
| mCachedDevice.getAddress()); |
| AudioDeviceAttributes bleHeadsetDevice = new AudioDeviceAttributes( |
| AudioDeviceAttributes.ROLE_OUTPUT, |
| AudioDeviceInfo.TYPE_BLE_HEADSET, |
| mCachedDevice.getAddress()); |
| AudioDeviceAttributes bleSpeakerDevice = new AudioDeviceAttributes( |
| AudioDeviceAttributes.ROLE_OUTPUT, |
| AudioDeviceInfo.TYPE_BLE_SPEAKER, |
| mCachedDevice.getAddress()); |
| AudioDeviceAttributes bleBroadcastDevice = new AudioDeviceAttributes( |
| AudioDeviceAttributes.ROLE_OUTPUT, |
| AudioDeviceInfo.TYPE_BLE_BROADCAST, |
| mCachedDevice.getAddress()); |
| AudioDeviceAttributes hearingAidDevice = new AudioDeviceAttributes( |
| AudioDeviceAttributes.ROLE_OUTPUT, |
| AudioDeviceInfo.TYPE_HEARING_AID, |
| mCachedDevice.getAddress()); |
| |
| if (mSpatializer.isAvailableForDevice(bleHeadsetDevice)) { |
| mAudioDevice = bleHeadsetDevice; |
| } else if (mSpatializer.isAvailableForDevice(bleSpeakerDevice)) { |
| mAudioDevice = bleSpeakerDevice; |
| } else if (mSpatializer.isAvailableForDevice(bleBroadcastDevice)) { |
| mAudioDevice = bleBroadcastDevice; |
| } else if (mSpatializer.isAvailableForDevice(a2dpDevice)) { |
| mAudioDevice = a2dpDevice; |
| } else if (mSpatializer.isAvailableForDevice(hearingAidDevice)) { |
| mAudioDevice = hearingAidDevice; |
| } else { |
| mAudioDevice = null; |
| } |
| |
| Log.d(TAG, "getAvailableDevice() device : " |
| + mCachedDevice.getDevice().getAnonymizedAddress() |
| + ", is available : " + (mAudioDevice != null) |
| + ", type : " + (mAudioDevice == null ? "no type" : mAudioDevice.getType())); |
| } |
| |
| @VisibleForTesting |
| void setAvailableDevice(AudioDeviceAttributes audioDevice) { |
| mAudioDevice = audioDevice; |
| } |
| } |