Add a stylus controller to Bluetooth Device Details.
Feature is currently flagged behing SETTINGS_SHOW_STYLUS_PREFERENCES. Bug: 251201006 Test: SettingsRoboTests StylusDevicesControllerTest DD: go/stylus-connected-devices-doc Change-Id: I438b7fe5ca1c94f9dfb506c8918d0e6cb005ca33
This commit is contained in:
@@ -23,12 +23,15 @@ import android.app.settings.SettingsEnums;
|
||||
import android.bluetooth.BluetoothDevice;
|
||||
import android.content.Context;
|
||||
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;
|
||||
@@ -37,9 +40,11 @@ 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.overlay.FeatureFactory;
|
||||
@@ -72,6 +77,7 @@ public class BluetoothDeviceDetailsFragment extends RestrictedDashboardFragment
|
||||
CachedBluetoothDevice getDevice(String deviceAddress);
|
||||
|
||||
LocalBluetoothManager getManager(Context context);
|
||||
|
||||
UserManager getUserManager();
|
||||
}
|
||||
|
||||
@@ -85,6 +91,9 @@ public class BluetoothDeviceDetailsFragment extends RestrictedDashboardFragment
|
||||
@VisibleForTesting
|
||||
CachedBluetoothDevice mCachedDevice;
|
||||
|
||||
@Nullable
|
||||
InputDevice mInputDevice;
|
||||
|
||||
private UserManager mUserManager;
|
||||
|
||||
public BluetoothDeviceDetailsFragment() {
|
||||
@@ -118,6 +127,21 @@ public class BluetoothDeviceDetailsFragment extends RestrictedDashboardFragment
|
||||
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);
|
||||
@@ -132,6 +156,12 @@ public class BluetoothDeviceDetailsFragment extends RestrictedDashboardFragment
|
||||
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
|
||||
@@ -191,6 +221,12 @@ public class BluetoothDeviceDetailsFragment extends RestrictedDashboardFragment
|
||||
}
|
||||
};
|
||||
|
||||
@Override
|
||||
public void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
setTitleForInputDevice();
|
||||
}
|
||||
|
||||
@Override
|
||||
public View onCreateView(LayoutInflater inflater, ViewGroup container,
|
||||
Bundle savedInstanceState) {
|
||||
@@ -269,6 +305,7 @@ public class BluetoothDeviceDetailsFragment extends RestrictedDashboardFragment
|
||||
mCachedDevice, lifecycle));
|
||||
controllers.add(new BluetoothDetailsMacAddressController(context, this, mCachedDevice,
|
||||
lifecycle));
|
||||
controllers.add(new StylusDevicesController(context, mInputDevice, lifecycle));
|
||||
controllers.add(new BluetoothDetailsRelatedToolsController(context, this, mCachedDevice,
|
||||
lifecycle));
|
||||
controllers.add(new BluetoothDetailsPairOtherController(context, this, mCachedDevice,
|
||||
@@ -289,4 +326,16 @@ public class BluetoothDeviceDetailsFragment extends RestrictedDashboardFragment
|
||||
resolvedAttributes.recycle();
|
||||
return width;
|
||||
}
|
||||
|
||||
@VisibleForTesting
|
||||
void setTitleForInputDevice() {
|
||||
// TODO(b/254835745) once source filter for bt stylus merged
|
||||
// && mInputDevice.supportsSource(InputDevice.SOURCE_BLUETOOTH_STYLUS))
|
||||
if (mInputDevice != null) {
|
||||
// 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));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -0,0 +1,199 @@
|
||||
/*
|
||||
* 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.connecteddevice.stylus;
|
||||
|
||||
import android.app.role.RoleManager;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.pm.ApplicationInfo;
|
||||
import android.content.pm.PackageManager;
|
||||
import android.provider.Settings;
|
||||
import android.util.Log;
|
||||
import android.view.InputDevice;
|
||||
|
||||
import androidx.annotation.Nullable;
|
||||
import androidx.annotation.VisibleForTesting;
|
||||
import androidx.preference.Preference;
|
||||
import androidx.preference.PreferenceCategory;
|
||||
import androidx.preference.PreferenceScreen;
|
||||
import androidx.preference.SwitchPreference;
|
||||
|
||||
import com.android.settings.R;
|
||||
import com.android.settingslib.core.AbstractPreferenceController;
|
||||
import com.android.settingslib.core.lifecycle.Lifecycle;
|
||||
import com.android.settingslib.core.lifecycle.LifecycleObserver;
|
||||
import com.android.settingslib.core.lifecycle.events.OnResume;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* This class adds stylus preferences.
|
||||
*/
|
||||
public class StylusDevicesController extends AbstractPreferenceController implements
|
||||
Preference.OnPreferenceClickListener, LifecycleObserver, OnResume {
|
||||
|
||||
@VisibleForTesting
|
||||
static final String KEY_STYLUS = "device_stylus";
|
||||
@VisibleForTesting
|
||||
static final String KEY_HANDWRITING = "handwriting_switch";
|
||||
@VisibleForTesting
|
||||
static final String KEY_IGNORE_BUTTON = "ignore_button";
|
||||
@VisibleForTesting
|
||||
static final String KEY_DEFAULT_NOTES = "default_notes";
|
||||
|
||||
private static final String TAG = "StylusDevicesController";
|
||||
|
||||
@Nullable
|
||||
private final InputDevice mInputDevice;
|
||||
|
||||
@VisibleForTesting
|
||||
PreferenceCategory mPreferencesContainer;
|
||||
|
||||
public StylusDevicesController(Context context, InputDevice inputDevice, Lifecycle lifecycle) {
|
||||
super(context);
|
||||
mInputDevice = inputDevice;
|
||||
lifecycle.addObserver(this);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isAvailable() {
|
||||
return mInputDevice != null && mInputDevice.supportsSource(InputDevice.SOURCE_STYLUS);
|
||||
}
|
||||
|
||||
@Nullable
|
||||
private Preference createDefaultNotesPreference() {
|
||||
RoleManager rm = mContext.getSystemService(RoleManager.class);
|
||||
if (rm == null) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// TODO(b/254834764): replace with notes role once merged
|
||||
List<String> roleHolders = rm.getRoleHoldersAsUser(RoleManager.ROLE_ASSISTANT,
|
||||
mContext.getUser());
|
||||
if (roleHolders.isEmpty()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
String packageName = roleHolders.get(0);
|
||||
PackageManager pm = mContext.getPackageManager();
|
||||
String appName = packageName;
|
||||
try {
|
||||
ApplicationInfo ai = pm.getApplicationInfo(packageName,
|
||||
PackageManager.ApplicationInfoFlags.of(0));
|
||||
appName = ai == null ? packageName : pm.getApplicationLabel(ai).toString();
|
||||
} catch (PackageManager.NameNotFoundException e) {
|
||||
Log.e(TAG, "Notes role package not found.");
|
||||
}
|
||||
|
||||
Preference pref = new Preference(mContext);
|
||||
pref.setKey(KEY_DEFAULT_NOTES);
|
||||
pref.setTitle(mContext.getString(R.string.stylus_default_notes_app));
|
||||
pref.setIcon(R.drawable.ic_article);
|
||||
pref.setEnabled(true);
|
||||
pref.setSummary(appName);
|
||||
return pref;
|
||||
}
|
||||
|
||||
private SwitchPreference createHandwritingPreference() {
|
||||
SwitchPreference pref = new SwitchPreference(mContext);
|
||||
pref.setKey(KEY_HANDWRITING);
|
||||
pref.setTitle(mContext.getString(R.string.stylus_textfield_handwriting));
|
||||
pref.setIcon(R.drawable.ic_text_fields_alt);
|
||||
pref.setOnPreferenceClickListener(this);
|
||||
pref.setChecked(Settings.Global.getInt(mContext.getContentResolver(),
|
||||
Settings.Global.STYLUS_HANDWRITING_ENABLED, 0) == 1);
|
||||
return pref;
|
||||
}
|
||||
|
||||
private SwitchPreference createButtonPressPreference() {
|
||||
SwitchPreference pref = new SwitchPreference(mContext);
|
||||
pref.setKey(KEY_IGNORE_BUTTON);
|
||||
pref.setTitle(mContext.getString(R.string.stylus_ignore_button));
|
||||
pref.setIcon(R.drawable.ic_block);
|
||||
pref.setOnPreferenceClickListener(this);
|
||||
return pref;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onPreferenceClick(Preference preference) {
|
||||
String key = preference.getKey();
|
||||
|
||||
switch (key) {
|
||||
case KEY_DEFAULT_NOTES:
|
||||
PackageManager pm = mContext.getPackageManager();
|
||||
String packageName = pm.getPermissionControllerPackageName();
|
||||
// TODO(b/254834764): replace with notes role once merged
|
||||
Intent intent = new Intent(Intent.ACTION_MANAGE_DEFAULT_APP).setPackage(
|
||||
packageName).putExtra(Intent.EXTRA_ROLE_NAME, RoleManager.ROLE_ASSISTANT);
|
||||
mContext.startActivity(intent);
|
||||
break;
|
||||
case KEY_HANDWRITING:
|
||||
Settings.Global.putInt(mContext.getContentResolver(),
|
||||
Settings.Global.STYLUS_HANDWRITING_ENABLED,
|
||||
((SwitchPreference) preference).isChecked() ? 1 : 0);
|
||||
break;
|
||||
case KEY_IGNORE_BUTTON:
|
||||
// TODO(b/251199452): to turn off stylus button presses
|
||||
break;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public final void displayPreference(PreferenceScreen screen) {
|
||||
mPreferencesContainer = (PreferenceCategory) screen.findPreference(getPreferenceKey());
|
||||
super.displayPreference(screen);
|
||||
|
||||
refresh();
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getPreferenceKey() {
|
||||
return KEY_STYLUS;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onResume() {
|
||||
refresh();
|
||||
}
|
||||
|
||||
private void refresh() {
|
||||
if (!isAvailable()) return;
|
||||
|
||||
if (mInputDevice.getBluetoothAddress() != null) {
|
||||
Preference notesPref = mPreferencesContainer.findPreference(KEY_DEFAULT_NOTES);
|
||||
if (notesPref == null) {
|
||||
notesPref = createDefaultNotesPreference();
|
||||
if (notesPref != null) {
|
||||
mPreferencesContainer.addPreference(notesPref);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Preference handwritingPref = mPreferencesContainer.findPreference(KEY_HANDWRITING);
|
||||
// TODO(b/255732419): add proper InputMethodInfo conditional to show or hide
|
||||
// InputMethodManager imm = mContext.getSystemService(InputMethodManager.class);
|
||||
if (handwritingPref == null) {
|
||||
mPreferencesContainer.addPreference(createHandwritingPreference());
|
||||
}
|
||||
|
||||
Preference buttonPref = mPreferencesContainer.findPreference(KEY_IGNORE_BUTTON);
|
||||
if (buttonPref == null) {
|
||||
mPreferencesContainer.addPreference(createButtonPressPreference());
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user