Merge "Flicker in Fingerprint Enrollment" into tm-qpr-dev am: 2d343061d3
Original change: https://googleplex-android-review.googlesource.com/c/platform/packages/apps/Settings/+/20523590 Change-Id: I8fdd0725e4c0a1b0d2f7ced3ce058522566528f2 Signed-off-by: Automerger Merge Worker <android-build-automerger-merge-worker@system.gserviceaccount.com>
This commit is contained in:
@@ -350,8 +350,8 @@ public class FingerprintEnrollEnrolling extends BiometricsEnrollEnrolling {
|
||||
|
||||
@Override
|
||||
protected BiometricEnrollSidecar getSidecar() {
|
||||
final FingerprintEnrollSidecar sidecar = new FingerprintEnrollSidecar();
|
||||
sidecar.setEnrollReason(FingerprintManager.ENROLL_ENROLL);
|
||||
final FingerprintEnrollSidecar sidecar = new FingerprintEnrollSidecar(this,
|
||||
FingerprintManager.ENROLL_ENROLL);
|
||||
return sidecar;
|
||||
}
|
||||
|
||||
|
||||
@@ -293,8 +293,8 @@ public class FingerprintEnrollFindSensor extends BiometricEnrollBase implements
|
||||
mSidecar = (FingerprintEnrollSidecar) getSupportFragmentManager().findFragmentByTag(
|
||||
FingerprintEnrollEnrolling.TAG_SIDECAR);
|
||||
if (mSidecar == null) {
|
||||
mSidecar = new FingerprintEnrollSidecar();
|
||||
mSidecar.setEnrollReason(FingerprintManager.ENROLL_FIND_SENSOR);
|
||||
mSidecar = new FingerprintEnrollSidecar(this,
|
||||
FingerprintManager.ENROLL_FIND_SENSOR);
|
||||
getSupportFragmentManager().beginTransaction()
|
||||
.add(mSidecar, FingerprintEnrollEnrolling.TAG_SIDECAR)
|
||||
.commitAllowingStateLoss();
|
||||
|
||||
@@ -16,19 +16,19 @@
|
||||
|
||||
package com.android.settings.biometrics.fingerprint;
|
||||
|
||||
import static android.hardware.fingerprint.FingerprintManager.ENROLL_ENROLL;
|
||||
|
||||
import android.app.Activity;
|
||||
import android.app.settings.SettingsEnums;
|
||||
import android.content.Context;
|
||||
import android.hardware.fingerprint.FingerprintManager;
|
||||
import android.os.SystemClock;
|
||||
import android.util.Log;
|
||||
|
||||
import com.android.internal.annotations.VisibleForTesting;
|
||||
import com.android.settings.R;
|
||||
import com.android.settings.biometrics.BiometricEnrollSidecar;
|
||||
|
||||
import java.util.Collections;
|
||||
import java.util.HashSet;
|
||||
import java.util.Set;
|
||||
|
||||
/**
|
||||
* Sidecar fragment to handle the state around fingerprint enrollment.
|
||||
*/
|
||||
@@ -37,19 +37,41 @@ public class FingerprintEnrollSidecar extends BiometricEnrollSidecar {
|
||||
|
||||
private FingerprintUpdater mFingerprintUpdater;
|
||||
private @FingerprintManager.EnrollReason int mEnrollReason;
|
||||
private Set<Integer> mHelpIgnore;
|
||||
private final MessageDisplayController mMessageDisplayController;
|
||||
private final boolean mMessageDisplayControllerFlag;
|
||||
|
||||
/**
|
||||
* Create a new FingerprintEnrollSidecar object.
|
||||
* @param context associated context
|
||||
* @param enrollReason reason for enrollment
|
||||
*/
|
||||
public FingerprintEnrollSidecar(Context context,
|
||||
@FingerprintManager.EnrollReason int enrollReason) {
|
||||
mEnrollReason = enrollReason;
|
||||
|
||||
int helpMinimumDisplayTime = context.getResources().getInteger(
|
||||
R.integer.enrollment_help_minimum_time_display);
|
||||
int progressMinimumDisplayTime = context.getResources().getInteger(
|
||||
R.integer.enrollment_progress_minimum_time_display);
|
||||
boolean progressPriorityOverHelp = context.getResources().getBoolean(
|
||||
R.bool.enrollment_progress_priority_over_help);
|
||||
boolean prioritizeAcquireMessages = context.getResources().getBoolean(
|
||||
R.bool.enrollment_prioritize_acquire_messages);
|
||||
int collectTime = context.getResources().getInteger(
|
||||
R.integer.enrollment_collect_time);
|
||||
mMessageDisplayControllerFlag = context.getResources().getBoolean(
|
||||
R.bool.enrollment_message_display_controller_flag);
|
||||
|
||||
mMessageDisplayController = new MessageDisplayController(context.getMainThreadHandler(),
|
||||
mEnrollmentCallback, SystemClock.elapsedRealtimeClock(), helpMinimumDisplayTime,
|
||||
progressMinimumDisplayTime, progressPriorityOverHelp, prioritizeAcquireMessages,
|
||||
collectTime);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onAttach(Activity activity) {
|
||||
super.onAttach(activity);
|
||||
mFingerprintUpdater = new FingerprintUpdater(activity);
|
||||
final int[] ignoreAcquiredInfo = getResources().getIntArray(
|
||||
R.array.fingerprint_acquired_ignore_list);
|
||||
mHelpIgnore = new HashSet<>();
|
||||
for (int acquiredInfo: ignoreAcquiredInfo) {
|
||||
mHelpIgnore.add(acquiredInfo);
|
||||
}
|
||||
mHelpIgnore = Collections.unmodifiableSet(mHelpIgnore);
|
||||
}
|
||||
|
||||
@Override
|
||||
@@ -62,8 +84,16 @@ public class FingerprintEnrollSidecar extends BiometricEnrollSidecar {
|
||||
getString(R.string.fingerprint_intro_error_unknown));
|
||||
return;
|
||||
}
|
||||
mFingerprintUpdater.enroll(mToken, mEnrollmentCancel, mUserId, mEnrollmentCallback,
|
||||
mEnrollReason);
|
||||
|
||||
if (mEnrollReason == ENROLL_ENROLL && mMessageDisplayControllerFlag) {
|
||||
//API calls need to be processed for {@link FingerprintEnrollEnrolling}
|
||||
mFingerprintUpdater.enroll(mToken, mEnrollmentCancel, mUserId,
|
||||
mMessageDisplayController, mEnrollReason);
|
||||
} else {
|
||||
//No processing required for {@link FingerprintEnrollFindSensor}
|
||||
mFingerprintUpdater.enroll(mToken, mEnrollmentCancel, mUserId, mEnrollmentCallback,
|
||||
mEnrollReason);
|
||||
}
|
||||
}
|
||||
|
||||
public void setEnrollReason(@FingerprintManager.EnrollReason int enrollReason) {
|
||||
@@ -80,9 +110,6 @@ public class FingerprintEnrollSidecar extends BiometricEnrollSidecar {
|
||||
|
||||
@Override
|
||||
public void onEnrollmentHelp(int helpMsgId, CharSequence helpString) {
|
||||
if (mHelpIgnore.contains(helpMsgId)) {
|
||||
return;
|
||||
}
|
||||
FingerprintEnrollSidecar.super.onEnrollmentHelp(helpMsgId, helpString);
|
||||
}
|
||||
|
||||
|
||||
@@ -0,0 +1,264 @@
|
||||
/*
|
||||
* Copyright (C) 2022 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.fingerprint;
|
||||
|
||||
import android.annotation.NonNull;
|
||||
import android.annotation.Nullable;
|
||||
import android.hardware.fingerprint.FingerprintManager;
|
||||
import android.os.Handler;
|
||||
|
||||
import java.time.Clock;
|
||||
import java.util.ArrayDeque;
|
||||
import java.util.Deque;
|
||||
import java.util.HashMap;
|
||||
|
||||
/**
|
||||
* Processes message provided from the enrollment callback and filters them based
|
||||
* on the below configurable flags. This is primarily used to reduce the rate
|
||||
* at which messages come through, which in turns eliminates UI flicker.
|
||||
*/
|
||||
public class MessageDisplayController extends FingerprintManager.EnrollmentCallback {
|
||||
|
||||
private final int mHelpMinimumDisplayTime;
|
||||
private final int mProgressMinimumDisplayTime;
|
||||
private final boolean mProgressPriorityOverHelp;
|
||||
private final boolean mPrioritizeAcquireMessages;
|
||||
private final int mCollectTime;
|
||||
@NonNull
|
||||
private final Deque<HelpMessage> mHelpMessageList;
|
||||
@NonNull
|
||||
private final Deque<ProgressMessage> mProgressMessageList;
|
||||
@NonNull
|
||||
private final Handler mHandler;
|
||||
@NonNull
|
||||
private final Clock mClock;
|
||||
@NonNull
|
||||
private final Runnable mDisplayMessageRunnable;
|
||||
|
||||
@Nullable
|
||||
private ProgressMessage mLastProgressMessageDisplayed;
|
||||
private boolean mMustDisplayProgress;
|
||||
private boolean mWaitingForMessage;
|
||||
@NonNull FingerprintManager.EnrollmentCallback mEnrollmentCallback;
|
||||
|
||||
private abstract static class Message {
|
||||
long mTimeStamp = 0;
|
||||
abstract void display();
|
||||
}
|
||||
|
||||
private class HelpMessage extends Message {
|
||||
private final int mHelpMsgId;
|
||||
private final CharSequence mHelpString;
|
||||
|
||||
HelpMessage(int helpMsgId, CharSequence helpString) {
|
||||
mHelpMsgId = helpMsgId;
|
||||
mHelpString = helpString;
|
||||
mTimeStamp = mClock.millis();
|
||||
}
|
||||
|
||||
@Override
|
||||
void display() {
|
||||
mEnrollmentCallback.onEnrollmentHelp(mHelpMsgId, mHelpString);
|
||||
mHandler.postDelayed(mDisplayMessageRunnable, mHelpMinimumDisplayTime);
|
||||
}
|
||||
}
|
||||
|
||||
private class ProgressMessage extends Message {
|
||||
private final int mRemaining;
|
||||
|
||||
ProgressMessage(int remaining) {
|
||||
mRemaining = remaining;
|
||||
mTimeStamp = mClock.millis();
|
||||
}
|
||||
|
||||
@Override
|
||||
void display() {
|
||||
mEnrollmentCallback.onEnrollmentProgress(mRemaining);
|
||||
mLastProgressMessageDisplayed = this;
|
||||
mHandler.postDelayed(mDisplayMessageRunnable, mProgressMinimumDisplayTime);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Creating a MessageDisplayController object.
|
||||
* @param handler main handler to run message queue
|
||||
* @param enrollmentCallback callback to display messages
|
||||
* @param clock real time system clock
|
||||
* @param helpMinimumDisplayTime the minimum duration (in millis) that
|
||||
* a help message needs to be displayed for
|
||||
* @param progressMinimumDisplayTime the minimum duration (in millis) that
|
||||
* a progress message needs to be displayed for
|
||||
* @param progressPriorityOverHelp if true, then progress message is displayed
|
||||
* when both help and progress message APIs have been called
|
||||
* @param prioritizeAcquireMessages if true, then displays the help message
|
||||
* which has occurred the most after the last display message
|
||||
* @param collectTime the waiting time (in millis) to collect messages when it is idle
|
||||
*/
|
||||
public MessageDisplayController(@NonNull Handler handler,
|
||||
FingerprintManager.EnrollmentCallback enrollmentCallback,
|
||||
@NonNull Clock clock, int helpMinimumDisplayTime, int progressMinimumDisplayTime,
|
||||
boolean progressPriorityOverHelp, boolean prioritizeAcquireMessages,
|
||||
int collectTime) {
|
||||
mClock = clock;
|
||||
mWaitingForMessage = false;
|
||||
mHelpMessageList = new ArrayDeque<>();
|
||||
mProgressMessageList = new ArrayDeque<>();
|
||||
mHandler = handler;
|
||||
mEnrollmentCallback = enrollmentCallback;
|
||||
|
||||
mHelpMinimumDisplayTime = helpMinimumDisplayTime;
|
||||
mProgressMinimumDisplayTime = progressMinimumDisplayTime;
|
||||
mProgressPriorityOverHelp = progressPriorityOverHelp;
|
||||
mPrioritizeAcquireMessages = prioritizeAcquireMessages;
|
||||
mCollectTime = collectTime;
|
||||
|
||||
mDisplayMessageRunnable = () -> {
|
||||
long timeStamp = mClock.millis();
|
||||
Message messageToDisplay = getMessageToDisplay(timeStamp);
|
||||
|
||||
if (messageToDisplay != null) {
|
||||
messageToDisplay.display();
|
||||
} else {
|
||||
mWaitingForMessage = true;
|
||||
}
|
||||
};
|
||||
|
||||
mHandler.postDelayed(mDisplayMessageRunnable, 0);
|
||||
}
|
||||
|
||||
/**
|
||||
* Adds help message to the queue to be processed later.
|
||||
*
|
||||
* @param helpMsgId message Id associated with the help message
|
||||
* @param helpString string associated with the help message
|
||||
*/
|
||||
@Override
|
||||
public void onEnrollmentHelp(int helpMsgId, CharSequence helpString) {
|
||||
mHelpMessageList.add(new HelpMessage(helpMsgId, helpString));
|
||||
|
||||
if (mWaitingForMessage) {
|
||||
mWaitingForMessage = false;
|
||||
mHandler.postDelayed(mDisplayMessageRunnable, mCollectTime);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Adds progress change message to the queue to be processed later.
|
||||
*
|
||||
* @param remaining remaining number of steps to complete enrollment
|
||||
*/
|
||||
@Override
|
||||
public void onEnrollmentProgress(int remaining) {
|
||||
mProgressMessageList.add(new ProgressMessage(remaining));
|
||||
|
||||
if (mWaitingForMessage) {
|
||||
mWaitingForMessage = false;
|
||||
mHandler.postDelayed(mDisplayMessageRunnable, mCollectTime);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onEnrollmentError(int errMsgId, CharSequence errString) {
|
||||
mEnrollmentCallback.onEnrollmentError(errMsgId, errString);
|
||||
}
|
||||
|
||||
private Message getMessageToDisplay(long timeStamp) {
|
||||
ProgressMessage progressMessageToDisplay = getProgressMessageToDisplay(timeStamp);
|
||||
if (mMustDisplayProgress) {
|
||||
mMustDisplayProgress = false;
|
||||
if (progressMessageToDisplay != null) {
|
||||
return progressMessageToDisplay;
|
||||
}
|
||||
if (mLastProgressMessageDisplayed != null) {
|
||||
return mLastProgressMessageDisplayed;
|
||||
}
|
||||
}
|
||||
|
||||
Message helpMessageToDisplay = getHelpMessageToDisplay(timeStamp);
|
||||
if (helpMessageToDisplay != null || progressMessageToDisplay != null) {
|
||||
if (mProgressPriorityOverHelp && progressMessageToDisplay != null) {
|
||||
return progressMessageToDisplay;
|
||||
} else if (helpMessageToDisplay != null) {
|
||||
if (progressMessageToDisplay != null) {
|
||||
mMustDisplayProgress = true;
|
||||
mLastProgressMessageDisplayed = progressMessageToDisplay;
|
||||
}
|
||||
return helpMessageToDisplay;
|
||||
} else {
|
||||
return progressMessageToDisplay;
|
||||
}
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
private ProgressMessage getProgressMessageToDisplay(long timeStamp) {
|
||||
ProgressMessage finalProgressMessage = null;
|
||||
while (mProgressMessageList != null && !mProgressMessageList.isEmpty()) {
|
||||
Message message = mProgressMessageList.peekFirst();
|
||||
if (message.mTimeStamp <= timeStamp) {
|
||||
ProgressMessage progressMessage = mProgressMessageList.pollFirst();
|
||||
if (mLastProgressMessageDisplayed != null
|
||||
&& mLastProgressMessageDisplayed.mRemaining == progressMessage.mRemaining) {
|
||||
continue;
|
||||
}
|
||||
finalProgressMessage = progressMessage;
|
||||
} else {
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return finalProgressMessage;
|
||||
}
|
||||
|
||||
private HelpMessage getHelpMessageToDisplay(long timeStamp) {
|
||||
HashMap<CharSequence, Integer> messageCount = new HashMap<>();
|
||||
HelpMessage finalHelpMessage = null;
|
||||
|
||||
while (mHelpMessageList != null && !mHelpMessageList.isEmpty()) {
|
||||
Message message = mHelpMessageList.peekFirst();
|
||||
if (message.mTimeStamp <= timeStamp) {
|
||||
finalHelpMessage = mHelpMessageList.pollFirst();
|
||||
CharSequence errString = finalHelpMessage.mHelpString;
|
||||
messageCount.put(errString, messageCount.getOrDefault(errString, 0) + 1);
|
||||
} else {
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (mPrioritizeAcquireMessages) {
|
||||
finalHelpMessage = prioritizeHelpMessageByCount(messageCount);
|
||||
}
|
||||
|
||||
return finalHelpMessage;
|
||||
}
|
||||
|
||||
private HelpMessage prioritizeHelpMessageByCount(HashMap<CharSequence, Integer> messageCount) {
|
||||
int maxCount = 0;
|
||||
CharSequence maxCountMessage = null;
|
||||
|
||||
for (CharSequence key :
|
||||
messageCount.keySet()) {
|
||||
if (maxCount < messageCount.get(key)) {
|
||||
maxCountMessage = key;
|
||||
maxCount = messageCount.get(key);
|
||||
}
|
||||
}
|
||||
|
||||
return maxCountMessage != null ? new HelpMessage(0 /* errMsgId */,
|
||||
maxCountMessage) : null;
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user