/* * Copyright (C) 2021 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.face; import android.app.admin.DevicePolicyManager; import android.app.settings.SettingsEnums; import android.content.Intent; import android.hardware.SensorPrivacyManager; import android.hardware.biometrics.BiometricAuthenticator; import android.hardware.face.FaceManager; import android.hardware.face.FaceSensorPropertiesInternal; import android.os.Bundle; import android.util.Log; import android.view.View; import android.widget.ImageView; import android.widget.LinearLayout; import android.widget.TextView; import androidx.annotation.NonNull; import androidx.annotation.Nullable; import androidx.annotation.StringRes; import com.android.settings.R; import com.android.settings.Utils; import com.android.settings.biometrics.BiometricEnrollActivity; import com.android.settings.biometrics.BiometricEnrollIntroduction; import com.android.settings.biometrics.BiometricUtils; import com.android.settings.overlay.FeatureFactory; import com.android.settings.password.ChooseLockSettingsHelper; import com.android.settings.utils.SensorPrivacyManagerHelper; import com.android.settingslib.RestrictedLockUtilsInternal; import com.google.android.setupcompat.template.FooterButton; import com.google.android.setupcompat.util.WizardManagerHelper; import com.google.android.setupdesign.span.LinkSpan; import java.util.List; /** * Provides introductory info about face unlock and prompts the user to agree before starting face * enrollment. */ public class FaceEnrollIntroduction extends BiometricEnrollIntroduction { private static final String TAG = "FaceEnrollIntroduction"; private FaceManager mFaceManager; private FaceFeatureProvider mFaceFeatureProvider; @Nullable private FooterButton mPrimaryFooterButton; @Nullable private FooterButton mSecondaryFooterButton; @Nullable private SensorPrivacyManager mSensorPrivacyManager; @Override protected void onCancelButtonClick(View view) { if (!BiometricUtils.tryStartingNextBiometricEnroll(this, ENROLL_NEXT_BIOMETRIC_REQUEST, "cancel")) { super.onCancelButtonClick(view); } } @Override protected void onSkipButtonClick(View view) { if (!BiometricUtils.tryStartingNextBiometricEnroll(this, ENROLL_NEXT_BIOMETRIC_REQUEST, "skip")) { super.onSkipButtonClick(view); } } @Override protected void onEnrollmentSkipped(@Nullable Intent data) { if (!BiometricUtils.tryStartingNextBiometricEnroll(this, ENROLL_NEXT_BIOMETRIC_REQUEST, "skipped")) { super.onEnrollmentSkipped(data); } } @Override protected void onFinishedEnrolling(@Nullable Intent data) { if (!BiometricUtils.tryStartingNextBiometricEnroll(this, ENROLL_NEXT_BIOMETRIC_REQUEST, "finished")) { super.onFinishedEnrolling(data); } } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // Apply extracted theme color to icons. final ImageView iconGlasses = findViewById(R.id.icon_glasses); final ImageView iconLooking = findViewById(R.id.icon_looking); iconGlasses.getBackground().setColorFilter(getIconColorFilter()); iconLooking.getBackground().setColorFilter(getIconColorFilter()); // Set text for views with multiple variations. final TextView infoMessageGlasses = findViewById(R.id.info_message_glasses); final TextView infoMessageLooking = findViewById(R.id.info_message_looking); final TextView howMessage = findViewById(R.id.how_message); final TextView inControlTitle = findViewById(R.id.title_in_control); final TextView inControlMessage = findViewById(R.id.message_in_control); infoMessageGlasses.setText(getInfoMessageGlasses()); infoMessageLooking.setText(getInfoMessageLooking()); inControlTitle.setText(getInControlTitle()); howMessage.setText(getHowMessage()); inControlMessage.setText(getInControlMessage()); // Set up and show the "less secure" info section if necessary. if (getResources().getBoolean(R.bool.config_face_intro_show_less_secure)) { final LinearLayout infoRowLessSecure = findViewById(R.id.info_row_less_secure); final ImageView iconLessSecure = findViewById(R.id.icon_less_secure); infoRowLessSecure.setVisibility(View.VISIBLE); iconLessSecure.getBackground().setColorFilter(getIconColorFilter()); } // Set up and show the "require eyes" info section if necessary. if (getResources().getBoolean(R.bool.config_face_intro_show_require_eyes)) { final LinearLayout infoRowRequireEyes = findViewById(R.id.info_row_require_eyes); final ImageView iconRequireEyes = findViewById(R.id.icon_require_eyes); final TextView infoMessageRequireEyes = findViewById(R.id.info_message_require_eyes); infoRowRequireEyes.setVisibility(View.VISIBLE); iconRequireEyes.getBackground().setColorFilter(getIconColorFilter()); infoMessageRequireEyes.setText(getInfoMessageRequireEyes()); } mFaceManager = Utils.getFaceManagerOrNull(this); mFaceFeatureProvider = FeatureFactory.getFactory(getApplicationContext()) .getFaceFeatureProvider(); // This path is an entry point for SetNewPasswordController, e.g. // adb shell am start -a android.app.action.SET_NEW_PASSWORD if (mToken == null && BiometricUtils.containsGatekeeperPasswordHandle(getIntent())) { if (generateChallengeOnCreate()) { mFooterBarMixin.getPrimaryButton().setEnabled(false); // We either block on generateChallenge, or need to gray out the "next" button until // the challenge is ready. Let's just do this for now. mFaceManager.generateChallenge(mUserId, (sensorId, userId, challenge) -> { mToken = BiometricUtils.requestGatekeeperHat(this, getIntent(), mUserId, challenge); mSensorId = sensorId; mChallenge = challenge; mFooterBarMixin.getPrimaryButton().setEnabled(true); }); } } mSensorPrivacyManager = getApplicationContext() .getSystemService(SensorPrivacyManager.class); final SensorPrivacyManagerHelper helper = SensorPrivacyManagerHelper .getInstance(getApplicationContext()); final boolean cameraPrivacyEnabled = helper .isSensorBlocked(SensorPrivacyManager.Sensors.CAMERA, mUserId); Log.v(TAG, "cameraPrivacyEnabled : " + cameraPrivacyEnabled); } protected boolean generateChallengeOnCreate() { return true; } @StringRes protected int getInfoMessageGlasses() { return R.string.security_settings_face_enroll_introduction_info_glasses; } @StringRes protected int getInfoMessageLooking() { return R.string.security_settings_face_enroll_introduction_info_looking; } @StringRes protected int getInfoMessageRequireEyes() { return R.string.security_settings_face_enroll_introduction_info_gaze; } @StringRes protected int getHowMessage() { return R.string.security_settings_face_enroll_introduction_how_message; } @StringRes protected int getInControlTitle() { return R.string.security_settings_face_enroll_introduction_control_title; } @StringRes protected int getInControlMessage() { return R.string.security_settings_face_enroll_introduction_control_message; } @Override protected boolean isDisabledByAdmin() { return RestrictedLockUtilsInternal.checkIfKeyguardFeaturesDisabled( this, DevicePolicyManager.KEYGUARD_DISABLE_FACE, mUserId) != null; } @Override protected int getLayoutResource() { return R.layout.face_enroll_introduction; } @Override protected int getHeaderResDisabledByAdmin() { return R.string.security_settings_face_enroll_introduction_title_unlock_disabled; } @Override protected int getHeaderResDefault() { return R.string.security_settings_face_enroll_introduction_title; } @Override protected int getDescriptionResDisabledByAdmin() { return R.string.security_settings_face_enroll_introduction_message_unlock_disabled; } @Override protected FooterButton getCancelButton() { if (mFooterBarMixin != null) { return mFooterBarMixin.getSecondaryButton(); } return null; } @Override protected FooterButton getNextButton() { if (mFooterBarMixin != null) { return mFooterBarMixin.getPrimaryButton(); } return null; } @Override protected TextView getErrorTextView() { return findViewById(R.id.error_text); } private boolean maxFacesEnrolled() { final boolean isSetupWizard = WizardManagerHelper.isAnySetupWizard(getIntent()); if (mFaceManager != null) { final List props = mFaceManager.getSensorPropertiesInternal(); // This will need to be updated for devices with multiple face sensors. final int max = props.get(0).maxEnrollmentsPerUser; final int numEnrolledFaces = mFaceManager.getEnrolledFaces(mUserId).size(); final int maxFacesEnrollableIfSUW = getApplicationContext().getResources() .getInteger(R.integer.suw_max_faces_enrollable); if (isSetupWizard) { return numEnrolledFaces >= maxFacesEnrollableIfSUW; } else { return numEnrolledFaces >= max; } } else { return false; } } //TODO: Refactor this to something that conveys it is used for getting a string ID. @Override protected int checkMaxEnrolled() { if (mFaceManager != null) { if (maxFacesEnrolled()) { return R.string.face_intro_error_max; } } else { return R.string.face_intro_error_unknown; } return 0; } @Override protected void getChallenge(GenerateChallengeCallback callback) { mFaceManager = Utils.getFaceManagerOrNull(this); if (mFaceManager == null) { callback.onChallengeGenerated(0, 0, 0L); return; } mFaceManager.generateChallenge(mUserId, callback::onChallengeGenerated); } @Override protected String getExtraKeyForBiometric() { return ChooseLockSettingsHelper.EXTRA_KEY_FOR_FACE; } @Override protected Intent getEnrollingIntent() { Intent intent = new Intent(this, FaceEnrollEducation.class); WizardManagerHelper.copyWizardManagerExtras(getIntent(), intent); return intent; } @Override protected int getConfirmLockTitleResId() { return R.string.security_settings_face_preference_title; } @Override public int getMetricsCategory() { return SettingsEnums.FACE_ENROLL_INTRO; } @Override public void onClick(LinkSpan span) { // TODO(b/110906762) } @Override public @BiometricAuthenticator.Modality int getModality() { return BiometricAuthenticator.TYPE_FACE; } @Override protected void onNextButtonClick(View view) { final boolean parentelConsentRequired = getIntent() .getBooleanExtra(BiometricEnrollActivity.EXTRA_REQUIRE_PARENTAL_CONSENT, false); final boolean cameraPrivacyEnabled = SensorPrivacyManagerHelper .getInstance(getApplicationContext()) .isSensorBlocked(SensorPrivacyManager.Sensors.CAMERA, mUserId); final boolean isSetupWizard = WizardManagerHelper.isAnySetupWizard(getIntent()); final boolean isSettingUp = isSetupWizard || (parentelConsentRequired && !WizardManagerHelper.isUserSetupComplete(this)); if (cameraPrivacyEnabled && !isSettingUp) { if (mSensorPrivacyManager == null) { mSensorPrivacyManager = getApplicationContext() .getSystemService(SensorPrivacyManager.class); } mSensorPrivacyManager.showSensorUseDialog(SensorPrivacyManager.Sensors.CAMERA); } else { super.onNextButtonClick(view); } } @Override @NonNull protected FooterButton getPrimaryFooterButton() { if (mPrimaryFooterButton == null) { mPrimaryFooterButton = new FooterButton.Builder(this) .setText(R.string.security_settings_face_enroll_introduction_agree) .setButtonType(FooterButton.ButtonType.OPT_IN) .setListener(this::onNextButtonClick) .setTheme(R.style.SudGlifButton_Primary) .build(); } return mPrimaryFooterButton; } @Override @NonNull protected FooterButton getSecondaryFooterButton() { if (mSecondaryFooterButton == null) { mSecondaryFooterButton = new FooterButton.Builder(this) .setText(R.string.security_settings_face_enroll_introduction_no_thanks) .setListener(this::onSkipButtonClick) .setButtonType(FooterButton.ButtonType.NEXT) .setTheme(R.style.SudGlifButton_Primary) .build(); } return mSecondaryFooterButton; } @Override @StringRes protected int getAgreeButtonTextRes() { return R.string.security_settings_fingerprint_enroll_introduction_agree; } @Override @StringRes protected int getMoreButtonTextRes() { return R.string.security_settings_face_enroll_introduction_more; } }