Merge "Update face & fingerprint pref logic" into main

This commit is contained in:
Joshua Mccloskey
2024-05-08 18:08:06 +00:00
committed by Android (Google) Code Review
18 changed files with 213 additions and 13 deletions

View File

@@ -15,12 +15,17 @@
*/ */
package com.android.settings.biometrics.combination; package com.android.settings.biometrics.combination;
import android.app.admin.DevicePolicyManager;
import android.content.Context; import android.content.Context;
import androidx.lifecycle.Lifecycle; import androidx.lifecycle.Lifecycle;
import androidx.preference.Preference;
import com.android.settings.Utils; import com.android.settings.Utils;
import com.android.settings.biometrics.face.FaceStatusPreferenceController; import com.android.settings.biometrics.face.FaceStatusPreferenceController;
import com.android.settingslib.RestrictedLockUtils;
import com.android.settingslib.RestrictedLockUtilsInternal;
import com.android.settingslib.RestrictedPreference;
/** /**
* Preference controller for face settings within the biometrics settings page, that controls the * Preference controller for face settings within the biometrics settings page, that controls the
@@ -37,6 +42,20 @@ public class BiometricFaceStatusPreferenceController extends FaceStatusPreferenc
super(context, key, lifecycle); super(context, key, lifecycle);
} }
@Override
public void updateState(Preference preference) {
super.updateState(preference);
final boolean isFaceEnrolled = mFaceStatusUtils.hasEnrolled();
final RestrictedLockUtils.EnforcedAdmin admin =
RestrictedLockUtilsInternal.checkIfKeyguardFeaturesDisabled(
mContext, DevicePolicyManager.KEYGUARD_DISABLE_FACE, getUserId());
if (admin != null && !isFaceEnrolled) {
((RestrictedPreference) preference).setDisabledByAdmin(admin);
} else {
preference.setEnabled(true);
}
}
@Override @Override
protected boolean isDeviceSupported() { protected boolean isDeviceSupported() {
return Utils.isMultipleBiometricsSupported(mContext) && isHardwareSupported(); return Utils.isMultipleBiometricsSupported(mContext) && isHardwareSupported();

View File

@@ -15,12 +15,17 @@
*/ */
package com.android.settings.biometrics.combination; package com.android.settings.biometrics.combination;
import android.app.admin.DevicePolicyManager;
import android.content.Context; import android.content.Context;
import androidx.lifecycle.Lifecycle; import androidx.lifecycle.Lifecycle;
import androidx.preference.Preference;
import com.android.settings.Utils; import com.android.settings.Utils;
import com.android.settings.biometrics.fingerprint.FingerprintStatusPreferenceController; import com.android.settings.biometrics.fingerprint.FingerprintStatusPreferenceController;
import com.android.settingslib.RestrictedLockUtils;
import com.android.settingslib.RestrictedLockUtilsInternal;
import com.android.settingslib.RestrictedPreference;
/** /**
* Preference controller for fingerprint settings within the biometrics settings page of work * Preference controller for fingerprint settings within the biometrics settings page of work
@@ -38,6 +43,20 @@ public class BiometricFingerprintStatusPreferenceController extends
super(context, key, lifecycle); super(context, key, lifecycle);
} }
@Override
public void updateState(Preference preference) {
super.updateState(preference);
final boolean isFingerprintEnrolled = mFingerprintStatusUtils.hasEnrolled();
final RestrictedLockUtils.EnforcedAdmin admin =
RestrictedLockUtilsInternal.checkIfKeyguardFeaturesDisabled(
mContext, DevicePolicyManager.KEYGUARD_DISABLE_FINGERPRINT, getUserId());
if (admin != null && !isFingerprintEnrolled) {
((RestrictedPreference) preference).setDisabledByAdmin(admin);
} else {
preference.setEnabled(true);
}
}
@Override @Override
protected boolean isDeviceSupported() { protected boolean isDeviceSupported() {
return Utils.isMultipleBiometricsSupported(mContext) && isHardwareSupported(); return Utils.isMultipleBiometricsSupported(mContext) && isHardwareSupported();

View File

@@ -36,6 +36,7 @@ import android.util.Log;
import android.widget.Button; import android.widget.Button;
import androidx.preference.Preference; import androidx.preference.Preference;
import androidx.preference.PreferenceCategory;
import com.android.settings.R; import com.android.settings.R;
import com.android.settings.SettingsActivity; import com.android.settings.SettingsActivity;
@@ -46,6 +47,7 @@ import com.android.settings.dashboard.DashboardFragment;
import com.android.settings.overlay.FeatureFactory; import com.android.settings.overlay.FeatureFactory;
import com.android.settings.password.ChooseLockSettingsHelper; import com.android.settings.password.ChooseLockSettingsHelper;
import com.android.settings.search.BaseSearchIndexProvider; import com.android.settings.search.BaseSearchIndexProvider;
import com.android.settingslib.RestrictedLockUtilsInternal;
import com.android.settingslib.core.AbstractPreferenceController; import com.android.settingslib.core.AbstractPreferenceController;
import com.android.settingslib.search.SearchIndexable; import com.android.settingslib.search.SearchIndexable;
import com.android.settingslib.widget.LayoutPreference; import com.android.settingslib.widget.LayoutPreference;
@@ -68,6 +70,8 @@ public class FaceSettings extends DashboardFragment {
"security_settings_face_delete_faces_container"; "security_settings_face_delete_faces_container";
private static final String PREF_KEY_ENROLL_FACE_UNLOCK = private static final String PREF_KEY_ENROLL_FACE_UNLOCK =
"security_settings_face_enroll_faces_container"; "security_settings_face_enroll_faces_container";
public static final String SECURITY_SETTINGS_FACE_MANAGE_CATEGORY =
"security_settings_face_manage_category";
private UserManager mUserManager; private UserManager mUserManager;
private FaceManager mFaceManager; private FaceManager mFaceManager;
@@ -175,6 +179,8 @@ public class FaceSettings extends DashboardFragment {
: use(FaceSettingsLockscreenBypassPreferenceController.class); : use(FaceSettingsLockscreenBypassPreferenceController.class);
mLockscreenController.setUserId(mUserId); mLockscreenController.setUserId(mUserId);
final PreferenceCategory managePref =
findPreference(SECURITY_SETTINGS_FACE_MANAGE_CATEGORY);
Preference keyguardPref = findPreference(FaceSettingsKeyguardPreferenceController.KEY); Preference keyguardPref = findPreference(FaceSettingsKeyguardPreferenceController.KEY);
Preference appPref = findPreference(FaceSettingsAppPreferenceController.KEY); Preference appPref = findPreference(FaceSettingsAppPreferenceController.KEY);
Preference attentionPref = findPreference(FaceSettingsAttentionPreferenceController.KEY); Preference attentionPref = findPreference(FaceSettingsAttentionPreferenceController.KEY);
@@ -184,6 +190,14 @@ public class FaceSettings extends DashboardFragment {
mTogglePreferences = new ArrayList<>( mTogglePreferences = new ArrayList<>(
Arrays.asList(keyguardPref, appPref, attentionPref, confirmPref, bypassPref)); Arrays.asList(keyguardPref, appPref, attentionPref, confirmPref, bypassPref));
if (RestrictedLockUtilsInternal.checkIfKeyguardFeaturesDisabled(
getContext(), DevicePolicyManager.KEYGUARD_DISABLE_FACE, mUserId) != null) {
managePref.setTitle(getString(
com.android.settingslib.widget.restricted.R.string.disabled_by_admin));
} else {
managePref.setTitle(R.string.security_settings_face_settings_preferences_category);
}
mRemoveButton = findPreference(FaceSettingsRemoveButtonPreferenceController.KEY); mRemoveButton = findPreference(FaceSettingsRemoveButtonPreferenceController.KEY);
mEnrollButton = findPreference(FaceSettingsEnrollButtonPreferenceController.KEY); mEnrollButton = findPreference(FaceSettingsEnrollButtonPreferenceController.KEY);

View File

@@ -24,6 +24,7 @@ import android.hardware.face.FaceManager.GetFeatureCallback;
import android.hardware.face.FaceManager.SetFeatureCallback; import android.hardware.face.FaceManager.SetFeatureCallback;
import android.provider.Settings; import android.provider.Settings;
import androidx.preference.Preference;
import androidx.preference.PreferenceScreen; import androidx.preference.PreferenceScreen;
import androidx.preference.TwoStatePreference; import androidx.preference.TwoStatePreference;
@@ -66,8 +67,12 @@ public class FaceSettingsAttentionPreferenceController extends FaceSettingsPrefe
requireAttentionEnabled = featureState[i]; requireAttentionEnabled = featureState[i];
} }
} }
mPreference.setEnabled(success);
mPreference.setChecked(requireAttentionEnabled); mPreference.setChecked(requireAttentionEnabled);
if (getRestrictingAdmin() != null) {
mPreference.setEnabled(false);
} else {
mPreference.setEnabled(success);
}
} }
}; };

View File

@@ -72,6 +72,8 @@ public class FaceSettingsConfirmPreferenceController extends FaceSettingsPrefere
preference.setEnabled(false); preference.setEnabled(false);
} else if (!mFaceManager.hasEnrolledTemplates(getUserId())) { } else if (!mFaceManager.hasEnrolledTemplates(getUserId())) {
preference.setEnabled(false); preference.setEnabled(false);
} else if (getRestrictingAdmin() != null) {
preference.setEnabled(false);
} else { } else {
preference.setEnabled(true); preference.setEnabled(true);
} }

View File

@@ -18,6 +18,7 @@ package com.android.settings.biometrics.face;
import static com.android.settings.Utils.SETTINGS_PACKAGE_NAME; import static com.android.settings.Utils.SETTINGS_PACKAGE_NAME;
import android.app.admin.DevicePolicyManager;
import android.content.Context; import android.content.Context;
import android.content.Intent; import android.content.Intent;
import android.view.View; import android.view.View;
@@ -28,6 +29,7 @@ import androidx.preference.Preference;
import com.android.settings.R; import com.android.settings.R;
import com.android.settings.core.BasePreferenceController; import com.android.settings.core.BasePreferenceController;
import com.android.settings.password.ChooseLockSettingsHelper; import com.android.settings.password.ChooseLockSettingsHelper;
import com.android.settingslib.RestrictedLockUtilsInternal;
import com.android.settingslib.widget.LayoutPreference; import com.android.settingslib.widget.LayoutPreference;
import com.google.android.setupdesign.util.ButtonStyler; import com.google.android.setupdesign.util.ButtonStyler;
@@ -71,6 +73,10 @@ public class FaceSettingsEnrollButtonPreferenceController extends BasePreference
} }
mButton.setOnClickListener(this); mButton.setOnClickListener(this);
final boolean isDeviceOwnerBlockingAuth =
RestrictedLockUtilsInternal.checkIfKeyguardFeaturesDisabled(
mContext, DevicePolicyManager.KEYGUARD_DISABLE_FACE, mUserId) != null;
mButton.setEnabled(!isDeviceOwnerBlockingAuth);
} }
@Override @Override

View File

@@ -27,8 +27,6 @@ import androidx.preference.Preference;
import com.android.internal.annotations.VisibleForTesting; import com.android.internal.annotations.VisibleForTesting;
import com.android.settings.Utils; import com.android.settings.Utils;
import com.android.settingslib.RestrictedLockUtils.EnforcedAdmin;
import com.android.settingslib.RestrictedSwitchPreference;
public class FaceSettingsLockscreenBypassPreferenceController public class FaceSettingsLockscreenBypassPreferenceController
extends FaceSettingsPreferenceController { extends FaceSettingsPreferenceController {
@@ -68,12 +66,11 @@ public class FaceSettingsLockscreenBypassPreferenceController
@Override @Override
public void updateState(Preference preference) { public void updateState(Preference preference) {
EnforcedAdmin admin;
super.updateState(preference); super.updateState(preference);
if (!FaceSettings.isFaceHardwareDetected(mContext)) { if (!FaceSettings.isFaceHardwareDetected(mContext)) {
preference.setEnabled(false); preference.setEnabled(false);
} else if ((admin = getRestrictingAdmin()) != null) { } else if (getRestrictingAdmin() != null) {
((RestrictedSwitchPreference) preference).setDisabledByAdmin(admin); preference.setEnabled(false);
} else if (!mFaceManager.hasEnrolledTemplates(getUserId())) { } else if (!mFaceManager.hasEnrolledTemplates(getUserId())) {
preference.setEnabled(false); preference.setEnabled(false);
} else { } else {

View File

@@ -42,7 +42,7 @@ public class FaceStatusPreferenceController extends BiometricStatusPreferenceCon
@VisibleForTesting @VisibleForTesting
RestrictedPreference mPreference; RestrictedPreference mPreference;
private PreferenceScreen mPreferenceScreen; private PreferenceScreen mPreferenceScreen;
private final FaceStatusUtils mFaceStatusUtils; protected final FaceStatusUtils mFaceStatusUtils;
public FaceStatusPreferenceController(Context context) { public FaceStatusPreferenceController(Context context) {
this(context, KEY_FACE_SETTINGS, null /* lifecycle */); this(context, KEY_FACE_SETTINGS, null /* lifecycle */);

