1. Change logs in error callback from debug to warning level 2. Remove redundant logs Bug: 334000375 Test: simple change, no tests needed Change-Id: I0fa0d1a693d43b8f774dd87b72839370840f65a8
373 lines
14 KiB
Java
373 lines
14 KiB
Java
/*
|
|
* Copyright (C) 2024 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 com.android.settings.bluetooth.BluetoothDetailsHearingDeviceController.KEY_HEARING_DEVICE_GROUP;
|
|
import static com.android.settings.bluetooth.BluetoothDetailsHearingDeviceController.ORDER_HEARING_AIDS_PRESETS;
|
|
|
|
import android.bluetooth.BluetoothCsipSetCoordinator;
|
|
import android.bluetooth.BluetoothDevice;
|
|
import android.bluetooth.BluetoothHapClient;
|
|
import android.bluetooth.BluetoothHapPresetInfo;
|
|
import android.content.Context;
|
|
import android.text.TextUtils;
|
|
import android.util.Log;
|
|
import android.widget.Toast;
|
|
|
|
import androidx.annotation.NonNull;
|
|
import androidx.annotation.Nullable;
|
|
import androidx.annotation.VisibleForTesting;
|
|
import androidx.preference.ListPreference;
|
|
import androidx.preference.Preference;
|
|
import androidx.preference.PreferenceCategory;
|
|
import androidx.preference.PreferenceFragmentCompat;
|
|
import androidx.preference.PreferenceScreen;
|
|
|
|
import com.android.settings.R;
|
|
import com.android.settingslib.bluetooth.CachedBluetoothDevice;
|
|
import com.android.settingslib.bluetooth.HapClientProfile;
|
|
import com.android.settingslib.bluetooth.LocalBluetoothManager;
|
|
import com.android.settingslib.bluetooth.LocalBluetoothProfileManager;
|
|
import com.android.settingslib.core.lifecycle.Lifecycle;
|
|
import com.android.settingslib.core.lifecycle.events.OnPause;
|
|
import com.android.settingslib.core.lifecycle.events.OnResume;
|
|
import com.android.settingslib.core.lifecycle.events.OnStart;
|
|
import com.android.settingslib.core.lifecycle.events.OnStop;
|
|
import com.android.settingslib.utils.ThreadUtils;
|
|
|
|
import java.util.List;
|
|
|
|
/**
|
|
* The controller of the hearing aid presets.
|
|
*/
|
|
public class BluetoothDetailsHearingAidsPresetsController extends
|
|
BluetoothDetailsController implements Preference.OnPreferenceChangeListener,
|
|
BluetoothHapClient.Callback, LocalBluetoothProfileManager.ServiceListener,
|
|
OnStart, OnResume, OnPause, OnStop {
|
|
|
|
private static final boolean DEBUG = true;
|
|
private static final String TAG = "BluetoothDetailsHearingAidsPresetsController";
|
|
static final String KEY_HEARING_AIDS_PRESETS = "hearing_aids_presets";
|
|
|
|
private final LocalBluetoothProfileManager mProfileManager;
|
|
private final HapClientProfile mHapClientProfile;
|
|
|
|
@Nullable
|
|
private ListPreference mPreference;
|
|
|
|
public BluetoothDetailsHearingAidsPresetsController(@NonNull Context context,
|
|
@NonNull PreferenceFragmentCompat fragment,
|
|
@NonNull LocalBluetoothManager manager,
|
|
@NonNull CachedBluetoothDevice device,
|
|
@NonNull Lifecycle lifecycle) {
|
|
super(context, fragment, device, lifecycle);
|
|
mProfileManager = manager.getProfileManager();
|
|
mHapClientProfile = mProfileManager.getHapClientProfile();
|
|
}
|
|
|
|
@Override
|
|
public void onStart() {
|
|
if (mHapClientProfile != null && !mHapClientProfile.isProfileReady()) {
|
|
mProfileManager.addServiceListener(this);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onResume() {
|
|
registerHapCallback();
|
|
super.onResume();
|
|
}
|
|
|
|
@Override
|
|
public void onPause() {
|
|
unregisterHapCallback();
|
|
super.onPause();
|
|
}
|
|
|
|
@Override
|
|
public void onStop() {
|
|
mProfileManager.removeServiceListener(this);
|
|
}
|
|
|
|
@Override
|
|
public boolean onPreferenceChange(@NonNull Preference preference, @Nullable Object newValue) {
|
|
if (TextUtils.equals(preference.getKey(), getPreferenceKey())) {
|
|
if (newValue instanceof final String value
|
|
&& preference instanceof final ListPreference listPreference) {
|
|
final int index = listPreference.findIndexOfValue(value);
|
|
final String presetName = listPreference.getEntries()[index].toString();
|
|
final int presetIndex = Integer.parseInt(value);
|
|
listPreference.setSummary(presetName);
|
|
if (DEBUG) {
|
|
Log.d(TAG, "onPreferenceChange"
|
|
+ ", presetIndex: " + presetIndex
|
|
+ ", presetName: " + presetName);
|
|
}
|
|
boolean supportSynchronizedPresets = mHapClientProfile.supportsSynchronizedPresets(
|
|
mCachedDevice.getDevice());
|
|
int hapGroupId = mHapClientProfile.getHapGroup(mCachedDevice.getDevice());
|
|
if (supportSynchronizedPresets) {
|
|
if (hapGroupId != BluetoothCsipSetCoordinator.GROUP_ID_INVALID) {
|
|
selectPresetSynchronously(hapGroupId, presetIndex);
|
|
} else {
|
|
Log.w(TAG, "supportSynchronizedPresets but hapGroupId is invalid.");
|
|
selectPresetIndependently(presetIndex);
|
|
}
|
|
} else {
|
|
selectPresetIndependently(presetIndex);
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Nullable
|
|
@Override
|
|
public String getPreferenceKey() {
|
|
return KEY_HEARING_AIDS_PRESETS;
|
|
}
|
|
|
|
@Override
|
|
protected void init(PreferenceScreen screen) {
|
|
PreferenceCategory deviceControls = screen.findPreference(KEY_HEARING_DEVICE_GROUP);
|
|
if (deviceControls != null) {
|
|
mPreference = createPresetPreference(deviceControls.getContext());
|
|
deviceControls.addPreference(mPreference);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void refresh() {
|
|
if (!isAvailable() || mPreference == null) {
|
|
return;
|
|
}
|
|
mPreference.setEnabled(mCachedDevice.isConnectedHapClientDevice());
|
|
|
|
loadAllPresetInfo();
|
|
if (mPreference.getEntries().length == 0) {
|
|
if (DEBUG) {
|
|
Log.w(TAG, "Disable the preference since preset info size = 0");
|
|
}
|
|
mPreference.setEnabled(false);
|
|
} else {
|
|
int activePresetIndex = mHapClientProfile.getActivePresetIndex(
|
|
mCachedDevice.getDevice());
|
|
if (activePresetIndex != BluetoothHapClient.PRESET_INDEX_UNAVAILABLE) {
|
|
mPreference.setValue(Integer.toString(activePresetIndex));
|
|
mPreference.setSummary(mPreference.getEntry());
|
|
} else {
|
|
mPreference.setSummary(null);
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean isAvailable() {
|
|
if (mHapClientProfile == null) {
|
|
return false;
|
|
}
|
|
return mCachedDevice.getProfiles().stream().anyMatch(
|
|
profile -> profile instanceof HapClientProfile);
|
|
}
|
|
|
|
@Override
|
|
public void onPresetSelected(@NonNull BluetoothDevice device, int presetIndex, int reason) {
|
|
if (device.equals(mCachedDevice.getDevice())) {
|
|
if (DEBUG) {
|
|
Log.d(TAG, "onPresetSelected, device: " + device.getAddress()
|
|
+ ", presetIndex: " + presetIndex + ", reason: " + reason);
|
|
}
|
|
mContext.getMainExecutor().execute(this::refresh);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onPresetSelectionFailed(@NonNull BluetoothDevice device, int reason) {
|
|
if (device.equals(mCachedDevice.getDevice())) {
|
|
Log.w(TAG, "onPresetSelectionFailed, device: " + device.getAddress()
|
|
+ ", reason: " + reason);
|
|
mContext.getMainExecutor().execute(() -> {
|
|
refresh();
|
|
showErrorToast();
|
|
});
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onPresetSelectionForGroupFailed(int hapGroupId, int reason) {
|
|
if (hapGroupId == mHapClientProfile.getHapGroup(mCachedDevice.getDevice())) {
|
|
Log.w(TAG, "onPresetSelectionForGroupFailed, group: " + hapGroupId
|
|
+ ", reason: " + reason);
|
|
// Try to set the preset independently if group operation failed
|
|
if (mPreference != null) {
|
|
selectPresetIndependently(Integer.parseInt(mPreference.getValue()));
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onPresetInfoChanged(@NonNull BluetoothDevice device,
|
|
@NonNull List<BluetoothHapPresetInfo> presetInfoList, int reason) {
|
|
if (device.equals(mCachedDevice.getDevice())) {
|
|
if (DEBUG) {
|
|
Log.d(TAG, "onPresetInfoChanged, device: " + device.getAddress()
|
|
+ ", reason: " + reason);
|
|
for (BluetoothHapPresetInfo info: presetInfoList) {
|
|
Log.d(TAG, " preset " + info.getIndex() + ": " + info.getName());
|
|
}
|
|
}
|
|
mContext.getMainExecutor().execute(this::refresh);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onSetPresetNameFailed(@NonNull BluetoothDevice device, int reason) {
|
|
if (device.equals(mCachedDevice.getDevice())) {
|
|
Log.w(TAG, "onSetPresetNameFailed, device: " + device.getAddress()
|
|
+ ", reason: " + reason);
|
|
mContext.getMainExecutor().execute(() -> {
|
|
refresh();
|
|
showErrorToast();
|
|
});
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onSetPresetNameForGroupFailed(int hapGroupId, int reason) {
|
|
if (hapGroupId == mHapClientProfile.getHapGroup(mCachedDevice.getDevice())) {
|
|
Log.w(TAG, "onSetPresetNameForGroupFailed, group: " + hapGroupId
|
|
+ ", reason: " + reason);
|
|
mContext.getMainExecutor().execute(() -> {
|
|
refresh();
|
|
showErrorToast();
|
|
});
|
|
}
|
|
}
|
|
|
|
private ListPreference createPresetPreference(Context context) {
|
|
ListPreference preference = new ListPreference(context);
|
|
preference.setKey(KEY_HEARING_AIDS_PRESETS);
|
|
preference.setOrder(ORDER_HEARING_AIDS_PRESETS);
|
|
preference.setTitle(context.getString(R.string.bluetooth_hearing_aids_presets));
|
|
preference.setOnPreferenceChangeListener(this);
|
|
return preference;
|
|
}
|
|
|
|
private void loadAllPresetInfo() {
|
|
if (mPreference == null) {
|
|
return;
|
|
}
|
|
List<BluetoothHapPresetInfo> infoList = mHapClientProfile.getAllPresetInfo(
|
|
mCachedDevice.getDevice());
|
|
CharSequence[] presetNames = new CharSequence[infoList.size()];
|
|
CharSequence[] presetIndexes = new CharSequence[infoList.size()];
|
|
for (int i = 0; i < infoList.size(); i++) {
|
|
presetNames[i] = infoList.get(i).getName();
|
|
presetIndexes[i] = Integer.toString(infoList.get(i).getIndex());
|
|
}
|
|
mPreference.setEntries(presetNames);
|
|
mPreference.setEntryValues(presetIndexes);
|
|
}
|
|
|
|
@VisibleForTesting
|
|
@Nullable
|
|
ListPreference getPreference() {
|
|
return mPreference;
|
|
}
|
|
|
|
void showErrorToast() {
|
|
Toast.makeText(mContext, R.string.bluetooth_hearing_aids_presets_error,
|
|
Toast.LENGTH_SHORT).show();
|
|
}
|
|
|
|
private void registerHapCallback() {
|
|
if (mHapClientProfile != null) {
|
|
try {
|
|
mHapClientProfile.registerCallback(ThreadUtils.getBackgroundExecutor(), this);
|
|
} catch (IllegalArgumentException e) {
|
|
// The callback was already registered
|
|
Log.w(TAG, "Cannot register callback: " + e.getMessage());
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
private void unregisterHapCallback() {
|
|
if (mHapClientProfile != null) {
|
|
try {
|
|
mHapClientProfile.unregisterCallback(this);
|
|
} catch (IllegalArgumentException e) {
|
|
// The callback was never registered or was already unregistered
|
|
Log.w(TAG, "Cannot unregister callback: " + e.getMessage());
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onServiceConnected() {
|
|
if (mHapClientProfile != null && mHapClientProfile.isProfileReady()) {
|
|
mProfileManager.removeServiceListener(this);
|
|
registerHapCallback();
|
|
refresh();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onServiceDisconnected() {
|
|
// Do nothing
|
|
}
|
|
|
|
private void selectPresetSynchronously(int groupId, int presetIndex) {
|
|
if (mPreference == null) {
|
|
return;
|
|
}
|
|
if (DEBUG) {
|
|
Log.d(TAG, "selectPresetSynchronously"
|
|
+ ", presetIndex: " + presetIndex
|
|
+ ", groupId: " + groupId
|
|
+ ", device: " + mCachedDevice.getAddress());
|
|
}
|
|
mHapClientProfile.selectPresetForGroup(groupId, presetIndex);
|
|
}
|
|
private void selectPresetIndependently(int presetIndex) {
|
|
if (mPreference == null) {
|
|
return;
|
|
}
|
|
if (DEBUG) {
|
|
Log.d(TAG, "selectPresetIndependently"
|
|
+ ", presetIndex: " + presetIndex
|
|
+ ", device: " + mCachedDevice.getAddress());
|
|
}
|
|
mHapClientProfile.selectPreset(mCachedDevice.getDevice(), presetIndex);
|
|
final CachedBluetoothDevice subDevice = mCachedDevice.getSubDevice();
|
|
if (subDevice != null) {
|
|
if (DEBUG) {
|
|
Log.d(TAG, "selectPreset for subDevice, device: " + subDevice);
|
|
}
|
|
mHapClientProfile.selectPreset(subDevice.getDevice(), presetIndex);
|
|
}
|
|
for (final CachedBluetoothDevice memberDevice :
|
|
mCachedDevice.getMemberDevice()) {
|
|
if (DEBUG) {
|
|
Log.d(TAG, "selectPreset for memberDevice, device: " + memberDevice);
|
|
}
|
|
mHapClientProfile.selectPreset(memberDevice.getDevice(), presetIndex);
|
|
}
|
|
}
|
|
}
|