1 /*
2  * Copyright (C) 2017 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 com.android.car.settings.accounts;
17 
18 import static android.content.Intent.EXTRA_USER;
19 
20 import android.accounts.AccountManager;
21 import android.accounts.AccountManagerCallback;
22 import android.accounts.AccountManagerFuture;
23 import android.accounts.AuthenticatorException;
24 import android.accounts.OperationCanceledException;
25 import android.app.Activity;
26 import android.app.PendingIntent;
27 import android.content.ComponentName;
28 import android.content.Context;
29 import android.content.Intent;
30 import android.os.Bundle;
31 import android.os.UserHandle;
32 import android.os.UserManager;
33 import android.widget.Toast;
34 
35 import com.android.car.settings.R;
36 import com.android.car.settings.common.Logger;
37 
38 import java.io.IOException;
39 
40 /**
41  * Entry point Activity for account setup. Works as follows
42  *
43  * <ol>
44  * <li> After receiving an account type from ChooseAccountFragment, this Activity launches the
45  * account setup specified by AccountManager.
46  * <li> After the account setup, this Activity finishes without showing anything.
47  * </ol>
48  */
49 public class AddAccountActivity extends Activity {
50     /**
51      * A boolean to keep the state of whether add account has already been called.
52      */
53     private static final String KEY_ADD_CALLED = "AddAccountCalled";
54     /**
55      * Extra parameter to identify the caller. Applications may display a
56      * different UI if the call is made from Settings or from a specific
57      * application.
58      */
59     private static final String KEY_CALLER_IDENTITY = "pendingIntent";
60     private static final String SHOULD_NOT_RESOLVE = "SHOULDN'T RESOLVE!";
61 
62     private static final Logger LOG = new Logger(AddAccountActivity.class);
63     private static final String ALLOW_SKIP = "allowSkip";
64 
65     /* package */ static final String EXTRA_SELECTED_ACCOUNT = "selected_account";
66 
67     // Show additional info regarding the use of a device with multiple users
68     static final String EXTRA_HAS_MULTIPLE_USERS = "hasMultipleUsers";
69 
70     // Need a specific request code for add account activity.
71     private static final int ADD_ACCOUNT_REQUEST = 2001;
72 
73     private UserManager mUserManager;
74     private UserHandle mUserHandle;
75     private PendingIntent mPendingIntent;
76     private boolean mAddAccountCalled;
77 
78     private final AccountManagerCallback<Bundle> mCallback = new AccountManagerCallback<Bundle>() {
79         @Override
80         public void run(AccountManagerFuture<Bundle> future) {
81             if (!future.isDone()) {
82                 LOG.v("Account manager future is not done.");
83                 finish();
84             }
85             boolean done = true;
86             try {
87                 Bundle result = future.getResult();
88                 Intent intent = result.getParcelable(AccountManager.KEY_INTENT);
89                 if (intent != null) {
90                     done = false;
91                     Bundle addAccountOptions = new Bundle();
92                     addAccountOptions.putBoolean(EXTRA_HAS_MULTIPLE_USERS,
93                             hasMultipleUsers(AddAccountActivity.this));
94                     addAccountOptions.putParcelable(EXTRA_USER, mUserHandle);
95                     intent.putExtras(addAccountOptions);
96                     intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
97                     startActivityForResultAsUser(
98                             new Intent(intent), ADD_ACCOUNT_REQUEST, mUserHandle);
99                 } else {
100                     setResult(RESULT_OK);
101                     if (mPendingIntent != null) {
102                         mPendingIntent.cancel();
103                         mPendingIntent = null;
104                     }
105                 }
106                 LOG.v("account added: " + result);
107             } catch (OperationCanceledException | IOException | AuthenticatorException e) {
108                 LOG.v("addAccount error: " + e);
109             } finally {
110                 if (done) {
111                     finish();
112                 }
113             }
114         }
115     };
116 
117     /**
118      * Creates an intent to start the {@link AddAccountActivity} to add an account of the given
119      * account type.
120      */
createAddAccountActivityIntent(Context context, String accountType)121     public static Intent createAddAccountActivityIntent(Context context, String accountType) {
122         Intent intent = new Intent(context, AddAccountActivity.class);
123         intent.putExtra(EXTRA_SELECTED_ACCOUNT, accountType);
124         return intent;
125     }
126 
127     @Override
onSaveInstanceState(Bundle outState)128     protected void onSaveInstanceState(Bundle outState) {
129         super.onSaveInstanceState(outState);
130         outState.putBoolean(KEY_ADD_CALLED, mAddAccountCalled);
131         LOG.v("saved");
132     }
133 
134     @Override
onCreate(Bundle savedInstanceState)135     public void onCreate(Bundle savedInstanceState) {
136         super.onCreate(savedInstanceState);
137         if (savedInstanceState != null) {
138             mAddAccountCalled = savedInstanceState.getBoolean(KEY_ADD_CALLED);
139             LOG.v("Restored from previous add account call: " + mAddAccountCalled);
140         }
141 
142         mUserManager = UserManager.get(getApplicationContext());
143         mUserHandle = UserHandle.of(UserHandle.myUserId());
144 
145         if (mUserManager.hasUserRestriction(UserManager.DISALLOW_MODIFY_ACCOUNTS)) {
146             // We aren't allowed to add an account.
147             Toast.makeText(
148                     this, R.string.user_cannot_add_accounts_message, Toast.LENGTH_LONG)
149                     .show();
150             finish();
151             return;
152         }
153 
154         if (mAddAccountCalled) {
155             // We already called add account - maybe the callback was lost.
156             finish();
157             return;
158         }
159         addAccount(getIntent().getStringExtra(EXTRA_SELECTED_ACCOUNT));
160     }
161 
162     @Override
onActivityResult(int requestCode, int resultCode, Intent data)163     public void onActivityResult(int requestCode, int resultCode, Intent data) {
164         setResult(resultCode);
165         if (mPendingIntent != null) {
166             mPendingIntent.cancel();
167             mPendingIntent = null;
168         }
169         finish();
170     }
171 
addAccount(String accountType)172     private void addAccount(String accountType) {
173         Bundle addAccountOptions = new Bundle();
174         addAccountOptions.putBoolean(EXTRA_HAS_MULTIPLE_USERS, hasMultipleUsers(this));
175         addAccountOptions.putBoolean(ALLOW_SKIP, true);
176 
177         /*
178          * The identityIntent is for the purposes of establishing the identity
179          * of the caller and isn't intended for launching activities, services
180          * or broadcasts.
181          */
182         Intent identityIntent = new Intent();
183         identityIntent.setComponent(new ComponentName(SHOULD_NOT_RESOLVE, SHOULD_NOT_RESOLVE));
184         identityIntent.setAction(SHOULD_NOT_RESOLVE);
185         identityIntent.addCategory(SHOULD_NOT_RESOLVE);
186 
187         mPendingIntent =
188                 PendingIntent.getBroadcast(this, 0, identityIntent, PendingIntent.FLAG_IMMUTABLE);
189         addAccountOptions.putParcelable(KEY_CALLER_IDENTITY, mPendingIntent);
190 
191         AccountManager.get(this).addAccountAsUser(
192                 accountType,
193                 /* authTokenType= */ null,
194                 /* requiredFeatures= */ null,
195                 addAccountOptions,
196                 null,
197                 mCallback,
198                 /* handler= */ null,
199                 mUserHandle);
200         mAddAccountCalled = true;
201     }
202 
hasMultipleUsers(Context context)203     private boolean hasMultipleUsers(Context context) {
204         return ((UserManager) context.getSystemService(Context.USER_SERVICE))
205                 .getUsers().size() > 1;
206     }
207 }
208