View File

@@ -16,6 +16,7 @@
package com.android.settings.biometrics.face; package com.android.settings.biometrics.face;
import android.app.admin.DevicePolicyManager;
import android.content.Context; import android.content.Context;
import android.hardware.biometrics.BiometricAuthenticator; import android.hardware.biometrics.BiometricAuthenticator;
import android.hardware.face.FaceManager; import android.hardware.face.FaceManager;
@@ -26,6 +27,7 @@ import com.android.settings.Settings;
import com.android.settings.Utils; import com.android.settings.Utils;
import com.android.settings.biometrics.ParentalControlsUtils; import com.android.settings.biometrics.ParentalControlsUtils;
import com.android.settingslib.RestrictedLockUtils.EnforcedAdmin; import com.android.settingslib.RestrictedLockUtils.EnforcedAdmin;
import com.android.settingslib.RestrictedLockUtilsInternal;
/** /**
* Utilities for face details shared between Security Settings and Safety Center. * Utilities for face details shared between Security Settings and Safety Center.
@@ -75,10 +77,15 @@ public class FaceStatusUtils {
* Returns the summary of face settings entity. * Returns the summary of face settings entity.
*/ */
public String getSummary() { public String getSummary() {
if (shouldShowDisabledByAdminStr()) {
return mContext.getString(
com.android.settingslib.widget.restricted.R.string.disabled_by_admin);
} else {
return mContext.getResources().getString(hasEnrolled() return mContext.getResources().getString(hasEnrolled()
? R.string.security_settings_face_preference_summary ? R.string.security_settings_face_preference_summary
: R.string.security_settings_face_preference_summary_none); : R.string.security_settings_face_preference_summary_none);
} }
}
/** /**
* Returns the class name of the Settings page corresponding to face settings. * Returns the class name of the Settings page corresponding to face settings.
@@ -94,4 +101,12 @@ public class FaceStatusUtils {
public boolean hasEnrolled() { public boolean hasEnrolled() {
return mFaceManager.hasEnrolledTemplates(mUserId); return mFaceManager.hasEnrolledTemplates(mUserId);
} }
/**
* Indicates if the face feature is enabled or disabled by the Device Admin.
*/
private boolean shouldShowDisabledByAdminStr() {
return RestrictedLockUtilsInternal.checkIfKeyguardFeaturesDisabled(
mContext, DevicePolicyManager.KEYGUARD_DISABLE_FACE, mUserId) != null;
}
} }

