Create a mechanism to allow OEM config posture guidance with 'config_face_enroll_guidance_page', and customize the config 'config_face_enroll_supported_posture' with standard postures 0 : DEVICE_POSTURE_UNKNOWN 1 : DEVICE_POSTURE_CLOSED 2 : DEVICE_POSTURE_HALF_OPENED 3 : DEVICE_POSTURE_OPENED 4 : DEVICE_POSTURE_FLIPPED For example, if we set 1 for the device, then device only allow to enroll face in closed(folded) state, if device do not in the allow state, we will prompt specific guidance page activity defined in config_face_enroll_guidance_page. At this stage , we only integrate 2 states OPENED/CLOSED through ScreenSizeFoldProvider and register for onFoldUpdated() callback - isFold(DEVICE_POSTURE_CLOSED): finish posture guidance - !isFold(DEVICE_POSTURE_OPENED): launch posture guidance - onActivityResult : reset mOnGuidanceShown false 1. Fix A11y lottie animation bug 2. Impl FoldProvider.FoldCallback 3. Register callback to ScreenSizeFoldProvider 4. Integrate back stack, skip, cancel events - Back key : RESULT_CANCELED - Skip btn : RESULT_SKIP - Posture changed : RESULT_FINISHED 5. Set single instance for relative activities 6. FaceEnrollFoldPage listen for onConfigurationChanged() 7. Add empty face_posture_guidance_lottie.json for overlay Test: atest SettingsGoogleUnitTests Test: m -j SettingsGoogleRoboTests RunSettingsGoogleRoboTests Test: m RunSettingsRoboTests ROBOTEST_FILTER= \ "com.android.settings.biometrics.face.FaceEnrollEducationTest" Test: m RunSettingsRoboTests ROBOTEST_FILTER= \ "com.android.settings.biometrics.face.FaceEnrollIntroductionTest" Test: Manual launch security settings face enroll, unfold device and observe posture guidance showing fullscreen on top Test: Fold device ensure the posture guidance activity finish Bug: 261141826 Fixes: 231908496 Change-Id: Ib9f43f82f7d19f3f187c2f6f8984e76cd843afbc Merged-In: Ib9f43f82f7d19f3f187c2f6f8984e76cd843afbc
521 lines
19 KiB
Java
521 lines
19 KiB
Java
/*
|
|
* 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 static android.app.admin.DevicePolicyResources.Strings.Settings.FACE_UNLOCK_DISABLED;
|
|
|
|
import static com.android.settings.biometrics.BiometricUtils.GatekeeperCredentialNotMatchException;
|
|
|
|
import android.app.admin.DevicePolicyManager;
|
|
import android.app.settings.SettingsEnums;
|
|
import android.content.Intent;
|
|
import android.content.res.Configuration;
|
|
import android.hardware.SensorPrivacyManager;
|
|
import android.hardware.biometrics.BiometricAuthenticator;
|
|
import android.hardware.face.FaceManager;
|
|
import android.os.Bundle;
|
|
import android.text.Html;
|
|
import android.text.method.LinkMovementMethod;
|
|
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 androidx.annotation.VisibleForTesting;
|
|
|
|
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.biometrics.MultiBiometricEnrollHelper;
|
|
import com.android.settings.password.ChooseLockSettingsHelper;
|
|
import com.android.settings.password.SetupSkipDialog;
|
|
import com.android.settings.utils.SensorPrivacyManagerHelper;
|
|
import com.android.settingslib.RestrictedLockUtilsInternal;
|
|
import com.android.systemui.unfold.compat.ScreenSizeFoldProvider;
|
|
import com.android.systemui.unfold.updates.FoldProvider;
|
|
|
|
import com.google.android.setupcompat.template.FooterButton;
|
|
import com.google.android.setupcompat.util.WizardManagerHelper;
|
|
import com.google.android.setupdesign.span.LinkSpan;
|
|
|
|
/**
|
|
* 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;
|
|
@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 boolean shouldFinishWhenBackgrounded() {
|
|
return super.shouldFinishWhenBackgrounded() && !BiometricUtils.isPostureGuidanceShowing(
|
|
mDevicePostureState, mLaunchedPostureGuidance);
|
|
}
|
|
|
|
@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);
|
|
final TextView lessSecure = findViewById(R.id.info_message_less_secure);
|
|
infoMessageGlasses.setText(getInfoMessageGlasses());
|
|
infoMessageLooking.setText(getInfoMessageLooking());
|
|
inControlTitle.setText(getInControlTitle());
|
|
howMessage.setText(getHowMessage());
|
|
inControlMessage.setText(Html.fromHtml(getString(getInControlMessage()),
|
|
Html.FROM_HTML_MODE_LEGACY));
|
|
inControlMessage.setMovementMethod(LinkMovementMethod.getInstance());
|
|
lessSecure.setText(getLessSecureMessage());
|
|
|
|
// 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 = getFaceManager();
|
|
|
|
// 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) -> {
|
|
if (isFinishing()) {
|
|
// Do nothing if activity is finishing
|
|
Log.w(TAG, "activity finished before challenge callback launched.");
|
|
return;
|
|
}
|
|
|
|
try {
|
|
mToken = requestGatekeeperHat(challenge);
|
|
mSensorId = sensorId;
|
|
mChallenge = challenge;
|
|
mFooterBarMixin.getPrimaryButton().setEnabled(true);
|
|
} catch (GatekeeperCredentialNotMatchException e) {
|
|
// Let BiometricEnrollBase#onCreate() to trigger confirmLock()
|
|
getIntent().removeExtra(ChooseLockSettingsHelper.EXTRA_KEY_GK_PW_HANDLE);
|
|
recreate();
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
@VisibleForTesting
|
|
@Nullable
|
|
protected FaceManager getFaceManager() {
|
|
return Utils.getFaceManagerOrNull(this);
|
|
}
|
|
|
|
@VisibleForTesting
|
|
@Nullable
|
|
protected Intent getPostureGuidanceIntent() {
|
|
return mPostureGuidanceIntent;
|
|
}
|
|
|
|
@VisibleForTesting
|
|
@Nullable
|
|
protected FoldProvider.FoldCallback getPostureCallback() {
|
|
return mFoldCallback;
|
|
}
|
|
|
|
@VisibleForTesting
|
|
@BiometricUtils.DevicePostureInt
|
|
protected int getDevicePostureState() {
|
|
return mDevicePostureState;
|
|
}
|
|
|
|
@VisibleForTesting
|
|
@Nullable
|
|
protected byte[] requestGatekeeperHat(long challenge) {
|
|
return BiometricUtils.requestGatekeeperHat(this, getIntent(), mUserId, challenge);
|
|
}
|
|
|
|
@Override
|
|
public void onConfigurationChanged(@NonNull Configuration newConfig) {
|
|
super.onConfigurationChanged(newConfig);
|
|
if (mScreenSizeFoldProvider != null && getPostureCallback() != null) {
|
|
mScreenSizeFoldProvider.onConfigurationChange(newConfig);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void onStart() {
|
|
super.onStart();
|
|
if (getPostureGuidanceIntent() == null) {
|
|
Log.d(TAG, "Device do not support posture guidance");
|
|
return;
|
|
}
|
|
|
|
BiometricUtils.setDevicePosturesAllowEnroll(
|
|
getResources().getInteger(R.integer.config_face_enroll_supported_posture));
|
|
|
|
if (getPostureCallback() == null) {
|
|
mFoldCallback = isFolded -> {
|
|
mDevicePostureState = isFolded ? BiometricUtils.DEVICE_POSTURE_CLOSED
|
|
: BiometricUtils.DEVICE_POSTURE_OPENED;
|
|
if (BiometricUtils.shouldShowPostureGuidance(mDevicePostureState,
|
|
mLaunchedPostureGuidance) && !mNextLaunched) {
|
|
launchPostureGuidance();
|
|
}
|
|
};
|
|
}
|
|
|
|
if (mScreenSizeFoldProvider == null) {
|
|
mScreenSizeFoldProvider = new ScreenSizeFoldProvider(getApplicationContext());
|
|
mScreenSizeFoldProvider.registerCallback(mFoldCallback, getMainExecutor());
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
|
|
if (requestCode == REQUEST_POSTURE_GUIDANCE) {
|
|
mLaunchedPostureGuidance = false;
|
|
if (resultCode == RESULT_CANCELED || resultCode == RESULT_SKIP) {
|
|
onSkipButtonClick(getCurrentFocus());
|
|
}
|
|
return;
|
|
}
|
|
|
|
// If user has skipped or finished enrolling, don't restart enrollment.
|
|
final boolean isEnrollRequest = requestCode == BIOMETRIC_FIND_SENSOR_REQUEST
|
|
|| requestCode == ENROLL_NEXT_BIOMETRIC_REQUEST;
|
|
final boolean isResultSkipOrFinished = resultCode == RESULT_SKIP
|
|
|| resultCode == SetupSkipDialog.RESULT_SKIP || resultCode == RESULT_FINISHED;
|
|
boolean hasEnrolledFace = false;
|
|
if (data != null) {
|
|
hasEnrolledFace = data.getBooleanExtra(EXTRA_FINISHED_ENROLL_FACE, false);
|
|
}
|
|
|
|
if (resultCode == RESULT_CANCELED) {
|
|
if (hasEnrolledFace || !BiometricUtils.isPostureAllowEnrollment(mDevicePostureState)) {
|
|
setResult(resultCode, data);
|
|
finish();
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (isEnrollRequest && isResultSkipOrFinished || hasEnrolledFace) {
|
|
data = setSkipPendingEnroll(data);
|
|
}
|
|
super.onActivityResult(requestCode, resultCode, data);
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
@StringRes
|
|
protected int getLessSecureMessage() {
|
|
return R.string.security_settings_face_enroll_introduction_info_less_secure;
|
|
}
|
|
|
|
@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 String getDescriptionDisabledByAdmin() {
|
|
DevicePolicyManager devicePolicyManager = getSystemService(DevicePolicyManager.class);
|
|
return devicePolicyManager.getResources().getString(
|
|
FACE_UNLOCK_DISABLED,
|
|
() -> getString(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() {
|
|
if (mFaceManager != null) {
|
|
// This will need to be updated for devices with multiple face sensors.
|
|
final int numEnrolledFaces = mFaceManager.getEnrolledFaces(mUserId).size();
|
|
final int maxFacesEnrollable = getApplicationContext().getResources()
|
|
.getInteger(R.integer.suw_max_faces_enrollable);
|
|
return numEnrolledFaces >= maxFacesEnrollable;
|
|
} 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;
|
|
}
|
|
|
|
@NonNull
|
|
protected static Intent setSkipPendingEnroll(@Nullable Intent data) {
|
|
if (data == null) {
|
|
data = new Intent();
|
|
}
|
|
data.putExtra(MultiBiometricEnrollHelper.EXTRA_SKIP_PENDING_ENROLL, true);
|
|
return data;
|
|
}
|
|
}
|