[Screen off unlock UDFPS] Fingerprint Settings integration 2/2
1. Integrate FingerprintSettings with Toggle
2. Sync the Toggle state with SettingProvider key
"screen_off_unlock_udfps"
Reference: go/udfps-aof #Settings UI design(Deck)
Bug: 373792870
Bug: 369939804
Bug: 369938501
Flag: android.hardware.biometrics.screen_off_unlock_udfps
Test: atest FingerprintSettingsFragmentTest
atest DevelopmentSettingsDashboardFragmentTest
atest FingerprintSettingsUnlockCategoryControllerTest
atest FingerprintSettingsScreenOffUnlockUdfpsPreferenceControllerTest
Test: adb shell settings put secure
screen_off_unlock_udfps <1|0>
Change-Id: I03794f53684bfb60b4a854e14507e67f60c55a7d
This commit is contained in:
@@ -20,6 +20,7 @@ package com.android.settings.biometrics.fingerprint;
|
||||
import static android.app.admin.DevicePolicyResources.Strings.Settings.FINGERPRINT_UNLOCK_DISABLED_EXPLANATION;
|
||||
import static android.app.admin.DevicePolicyResources.Strings.Settings.WORK_PROFILE_FINGERPRINT_LAST_DELETE_MESSAGE;
|
||||
import static android.app.admin.DevicePolicyResources.UNDEFINED;
|
||||
import static android.hardware.biometrics.Flags.screenOffUnlockUdfps;
|
||||
|
||||
import static com.android.settings.Utils.SETTINGS_PACKAGE_NAME;
|
||||
import static com.android.settings.Utils.isPrivateProfile;
|
||||
@@ -207,6 +208,17 @@ public class FingerprintSettings extends SubSettings {
|
||||
context,
|
||||
KEY_REQUIRE_SCREEN_ON_TO_AUTH
|
||||
));
|
||||
} else if (screenOffUnlockUdfps()) {
|
||||
controllers.add(
|
||||
new FingerprintUnlockCategoryController(
|
||||
context,
|
||||
KEY_FINGERPRINT_UNLOCK_CATEGORY
|
||||
));
|
||||
controllers.add(
|
||||
new FingerprintSettingsScreenOffUnlockUdfpsPreferenceController(
|
||||
context,
|
||||
KEY_SCREEN_OFF_FINGERPRINT_UNLOCK
|
||||
));
|
||||
}
|
||||
controllers.add(new FingerprintsEnrolledCategoryPreferenceController(context,
|
||||
KEY_FINGERPRINTS_ENROLLED_CATEGORY));
|
||||
@@ -233,6 +245,9 @@ public class FingerprintSettings extends SubSettings {
|
||||
@VisibleForTesting
|
||||
static final String KEY_REQUIRE_SCREEN_ON_TO_AUTH =
|
||||
"security_settings_require_screen_on_to_auth";
|
||||
@VisibleForTesting
|
||||
static final String KEY_SCREEN_OFF_FINGERPRINT_UNLOCK =
|
||||
"security_settings_screen_off_unlock_udfps";
|
||||
private static final String KEY_FINGERPRINTS_ENROLLED_CATEGORY =
|
||||
"security_settings_fingerprints_enrolled";
|
||||
private static final String KEY_FINGERPRINT_UNLOCK_CATEGORY =
|
||||
@@ -263,8 +278,11 @@ public class FingerprintSettings extends SubSettings {
|
||||
mFingerprintUnlockCategoryPreferenceController;
|
||||
private FingerprintSettingsRequireScreenOnToAuthPreferenceController
|
||||
mRequireScreenOnToAuthPreferenceController;
|
||||
private FingerprintSettingsScreenOffUnlockUdfpsPreferenceController
|
||||
mScreenOffUnlockUdfpsPreferenceController;
|
||||
private Preference mAddFingerprintPreference;
|
||||
private RestrictedSwitchPreference mRequireScreenOnToAuthPreference;
|
||||
private RestrictedSwitchPreference mScreenOffUnlockUdfpsPreference;
|
||||
private PreferenceCategory mFingerprintsEnrolledCategory;
|
||||
private PreferenceCategory mFingerprintUnlockCategory;
|
||||
private PreferenceCategory mFingerprintUnlockFooter;
|
||||
@@ -621,7 +639,7 @@ public class FingerprintSettings extends SubSettings {
|
||||
// This needs to be after setting ids, otherwise
|
||||
// |mRequireScreenOnToAuthPreferenceController.isChecked| is always checking the primary
|
||||
// user instead of the user with |mUserId|.
|
||||
if (isSfps()) {
|
||||
if (isSfps() || screenOffUnlockUdfps()) {
|
||||
scrollToPreference(fpPrefKey);
|
||||
addFingerprintUnlockCategory();
|
||||
}
|
||||
@@ -671,33 +689,46 @@ public class FingerprintSettings extends SubSettings {
|
||||
|
||||
private void addFingerprintUnlockCategory() {
|
||||
mFingerprintUnlockCategory = findPreference(KEY_FINGERPRINT_UNLOCK_CATEGORY);
|
||||
setupFingerprintUnlockCategoryPreferences();
|
||||
final Preference restToUnlockPreference = FeatureFactory.getFeatureFactory()
|
||||
.getFingerprintFeatureProvider()
|
||||
.getSfpsRestToUnlockFeature(getContext())
|
||||
.getRestToUnlockPreference(getContext());
|
||||
if (restToUnlockPreference != null) {
|
||||
// Use custom featured preference if any.
|
||||
mRequireScreenOnToAuthPreference.setTitle(restToUnlockPreference.getTitle());
|
||||
mRequireScreenOnToAuthPreference.setSummary(restToUnlockPreference.getSummary());
|
||||
mRequireScreenOnToAuthPreference.setChecked(
|
||||
((TwoStatePreference) restToUnlockPreference).isChecked());
|
||||
mRequireScreenOnToAuthPreference.setOnPreferenceChangeListener(
|
||||
restToUnlockPreference.getOnPreferenceChangeListener());
|
||||
if (isSfps()) {
|
||||
// For both SFPS "screen on to auth" and "rest to unlock"
|
||||
final Preference restToUnlockPreference = FeatureFactory.getFeatureFactory()
|
||||
.getFingerprintFeatureProvider()
|
||||
.getSfpsRestToUnlockFeature(getContext())
|
||||
.getRestToUnlockPreference(getContext());
|
||||
if (restToUnlockPreference != null) {
|
||||
// Use custom featured preference if any.
|
||||
mRequireScreenOnToAuthPreference.setTitle(restToUnlockPreference.getTitle());
|
||||
mRequireScreenOnToAuthPreference.setSummary(
|
||||
restToUnlockPreference.getSummary());
|
||||
mRequireScreenOnToAuthPreference.setChecked(
|
||||
((TwoStatePreference) restToUnlockPreference).isChecked());
|
||||
mRequireScreenOnToAuthPreference.setOnPreferenceChangeListener(
|
||||
restToUnlockPreference.getOnPreferenceChangeListener());
|
||||
}
|
||||
setupFingerprintUnlockCategoryPreferencesForScreenOnToAuth();
|
||||
} else if (screenOffUnlockUdfps()) {
|
||||
setupFingerprintUnlockCategoryPreferencesForScreenOffUnlock();
|
||||
}
|
||||
updateFingerprintUnlockCategoryVisibility();
|
||||
}
|
||||
|
||||
private void updateFingerprintUnlockCategoryVisibility() {
|
||||
final boolean mFingerprintUnlockCategoryAvailable =
|
||||
final boolean fingerprintUnlockCategoryAvailable =
|
||||
mFingerprintUnlockCategoryPreferenceController.isAvailable();
|
||||
if (mFingerprintUnlockCategory.isVisible() != mFingerprintUnlockCategoryAvailable) {
|
||||
mFingerprintUnlockCategory.setVisible(
|
||||
mFingerprintUnlockCategoryAvailable);
|
||||
if (mFingerprintUnlockCategory.isVisible() != fingerprintUnlockCategoryAvailable) {
|
||||
mFingerprintUnlockCategory.setVisible(fingerprintUnlockCategoryAvailable);
|
||||
}
|
||||
if (mRequireScreenOnToAuthPreferenceController != null) {
|
||||
mRequireScreenOnToAuthPreference.setVisible(
|
||||
mRequireScreenOnToAuthPreferenceController.isAvailable());
|
||||
}
|
||||
if (mScreenOffUnlockUdfpsPreferenceController != null) {
|
||||
mScreenOffUnlockUdfpsPreference.setVisible(
|
||||
mScreenOffUnlockUdfpsPreferenceController.isAvailable());
|
||||
}
|
||||
}
|
||||
|
||||
private void setupFingerprintUnlockCategoryPreferences() {
|
||||
private void setupFingerprintUnlockCategoryPreferencesForScreenOnToAuth() {
|
||||
mRequireScreenOnToAuthPreference = findPreference(KEY_REQUIRE_SCREEN_ON_TO_AUTH);
|
||||
mRequireScreenOnToAuthPreference.setChecked(
|
||||
mRequireScreenOnToAuthPreferenceController.isChecked());
|
||||
@@ -709,9 +740,21 @@ public class FingerprintSettings extends SubSettings {
|
||||
});
|
||||
}
|
||||
|
||||
private void setupFingerprintUnlockCategoryPreferencesForScreenOffUnlock() {
|
||||
mScreenOffUnlockUdfpsPreference = findPreference(KEY_SCREEN_OFF_FINGERPRINT_UNLOCK);
|
||||
mScreenOffUnlockUdfpsPreference.setChecked(
|
||||
mScreenOffUnlockUdfpsPreferenceController.isChecked());
|
||||
mScreenOffUnlockUdfpsPreference.setOnPreferenceChangeListener(
|
||||
(preference, newValue) -> {
|
||||
final boolean isChecked = ((TwoStatePreference) preference).isChecked();
|
||||
mScreenOffUnlockUdfpsPreferenceController.setChecked(!isChecked);
|
||||
return true;
|
||||
});
|
||||
}
|
||||
|
||||
private void updatePreferencesAfterFingerprintRemoved() {
|
||||
updateAddPreference();
|
||||
if (isSfps()) {
|
||||
if (isSfps() || screenOffUnlockUdfps()) {
|
||||
updateFingerprintUnlockCategoryVisibility();
|
||||
}
|
||||
updatePreferences();
|
||||
@@ -954,6 +997,18 @@ public class FingerprintSettings extends SubSettings {
|
||||
controller;
|
||||
}
|
||||
|
||||
}
|
||||
} else if (screenOffUnlockUdfps()) {
|
||||
for (AbstractPreferenceController controller : controllers) {
|
||||
if (controller.getPreferenceKey() == KEY_FINGERPRINT_UNLOCK_CATEGORY) {
|
||||
mFingerprintUnlockCategoryPreferenceController =
|
||||
(FingerprintUnlockCategoryController) controller;
|
||||
} else if (controller.getPreferenceKey() == KEY_SCREEN_OFF_FINGERPRINT_UNLOCK) {
|
||||
mScreenOffUnlockUdfpsPreferenceController =
|
||||
(FingerprintSettingsScreenOffUnlockUdfpsPreferenceController)
|
||||
controller;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
return controllers;
|
||||
@@ -1070,7 +1125,8 @@ public class FingerprintSettings extends SubSettings {
|
||||
} else if (requestCode == BIOMETRIC_AUTH_REQUEST) {
|
||||
mBiometricsAuthenticationRequested = false;
|
||||
if (resultCode != RESULT_OK) {
|
||||
if (resultCode == ConfirmDeviceCredentialActivity.BIOMETRIC_LOCKOUT_ERROR_RESULT) {
|
||||
if (resultCode
|
||||
== ConfirmDeviceCredentialActivity.BIOMETRIC_LOCKOUT_ERROR_RESULT) {
|
||||
IdentityCheckBiometricErrorDialog
|
||||
.showBiometricErrorDialogAndFinishActivityOnDismiss(getActivity(),
|
||||
Utils.BiometricStatus.LOCKOUT);
|
||||
@@ -1408,7 +1464,7 @@ public class FingerprintSettings extends SubSettings {
|
||||
getContext().getSystemService(DevicePolicyManager.class);
|
||||
String messageId =
|
||||
isProfileChallengeUser ? WORK_PROFILE_FINGERPRINT_LAST_DELETE_MESSAGE
|
||||
: UNDEFINED;
|
||||
: UNDEFINED;
|
||||
int defaultMessageId = isProfileChallengeUser
|
||||
? R.string.fingerprint_last_delete_message_profile_challenge
|
||||
: R.string.fingerprint_last_delete_message;
|
||||
@@ -1417,7 +1473,7 @@ public class FingerprintSettings extends SubSettings {
|
||||
.setTitle(title)
|
||||
.setMessage(devicePolicyManager.getResources().getString(
|
||||
messageId,
|
||||
() -> message + "\n\n" + getContext().getString(defaultMessageId)))
|
||||
() -> message + "\n\n" + getContext().getString(defaultMessageId)))
|
||||
.setPositiveButton(
|
||||
R.string.security_settings_fingerprint_enroll_dialog_delete,
|
||||
new DialogInterface.OnClickListener() {
|
||||
|
||||
@@ -0,0 +1,119 @@
|
||||
/*
|
||||
* 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.biometrics.fingerprint;
|
||||
|
||||
import static android.hardware.biometrics.Flags.screenOffUnlockUdfps;
|
||||
|
||||
import android.annotation.SuppressLint;
|
||||
import android.content.Context;
|
||||
import android.hardware.fingerprint.FingerprintManager;
|
||||
import android.os.UserHandle;
|
||||
import android.provider.Settings;
|
||||
|
||||
import androidx.annotation.NonNull;
|
||||
import androidx.preference.Preference;
|
||||
|
||||
import com.android.internal.annotations.VisibleForTesting;
|
||||
import com.android.settings.Utils;
|
||||
import com.android.settings.search.BaseSearchIndexProvider;
|
||||
import com.android.settingslib.search.SearchIndexable;
|
||||
|
||||
/**
|
||||
* Preference controller that controls whether show screen off UDFPS unlock toggle for users to
|
||||
* turn this feature ON or OFF
|
||||
*/
|
||||
@SearchIndexable
|
||||
public class FingerprintSettingsScreenOffUnlockUdfpsPreferenceController
|
||||
extends FingerprintSettingsPreferenceController {
|
||||
private static final String TAG =
|
||||
"FingerprintSettingsScreenOffUnlockUdfpsPreferenceController";
|
||||
|
||||
@VisibleForTesting
|
||||
protected FingerprintManager mFingerprintManager;
|
||||
|
||||
public FingerprintSettingsScreenOffUnlockUdfpsPreferenceController(
|
||||
@NonNull Context context, @NonNull String prefKey) {
|
||||
super(context, prefKey);
|
||||
mFingerprintManager = Utils.getFingerprintManagerOrNull(context);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isChecked() {
|
||||
if (!FingerprintSettings.isFingerprintHardwareDetected(mContext)) {
|
||||
return false;
|
||||
} else if (getRestrictingAdmin() != null) {
|
||||
return false;
|
||||
}
|
||||
final boolean defEnabled = mContext.getResources().getBoolean(
|
||||
com.android.internal.R.bool.config_screen_off_udfps_enabled);
|
||||
final int value = Settings.Secure.getIntForUser(
|
||||
mContext.getContentResolver(),
|
||||
Settings.Secure.SCREEN_OFF_UNLOCK_UDFPS_ENABLED,
|
||||
defEnabled ? 1 : 0 /* config_screen_off_udfps_enabled */,
|
||||
getUserHandle());
|
||||
return value == 1;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean setChecked(boolean isChecked) {
|
||||
Settings.Secure.putIntForUser(
|
||||
mContext.getContentResolver(),
|
||||
Settings.Secure.SCREEN_OFF_UNLOCK_UDFPS_ENABLED,
|
||||
isChecked ? 1 : 0,
|
||||
getUserHandle());
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void updateState(@NonNull Preference preference) {
|
||||
super.updateState(preference);
|
||||
if (!FingerprintSettings.isFingerprintHardwareDetected(mContext)) {
|
||||
preference.setEnabled(false);
|
||||
} else if (!mFingerprintManager.hasEnrolledTemplates(getUserId())) {
|
||||
preference.setEnabled(false);
|
||||
} else if (getRestrictingAdmin() != null) {
|
||||
preference.setEnabled(false);
|
||||
} else {
|
||||
preference.setEnabled(true);
|
||||
}
|
||||
}
|
||||
|
||||
@SuppressLint("MissingPermission")
|
||||
@Override
|
||||
public int getAvailabilityStatus() {
|
||||
if (mFingerprintManager != null
|
||||
&& mFingerprintManager.isHardwareDetected()
|
||||
&& screenOffUnlockUdfps()
|
||||
&& !mFingerprintManager.isPowerbuttonFps()) {
|
||||
return mFingerprintManager.hasEnrolledTemplates(getUserId())
|
||||
? AVAILABLE : CONDITIONALLY_UNAVAILABLE;
|
||||
} else {
|
||||
return UNSUPPORTED_ON_DEVICE;
|
||||
}
|
||||
}
|
||||
|
||||
private int getUserHandle() {
|
||||
return UserHandle.of(getUserId()).getIdentifier();
|
||||
}
|
||||
|
||||
/**
|
||||
* This feature is not directly searchable.
|
||||
*/
|
||||
public static final BaseSearchIndexProvider SEARCH_INDEX_DATA_PROVIDER =
|
||||
new BaseSearchIndexProvider() {};
|
||||
|
||||
}
|
||||
@@ -16,6 +16,8 @@
|
||||
|
||||
package com.android.settings.biometrics.fingerprint;
|
||||
|
||||
import static android.hardware.biometrics.Flags.screenOffUnlockUdfps;
|
||||
|
||||
import android.content.Context;
|
||||
import android.hardware.fingerprint.FingerprintManager;
|
||||
|
||||
@@ -42,7 +44,7 @@ public class FingerprintUnlockCategoryController extends BasePreferenceControlle
|
||||
public int getAvailabilityStatus() {
|
||||
if (mFingerprintManager != null
|
||||
&& mFingerprintManager.isHardwareDetected()
|
||||
&& mFingerprintManager.isPowerbuttonFps()) {
|
||||
&& (mFingerprintManager.isPowerbuttonFps() || screenOffUnlockUdfps())) {
|
||||
return mFingerprintManager.hasEnrolledTemplates(getUserId())
|
||||
? AVAILABLE : CONDITIONALLY_UNAVAILABLE;
|
||||
} else {
|
||||
|
||||
Reference in New Issue
Block a user