View File

@@ -686,10 +686,16 @@ public class FingerprintSettings extends SubSettings {
// retryFingerprint() will be called when remove finishes // retryFingerprint() will be called when remove finishes
// need to disable enroll or have a way to determine if enroll is in progress // need to disable enroll or have a way to determine if enroll is in progress
final boolean removalInProgress = mRemovalSidecar.inProgress(); final boolean removalInProgress = mRemovalSidecar.inProgress();
final boolean isDeviceOwnerBlockingAuth =
RestrictedLockUtilsInternal.checkIfKeyguardFeaturesDisabled(
getContext(), DevicePolicyManager.KEYGUARD_DISABLE_FINGERPRINT,
mUserId) != null;
CharSequence maxSummary = tooMany ? CharSequence maxSummary = tooMany ?
getContext().getString(R.string.fingerprint_add_max, max) : ""; getContext().getString(R.string.fingerprint_add_max, max) : "";
mAddFingerprintPreference.setSummary(maxSummary); mAddFingerprintPreference.setSummary(maxSummary);
mAddFingerprintPreference.setEnabled(!tooMany && !removalInProgress && mToken != null); mAddFingerprintPreference.setEnabled(!isDeviceOwnerBlockingAuth
&& !tooMany && !removalInProgress && mToken != null);
} }
private void createFooterPreference(PreferenceGroup root) { private void createFooterPreference(PreferenceGroup root) {

View File

@@ -86,6 +86,8 @@ public class FingerprintSettingsRequireScreenOnToAuthPreferenceController
preference.setEnabled(false); preference.setEnabled(false);
} else if (!mFingerprintManager.hasEnrolledTemplates(getUserId())) { } else if (!mFingerprintManager.hasEnrolledTemplates(getUserId())) {
preference.setEnabled(false); preference.setEnabled(false);
} else if (getRestrictingAdmin() != null) {
preference.setEnabled(false);
} else { } else {
preference.setEnabled(true); preference.setEnabled(true);
} }

