Bug: 306658427 Test: manual - on Settings pages Change-Id: I258a8c63d5c46403c6d333a9d2f34c33cb35e27f
239 lines
9.6 KiB
Java
239 lines
9.6 KiB
Java
/*
|
|
* 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.AudioManager;
|
|
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.settingslib.bluetooth.CachedBluetoothDevice;
|
|
import com.android.settingslib.core.lifecycle.Lifecycle;
|
|
|
|
/**
|
|
* 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;
|
|
|
|
public BluetoothDetailsSpatialAudioController(
|
|
Context context,
|
|
PreferenceFragmentCompat fragment,
|
|
CachedBluetoothDevice device,
|
|
Lifecycle lifecycle) {
|
|
super(context, fragment, device, lifecycle);
|
|
AudioManager audioManager = context.getSystemService(AudioManager.class);
|
|
mSpatializer = audioManager.getSpatializer();
|
|
}
|
|
|
|
@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());
|
|
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();
|
|
}
|
|
|
|
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() && mSpatializer.hasHeadTracker(mAudioDevice);
|
|
Log.d(TAG, "refresh() has head tracker : " + mSpatializer.hasHeadTracker(mAudioDevice));
|
|
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;
|
|
}
|
|
}
|