/* * Copyright (C) 2018 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.Activity.RESULT_OK; import static com.android.settings.biometrics.BiometricEnrollBase.CONFIRM_REQUEST; import static com.android.settings.biometrics.BiometricEnrollBase.RESULT_FINISHED; import android.content.Context; import android.content.Intent; import android.hardware.face.FaceManager; import android.os.Bundle; import android.os.UserHandle; import android.provider.SearchIndexableResource; import android.util.Log; import com.android.internal.logging.nano.MetricsProto; import com.android.settings.R; import com.android.settings.Utils; import com.android.settings.dashboard.DashboardFragment; import com.android.settings.password.ChooseLockSettingsHelper; import com.android.settings.search.BaseSearchIndexProvider; import com.android.settings.widget.VideoPreferenceController; import com.android.settingslib.core.AbstractPreferenceController; import com.android.settingslib.core.lifecycle.Lifecycle; import com.android.settingslib.search.SearchIndexable; import java.util.ArrayList; import java.util.Arrays; import java.util.List; /** * Settings screen for face authentication. */ @SearchIndexable public class FaceSettings extends DashboardFragment { private static final String TAG = "FaceSettings"; private static final String KEY_LAUNCHED_CONFIRM = "key_launched_confirm"; private FaceManager mFaceManager; private int mUserId; private boolean mLaunchedConfirm; private byte[] mToken; private FaceSettingsAttentionPreferenceController mAttentionController; public static boolean isAvailable(Context context) { FaceManager manager = Utils.getFaceManagerOrNull(context); return manager != null && manager.isHardwareDetected(); } @Override public int getMetricsCategory() { return MetricsProto.MetricsEvent.FACE; } @Override protected int getPreferenceScreenResId() { return R.xml.security_settings_face; } @Override protected String getLogTag() { return TAG; } @Override public void onSaveInstanceState(Bundle outState) { super.onSaveInstanceState(outState); outState.putBoolean(KEY_LAUNCHED_CONFIRM, mLaunchedConfirm); } @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); mFaceManager = getPrefContext().getSystemService(FaceManager.class); mUserId = getActivity().getIntent().getIntExtra( Intent.EXTRA_USER_ID, UserHandle.myUserId()); if (savedInstanceState != null) { mLaunchedConfirm = savedInstanceState.getBoolean(KEY_LAUNCHED_CONFIRM, false); } if (!mLaunchedConfirm) { final long challenge = mFaceManager.generateChallenge(); ChooseLockSettingsHelper helper = new ChooseLockSettingsHelper(getActivity(), this); if (!helper.launchConfirmationActivity(CONFIRM_REQUEST, getString(R.string.security_settings_face_preference_title), null, null, challenge, mUserId)) { Log.e(TAG, "Password not set"); finish(); } } } @Override public void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); if (requestCode == CONFIRM_REQUEST) { if (resultCode == RESULT_FINISHED || resultCode == RESULT_OK) { // The pin/pattern/password was set. if (data != null) { mToken = data.getByteArrayExtra( ChooseLockSettingsHelper.EXTRA_KEY_CHALLENGE_TOKEN); if (mToken != null) { mAttentionController.setToken(mToken); } } } } if (mToken == null) { // Didn't get an authentication, finishing getActivity().finish(); } } @Override public void onDestroy() { super.onDestroy(); if (getActivity().isFinishing()) { final int result = mFaceManager.revokeChallenge(); if (result < 0) { Log.w(TAG, "revokeChallenge failed, result: " + result); } } } @Override protected List createPreferenceControllers(Context context) { final List controllers = buildPreferenceControllers(context, getSettingsLifecycle()); for (AbstractPreferenceController controller : controllers) { if (controller instanceof FaceSettingsAttentionPreferenceController) { mAttentionController = (FaceSettingsAttentionPreferenceController) controller; break; } } return controllers; } private static List buildPreferenceControllers(Context context, Lifecycle lifecycle) { final List controllers = new ArrayList<>(); controllers.add(new FaceSettingsImprovePreferenceController(context)); controllers.add(new FaceSettingsKeyguardPreferenceController(context)); controllers.add(new FaceSettingsAppPreferenceController(context)); controllers.add(new FaceSettingsAttentionPreferenceController(context)); controllers.add(new FaceSettingsRemoveButtonPreferenceController(context)); controllers.add(new FaceSettingsFooterPreferenceController(context)); return controllers; } public static final SearchIndexProvider SEARCH_INDEX_DATA_PROVIDER = new BaseSearchIndexProvider() { @Override public List getXmlResourcesToIndex( Context context, boolean enabled) { final SearchIndexableResource sir = new SearchIndexableResource(context); sir.xmlResId = R.xml.security_settings_face; return Arrays.asList(sir); } @Override public List createPreferenceControllers( Context context) { return buildPreferenceControllers(context, null /* lifecycle */); } @Override protected boolean isPageSearchEnabled(Context context) { return isAvailable(context); } }; }