View File

@@ -40,7 +40,7 @@ public class FingerprintStatusPreferenceController extends BiometricStatusPrefer
protected final FingerprintManager mFingerprintManager; protected final FingerprintManager mFingerprintManager;
@VisibleForTesting @VisibleForTesting
RestrictedPreference mPreference; RestrictedPreference mPreference;
private final FingerprintStatusUtils mFingerprintStatusUtils; protected final FingerprintStatusUtils mFingerprintStatusUtils;
private PreferenceScreen mPreferenceScreen; private PreferenceScreen mPreferenceScreen;
public FingerprintStatusPreferenceController(Context context) { public FingerprintStatusPreferenceController(Context context) {

View File

@@ -16,6 +16,7 @@
package com.android.settings.biometrics.fingerprint; package com.android.settings.biometrics.fingerprint;
import android.app.admin.DevicePolicyManager;
import android.content.Context; import android.content.Context;
import android.hardware.biometrics.BiometricAuthenticator; import android.hardware.biometrics.BiometricAuthenticator;
import android.hardware.fingerprint.FingerprintManager; import android.hardware.fingerprint.FingerprintManager;
@@ -25,6 +26,7 @@ import com.android.settings.R;
import com.android.settings.Utils; import com.android.settings.Utils;
import com.android.settings.biometrics.ParentalControlsUtils; import com.android.settings.biometrics.ParentalControlsUtils;
import com.android.settingslib.RestrictedLockUtils.EnforcedAdmin; import com.android.settingslib.RestrictedLockUtils.EnforcedAdmin;
import com.android.settingslib.RestrictedLockUtilsInternal;
import com.android.settingslib.utils.StringUtil; import com.android.settingslib.utils.StringUtil;
/** /**
@@ -76,6 +78,10 @@ public class FingerprintStatusUtils {
* Returns the summary of fingerprint settings entity. * Returns the summary of fingerprint settings entity.
*/ */
public String getSummary() { public String getSummary() {
if (shouldShowDisabledByAdminStr()) {
return mContext.getString(
com.android.settingslib.widget.restricted.R.string.disabled_by_admin);
}
if (hasEnrolled()) { if (hasEnrolled()) {
final int numEnrolled = mFingerprintManager.getEnrolledFingerprints(mUserId).size(); final int numEnrolled = mFingerprintManager.getEnrolledFingerprints(mUserId).size();
return StringUtil.getIcuPluralsString(mContext, numEnrolled, return StringUtil.getIcuPluralsString(mContext, numEnrolled,
@@ -99,4 +105,12 @@ public class FingerprintStatusUtils {
public boolean hasEnrolled() { public boolean hasEnrolled() {
return mFingerprintManager.hasEnrolledFingerprints(mUserId); return mFingerprintManager.hasEnrolledFingerprints(mUserId);
} }
/**
* Indicates if the fingerprint feature should show the "Disabled by Admin" string.
*/
private boolean shouldShowDisabledByAdminStr() {
return RestrictedLockUtilsInternal.checkIfKeyguardFeaturesDisabled(
mContext, DevicePolicyManager.KEYGUARD_DISABLE_FINGERPRINT, mUserId) != null;
}
} }

View File

@@ -18,10 +18,15 @@ package com.android.settings.biometrics.combination;
import static com.google.common.truth.Truth.assertThat; import static com.google.common.truth.Truth.assertThat;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.ArgumentMatchers.anyInt; import static org.mockito.ArgumentMatchers.anyInt;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.never;
import static org.mockito.Mockito.spy; import static org.mockito.Mockito.spy;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when; import static org.mockito.Mockito.when;
import android.app.admin.DevicePolicyManager;
import android.content.Context; import android.content.Context;
import android.content.pm.PackageManager; import android.content.pm.PackageManager;
import android.hardware.face.FaceManager; import android.hardware.face.FaceManager;
@@ -30,6 +35,7 @@ import android.os.UserManager;
import com.android.settings.testutils.ActiveUnlockTestUtils; import com.android.settings.testutils.ActiveUnlockTestUtils;
import com.android.settings.testutils.shadow.ShadowDeviceConfig; import com.android.settings.testutils.shadow.ShadowDeviceConfig;
import com.android.settings.testutils.shadow.ShadowRestrictedLockUtilsInternal;
import com.android.settingslib.RestrictedPreference; import com.android.settingslib.RestrictedPreference;
import org.junit.After; import org.junit.After;
@@ -46,7 +52,7 @@ import org.robolectric.annotation.Config;
import org.robolectric.shadows.ShadowApplication; import org.robolectric.shadows.ShadowApplication;
@RunWith(RobolectricTestRunner.class) @RunWith(RobolectricTestRunner.class)
@Config(shadows = {ShadowDeviceConfig.class}) @Config(shadows = {ShadowDeviceConfig.class, ShadowRestrictedLockUtilsInternal.class})
public class BiometricFaceStatusPreferenceControllerTest { public class BiometricFaceStatusPreferenceControllerTest {
@Rule public final MockitoRule mMocks = MockitoJUnit.rule(); @Rule public final MockitoRule mMocks = MockitoJUnit.rule();
@@ -78,6 +84,7 @@ public class BiometricFaceStatusPreferenceControllerTest {
@After @After
public void tearDown() { public void tearDown() {
ActiveUnlockTestUtils.disable(mContext); ActiveUnlockTestUtils.disable(mContext);
ShadowRestrictedLockUtilsInternal.reset();
} }
@Test @Test
@@ -110,4 +117,43 @@ public class BiometricFaceStatusPreferenceControllerTest {
assertThat(mPreference.isVisible()).isTrue(); assertThat(mPreference.isVisible()).isTrue();
} }
@Test
public void faceDisabled_whenAdminAndNoFingerprintsEnrolled() {
when(mFaceManager.isHardwareDetected()).thenReturn(true);
when(mFaceManager.hasEnrolledTemplates(anyInt())).thenReturn(false);
ShadowRestrictedLockUtilsInternal
.setKeyguardDisabledFeatures(DevicePolicyManager.KEYGUARD_DISABLE_FACE);
final RestrictedPreference restrictedPreference = mock(RestrictedPreference.class);
mController.updateState(restrictedPreference);
verify(restrictedPreference).setDisabledByAdmin(any());
}
@Test
public void faceNotDisabled_whenAdminAndFingerprintsEnrolled() {
when(mFaceManager.isHardwareDetected()).thenReturn(true);
when(mFaceManager.hasEnrolledTemplates(anyInt())).thenReturn(true);
ShadowRestrictedLockUtilsInternal
.setKeyguardDisabledFeatures(DevicePolicyManager.KEYGUARD_DISABLE_FACE);
final RestrictedPreference restrictedPreference = mock(RestrictedPreference.class);
mController.updateState(restrictedPreference);
verify(restrictedPreference, never()).setDisabledByAdmin(any());
verify(restrictedPreference).setEnabled(true);
}
@Test
public void faceNotDisabled_whenNoAdmin() {
when(mFaceManager.isHardwareDetected()).thenReturn(true);
final RestrictedPreference restrictedPreference = mock(RestrictedPreference.class);
mController.updateState(restrictedPreference);
verify(restrictedPreference, never()).setDisabledByAdmin(any());
verify(restrictedPreference).setEnabled(true);
}
} }

View File

@@ -18,10 +18,15 @@ package com.android.settings.biometrics.combination;
import static com.google.common.truth.Truth.assertThat; import static com.google.common.truth.Truth.assertThat;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.ArgumentMatchers.anyInt; import static org.mockito.ArgumentMatchers.anyInt;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.never;
import static org.mockito.Mockito.spy; import static org.mockito.Mockito.spy;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when; import static org.mockito.Mockito.when;
import android.app.admin.DevicePolicyManager;
import android.content.Context; import android.content.Context;
import android.content.pm.PackageManager; import android.content.pm.PackageManager;
import android.hardware.face.FaceManager; import android.hardware.face.FaceManager;
@@ -30,6 +35,7 @@ import android.os.UserManager;
import com.android.settings.testutils.ActiveUnlockTestUtils; import com.android.settings.testutils.ActiveUnlockTestUtils;
import com.android.settings.testutils.shadow.ShadowDeviceConfig; import com.android.settings.testutils.shadow.ShadowDeviceConfig;
import com.android.settings.testutils.shadow.ShadowRestrictedLockUtilsInternal;
import com.android.settingslib.RestrictedPreference; import com.android.settingslib.RestrictedPreference;
import org.junit.After; import org.junit.After;
@@ -46,7 +52,7 @@ import org.robolectric.annotation.Config;
import org.robolectric.shadows.ShadowApplication; import org.robolectric.shadows.ShadowApplication;
@RunWith(RobolectricTestRunner.class) @RunWith(RobolectricTestRunner.class)
@Config(shadows = {ShadowDeviceConfig.class}) @Config(shadows = {ShadowDeviceConfig.class, ShadowRestrictedLockUtilsInternal.class})
public class BiometricFingerprintStatusPreferenceControllerTest { public class BiometricFingerprintStatusPreferenceControllerTest {
@Rule public final MockitoRule mMocks = MockitoJUnit.rule(); @Rule public final MockitoRule mMocks = MockitoJUnit.rule();
@@ -78,6 +84,7 @@ public class BiometricFingerprintStatusPreferenceControllerTest {
@After @After
public void tearDown() { public void tearDown() {
ActiveUnlockTestUtils.disable(mContext); ActiveUnlockTestUtils.disable(mContext);
ShadowRestrictedLockUtilsInternal.reset();
} }
@Test @Test
@@ -110,4 +117,45 @@ public class BiometricFingerprintStatusPreferenceControllerTest {
assertThat(mPreference.isVisible()).isTrue(); assertThat(mPreference.isVisible()).isTrue();
} }
@Test
public void fingerprintDisabled_whenAdminAndNoFingerprintsEnrolled() {
when(mFingerprintManager.isHardwareDetected()).thenReturn(true);
when(mFingerprintManager.hasEnrolledFingerprints(anyInt())).thenReturn(false);
ShadowRestrictedLockUtilsInternal
.setKeyguardDisabledFeatures(DevicePolicyManager.KEYGUARD_DISABLE_FINGERPRINT);
final RestrictedPreference restrictedPreference = mock(RestrictedPreference.class);
mController.updateState(restrictedPreference);
verify(restrictedPreference).setDisabledByAdmin(any());
}
@Test
public void fingerprintNotDisabled_whenAdminAndFingerprintsEnrolled() {
when(mFingerprintManager.isHardwareDetected()).thenReturn(true);
when(mFingerprintManager.hasEnrolledFingerprints(anyInt())).thenReturn(true);
ShadowRestrictedLockUtilsInternal
.setKeyguardDisabledFeatures(DevicePolicyManager.KEYGUARD_DISABLE_FINGERPRINT);
final RestrictedPreference restrictedPreference = mock(RestrictedPreference.class);
mController.updateState(restrictedPreference);
verify(restrictedPreference, never()).setDisabledByAdmin(any());
verify(restrictedPreference).setEnabled(true);
}
@Test
public void fingerprintNotDisabled_whenNoAdmin() {
when(mFingerprintManager.isHardwareDetected()).thenReturn(true);
final RestrictedPreference restrictedPreference = mock(RestrictedPreference.class);
mController.updateState(restrictedPreference);
verify(restrictedPreference, never()).setDisabledByAdmin(any());
verify(restrictedPreference).setEnabled(true);
}
} }

View File

@@ -127,6 +127,7 @@ public class FaceSettingsLockscreenBypassPreferenceControllerTest {
EnforcedAdmin admin = new EnforcedAdmin(); EnforcedAdmin admin = new EnforcedAdmin();
doReturn(admin).when(mController).getRestrictingAdmin(); doReturn(admin).when(mController).getRestrictingAdmin();
mController.updateState(mPreference); mController.updateState(mPreference);
verify(mPreference).setDisabledByAdmin(admin);
verify(mPreference).setEnabled(false);
} }
} }

View File

@@ -43,6 +43,7 @@ import androidx.preference.Preference;
import com.android.internal.widget.LockPatternUtils; import com.android.internal.widget.LockPatternUtils;
import com.android.settings.R; import com.android.settings.R;
import com.android.settings.testutils.FakeFeatureFactory; import com.android.settings.testutils.FakeFeatureFactory;
import com.android.settings.testutils.shadow.ShadowRestrictedLockUtilsInternal;
import com.android.settingslib.RestrictedLockUtils; import com.android.settingslib.RestrictedLockUtils;
import com.android.settingslib.RestrictedPreference; import com.android.settingslib.RestrictedPreference;
import com.android.settingslib.core.lifecycle.Lifecycle; import com.android.settingslib.core.lifecycle.Lifecycle;
@@ -55,11 +56,13 @@ import org.mockito.Mock;
import org.mockito.MockitoAnnotations; import org.mockito.MockitoAnnotations;
import org.robolectric.RobolectricTestRunner; import org.robolectric.RobolectricTestRunner;
import org.robolectric.RuntimeEnvironment; import org.robolectric.RuntimeEnvironment;
import org.robolectric.annotation.Config;
import org.robolectric.shadows.ShadowApplication; import org.robolectric.shadows.ShadowApplication;
import java.util.Collections; import java.util.Collections;
@RunWith(RobolectricTestRunner.class) @RunWith(RobolectricTestRunner.class)
@Config(shadows = {ShadowRestrictedLockUtilsInternal.class})
public class FaceStatusPreferenceControllerTest { public class FaceStatusPreferenceControllerTest {
private static final String TEST_PREF_KEY = "baz"; private static final String TEST_PREF_KEY = "baz";

View File

@@ -43,6 +43,7 @@ import androidx.preference.Preference;
import com.android.internal.widget.LockPatternUtils; import com.android.internal.widget.LockPatternUtils;
import com.android.settings.R; import com.android.settings.R;
import com.android.settings.testutils.FakeFeatureFactory; import com.android.settings.testutils.FakeFeatureFactory;
import com.android.settings.testutils.shadow.ShadowRestrictedLockUtilsInternal;
import com.android.settingslib.RestrictedLockUtils; import com.android.settingslib.RestrictedLockUtils;
import com.android.settingslib.RestrictedPreference; import com.android.settingslib.RestrictedPreference;
import com.android.settingslib.core.lifecycle.Lifecycle; import com.android.settingslib.core.lifecycle.Lifecycle;
@@ -56,11 +57,13 @@ import org.mockito.Mock;
import org.mockito.MockitoAnnotations; import org.mockito.MockitoAnnotations;
import org.robolectric.RobolectricTestRunner; import org.robolectric.RobolectricTestRunner;
import org.robolectric.RuntimeEnvironment; import org.robolectric.RuntimeEnvironment;
import org.robolectric.annotation.Config;
import org.robolectric.shadows.ShadowApplication; import org.robolectric.shadows.ShadowApplication;
import java.util.Collections; import java.util.Collections;
@RunWith(RobolectricTestRunner.class) @RunWith(RobolectricTestRunner.class)
@Config(shadows = {ShadowRestrictedLockUtilsInternal.class})
public class FingerprintStatusPreferenceControllerTest { public class FingerprintStatusPreferenceControllerTest {
@Mock @Mock