| /* |
| * Copyright (C) 2018 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.accessibility; |
| |
| import android.bluetooth.BluetoothAdapter; |
| import android.bluetooth.BluetoothDevice; |
| import android.bluetooth.BluetoothHapClient; |
| import android.bluetooth.BluetoothHearingAid; |
| import android.bluetooth.BluetoothLeAudio; |
| import android.bluetooth.BluetoothProfile; |
| import android.content.BroadcastReceiver; |
| import android.content.Context; |
| import android.content.Intent; |
| import android.content.IntentFilter; |
| import android.os.Bundle; |
| import android.text.TextUtils; |
| import android.util.FeatureFlagUtils; |
| import android.util.Log; |
| |
| import androidx.annotation.VisibleForTesting; |
| import androidx.fragment.app.FragmentManager; |
| import androidx.preference.Preference; |
| import androidx.preference.PreferenceScreen; |
| |
| import com.android.settings.R; |
| import com.android.settings.bluetooth.BluetoothDeviceDetailsFragment; |
| import com.android.settings.core.BasePreferenceController; |
| import com.android.settings.core.SubSettingLauncher; |
| import com.android.settingslib.bluetooth.BluetoothCallback; |
| import com.android.settingslib.bluetooth.CachedBluetoothDevice; |
| import com.android.settingslib.bluetooth.CachedBluetoothDeviceManager; |
| import com.android.settingslib.bluetooth.HapClientProfile; |
| import com.android.settingslib.bluetooth.HearingAidInfo; |
| import com.android.settingslib.bluetooth.HearingAidProfile; |
| import com.android.settingslib.bluetooth.LocalBluetoothManager; |
| import com.android.settingslib.bluetooth.LocalBluetoothProfileManager; |
| import com.android.settingslib.core.lifecycle.LifecycleObserver; |
| import com.android.settingslib.core.lifecycle.events.OnStart; |
| import com.android.settingslib.core.lifecycle.events.OnStop; |
| |
| import java.util.ArrayList; |
| import java.util.List; |
| import java.util.Set; |
| import java.util.concurrent.ExecutionException; |
| import java.util.concurrent.FutureTask; |
| import java.util.stream.Collectors; |
| |
| /** |
| * Controller that shows and updates the bluetooth device name |
| */ |
| public class AccessibilityHearingAidPreferenceController extends BasePreferenceController |
| implements LifecycleObserver, OnStart, OnStop, BluetoothCallback, |
| LocalBluetoothProfileManager.ServiceListener { |
| private static final String TAG = "AccessibilityHearingAidPreferenceController"; |
| private Preference mHearingAidPreference; |
| |
| private final BroadcastReceiver mHearingAidChangedReceiver = new BroadcastReceiver() { |
| @Override |
| public void onReceive(Context context, Intent intent) { |
| updateState(mHearingAidPreference); |
| } |
| }; |
| |
| private final LocalBluetoothManager mLocalBluetoothManager; |
| private final BluetoothAdapter mBluetoothAdapter; |
| private final LocalBluetoothProfileManager mProfileManager; |
| private final CachedBluetoothDeviceManager mCachedDeviceManager; |
| |
| private FragmentManager mFragmentManager; |
| |
| public AccessibilityHearingAidPreferenceController(Context context, String preferenceKey) { |
| super(context, preferenceKey); |
| mLocalBluetoothManager = getLocalBluetoothManager(); |
| mProfileManager = mLocalBluetoothManager.getProfileManager(); |
| mCachedDeviceManager = mLocalBluetoothManager.getCachedDeviceManager(); |
| mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter(); |
| } |
| |
| @Override |
| public void displayPreference(PreferenceScreen screen) { |
| super.displayPreference(screen); |
| mHearingAidPreference = screen.findPreference(getPreferenceKey()); |
| } |
| |
| @Override |
| public int getAvailabilityStatus() { |
| return isHearingAidSupported() ? AVAILABLE : UNSUPPORTED_ON_DEVICE; |
| } |
| |
| @Override |
| public void onStart() { |
| IntentFilter filter = new IntentFilter(); |
| filter.addAction(BluetoothHearingAid.ACTION_CONNECTION_STATE_CHANGED); |
| filter.addAction(BluetoothHapClient.ACTION_HAP_CONNECTION_STATE_CHANGED); |
| filter.addAction(BluetoothLeAudio.ACTION_LE_AUDIO_CONNECTION_STATE_CHANGED); |
| filter.addAction(BluetoothAdapter.ACTION_STATE_CHANGED); |
| mContext.registerReceiver(mHearingAidChangedReceiver, filter); |
| mLocalBluetoothManager.getEventManager().registerCallback(this); |
| // Can't get connected hearing aids when hearing aids related profiles are not ready. The |
| // profiles will be ready after the services are connected. Needs to add listener and |
| // updates the information when all hearing aids related services are connected. |
| if (isAnyHearingAidRelatedProfilesNotReady()) { |
| mProfileManager.addServiceListener(this); |
| } |
| } |
| |
| @Override |
| public void onStop() { |
| mContext.unregisterReceiver(mHearingAidChangedReceiver); |
| mLocalBluetoothManager.getEventManager().unregisterCallback(this); |
| mProfileManager.removeServiceListener(this); |
| } |
| |
| @Override |
| public boolean handlePreferenceTreeClick(Preference preference) { |
| if (TextUtils.equals(preference.getKey(), getPreferenceKey())) { |
| final CachedBluetoothDevice device = getConnectedHearingAidDevice(); |
| if (FeatureFlagUtils.isEnabled(mContext, |
| FeatureFlagUtils.SETTINGS_ACCESSIBILITY_HEARING_AID_PAGE)) { |
| launchHearingAidPage(); |
| return true; |
| } |
| if (device == null) { |
| launchHearingAidInstructionDialog(); |
| } else { |
| launchBluetoothDeviceDetailSetting(device); |
| } |
| return true; |
| } |
| return false; |
| } |
| |
| @Override |
| public CharSequence getSummary() { |
| if (mBluetoothAdapter == null || !mBluetoothAdapter.isEnabled()) { |
| return mContext.getText(R.string.accessibility_hearingaid_not_connected_summary); |
| } |
| final CachedBluetoothDevice device = getConnectedHearingAidDevice(); |
| if (device == null) { |
| return mContext.getText(R.string.accessibility_hearingaid_not_connected_summary); |
| } |
| |
| final int connectedNum = getConnectedHearingAidDeviceNum(); |
| final CharSequence name = device.getName(); |
| if (connectedNum > 1) { |
| return mContext.getString(R.string.accessibility_hearingaid_more_device_summary, name); |
| } |
| |
| // Check if another side of LE audio hearing aid is connected as a pair |
| final Set<CachedBluetoothDevice> memberDevices = device.getMemberDevice(); |
| if (memberDevices.stream().anyMatch(m -> m.isConnected())) { |
| return mContext.getString( |
| R.string.accessibility_hearingaid_left_and_right_side_device_summary, |
| name); |
| } |
| |
| // Check if another side of ASHA hearing aid is connected as a pair |
| final CachedBluetoothDevice subDevice = device.getSubDevice(); |
| if (subDevice != null && subDevice.isConnected()) { |
| return mContext.getString( |
| R.string.accessibility_hearingaid_left_and_right_side_device_summary, name); |
| } |
| |
| final int side = device.getDeviceSide(); |
| if (side == HearingAidInfo.DeviceSide.SIDE_LEFT_AND_RIGHT) { |
| return mContext.getString( |
| R.string.accessibility_hearingaid_left_and_right_side_device_summary, name); |
| } else if (side == HearingAidInfo.DeviceSide.SIDE_LEFT) { |
| return mContext.getString( |
| R.string.accessibility_hearingaid_left_side_device_summary, name); |
| } else if (side == HearingAidInfo.DeviceSide.SIDE_RIGHT) { |
| return mContext.getString( |
| R.string.accessibility_hearingaid_right_side_device_summary, name); |
| } |
| |
| // Invalid side |
| return mContext.getString( |
| R.string.accessibility_hearingaid_active_device_summary, name); |
| } |
| |
| @Override |
| public void onActiveDeviceChanged(CachedBluetoothDevice activeDevice, int bluetoothProfile) { |
| if (activeDevice == null) { |
| return; |
| } |
| |
| if (bluetoothProfile == BluetoothProfile.HEARING_AID) { |
| HearingAidUtils.launchHearingAidPairingDialog(mFragmentManager, activeDevice); |
| } |
| } |
| |
| @Override |
| public void onServiceConnected() { |
| if (!isAnyHearingAidRelatedProfilesNotReady()) { |
| updateState(mHearingAidPreference); |
| mProfileManager.removeServiceListener(this); |
| } |
| } |
| |
| @Override |
| public void onServiceDisconnected() { |
| // Do nothing |
| } |
| |
| public void setFragmentManager(FragmentManager fragmentManager) { |
| mFragmentManager = fragmentManager; |
| } |
| |
| @VisibleForTesting |
| CachedBluetoothDevice getConnectedHearingAidDevice() { |
| final List<BluetoothDevice> deviceList = getConnectedHearingAidDeviceList(); |
| return deviceList.isEmpty() ? null : mCachedDeviceManager.findDevice(deviceList.get(0)); |
| } |
| |
| private int getConnectedHearingAidDeviceNum() { |
| return getConnectedHearingAidDeviceList().size(); |
| } |
| |
| private List<BluetoothDevice> getConnectedHearingAidDeviceList() { |
| if (!isHearingAidSupported()) { |
| return new ArrayList<>(); |
| } |
| final List<BluetoothDevice> deviceList = new ArrayList<>(); |
| final HapClientProfile hapClientProfile = mProfileManager.getHapClientProfile(); |
| if (hapClientProfile != null) { |
| deviceList.addAll(hapClientProfile.getConnectedDevices()); |
| } |
| final HearingAidProfile hearingAidProfile = mProfileManager.getHearingAidProfile(); |
| if (hearingAidProfile != null) { |
| deviceList.addAll(hearingAidProfile.getConnectedDevices()); |
| } |
| return deviceList.stream() |
| .distinct() |
| .filter(d -> !mCachedDeviceManager.isSubDevice(d)).collect(Collectors.toList()); |
| } |
| |
| private boolean isHearingAidSupported() { |
| if (mBluetoothAdapter == null || !mBluetoothAdapter.isEnabled()) { |
| return false; |
| } |
| final List<Integer> supportedList = mBluetoothAdapter.getSupportedProfiles(); |
| return supportedList.contains(BluetoothProfile.HEARING_AID) |
| || supportedList.contains(BluetoothProfile.HAP_CLIENT); |
| } |
| |
| private boolean isAnyHearingAidRelatedProfilesNotReady() { |
| HearingAidProfile hearingAidProfile = mProfileManager.getHearingAidProfile(); |
| if (hearingAidProfile != null && !hearingAidProfile.isProfileReady()) { |
| return true; |
| } |
| HapClientProfile hapClientProfile = mProfileManager.getHapClientProfile(); |
| if (hapClientProfile != null && !hapClientProfile.isProfileReady()) { |
| return true; |
| } |
| return false; |
| } |
| |
| private LocalBluetoothManager getLocalBluetoothManager() { |
| final FutureTask<LocalBluetoothManager> localBtManagerFutureTask = new FutureTask<>( |
| // Avoid StrictMode ThreadPolicy violation |
| () -> com.android.settings.bluetooth.Utils.getLocalBtManager(mContext)); |
| try { |
| localBtManagerFutureTask.run(); |
| return localBtManagerFutureTask.get(); |
| } catch (InterruptedException | ExecutionException e) { |
| Log.w(TAG, "Error getting LocalBluetoothManager.", e); |
| return null; |
| } |
| } |
| |
| @VisibleForTesting(otherwise = VisibleForTesting.NONE) |
| void setPreference(Preference preference) { |
| mHearingAidPreference = preference; |
| } |
| |
| @VisibleForTesting |
| void launchBluetoothDeviceDetailSetting(final CachedBluetoothDevice device) { |
| if (device == null) { |
| return; |
| } |
| final Bundle args = new Bundle(); |
| args.putString(BluetoothDeviceDetailsFragment.KEY_DEVICE_ADDRESS, |
| device.getDevice().getAddress()); |
| |
| new SubSettingLauncher(mContext) |
| .setDestination(BluetoothDeviceDetailsFragment.class.getName()) |
| .setArguments(args) |
| .setTitleRes(R.string.device_details_title) |
| .setSourceMetricsCategory(getMetricsCategory()) |
| .launch(); |
| } |
| |
| @VisibleForTesting |
| void launchHearingAidInstructionDialog() { |
| HearingAidDialogFragment fragment = HearingAidDialogFragment.newInstance(); |
| fragment.show(mFragmentManager, HearingAidDialogFragment.class.toString()); |
| } |
| |
| private void launchHearingAidPage() { |
| new SubSettingLauncher(mContext) |
| .setDestination(AccessibilityHearingAidsFragment.class.getName()) |
| .setSourceMetricsCategory(getMetricsCategory()) |
| .launch(); |
| } |
| } |