Fix face delete button rotation not working. am: 94f759230a

Original change: https://googleplex-android-review.googlesource.com/c/platform/packages/apps/Settings/+/24484929

Change-Id: I97275e1f64fa87f615444870d894e05f386c5eda
Signed-off-by: Automerger Merge Worker <android-build-automerger-merge-worker@system.gserviceaccount.com>
This commit is contained in:
Hao Dong
2023-08-18 20:40:19 +00:00
committed by Automerger Merge Worker
2 changed files with 136 additions and 14 deletions

View File

@@ -30,6 +30,7 @@ import android.view.View;
import android.widget.Button; import android.widget.Button;
import android.widget.Toast; import android.widget.Toast;
import androidx.annotation.VisibleForTesting;
import androidx.preference.Preference; import androidx.preference.Preference;
import com.android.settings.R; import com.android.settings.R;
@@ -57,10 +58,18 @@ public class FaceSettingsRemoveButtonPreferenceController extends BasePreference
static final String KEY = "security_settings_face_delete_faces_container"; static final String KEY = "security_settings_face_delete_faces_container";
public static class ConfirmRemoveDialog extends InstrumentedDialogFragment { public static class ConfirmRemoveDialog extends InstrumentedDialogFragment {
private static final String KEY_IS_CONVENIENCE = "is_convenience";
private boolean mIsConvenience;
private DialogInterface.OnClickListener mOnClickListener; private DialogInterface.OnClickListener mOnClickListener;
/** Returns the new instance of the class */
public static ConfirmRemoveDialog newInstance(boolean isConvenience) {
final ConfirmRemoveDialog dialog = new ConfirmRemoveDialog();
final Bundle args = new Bundle();
args.putBoolean(KEY_IS_CONVENIENCE, isConvenience);
dialog.setArguments(args);
return dialog;
}
@Override @Override
public int getMetricsCategory() { public int getMetricsCategory() {
return SettingsEnums.DIALOG_FACE_REMOVE; return SettingsEnums.DIALOG_FACE_REMOVE;
@@ -68,6 +77,8 @@ public class FaceSettingsRemoveButtonPreferenceController extends BasePreference
@Override @Override
public Dialog onCreateDialog(Bundle savedInstanceState) { public Dialog onCreateDialog(Bundle savedInstanceState) {
boolean isConvenience = getArguments().getBoolean(KEY_IS_CONVENIENCE);
AlertDialog.Builder builder = new AlertDialog.Builder(getActivity()); AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
final PackageManager pm = getContext().getPackageManager(); final PackageManager pm = getContext().getPackageManager();
@@ -75,11 +86,11 @@ public class FaceSettingsRemoveButtonPreferenceController extends BasePreference
final int dialogMessageRes; final int dialogMessageRes;
if (hasFingerprint) { if (hasFingerprint) {
dialogMessageRes = mIsConvenience dialogMessageRes = isConvenience
? R.string.security_settings_face_remove_dialog_details_fingerprint_conv ? R.string.security_settings_face_remove_dialog_details_fingerprint_conv
: R.string.security_settings_face_remove_dialog_details_fingerprint; : R.string.security_settings_face_remove_dialog_details_fingerprint;
} else { } else {
dialogMessageRes = mIsConvenience dialogMessageRes = isConvenience
? R.string.security_settings_face_settings_remove_dialog_details_convenience ? R.string.security_settings_face_settings_remove_dialog_details_convenience
: R.string.security_settings_face_settings_remove_dialog_details; : R.string.security_settings_face_settings_remove_dialog_details;
} }
@@ -93,10 +104,6 @@ public class FaceSettingsRemoveButtonPreferenceController extends BasePreference
return dialog; return dialog;
} }
public void setIsConvenience(boolean isConvenience) {
mIsConvenience = isConvenience;
}
public void setOnClickListener(DialogInterface.OnClickListener listener) { public void setOnClickListener(DialogInterface.OnClickListener listener) {
mOnClickListener = listener; mOnClickListener = listener;
} }
@@ -111,7 +118,8 @@ public class FaceSettingsRemoveButtonPreferenceController extends BasePreference
private Listener mListener; private Listener mListener;
private SettingsActivity mActivity; private SettingsActivity mActivity;
private int mUserId; private int mUserId;
private boolean mRemoving; @VisibleForTesting
boolean mRemoving;
private final MetricsFeatureProvider mMetricsFeatureProvider; private final MetricsFeatureProvider mMetricsFeatureProvider;
private final Context mContext; private final Context mContext;
@@ -142,7 +150,7 @@ public class FaceSettingsRemoveButtonPreferenceController extends BasePreference
} }
}; };
private final DialogInterface.OnClickListener mOnClickListener private final DialogInterface.OnClickListener mOnConfirmDialogClickListener
= new DialogInterface.OnClickListener() { = new DialogInterface.OnClickListener() {
@Override @Override
public void onClick(DialogInterface dialog, int which) { public void onClick(DialogInterface dialog, int which) {
@@ -196,6 +204,16 @@ public class FaceSettingsRemoveButtonPreferenceController extends BasePreference
mButton.setOnClickListener(this); mButton.setOnClickListener(this);
// If there is already a ConfirmRemoveDialog showing, reset the listener since the
// controller has been recreated.
ConfirmRemoveDialog removeDialog =
(ConfirmRemoveDialog) mActivity.getSupportFragmentManager()
.findFragmentByTag(ConfirmRemoveDialog.class.getName());
if (removeDialog != null) {
mRemoving = true;
removeDialog.setOnClickListener(mOnConfirmDialogClickListener);
}
if (!FaceSettings.isFaceHardwareDetected(mContext)) { if (!FaceSettings.isFaceHardwareDetected(mContext)) {
mButton.setEnabled(false); mButton.setEnabled(false);
} else { } else {
@@ -218,10 +236,11 @@ public class FaceSettingsRemoveButtonPreferenceController extends BasePreference
if (v == mButton) { if (v == mButton) {
mMetricsFeatureProvider.logClickedPreference(mPreference, getMetricsCategory()); mMetricsFeatureProvider.logClickedPreference(mPreference, getMetricsCategory());
mRemoving = true; mRemoving = true;
ConfirmRemoveDialog dialog = new ConfirmRemoveDialog(); ConfirmRemoveDialog confirmRemoveDialog =
dialog.setOnClickListener(mOnClickListener); ConfirmRemoveDialog.newInstance(BiometricUtils.isConvenience(mFaceManager));
dialog.setIsConvenience(BiometricUtils.isConvenience(mFaceManager)); confirmRemoveDialog.setOnClickListener(mOnConfirmDialogClickListener);
dialog.show(mActivity.getSupportFragmentManager(), ConfirmRemoveDialog.class.getName()); confirmRemoveDialog.show(mActivity.getSupportFragmentManager(),
ConfirmRemoveDialog.class.getName());
} }
} }

View File

@@ -0,0 +1,103 @@
/*
* Copyright (C) 2023 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 com.google.common.truth.Truth.assertThat;
import static org.mockito.Mockito.spy;
import static org.mockito.Mockito.when;
import android.content.Context;
import android.content.pm.PackageManager;
import android.hardware.face.FaceManager;
import com.android.settings.R;
import com.android.settings.SettingsActivity;
import com.android.settings.biometrics.face.FaceSettingsRemoveButtonPreferenceController.ConfirmRemoveDialog;
import com.android.settings.testutils.shadow.ShadowUserManager;
import com.android.settingslib.widget.LayoutPreference;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.Mock;
import org.mockito.junit.MockitoJUnit;
import org.mockito.junit.MockitoRule;
import org.robolectric.Robolectric;
import org.robolectric.RobolectricTestRunner;
import org.robolectric.RuntimeEnvironment;
import org.robolectric.annotation.Config;
import org.robolectric.shadows.ShadowApplication;
@RunWith(RobolectricTestRunner.class)
@Config(shadows = {ShadowUserManager.class})
public class FaceSettingsRemoveButtonPreferenceControllerTest {
@Rule
public final MockitoRule mMockitoRule = MockitoJUnit.rule();
private static final String TEST_PREF_KEY = "baz";
@Mock
private FaceManager mFaceManager;
@Mock
private PackageManager mPackageManager;
private SettingsActivity mActivity;
private Context mContext;
private FaceSettingsRemoveButtonPreferenceController mController;
private LayoutPreference mPreference;
@Before
public void setUp() {
mContext = spy(RuntimeEnvironment.application);
when(mContext.getPackageManager()).thenReturn(mPackageManager);
when(mPackageManager.hasSystemFeature(PackageManager.FEATURE_FACE)).thenReturn(true);
ShadowApplication.getInstance().setSystemService(Context.FACE_SERVICE, mFaceManager);
mPreference = new LayoutPreference(mContext, R.layout.face_remove_button);
mController = new FaceSettingsRemoveButtonPreferenceController(mContext, TEST_PREF_KEY);
mActivity = spy(Robolectric.buildActivity(SettingsActivity.class).create().get());
mController.setActivity(mActivity);
}
@Test
public void testRotationConfirmRemoveDialog() {
// mController calls onClick(), the dialog is created.
mController.updateState(mPreference);
assertThat(mController.mRemoving).isFalse();
mController.onClick(
mPreference.findViewById(R.id.security_settings_face_settings_remove_button));
ConfirmRemoveDialog removeDialog =
(ConfirmRemoveDialog) mActivity.getSupportFragmentManager()
.findFragmentByTag(ConfirmRemoveDialog.class.getName());
assertThat(removeDialog).isNotNull();
assertThat(mController.mRemoving).isTrue();
// Simulate rotation, a new controller mController2 is created and updateState() is called.
// Since the dialog hasn't been dismissed, so mController2.mRemoving should be true
FaceSettingsRemoveButtonPreferenceController controller2 =
new FaceSettingsRemoveButtonPreferenceController(mContext, TEST_PREF_KEY);
controller2.setActivity(mActivity);
assertThat(controller2.mRemoving).isFalse();
controller2.updateState(mPreference);
assertThat(controller2.mRemoving).isTrue();
}
}