1 /*
2  * Copyright (C) 2011 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 package android.accounts;
17 
18 import com.google.android.collect.Sets;
19 
20 import android.app.Activity;
21 import android.app.ActivityManagerNative;
22 import android.content.Intent;
23 import android.os.Bundle;
24 import android.os.IBinder;
25 import android.os.Parcelable;
26 import android.os.RemoteException;
27 import android.os.UserHandle;
28 import android.os.UserManager;
29 import android.text.TextUtils;
30 import android.util.Log;
31 import android.view.View;
32 import android.view.Window;
33 import android.widget.AdapterView;
34 import android.widget.ArrayAdapter;
35 import android.widget.Button;
36 import android.widget.ListView;
37 import android.widget.TextView;
38 
39 import com.android.internal.R;
40 
41 import java.io.IOException;
42 import java.util.ArrayList;
43 import java.util.HashSet;
44 import java.util.Set;
45 
46 /**
47  * @hide
48  */
49 public class ChooseTypeAndAccountActivity extends Activity
50         implements AccountManagerCallback<Bundle> {
51     private static final String TAG = "AccountChooser";
52 
53     /**
54      * A Parcelable ArrayList of Account objects that limits the choosable accounts to those
55      * in this list, if this parameter is supplied.
56      */
57     public static final String EXTRA_ALLOWABLE_ACCOUNTS_ARRAYLIST = "allowableAccounts";
58 
59     /**
60      * A Parcelable ArrayList of String objects that limits the accounts to choose to those
61      * that match the types in this list, if this parameter is supplied. This list is also
62      * used to filter the allowable account types if add account is selected.
63      */
64     public static final String EXTRA_ALLOWABLE_ACCOUNT_TYPES_STRING_ARRAY = "allowableAccountTypes";
65 
66     /**
67      * This is passed as the addAccountOptions parameter in AccountManager.addAccount()
68      * if it is called.
69      */
70     public static final String EXTRA_ADD_ACCOUNT_OPTIONS_BUNDLE = "addAccountOptions";
71 
72     /**
73      * This is passed as the requiredFeatures parameter in AccountManager.addAccount()
74      * if it is called.
75      */
76     public static final String EXTRA_ADD_ACCOUNT_REQUIRED_FEATURES_STRING_ARRAY =
77             "addAccountRequiredFeatures";
78 
79     /**
80      * This is passed as the authTokenType string in AccountManager.addAccount()
81      * if it is called.
82      */
83     public static final String EXTRA_ADD_ACCOUNT_AUTH_TOKEN_TYPE_STRING = "authTokenType";
84 
85     /**
86      * If set then the specified account is already "selected".
87      */
88     public static final String EXTRA_SELECTED_ACCOUNT = "selectedAccount";
89 
90     /**
91      * If true then display the account selection list even if there is just
92      * one account to choose from. boolean.
93      */
94     public static final String EXTRA_ALWAYS_PROMPT_FOR_ACCOUNT =
95             "alwaysPromptForAccount";
96 
97     /**
98      * If set then this string willb e used as the description rather than
99      * the default.
100      */
101     public static final String EXTRA_DESCRIPTION_TEXT_OVERRIDE =
102             "descriptionTextOverride";
103 
104     public static final int REQUEST_NULL = 0;
105     public static final int REQUEST_CHOOSE_TYPE = 1;
106     public static final int REQUEST_ADD_ACCOUNT = 2;
107 
108     private static final String KEY_INSTANCE_STATE_PENDING_REQUEST = "pendingRequest";
109     private static final String KEY_INSTANCE_STATE_EXISTING_ACCOUNTS = "existingAccounts";
110     private static final String KEY_INSTANCE_STATE_SELECTED_ACCOUNT_NAME = "selectedAccountName";
111     private static final String KEY_INSTANCE_STATE_SELECTED_ADD_ACCOUNT = "selectedAddAccount";
112     private static final String KEY_INSTANCE_STATE_ACCOUNT_LIST = "accountList";
113 
114     private static final int SELECTED_ITEM_NONE = -1;
115 
116     private Set<Account> mSetOfAllowableAccounts;
117     private Set<String> mSetOfRelevantAccountTypes;
118     private String mSelectedAccountName = null;
119     private boolean mSelectedAddNewAccount = false;
120     private boolean mAlwaysPromptForAccount = false;
121     private String mDescriptionOverride;
122 
123     private ArrayList<Account> mAccounts;
124     private int mPendingRequest = REQUEST_NULL;
125     private Parcelable[] mExistingAccounts = null;
126     private int mSelectedItemIndex;
127     private Button mOkButton;
128     private int mCallingUid;
129     private String mCallingPackage;
130     private boolean mDisallowAddAccounts;
131     private boolean mDontShowPicker;
132 
133     @Override
onCreate(Bundle savedInstanceState)134     public void onCreate(Bundle savedInstanceState) {
135         super.onCreate(savedInstanceState);
136         if (Log.isLoggable(TAG, Log.VERBOSE)) {
137             Log.v(TAG, "ChooseTypeAndAccountActivity.onCreate(savedInstanceState="
138                     + savedInstanceState + ")");
139         }
140 
141         String message = null;
142 
143         try {
144             IBinder activityToken = getActivityToken();
145             mCallingUid = ActivityManagerNative.getDefault().getLaunchedFromUid(activityToken);
146             mCallingPackage = ActivityManagerNative.getDefault().getLaunchedFromPackage(
147                     activityToken);
148             if (mCallingUid != 0 && mCallingPackage != null) {
149                 Bundle restrictions = UserManager.get(this)
150                         .getUserRestrictions(new UserHandle(UserHandle.getUserId(mCallingUid)));
151                 mDisallowAddAccounts =
152                         restrictions.getBoolean(UserManager.DISALLOW_MODIFY_ACCOUNTS, false);
153             }
154         } catch (RemoteException re) {
155             // Couldn't figure out caller details
156             Log.w(getClass().getSimpleName(), "Unable to get caller identity \n" + re);
157         }
158 
159         // save some items we use frequently
160         final Intent intent = getIntent();
161 
162         if (savedInstanceState != null) {
163             mPendingRequest = savedInstanceState.getInt(KEY_INSTANCE_STATE_PENDING_REQUEST);
164             mExistingAccounts =
165                     savedInstanceState.getParcelableArray(KEY_INSTANCE_STATE_EXISTING_ACCOUNTS);
166 
167             // Makes sure that any user selection is preserved across orientation changes.
168             mSelectedAccountName = savedInstanceState.getString(
169                     KEY_INSTANCE_STATE_SELECTED_ACCOUNT_NAME);
170 
171             mSelectedAddNewAccount = savedInstanceState.getBoolean(
172                     KEY_INSTANCE_STATE_SELECTED_ADD_ACCOUNT, false);
173             mAccounts = savedInstanceState.getParcelableArrayList(KEY_INSTANCE_STATE_ACCOUNT_LIST);
174         } else {
175             mPendingRequest = REQUEST_NULL;
176             mExistingAccounts = null;
177             // If the selected account as specified in the intent matches one in the list we will
178             // show is as pre-selected.
179             Account selectedAccount = (Account) intent.getParcelableExtra(EXTRA_SELECTED_ACCOUNT);
180             if (selectedAccount != null) {
181                 mSelectedAccountName = selectedAccount.name;
182             }
183         }
184 
185         if (Log.isLoggable(TAG, Log.VERBOSE)) {
186             Log.v(TAG, "selected account name is " + mSelectedAccountName);
187         }
188 
189 
190         mSetOfAllowableAccounts = getAllowableAccountSet(intent);
191         mSetOfRelevantAccountTypes = getReleventAccountTypes(intent);
192         mAlwaysPromptForAccount = intent.getBooleanExtra(EXTRA_ALWAYS_PROMPT_FOR_ACCOUNT, false);
193         mDescriptionOverride = intent.getStringExtra(EXTRA_DESCRIPTION_TEXT_OVERRIDE);
194 
195         // Need to do this once here to request the window feature. Can't do it in onResume
196         mAccounts = getAcceptableAccountChoices(AccountManager.get(this));
197         if (mAccounts.isEmpty()
198                 && mDisallowAddAccounts) {
199             requestWindowFeature(Window.FEATURE_NO_TITLE);
200             setContentView(R.layout.app_not_authorized);
201             mDontShowPicker = true;
202         }
203     }
204 
205     @Override
onResume()206     protected void onResume() {
207         super.onResume();
208 
209         if (mDontShowPicker) return;
210 
211         final AccountManager accountManager = AccountManager.get(this);
212 
213         mAccounts = getAcceptableAccountChoices(accountManager);
214 
215         // In cases where the activity does not need to show an account picker, cut the chase
216         // and return the result directly. Eg:
217         // Single account -> select it directly
218         // No account -> launch add account activity directly
219         if (mPendingRequest == REQUEST_NULL) {
220             // If there are no relevant accounts and only one relevant account type go directly to
221             // add account. Otherwise let the user choose.
222             if (mAccounts.isEmpty()) {
223                 if (mSetOfRelevantAccountTypes.size() == 1) {
224                     runAddAccountForAuthenticator(mSetOfRelevantAccountTypes.iterator().next());
225                 } else {
226                     startChooseAccountTypeActivity();
227                 }
228                 return;
229             }
230 
231             // if there is only one allowable account return it
232             if (!mAlwaysPromptForAccount && mAccounts.size() == 1) {
233                 Account account = mAccounts.get(0);
234                 setResultAndFinish(account.name, account.type);
235                 return;
236             }
237         }
238 
239         String[] listItems = getListOfDisplayableOptions(mAccounts);
240         mSelectedItemIndex = getItemIndexToSelect(
241             mAccounts, mSelectedAccountName, mSelectedAddNewAccount);
242 
243         // Cannot set content view until we know that mPendingRequest is not null, otherwise
244         // would cause screen flicker.
245         setContentView(R.layout.choose_type_and_account);
246         overrideDescriptionIfSupplied(mDescriptionOverride);
247         populateUIAccountList(listItems);
248 
249         // Only enable "OK" button if something has been selected.
250         mOkButton = (Button) findViewById(android.R.id.button2);
251         mOkButton.setEnabled(mSelectedItemIndex != SELECTED_ITEM_NONE);
252     }
253 
254     @Override
onDestroy()255     protected void onDestroy() {
256         if (Log.isLoggable(TAG, Log.VERBOSE)) {
257             Log.v(TAG, "ChooseTypeAndAccountActivity.onDestroy()");
258         }
259         super.onDestroy();
260     }
261 
262     @Override
onSaveInstanceState(final Bundle outState)263     protected void onSaveInstanceState(final Bundle outState) {
264         super.onSaveInstanceState(outState);
265         outState.putInt(KEY_INSTANCE_STATE_PENDING_REQUEST, mPendingRequest);
266         if (mPendingRequest == REQUEST_ADD_ACCOUNT) {
267             outState.putParcelableArray(KEY_INSTANCE_STATE_EXISTING_ACCOUNTS, mExistingAccounts);
268         }
269         if (mSelectedItemIndex != SELECTED_ITEM_NONE) {
270             if (mSelectedItemIndex == mAccounts.size()) {
271                 outState.putBoolean(KEY_INSTANCE_STATE_SELECTED_ADD_ACCOUNT, true);
272             } else {
273                 outState.putBoolean(KEY_INSTANCE_STATE_SELECTED_ADD_ACCOUNT, false);
274                 outState.putString(KEY_INSTANCE_STATE_SELECTED_ACCOUNT_NAME,
275                         mAccounts.get(mSelectedItemIndex).name);
276             }
277         }
278         outState.putParcelableArrayList(KEY_INSTANCE_STATE_ACCOUNT_LIST, mAccounts);
279     }
280 
onCancelButtonClicked(View view)281     public void onCancelButtonClicked(View view) {
282         onBackPressed();
283     }
284 
onOkButtonClicked(View view)285     public void onOkButtonClicked(View view) {
286         if (mSelectedItemIndex == mAccounts.size()) {
287             // Selected "Add New Account" option
288             startChooseAccountTypeActivity();
289         } else if (mSelectedItemIndex != SELECTED_ITEM_NONE) {
290             onAccountSelected(mAccounts.get(mSelectedItemIndex));
291         }
292     }
293 
294     // Called when the choose account type activity (for adding an account) returns.
295     // If it was a success read the account and set it in the result. In all cases
296     // return the result and finish this activity.
297     @Override
onActivityResult(final int requestCode, final int resultCode, final Intent data)298     protected void onActivityResult(final int requestCode, final int resultCode,
299             final Intent data) {
300         if (Log.isLoggable(TAG, Log.VERBOSE)) {
301             if (data != null && data.getExtras() != null) data.getExtras().keySet();
302             Bundle extras = data != null ? data.getExtras() : null;
303             Log.v(TAG, "ChooseTypeAndAccountActivity.onActivityResult(reqCode=" + requestCode
304                     + ", resCode=" + resultCode + ", extras=" + extras + ")");
305         }
306 
307         // we got our result, so clear the fact that we had a pending request
308         mPendingRequest = REQUEST_NULL;
309 
310         if (resultCode == RESULT_CANCELED) {
311             // if canceling out of addAccount and the original state caused us to skip this,
312             // finish this activity
313             if (mAccounts.isEmpty()) {
314                 setResult(Activity.RESULT_CANCELED);
315                 finish();
316             }
317             return;
318         }
319 
320         if (resultCode == RESULT_OK) {
321             if (requestCode == REQUEST_CHOOSE_TYPE) {
322                 if (data != null) {
323                     String accountType = data.getStringExtra(AccountManager.KEY_ACCOUNT_TYPE);
324                     if (accountType != null) {
325                         runAddAccountForAuthenticator(accountType);
326                         return;
327                     }
328                 }
329                 Log.d(TAG, "ChooseTypeAndAccountActivity.onActivityResult: unable to find account "
330                         + "type, pretending the request was canceled");
331             } else if (requestCode == REQUEST_ADD_ACCOUNT) {
332                 String accountName = null;
333                 String accountType = null;
334 
335                 if (data != null) {
336                     accountName = data.getStringExtra(AccountManager.KEY_ACCOUNT_NAME);
337                     accountType = data.getStringExtra(AccountManager.KEY_ACCOUNT_TYPE);
338                 }
339 
340                 if (accountName == null || accountType == null) {
341                     Account[] currentAccounts = AccountManager.get(this).getAccountsForPackage(
342                             mCallingPackage, mCallingUid);
343                     Set<Account> preExistingAccounts = new HashSet<Account>();
344                     for (Parcelable accountParcel : mExistingAccounts) {
345                         preExistingAccounts.add((Account) accountParcel);
346                     }
347                     for (Account account : currentAccounts) {
348                         if (!preExistingAccounts.contains(account)) {
349                             accountName = account.name;
350                             accountType = account.type;
351                             break;
352                         }
353                     }
354                 }
355 
356                 if (accountName != null || accountType != null) {
357                     setResultAndFinish(accountName, accountType);
358                     return;
359                 }
360             }
361             Log.d(TAG, "ChooseTypeAndAccountActivity.onActivityResult: unable to find added "
362                     + "account, pretending the request was canceled");
363         }
364         if (Log.isLoggable(TAG, Log.VERBOSE)) {
365             Log.v(TAG, "ChooseTypeAndAccountActivity.onActivityResult: canceled");
366         }
367         setResult(Activity.RESULT_CANCELED);
368         finish();
369     }
370 
runAddAccountForAuthenticator(String type)371     protected void runAddAccountForAuthenticator(String type) {
372         if (Log.isLoggable(TAG, Log.VERBOSE)) {
373             Log.v(TAG, "runAddAccountForAuthenticator: " + type);
374         }
375         final Bundle options = getIntent().getBundleExtra(
376                 ChooseTypeAndAccountActivity.EXTRA_ADD_ACCOUNT_OPTIONS_BUNDLE);
377         final String[] requiredFeatures = getIntent().getStringArrayExtra(
378                 ChooseTypeAndAccountActivity.EXTRA_ADD_ACCOUNT_REQUIRED_FEATURES_STRING_ARRAY);
379         final String authTokenType = getIntent().getStringExtra(
380                 ChooseTypeAndAccountActivity.EXTRA_ADD_ACCOUNT_AUTH_TOKEN_TYPE_STRING);
381         AccountManager.get(this).addAccount(type, authTokenType, requiredFeatures,
382                 options, null /* activity */, this /* callback */, null /* Handler */);
383     }
384 
385     @Override
run(final AccountManagerFuture<Bundle> accountManagerFuture)386     public void run(final AccountManagerFuture<Bundle> accountManagerFuture) {
387         try {
388             final Bundle accountManagerResult = accountManagerFuture.getResult();
389             final Intent intent = (Intent)accountManagerResult.getParcelable(
390                     AccountManager.KEY_INTENT);
391             if (intent != null) {
392                 mPendingRequest = REQUEST_ADD_ACCOUNT;
393                 mExistingAccounts = AccountManager.get(this).getAccountsForPackage(mCallingPackage,
394                         mCallingUid);
395                 intent.setFlags(intent.getFlags() & ~Intent.FLAG_ACTIVITY_NEW_TASK);
396                 startActivityForResult(intent, REQUEST_ADD_ACCOUNT);
397                 return;
398             }
399         } catch (OperationCanceledException e) {
400             setResult(Activity.RESULT_CANCELED);
401             finish();
402             return;
403         } catch (IOException e) {
404         } catch (AuthenticatorException e) {
405         }
406         Bundle bundle = new Bundle();
407         bundle.putString(AccountManager.KEY_ERROR_MESSAGE, "error communicating with server");
408         setResult(Activity.RESULT_OK, new Intent().putExtras(bundle));
409         finish();
410     }
411 
onAccountSelected(Account account)412     private void onAccountSelected(Account account) {
413       Log.d(TAG, "selected account " + account);
414       setResultAndFinish(account.name, account.type);
415     }
416 
setResultAndFinish(final String accountName, final String accountType)417     private void setResultAndFinish(final String accountName, final String accountType) {
418         Bundle bundle = new Bundle();
419         bundle.putString(AccountManager.KEY_ACCOUNT_NAME, accountName);
420         bundle.putString(AccountManager.KEY_ACCOUNT_TYPE, accountType);
421         setResult(Activity.RESULT_OK, new Intent().putExtras(bundle));
422         if (Log.isLoggable(TAG, Log.VERBOSE)) {
423             Log.v(TAG, "ChooseTypeAndAccountActivity.setResultAndFinish: "
424                     + "selected account " + accountName + ", " + accountType);
425         }
426         finish();
427     }
428 
startChooseAccountTypeActivity()429     private void startChooseAccountTypeActivity() {
430         if (Log.isLoggable(TAG, Log.VERBOSE)) {
431             Log.v(TAG, "ChooseAccountTypeActivity.startChooseAccountTypeActivity()");
432         }
433         final Intent intent = new Intent(this, ChooseAccountTypeActivity.class);
434         intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_WHEN_TASK_RESET);
435         intent.putExtra(EXTRA_ALLOWABLE_ACCOUNT_TYPES_STRING_ARRAY,
436                 getIntent().getStringArrayExtra(EXTRA_ALLOWABLE_ACCOUNT_TYPES_STRING_ARRAY));
437         intent.putExtra(EXTRA_ADD_ACCOUNT_OPTIONS_BUNDLE,
438                 getIntent().getBundleExtra(EXTRA_ADD_ACCOUNT_OPTIONS_BUNDLE));
439         intent.putExtra(EXTRA_ADD_ACCOUNT_REQUIRED_FEATURES_STRING_ARRAY,
440                 getIntent().getStringArrayExtra(EXTRA_ADD_ACCOUNT_REQUIRED_FEATURES_STRING_ARRAY));
441         intent.putExtra(EXTRA_ADD_ACCOUNT_AUTH_TOKEN_TYPE_STRING,
442                 getIntent().getStringExtra(EXTRA_ADD_ACCOUNT_AUTH_TOKEN_TYPE_STRING));
443         startActivityForResult(intent, REQUEST_CHOOSE_TYPE);
444         mPendingRequest = REQUEST_CHOOSE_TYPE;
445     }
446 
447     /**
448      * @return a value between 0 (inclusive) and accounts.size() (inclusive) or SELECTED_ITEM_NONE.
449      *      An index value of accounts.size() indicates 'Add account' option.
450      */
getItemIndexToSelect(ArrayList<Account> accounts, String selectedAccountName, boolean selectedAddNewAccount)451     private int getItemIndexToSelect(ArrayList<Account> accounts, String selectedAccountName,
452         boolean selectedAddNewAccount) {
453       // If "Add account" option was previously selected by user, preserve it across
454       // orientation changes.
455       if (selectedAddNewAccount) {
456           return accounts.size();
457       }
458       // search for the selected account name if present
459       for (int i = 0; i < accounts.size(); i++) {
460         if (accounts.get(i).name.equals(selectedAccountName)) {
461           return i;
462         }
463       }
464       // no account selected.
465       return SELECTED_ITEM_NONE;
466     }
467 
getListOfDisplayableOptions(ArrayList<Account> accounts)468     private String[] getListOfDisplayableOptions(ArrayList<Account> accounts) {
469       // List of options includes all accounts found together with "Add new account" as the
470       // last item in the list.
471       String[] listItems = new String[accounts.size() + (mDisallowAddAccounts ? 0 : 1)];
472       for (int i = 0; i < accounts.size(); i++) {
473           listItems[i] = accounts.get(i).name;
474       }
475       if (!mDisallowAddAccounts) {
476           listItems[accounts.size()] = getResources().getString(
477                   R.string.add_account_button_label);
478       }
479       return listItems;
480     }
481 
482     /**
483      * Create a list of Account objects for each account that is acceptable. Filter out
484      * accounts that don't match the allowable types, if provided, or that don't match the
485      * allowable accounts, if provided.
486      */
getAcceptableAccountChoices(AccountManager accountManager)487     private ArrayList<Account> getAcceptableAccountChoices(AccountManager accountManager) {
488       final Account[] accounts = accountManager.getAccountsForPackage(mCallingPackage,
489               mCallingUid);
490       ArrayList<Account> accountsToPopulate = new ArrayList<Account>(accounts.length);
491       for (Account account : accounts) {
492           if (mSetOfAllowableAccounts != null
493                   && !mSetOfAllowableAccounts.contains(account)) {
494               continue;
495           }
496           if (mSetOfRelevantAccountTypes != null
497                   && !mSetOfRelevantAccountTypes.contains(account.type)) {
498               continue;
499           }
500           accountsToPopulate.add(account);
501       }
502       return accountsToPopulate;
503     }
504 
505     /**
506      * Return a set of account types speficied by the intent as well as supported by the
507      * AccountManager.
508      */
getReleventAccountTypes(final Intent intent)509     private Set<String> getReleventAccountTypes(final Intent intent) {
510       // An account type is relevant iff it is allowed by the caller and supported by the account
511       // manager.
512       Set<String> setOfRelevantAccountTypes = null;
513       final String[] allowedAccountTypes =
514               intent.getStringArrayExtra(EXTRA_ALLOWABLE_ACCOUNT_TYPES_STRING_ARRAY);
515       if (allowedAccountTypes != null) {
516           setOfRelevantAccountTypes = Sets.newHashSet(allowedAccountTypes);
517           AuthenticatorDescription[] descs = AccountManager.get(this).getAuthenticatorTypes();
518           Set<String> supportedAccountTypes = new HashSet<String>(descs.length);
519           for (AuthenticatorDescription desc : descs) {
520               supportedAccountTypes.add(desc.type);
521           }
522           setOfRelevantAccountTypes.retainAll(supportedAccountTypes);
523       }
524       return setOfRelevantAccountTypes;
525     }
526 
527     /**
528      * Returns a set of whitelisted accounts given by the intent or null if none specified by the
529      * intent.
530      */
getAllowableAccountSet(final Intent intent)531     private Set<Account> getAllowableAccountSet(final Intent intent) {
532       Set<Account> setOfAllowableAccounts = null;
533       final ArrayList<Parcelable> validAccounts =
534               intent.getParcelableArrayListExtra(EXTRA_ALLOWABLE_ACCOUNTS_ARRAYLIST);
535       if (validAccounts != null) {
536           setOfAllowableAccounts = new HashSet<Account>(validAccounts.size());
537           for (Parcelable parcelable : validAccounts) {
538               setOfAllowableAccounts.add((Account)parcelable);
539           }
540       }
541       return setOfAllowableAccounts;
542     }
543 
544     /**
545      * Overrides the description text view for the picker activity if specified by the intent.
546      * If not specified then makes the description invisible.
547      */
overrideDescriptionIfSupplied(String descriptionOverride)548     private void overrideDescriptionIfSupplied(String descriptionOverride) {
549       TextView descriptionView = (TextView) findViewById(R.id.description);
550       if (!TextUtils.isEmpty(descriptionOverride)) {
551           descriptionView.setText(descriptionOverride);
552       } else {
553           descriptionView.setVisibility(View.GONE);
554       }
555     }
556 
557     /**
558      * Populates the UI ListView with the given list of items and selects an item
559      * based on {@code mSelectedItemIndex} member variable.
560      */
populateUIAccountList(String[] listItems)561     private final void populateUIAccountList(String[] listItems) {
562       ListView list = (ListView) findViewById(android.R.id.list);
563       list.setAdapter(new ArrayAdapter<String>(this,
564               android.R.layout.simple_list_item_single_choice, listItems));
565       list.setChoiceMode(ListView.CHOICE_MODE_SINGLE);
566       list.setItemsCanFocus(false);
567       list.setOnItemClickListener(
568               new AdapterView.OnItemClickListener() {
569                   @Override
570                   public void onItemClick(AdapterView<?> parent, View v, int position, long id) {
571                       mSelectedItemIndex = position;
572                       mOkButton.setEnabled(true);
573                   }
574               });
575       if (mSelectedItemIndex != SELECTED_ITEM_NONE) {
576           list.setItemChecked(mSelectedItemIndex, true);
577           if (Log.isLoggable(TAG, Log.VERBOSE)) {
578               Log.v(TAG, "List item " + mSelectedItemIndex + " should be selected");
579           }
580       }
581     }
582 }
583