• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2012 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.keyguard;
18 
19 import com.android.internal.telephony.ITelephony;
20 import com.android.internal.telephony.IccCardConstants;
21 import com.android.internal.telephony.IccCardConstants.State;
22 import com.android.internal.telephony.PhoneConstants;
23 
24 import android.content.Context;
25 import android.content.res.ColorStateList;
26 import android.content.res.Configuration;
27 import android.content.res.Resources;
28 import android.app.AlertDialog;
29 import android.app.AlertDialog.Builder;
30 import android.app.Dialog;
31 import android.app.ProgressDialog;
32 import android.graphics.Color;
33 import android.os.RemoteException;
34 import android.os.ServiceManager;
35 import android.telephony.SubscriptionInfo;
36 import android.telephony.SubscriptionManager;
37 import android.telephony.TelephonyManager;
38 import android.telephony.euicc.EuiccManager;
39 import android.util.AttributeSet;
40 import android.util.Log;
41 import android.view.View;
42 import android.view.WindowManager;
43 import android.widget.ImageView;
44 
45 /**
46  * Displays a PIN pad for unlocking.
47  */
48 public class KeyguardSimPinView extends KeyguardPinBasedInputView {
49     private static final String LOG_TAG = "KeyguardSimPinView";
50     private static final boolean DEBUG = KeyguardConstants.DEBUG_SIM_STATES;
51     public static final String TAG = "KeyguardSimPinView";
52 
53     private ProgressDialog mSimUnlockProgressDialog = null;
54     private CheckSimPin mCheckSimPinThread;
55 
56     // Below flag is set to true during power-up or when a new SIM card inserted on device.
57     // When this is true and when SIM card is PIN locked state, on PIN lock screen, message would
58     // be displayed to inform user about the number of remaining PIN attempts left.
59     private boolean mShowDefaultMessage = true;
60     private int mRemainingAttempts = -1;
61     private AlertDialog mRemainingAttemptsDialog;
62     private int mSubId = SubscriptionManager.INVALID_SUBSCRIPTION_ID;
63     private ImageView mSimImageView;
64 
65     KeyguardUpdateMonitorCallback mUpdateMonitorCallback = new KeyguardUpdateMonitorCallback() {
66         @Override
67         public void onSimStateChanged(int subId, int slotId, State simState) {
68             if (DEBUG) Log.v(TAG, "onSimStateChanged(subId=" + subId + ",state=" + simState + ")");
69             switch(simState) {
70                 // If the SIM is removed, then we must remove the keyguard. It will be put up
71                 // again when the PUK locked SIM is re-entered.
72                 case ABSENT: {
73                     KeyguardUpdateMonitor.getInstance(getContext()).reportSimUnlocked(mSubId);
74                     // onSimStateChanged callback can fire when the SIM PIN lock is not currently
75                     // active and mCallback is null.
76                     if (mCallback != null) {
77                         mCallback.dismiss(true, KeyguardUpdateMonitor.getCurrentUser());
78                     }
79                     break;
80                 }
81                 case READY: {
82                     mRemainingAttempts = -1;
83                     resetState();
84                     break;
85                 }
86                 default:
87                     resetState();
88             }
89         }
90     };
91 
KeyguardSimPinView(Context context)92     public KeyguardSimPinView(Context context) {
93         this(context, null);
94     }
95 
KeyguardSimPinView(Context context, AttributeSet attrs)96     public KeyguardSimPinView(Context context, AttributeSet attrs) {
97         super(context, attrs);
98     }
99 
100     @Override
resetState()101     public void resetState() {
102         super.resetState();
103         if (DEBUG) Log.v(TAG, "Resetting state");
104         handleSubInfoChangeIfNeeded();
105         if (mShowDefaultMessage) {
106             showDefaultMessage();
107         }
108         boolean isEsimLocked = KeyguardEsimArea.isEsimLocked(mContext, mSubId);
109 
110         KeyguardEsimArea esimButton = findViewById(R.id.keyguard_esim_area);
111         esimButton.setVisibility(isEsimLocked ? View.VISIBLE : View.GONE);
112     }
113 
showDefaultMessage()114     private void showDefaultMessage() {
115         if (mRemainingAttempts >= 0) {
116             mSecurityMessageDisplay.setMessage(getPinPasswordErrorMessage(
117                     mRemainingAttempts, true));
118             return;
119         }
120 
121         boolean isEsimLocked = KeyguardEsimArea.isEsimLocked(mContext, mSubId);
122         int count = TelephonyManager.getDefault().getSimCount();
123         Resources rez = getResources();
124         String msg;
125         int color = Color.WHITE;
126         if (count < 2) {
127             msg = rez.getString(R.string.kg_sim_pin_instructions);
128         } else {
129             SubscriptionInfo info = KeyguardUpdateMonitor.getInstance(mContext).
130                     getSubscriptionInfoForSubId(mSubId);
131             CharSequence displayName = info != null ? info.getDisplayName() : ""; // don't crash
132             msg = rez.getString(R.string.kg_sim_pin_instructions_multi, displayName);
133             if (info != null) {
134                 color = info.getIconTint();
135             }
136         }
137 
138         if (isEsimLocked) {
139             msg = rez.getString(R.string.kg_sim_lock_esim_instructions, msg);
140         }
141 
142         mSecurityMessageDisplay.setMessage(msg);
143         mSimImageView.setImageTintList(ColorStateList.valueOf(color));
144 
145         // Sending empty PIN here to query the number of remaining PIN attempts
146         new CheckSimPin("", mSubId) {
147             void onSimCheckResponse(final int result, final int attemptsRemaining) {
148                 Log.d(LOG_TAG, "onSimCheckResponse " + " dummy One result" + result +
149                         " attemptsRemaining=" + attemptsRemaining);
150                 if (attemptsRemaining >= 0) {
151                     mRemainingAttempts = attemptsRemaining;
152                     mSecurityMessageDisplay.setMessage(
153                             getPinPasswordErrorMessage(attemptsRemaining, true));
154                 }
155             }
156         }.start();
157     }
158 
handleSubInfoChangeIfNeeded()159     private void handleSubInfoChangeIfNeeded() {
160         KeyguardUpdateMonitor monitor = KeyguardUpdateMonitor.getInstance(mContext);
161         int subId = monitor.getNextSubIdForState(IccCardConstants.State.PIN_REQUIRED);
162         if (subId != mSubId && SubscriptionManager.isValidSubscriptionId(subId)) {
163             mSubId = subId;
164             mShowDefaultMessage = true;
165             mRemainingAttempts = -1;
166         }
167     }
168 
169     @Override
onConfigurationChanged(Configuration newConfig)170     protected void onConfigurationChanged(Configuration newConfig) {
171         super.onConfigurationChanged(newConfig);
172         resetState();
173     }
174 
175     @Override
getPromptReasonStringRes(int reason)176     protected int getPromptReasonStringRes(int reason) {
177         // No message on SIM Pin
178         return 0;
179     }
180 
getPinPasswordErrorMessage(int attemptsRemaining, boolean isDefault)181     private String getPinPasswordErrorMessage(int attemptsRemaining, boolean isDefault) {
182         String displayMessage;
183         int msgId;
184         if (attemptsRemaining == 0) {
185             displayMessage = getContext().getString(R.string.kg_password_wrong_pin_code_pukked);
186         } else if (attemptsRemaining > 0) {
187             msgId = isDefault ? R.plurals.kg_password_default_pin_message :
188                      R.plurals.kg_password_wrong_pin_code;
189             displayMessage = getContext().getResources()
190                     .getQuantityString(msgId, attemptsRemaining, attemptsRemaining);
191         } else {
192             msgId = isDefault ? R.string.kg_sim_pin_instructions : R.string.kg_password_pin_failed;
193             displayMessage = getContext().getString(msgId);
194         }
195         if (KeyguardEsimArea.isEsimLocked(mContext, mSubId)) {
196             displayMessage = getResources()
197                     .getString(R.string.kg_sim_lock_esim_instructions, displayMessage);
198         }
199         if (DEBUG) Log.d(LOG_TAG, "getPinPasswordErrorMessage:"
200                 + " attemptsRemaining=" + attemptsRemaining + " displayMessage=" + displayMessage);
201         return displayMessage;
202     }
203 
204     @Override
shouldLockout(long deadline)205     protected boolean shouldLockout(long deadline) {
206         // SIM PIN doesn't have a timed lockout
207         return false;
208     }
209 
210     @Override
getPasswordTextViewId()211     protected int getPasswordTextViewId() {
212         return R.id.simPinEntry;
213     }
214 
215     @Override
onFinishInflate()216     protected void onFinishInflate() {
217         super.onFinishInflate();
218 
219         if (mEcaView instanceof EmergencyCarrierArea) {
220             ((EmergencyCarrierArea) mEcaView).setCarrierTextVisible(true);
221         }
222         mSimImageView = findViewById(R.id.keyguard_sim);
223     }
224 
225     @Override
onAttachedToWindow()226     protected void onAttachedToWindow() {
227         super.onAttachedToWindow();
228         KeyguardUpdateMonitor.getInstance(mContext).registerCallback(mUpdateMonitorCallback);
229     }
230 
231     @Override
onDetachedFromWindow()232     protected void onDetachedFromWindow() {
233         super.onDetachedFromWindow();
234         KeyguardUpdateMonitor.getInstance(mContext).removeCallback(mUpdateMonitorCallback);
235     }
236 
237     @Override
showUsabilityHint()238     public void showUsabilityHint() {
239     }
240 
241     @Override
onPause()242     public void onPause() {
243         // dismiss the dialog.
244         if (mSimUnlockProgressDialog != null) {
245             mSimUnlockProgressDialog.dismiss();
246             mSimUnlockProgressDialog = null;
247         }
248     }
249 
250     /**
251      * Since the IPC can block, we want to run the request in a separate thread
252      * with a callback.
253      */
254     private abstract class CheckSimPin extends Thread {
255         private final String mPin;
256         private int mSubId;
257 
CheckSimPin(String pin, int subId)258         protected CheckSimPin(String pin, int subId) {
259             mPin = pin;
260             mSubId = subId;
261         }
262 
onSimCheckResponse(final int result, final int attemptsRemaining)263         abstract void onSimCheckResponse(final int result, final int attemptsRemaining);
264 
265         @Override
run()266         public void run() {
267             try {
268                 if (DEBUG) {
269                     Log.v(TAG, "call supplyPinReportResultForSubscriber(subid=" + mSubId + ")");
270                 }
271                 final int[] result = ITelephony.Stub.asInterface(ServiceManager
272                         .checkService("phone")).supplyPinReportResultForSubscriber(mSubId, mPin);
273                 if (DEBUG) {
274                     Log.v(TAG, "supplyPinReportResult returned: " + result[0] + " " + result[1]);
275                 }
276                 post(new Runnable() {
277                     @Override
278                     public void run() {
279                         onSimCheckResponse(result[0], result[1]);
280                     }
281                 });
282             } catch (RemoteException e) {
283                 Log.e(TAG, "RemoteException for supplyPinReportResult:", e);
284                 post(new Runnable() {
285                     @Override
286                     public void run() {
287                         onSimCheckResponse(PhoneConstants.PIN_GENERAL_FAILURE, -1);
288                     }
289                 });
290             }
291         }
292     }
293 
getSimUnlockProgressDialog()294     private Dialog getSimUnlockProgressDialog() {
295         if (mSimUnlockProgressDialog == null) {
296             mSimUnlockProgressDialog = new ProgressDialog(mContext);
297             mSimUnlockProgressDialog.setMessage(
298                     mContext.getString(R.string.kg_sim_unlock_progress_dialog_message));
299             mSimUnlockProgressDialog.setIndeterminate(true);
300             mSimUnlockProgressDialog.setCancelable(false);
301             mSimUnlockProgressDialog.getWindow().setType(
302                     WindowManager.LayoutParams.TYPE_KEYGUARD_DIALOG);
303         }
304         return mSimUnlockProgressDialog;
305     }
306 
getSimRemainingAttemptsDialog(int remaining)307     private Dialog getSimRemainingAttemptsDialog(int remaining) {
308         String msg = getPinPasswordErrorMessage(remaining, false);
309         if (mRemainingAttemptsDialog == null) {
310             Builder builder = new AlertDialog.Builder(mContext);
311             builder.setMessage(msg);
312             builder.setCancelable(false);
313             builder.setNeutralButton(R.string.ok, null);
314             mRemainingAttemptsDialog = builder.create();
315             mRemainingAttemptsDialog.getWindow().setType(
316                     WindowManager.LayoutParams.TYPE_KEYGUARD_DIALOG);
317         } else {
318             mRemainingAttemptsDialog.setMessage(msg);
319         }
320         return mRemainingAttemptsDialog;
321     }
322 
323     @Override
verifyPasswordAndUnlock()324     protected void verifyPasswordAndUnlock() {
325         String entry = mPasswordEntry.getText();
326 
327         if (entry.length() < 4) {
328             // otherwise, display a message to the user, and don't submit.
329             mSecurityMessageDisplay.setMessage(R.string.kg_invalid_sim_pin_hint);
330             resetPasswordText(true /* animate */, true /* announce */);
331             mCallback.userActivity();
332             return;
333         }
334 
335         getSimUnlockProgressDialog().show();
336 
337         if (mCheckSimPinThread == null) {
338             mCheckSimPinThread = new CheckSimPin(mPasswordEntry.getText(), mSubId) {
339                 @Override
340                 void onSimCheckResponse(final int result, final int attemptsRemaining) {
341                     post(new Runnable() {
342                         @Override
343                         public void run() {
344                             mRemainingAttempts = attemptsRemaining;
345                             if (mSimUnlockProgressDialog != null) {
346                                 mSimUnlockProgressDialog.hide();
347                             }
348                             resetPasswordText(true /* animate */,
349                                     result != PhoneConstants.PIN_RESULT_SUCCESS /* announce */);
350                             if (result == PhoneConstants.PIN_RESULT_SUCCESS) {
351                                 KeyguardUpdateMonitor.getInstance(getContext())
352                                         .reportSimUnlocked(mSubId);
353                                 mRemainingAttempts = -1;
354                                 mShowDefaultMessage = true;
355                                 if (mCallback != null) {
356                                     mCallback.dismiss(true, KeyguardUpdateMonitor.getCurrentUser());
357                                 }
358                             } else {
359                                 mShowDefaultMessage = false;
360                                 if (result == PhoneConstants.PIN_PASSWORD_INCORRECT) {
361                                     if (attemptsRemaining <= 2) {
362                                         // this is getting critical - show dialog
363                                         getSimRemainingAttemptsDialog(attemptsRemaining).show();
364                                     } else {
365                                         // show message
366                                         mSecurityMessageDisplay.setMessage(
367                                                 getPinPasswordErrorMessage(attemptsRemaining, false));
368                                     }
369                                 } else {
370                                     // "PIN operation failed!" - no idea what this was and no way to
371                                     // find out. :/
372                                     mSecurityMessageDisplay.setMessage(getContext().getString(
373                                             R.string.kg_password_pin_failed));
374                                 }
375                                 if (DEBUG) Log.d(LOG_TAG, "verifyPasswordAndUnlock "
376                                         + " CheckSimPin.onSimCheckResponse: " + result
377                                         + " attemptsRemaining=" + attemptsRemaining);
378                             }
379                             mCallback.userActivity();
380                             mCheckSimPinThread = null;
381                         }
382                     });
383                 }
384             };
385             mCheckSimPinThread.start();
386         }
387     }
388 
389     @Override
startAppearAnimation()390     public void startAppearAnimation() {
391         // noop.
392     }
393 
394     @Override
startDisappearAnimation(Runnable finishRunnable)395     public boolean startDisappearAnimation(Runnable finishRunnable) {
396         return false;
397     }
398 
399     @Override
getTitle()400     public CharSequence getTitle() {
401         return getContext().getString(
402                 com.android.internal.R.string.keyguard_accessibility_sim_pin_unlock);
403     }
404 }
405 
406