Bug: 333836213 Test: atest BluetoothDeviceDetailsFragmentTest Change-Id: I4632c10a8f6f828885825e1ee46df6537a0c050c
401 lines
16 KiB
Java
401 lines
16 KiB
Java
/*
|
|
* Copyright (C) 2017 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.bluetooth.BluetoothDevice.BOND_NONE;
|
|
import static android.os.UserManager.DISALLOW_CONFIG_BLUETOOTH;
|
|
|
|
import android.app.Activity;
|
|
import android.app.settings.SettingsEnums;
|
|
import android.bluetooth.BluetoothAdapter;
|
|
import android.bluetooth.BluetoothDevice;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.content.res.TypedArray;
|
|
import android.hardware.input.InputManager;
|
|
import android.net.Uri;
|
|
import android.os.Bundle;
|
|
import android.os.UserManager;
|
|
import android.provider.DeviceConfig;
|
|
import android.text.TextUtils;
|
|
import android.util.FeatureFlagUtils;
|
|
import android.util.Log;
|
|
import android.view.InputDevice;
|
|
import android.view.LayoutInflater;
|
|
import android.view.Menu;
|
|
import android.view.MenuInflater;
|
|
import android.view.MenuItem;
|
|
import android.view.View;
|
|
import android.view.ViewGroup;
|
|
import android.view.ViewTreeObserver;
|
|
|
|
import androidx.annotation.Nullable;
|
|
import androidx.annotation.VisibleForTesting;
|
|
|
|
import com.android.settings.R;
|
|
import com.android.settings.connecteddevice.stylus.StylusDevicesController;
|
|
import com.android.settings.core.SettingsUIDeviceConfig;
|
|
import com.android.settings.dashboard.RestrictedDashboardFragment;
|
|
import com.android.settings.inputmethod.KeyboardSettingsPreferenceController;
|
|
import com.android.settings.overlay.FeatureFactory;
|
|
import com.android.settings.slices.SlicePreferenceController;
|
|
import com.android.settingslib.bluetooth.BluetoothCallback;
|
|
import com.android.settingslib.bluetooth.CachedBluetoothDevice;
|
|
import com.android.settingslib.bluetooth.LocalBluetoothManager;
|
|
import com.android.settingslib.core.AbstractPreferenceController;
|
|
import com.android.settingslib.core.instrumentation.MetricsFeatureProvider;
|
|
import com.android.settingslib.core.lifecycle.Lifecycle;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
|
|
public class BluetoothDeviceDetailsFragment extends RestrictedDashboardFragment {
|
|
public static final String KEY_DEVICE_ADDRESS = "device_address";
|
|
private static final String TAG = "BTDeviceDetailsFrg";
|
|
|
|
@VisibleForTesting
|
|
static int EDIT_DEVICE_NAME_ITEM_ID = Menu.FIRST;
|
|
|
|
/**
|
|
* An interface to let tests override the normal mechanism for looking up the
|
|
* CachedBluetoothDevice and LocalBluetoothManager, and substitute their own mocks instead.
|
|
* This is only needed in situations where you instantiate the fragment indirectly (eg via an
|
|
* intent) and can't use something like spying on an instance you construct directly via
|
|
* newInstance.
|
|
*/
|
|
@VisibleForTesting
|
|
interface TestDataFactory {
|
|
CachedBluetoothDevice getDevice(String deviceAddress);
|
|
|
|
LocalBluetoothManager getManager(Context context);
|
|
|
|
UserManager getUserManager();
|
|
}
|
|
|
|
@VisibleForTesting
|
|
static TestDataFactory sTestDataFactory;
|
|
|
|
@VisibleForTesting
|
|
String mDeviceAddress;
|
|
@VisibleForTesting
|
|
LocalBluetoothManager mManager;
|
|
@VisibleForTesting
|
|
CachedBluetoothDevice mCachedDevice;
|
|
|
|
@Nullable
|
|
InputDevice mInputDevice;
|
|
|
|
private UserManager mUserManager;
|
|
|
|
private final BluetoothCallback mBluetoothCallback =
|
|
new BluetoothCallback() {
|
|
@Override
|
|
public void onBluetoothStateChanged(int bluetoothState) {
|
|
if (bluetoothState == BluetoothAdapter.STATE_OFF) {
|
|
Log.i(TAG, "Bluetooth is off, exit activity.");
|
|
Activity activity = getActivity();
|
|
if (activity != null) {
|
|
activity.finish();
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
public BluetoothDeviceDetailsFragment() {
|
|
super(DISALLOW_CONFIG_BLUETOOTH);
|
|
}
|
|
|
|
@VisibleForTesting
|
|
LocalBluetoothManager getLocalBluetoothManager(Context context) {
|
|
if (sTestDataFactory != null) {
|
|
return sTestDataFactory.getManager(context);
|
|
}
|
|
return Utils.getLocalBtManager(context);
|
|
}
|
|
|
|
@VisibleForTesting
|
|
CachedBluetoothDevice getCachedDevice(String deviceAddress) {
|
|
if (sTestDataFactory != null) {
|
|
return sTestDataFactory.getDevice(deviceAddress);
|
|
}
|
|
BluetoothDevice remoteDevice =
|
|
mManager.getBluetoothAdapter().getRemoteDevice(deviceAddress);
|
|
return mManager.getCachedDeviceManager().findDevice(remoteDevice);
|
|
}
|
|
|
|
@VisibleForTesting
|
|
UserManager getUserManager() {
|
|
if (sTestDataFactory != null) {
|
|
return sTestDataFactory.getUserManager();
|
|
}
|
|
|
|
return getSystemService(UserManager.class);
|
|
}
|
|
|
|
@Nullable
|
|
@VisibleForTesting
|
|
InputDevice getInputDevice(Context context) {
|
|
InputManager im = context.getSystemService(InputManager.class);
|
|
|
|
for (int deviceId : im.getInputDeviceIds()) {
|
|
String btAddress = im.getInputDeviceBluetoothAddress(deviceId);
|
|
|
|
if (btAddress != null && btAddress.equals(mDeviceAddress)) {
|
|
return im.getInputDevice(deviceId);
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
public static BluetoothDeviceDetailsFragment newInstance(String deviceAddress) {
|
|
Bundle args = new Bundle(1);
|
|
args.putString(KEY_DEVICE_ADDRESS, deviceAddress);
|
|
BluetoothDeviceDetailsFragment fragment = new BluetoothDeviceDetailsFragment();
|
|
fragment.setArguments(args);
|
|
return fragment;
|
|
}
|
|
|
|
@Override
|
|
public void onAttach(Context context) {
|
|
mDeviceAddress = getArguments().getString(KEY_DEVICE_ADDRESS);
|
|
mManager = getLocalBluetoothManager(context);
|
|
mCachedDevice = getCachedDevice(mDeviceAddress);
|
|
mUserManager = getUserManager();
|
|
|
|
if (FeatureFlagUtils.isEnabled(context,
|
|
FeatureFlagUtils.SETTINGS_SHOW_STYLUS_PREFERENCES)) {
|
|
mInputDevice = getInputDevice(context);
|
|
}
|
|
|
|
super.onAttach(context);
|
|
if (mCachedDevice == null) {
|
|
// Close this page if device is null with invalid device mac address
|
|
Log.w(TAG, "onAttach() CachedDevice is null!");
|
|
finish();
|
|
return;
|
|
}
|
|
use(AdvancedBluetoothDetailsHeaderController.class).init(mCachedDevice);
|
|
use(LeAudioBluetoothDetailsHeaderController.class).init(mCachedDevice, mManager);
|
|
use(KeyboardSettingsPreferenceController.class).init(mCachedDevice);
|
|
|
|
final BluetoothFeatureProvider featureProvider =
|
|
FeatureFactory.getFeatureFactory().getBluetoothFeatureProvider();
|
|
final boolean sliceEnabled = DeviceConfig.getBoolean(DeviceConfig.NAMESPACE_SETTINGS_UI,
|
|
SettingsUIDeviceConfig.BT_SLICE_SETTINGS_ENABLED, true);
|
|
|
|
use(BlockingPrefWithSliceController.class).setSliceUri(sliceEnabled
|
|
? featureProvider.getBluetoothDeviceSettingsUri(mCachedDevice.getDevice())
|
|
: null);
|
|
|
|
mManager.getEventManager().registerCallback(mBluetoothCallback);
|
|
}
|
|
|
|
@Override
|
|
public void onDetach() {
|
|
super.onDetach();
|
|
mManager.getEventManager().unregisterCallback(mBluetoothCallback);
|
|
}
|
|
|
|
private void updateExtraControlUri(int viewWidth) {
|
|
BluetoothFeatureProvider featureProvider =
|
|
FeatureFactory.getFeatureFactory().getBluetoothFeatureProvider();
|
|
boolean sliceEnabled = DeviceConfig.getBoolean(DeviceConfig.NAMESPACE_SETTINGS_UI,
|
|
SettingsUIDeviceConfig.BT_SLICE_SETTINGS_ENABLED, true);
|
|
Uri controlUri = null;
|
|
String uri = featureProvider.getBluetoothDeviceControlUri(mCachedDevice.getDevice());
|
|
if (!TextUtils.isEmpty(uri)) {
|
|
try {
|
|
controlUri = Uri.parse(uri + viewWidth);
|
|
} catch (NullPointerException exception) {
|
|
Log.d(TAG, "unable to parse uri");
|
|
controlUri = null;
|
|
}
|
|
}
|
|
final SlicePreferenceController slicePreferenceController = use(
|
|
SlicePreferenceController.class);
|
|
slicePreferenceController.setSliceUri(sliceEnabled ? controlUri : null);
|
|
slicePreferenceController.onStart();
|
|
slicePreferenceController.displayPreference(getPreferenceScreen());
|
|
|
|
// Temporarily fix the issue that the page will be automatically scrolled to a wrong
|
|
// position when entering the page. This will make sure the bluetooth header is shown on top
|
|
// of the page.
|
|
use(LeAudioBluetoothDetailsHeaderController.class).displayPreference(
|
|
getPreferenceScreen());
|
|
use(AdvancedBluetoothDetailsHeaderController.class).displayPreference(
|
|
getPreferenceScreen());
|
|
use(BluetoothDetailsHeaderController.class).displayPreference(
|
|
getPreferenceScreen());
|
|
}
|
|
|
|
private final ViewTreeObserver.OnGlobalLayoutListener mOnGlobalLayoutListener =
|
|
new ViewTreeObserver.OnGlobalLayoutListener() {
|
|
@Override
|
|
public void onGlobalLayout() {
|
|
View view = getView();
|
|
if (view == null) {
|
|
return;
|
|
}
|
|
if (view.getWidth() <= 0) {
|
|
return;
|
|
}
|
|
updateExtraControlUri(view.getWidth() - getPaddingSize());
|
|
view.getViewTreeObserver().removeOnGlobalLayoutListener(
|
|
mOnGlobalLayoutListener);
|
|
}
|
|
};
|
|
|
|
@Override
|
|
public void onCreate(Bundle savedInstanceState) {
|
|
super.onCreate(savedInstanceState);
|
|
setTitleForInputDevice();
|
|
}
|
|
|
|
@Override
|
|
public View onCreateView(LayoutInflater inflater, ViewGroup container,
|
|
Bundle savedInstanceState) {
|
|
View view = super.onCreateView(inflater, container, savedInstanceState);
|
|
if (view != null) {
|
|
view.getViewTreeObserver().addOnGlobalLayoutListener(mOnGlobalLayoutListener);
|
|
}
|
|
return view;
|
|
}
|
|
|
|
@Override
|
|
public void onResume() {
|
|
super.onResume();
|
|
finishFragmentIfNecessary();
|
|
}
|
|
|
|
@VisibleForTesting
|
|
void finishFragmentIfNecessary() {
|
|
if (mCachedDevice.getBondState() == BOND_NONE) {
|
|
finish();
|
|
return;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public int getMetricsCategory() {
|
|
return SettingsEnums.BLUETOOTH_DEVICE_DETAILS;
|
|
}
|
|
|
|
@Override
|
|
protected String getLogTag() {
|
|
return TAG;
|
|
}
|
|
|
|
@Override
|
|
protected int getPreferenceScreenResId() {
|
|
return R.xml.bluetooth_device_details_fragment;
|
|
}
|
|
|
|
@Override
|
|
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
|
|
if (!mUserManager.isGuestUser()) {
|
|
MenuItem item = menu.add(0, EDIT_DEVICE_NAME_ITEM_ID, 0,
|
|
R.string.bluetooth_rename_button);
|
|
item.setIcon(com.android.internal.R.drawable.ic_mode_edit);
|
|
item.setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS);
|
|
}
|
|
super.onCreateOptionsMenu(menu, inflater);
|
|
}
|
|
|
|
@Override
|
|
public boolean onOptionsItemSelected(MenuItem menuItem) {
|
|
if (menuItem.getItemId() == EDIT_DEVICE_NAME_ITEM_ID) {
|
|
RemoteDeviceNameDialogFragment.newInstance(mCachedDevice).show(
|
|
getFragmentManager(), RemoteDeviceNameDialogFragment.TAG);
|
|
return true;
|
|
}
|
|
return super.onOptionsItemSelected(menuItem);
|
|
}
|
|
|
|
@Override
|
|
protected List<AbstractPreferenceController> createPreferenceControllers(Context context) {
|
|
ArrayList<AbstractPreferenceController> controllers = new ArrayList<>();
|
|
|
|
if (mCachedDevice != null) {
|
|
Lifecycle lifecycle = getSettingsLifecycle();
|
|
controllers.add(new BluetoothDetailsHeaderController(context, this, mCachedDevice,
|
|
lifecycle));
|
|
controllers.add(new BluetoothDetailsButtonsController(context, this, mCachedDevice,
|
|
lifecycle));
|
|
controllers.add(new BluetoothDetailsCompanionAppsController(context, this,
|
|
mCachedDevice, lifecycle));
|
|
controllers.add(new BluetoothDetailsAudioDeviceTypeController(context, this, mManager,
|
|
mCachedDevice, lifecycle));
|
|
controllers.add(new BluetoothDetailsSpatialAudioController(context, this, mCachedDevice,
|
|
lifecycle));
|
|
controllers.add(new BluetoothDetailsProfilesController(context, this, mManager,
|
|
mCachedDevice, lifecycle));
|
|
controllers.add(new BluetoothDetailsMacAddressController(context, this, mCachedDevice,
|
|
lifecycle));
|
|
controllers.add(new StylusDevicesController(context, mInputDevice, mCachedDevice,
|
|
lifecycle));
|
|
controllers.add(new BluetoothDetailsRelatedToolsController(context, this, mCachedDevice,
|
|
lifecycle));
|
|
controllers.add(new BluetoothDetailsPairOtherController(context, this, mCachedDevice,
|
|
lifecycle));
|
|
controllers.add(new BluetoothDetailsDataSyncController(context, this, mCachedDevice,
|
|
lifecycle));
|
|
controllers.add(new BluetoothDetailsExtraOptionsController(context, this, mCachedDevice,
|
|
lifecycle));
|
|
BluetoothDetailsHearingDeviceController hearingDeviceController =
|
|
new BluetoothDetailsHearingDeviceController(context, this, mManager,
|
|
mCachedDevice, lifecycle);
|
|
controllers.add(hearingDeviceController);
|
|
hearingDeviceController.initSubControllers(isLaunchFromHearingDevicePage());
|
|
controllers.addAll(hearingDeviceController.getSubControllers());
|
|
}
|
|
return controllers;
|
|
}
|
|
|
|
private int getPaddingSize() {
|
|
TypedArray resolvedAttributes =
|
|
getContext().obtainStyledAttributes(
|
|
new int[]{
|
|
android.R.attr.listPreferredItemPaddingStart,
|
|
android.R.attr.listPreferredItemPaddingEnd
|
|
});
|
|
int width = resolvedAttributes.getDimensionPixelSize(0, 0)
|
|
+ resolvedAttributes.getDimensionPixelSize(1, 0);
|
|
resolvedAttributes.recycle();
|
|
return width;
|
|
}
|
|
|
|
private boolean isLaunchFromHearingDevicePage() {
|
|
final Intent intent = getIntent();
|
|
if (intent == null) {
|
|
return false;
|
|
}
|
|
|
|
return intent.getIntExtra(MetricsFeatureProvider.EXTRA_SOURCE_METRICS_CATEGORY,
|
|
SettingsEnums.PAGE_UNKNOWN) == SettingsEnums.ACCESSIBILITY_HEARING_AID_SETTINGS;
|
|
}
|
|
|
|
@VisibleForTesting
|
|
void setTitleForInputDevice() {
|
|
if (StylusDevicesController.isDeviceStylus(mInputDevice, mCachedDevice)) {
|
|
// This will override the default R.string.device_details_title "Device Details"
|
|
// that will show on non-stylus bluetooth devices.
|
|
// That title is set via the manifest and also from BluetoothDeviceUpdater.
|
|
getActivity().setTitle(getContext().getString(R.string.stylus_device_details_title));
|
|
}
|
|
}
|
|
}
|