Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2018 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 |
Brad Stenning | 8d1a51c | 2018-11-20 17:34:16 -0800 | [diff] [blame] | 14 | * limitations under the License. |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 15 | */ |
| 16 | |
| 17 | package com.android.systemui.statusbar.car; |
| 18 | |
Aarthi Balachander | a709600 | 2018-05-21 18:12:25 -0700 | [diff] [blame] | 19 | import static android.content.DialogInterface.BUTTON_NEGATIVE; |
Aarthi Balachander | 0e0e921 | 2018-04-19 19:25:33 -0700 | [diff] [blame] | 20 | import static android.content.DialogInterface.BUTTON_POSITIVE; |
| 21 | |
| 22 | import android.app.AlertDialog; |
| 23 | import android.app.AlertDialog.Builder; |
| 24 | import android.app.Dialog; |
Ying Zheng | ad2773f | 2018-09-14 10:31:07 -0700 | [diff] [blame] | 25 | import android.car.userlib.CarUserManagerHelper; |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 26 | import android.content.Context; |
Aarthi Balachander | 0e0e921 | 2018-04-19 19:25:33 -0700 | [diff] [blame] | 27 | import android.content.DialogInterface; |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 28 | import android.content.pm.UserInfo; |
| 29 | import android.content.res.Resources; |
jovanak | e4ce3cc | 2018-04-19 12:17:12 -0700 | [diff] [blame] | 30 | import android.graphics.Bitmap; |
Heemin Seog | 0d5e018 | 2019-03-13 13:49:24 -0700 | [diff] [blame^] | 31 | import android.graphics.Rect; |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 32 | import android.os.AsyncTask; |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 33 | import android.util.AttributeSet; |
| 34 | import android.view.LayoutInflater; |
| 35 | import android.view.View; |
| 36 | import android.view.ViewGroup; |
| 37 | import android.widget.ImageView; |
| 38 | import android.widget.TextView; |
| 39 | |
Aarthi Balachander | e3110e4 | 2018-04-30 18:16:26 -0700 | [diff] [blame] | 40 | import androidx.core.graphics.drawable.RoundedBitmapDrawable; |
| 41 | import androidx.core.graphics.drawable.RoundedBitmapDrawableFactory; |
Heemin Seog | 0d5e018 | 2019-03-13 13:49:24 -0700 | [diff] [blame^] | 42 | import androidx.recyclerview.widget.GridLayoutManager; |
Aarthi Balachander | a709600 | 2018-05-21 18:12:25 -0700 | [diff] [blame] | 43 | import androidx.recyclerview.widget.RecyclerView; |
| 44 | |
jovanak | e4ce3cc | 2018-04-19 12:17:12 -0700 | [diff] [blame] | 45 | import com.android.internal.util.UserIcons; |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 46 | import com.android.systemui.R; |
Aarthi Balachander | 0e0e921 | 2018-04-19 19:25:33 -0700 | [diff] [blame] | 47 | import com.android.systemui.statusbar.phone.SystemUIDialog; |
Gus Prevas | ab33679 | 2018-11-14 13:52:20 -0500 | [diff] [blame] | 48 | |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 49 | import java.util.ArrayList; |
| 50 | import java.util.List; |
| 51 | |
| 52 | /** |
| 53 | * Displays a GridLayout with icons for the users in the system to allow switching between users. |
| 54 | * One of the uses of this is for the lock screen in auto. |
| 55 | */ |
Heemin Seog | 0d5e018 | 2019-03-13 13:49:24 -0700 | [diff] [blame^] | 56 | public class UserGridRecyclerView extends RecyclerView implements |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 57 | CarUserManagerHelper.OnUsersUpdateListener { |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 58 | private UserSelectionListener mUserSelectionListener; |
| 59 | private UserAdapter mAdapter; |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 60 | private CarUserManagerHelper mCarUserManagerHelper; |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 61 | private Context mContext; |
| 62 | |
| 63 | public UserGridRecyclerView(Context context, AttributeSet attrs) { |
| 64 | super(context, attrs); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 65 | mContext = context; |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 66 | mCarUserManagerHelper = new CarUserManagerHelper(mContext); |
Heemin Seog | 0d5e018 | 2019-03-13 13:49:24 -0700 | [diff] [blame^] | 67 | |
| 68 | addItemDecoration(new ItemSpacingDecoration(context.getResources().getDimensionPixelSize( |
| 69 | R.dimen.car_user_switcher_vertical_spacing_between_users))); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 70 | } |
| 71 | |
| 72 | /** |
| 73 | * Register listener for any update to the users |
| 74 | */ |
| 75 | @Override |
| 76 | public void onFinishInflate() { |
Aarthi Balachander | 608b6e3 | 2018-04-11 18:41:52 -0700 | [diff] [blame] | 77 | super.onFinishInflate(); |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 78 | mCarUserManagerHelper.registerOnUsersUpdateListener(this); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 79 | } |
| 80 | |
| 81 | /** |
| 82 | * Unregisters listener checking for any change to the users |
| 83 | */ |
| 84 | @Override |
| 85 | public void onDetachedFromWindow() { |
Aarthi Balachander | 608b6e3 | 2018-04-11 18:41:52 -0700 | [diff] [blame] | 86 | super.onDetachedFromWindow(); |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 87 | mCarUserManagerHelper.unregisterOnUsersUpdateListener(this); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 88 | } |
| 89 | |
| 90 | /** |
| 91 | * Initializes the adapter that populates the grid layout |
| 92 | * |
| 93 | * @return the adapter |
| 94 | */ |
| 95 | public void buildAdapter() { |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 96 | List<UserRecord> userRecords = createUserRecords(mCarUserManagerHelper |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 97 | .getAllUsers()); |
| 98 | mAdapter = new UserAdapter(mContext, userRecords); |
| 99 | super.setAdapter(mAdapter); |
| 100 | } |
| 101 | |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 102 | private List<UserRecord> createUserRecords(List<UserInfo> userInfoList) { |
| 103 | List<UserRecord> userRecords = new ArrayList<>(); |
jovanak | 2baf8d6 | 2018-08-09 12:59:50 -0700 | [diff] [blame] | 104 | |
| 105 | // If the foreground user CANNOT switch to other users, only display the foreground user. |
| 106 | if (!mCarUserManagerHelper.canForegroundUserSwitchUsers()) { |
| 107 | userRecords.add(createForegroundUserRecord()); |
| 108 | return userRecords; |
| 109 | } |
| 110 | |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 111 | for (UserInfo userInfo : userInfoList) { |
jovanak | 80b4859 | 2018-04-11 17:09:53 -0700 | [diff] [blame] | 112 | if (userInfo.isGuest()) { |
| 113 | // Don't display guests in the switcher. |
| 114 | continue; |
| 115 | } |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 116 | |
| 117 | boolean isForeground = |
| 118 | mCarUserManagerHelper.getCurrentForegroundUserId() == userInfo.id; |
jovanak | 80b4859 | 2018-04-11 17:09:53 -0700 | [diff] [blame] | 119 | UserRecord record = new UserRecord(userInfo, false /* isStartGuestSession */, |
jovanak | 0535abc | 2018-04-10 15:14:50 -0700 | [diff] [blame] | 120 | false /* isAddUser */, isForeground); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 121 | userRecords.add(record); |
| 122 | } |
| 123 | |
jovanak | 2d26ae3 | 2018-07-31 10:44:41 -0700 | [diff] [blame] | 124 | // Add button for starting guest session. |
| 125 | userRecords.add(createStartGuestUserRecord()); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 126 | |
jovanak | 0535abc | 2018-04-10 15:14:50 -0700 | [diff] [blame] | 127 | // Add add user record if the foreground user can add users |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 128 | if (mCarUserManagerHelper.canForegroundUserAddUsers()) { |
jovanak | 2d26ae3 | 2018-07-31 10:44:41 -0700 | [diff] [blame] | 129 | userRecords.add(createAddUserRecord()); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 130 | } |
| 131 | |
| 132 | return userRecords; |
| 133 | } |
| 134 | |
jovanak | 2baf8d6 | 2018-08-09 12:59:50 -0700 | [diff] [blame] | 135 | private UserRecord createForegroundUserRecord() { |
| 136 | return new UserRecord(mCarUserManagerHelper.getCurrentForegroundUserInfo(), |
| 137 | false /* isStartGuestSession */, false /* isAddUser */, true /* isForeground */); |
| 138 | } |
| 139 | |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 140 | /** |
| 141 | * Create guest user record |
| 142 | */ |
jovanak | 2d26ae3 | 2018-07-31 10:44:41 -0700 | [diff] [blame] | 143 | private UserRecord createStartGuestUserRecord() { |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 144 | UserInfo userInfo = new UserInfo(); |
jovanak | 2d26ae3 | 2018-07-31 10:44:41 -0700 | [diff] [blame] | 145 | userInfo.name = mContext.getString(R.string.start_guest_session); |
| 146 | return new UserRecord(userInfo, true /* isStartGuestSession */, false /* isAddUser */, |
| 147 | false /* isForeground */); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 148 | } |
| 149 | |
| 150 | /** |
| 151 | * Create add user record |
| 152 | */ |
jovanak | 2d26ae3 | 2018-07-31 10:44:41 -0700 | [diff] [blame] | 153 | private UserRecord createAddUserRecord() { |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 154 | UserInfo userInfo = new UserInfo(); |
| 155 | userInfo.name = mContext.getString(R.string.car_add_user); |
jovanak | 80b4859 | 2018-04-11 17:09:53 -0700 | [diff] [blame] | 156 | return new UserRecord(userInfo, false /* isStartGuestSession */, |
jovanak | 0535abc | 2018-04-10 15:14:50 -0700 | [diff] [blame] | 157 | true /* isAddUser */, false /* isForeground */); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 158 | } |
| 159 | |
| 160 | public void setUserSelectionListener(UserSelectionListener userSelectionListener) { |
| 161 | mUserSelectionListener = userSelectionListener; |
| 162 | } |
| 163 | |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 164 | @Override |
| 165 | public void onUsersUpdate() { |
| 166 | mAdapter.clearUsers(); |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 167 | mAdapter.updateUsers(createUserRecords(mCarUserManagerHelper.getAllUsers())); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 168 | mAdapter.notifyDataSetChanged(); |
| 169 | } |
| 170 | |
| 171 | /** |
| 172 | * Adapter to populate the grid layout with the available user profiles |
| 173 | */ |
Aarthi Balachander | 0e0e921 | 2018-04-19 19:25:33 -0700 | [diff] [blame] | 174 | public final class UserAdapter extends RecyclerView.Adapter<UserAdapter.UserAdapterViewHolder> |
Aarthi Balachander | c6d1366 | 2018-08-13 14:49:41 -0700 | [diff] [blame] | 175 | implements Dialog.OnClickListener, Dialog.OnCancelListener { |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 176 | |
| 177 | private final Context mContext; |
| 178 | private List<UserRecord> mUsers; |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 179 | private final Resources mRes; |
| 180 | private final String mGuestName; |
| 181 | private final String mNewUserName; |
Aarthi Balachander | 0e0e921 | 2018-04-19 19:25:33 -0700 | [diff] [blame] | 182 | // View that holds the add user button. Used to enable/disable the view |
| 183 | private View mAddUserView; |
Aarthi Balachander | a709600 | 2018-05-21 18:12:25 -0700 | [diff] [blame] | 184 | // User record for the add user. Need to call notifyUserSelected only if the user |
| 185 | // confirms adding a user |
| 186 | private UserRecord mAddUserRecord; |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 187 | |
| 188 | public UserAdapter(Context context, List<UserRecord> users) { |
| 189 | mRes = context.getResources(); |
| 190 | mContext = context; |
| 191 | updateUsers(users); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 192 | mGuestName = mRes.getString(R.string.car_guest); |
| 193 | mNewUserName = mRes.getString(R.string.car_new_user); |
| 194 | } |
| 195 | |
| 196 | public void clearUsers() { |
| 197 | mUsers.clear(); |
| 198 | } |
| 199 | |
| 200 | public void updateUsers(List<UserRecord> users) { |
| 201 | mUsers = users; |
| 202 | } |
| 203 | |
| 204 | @Override |
| 205 | public UserAdapterViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { |
| 206 | View view = LayoutInflater.from(mContext) |
| 207 | .inflate(R.layout.car_fullscreen_user_pod, parent, false); |
| 208 | view.setAlpha(1f); |
| 209 | view.bringToFront(); |
| 210 | return new UserAdapterViewHolder(view); |
| 211 | } |
| 212 | |
| 213 | @Override |
| 214 | public void onBindViewHolder(UserAdapterViewHolder holder, int position) { |
| 215 | UserRecord userRecord = mUsers.get(position); |
Aarthi Balachander | e3110e4 | 2018-04-30 18:16:26 -0700 | [diff] [blame] | 216 | RoundedBitmapDrawable circleIcon = RoundedBitmapDrawableFactory.create(mRes, |
Brad Stenning | 8d1a51c | 2018-11-20 17:34:16 -0800 | [diff] [blame] | 217 | getUserRecordIcon(userRecord)); |
Aarthi Balachander | e3110e4 | 2018-04-30 18:16:26 -0700 | [diff] [blame] | 218 | circleIcon.setCircular(true); |
| 219 | holder.mUserAvatarImageView.setImageDrawable(circleIcon); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 220 | holder.mUserNameTextView.setText(userRecord.mInfo.name); |
Aarthi Balachander | a709600 | 2018-05-21 18:12:25 -0700 | [diff] [blame] | 221 | |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 222 | holder.mView.setOnClickListener(v -> { |
| 223 | if (userRecord == null) { |
| 224 | return; |
| 225 | } |
| 226 | |
jovanak | 80b4859 | 2018-04-11 17:09:53 -0700 | [diff] [blame] | 227 | if (userRecord.mIsStartGuestSession) { |
Aarthi Balachander | a709600 | 2018-05-21 18:12:25 -0700 | [diff] [blame] | 228 | notifyUserSelected(userRecord); |
jovanak | 2469ca7 | 2018-09-19 16:30:04 -0700 | [diff] [blame] | 229 | mCarUserManagerHelper.startGuestSession(mGuestName); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 230 | return; |
| 231 | } |
| 232 | |
Aarthi Balachander | 0e0e921 | 2018-04-19 19:25:33 -0700 | [diff] [blame] | 233 | // If the user wants to add a user, show dialog to confirm adding a user |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 234 | if (userRecord.mIsAddUser) { |
Aarthi Balachander | 0e0e921 | 2018-04-19 19:25:33 -0700 | [diff] [blame] | 235 | // Disable button so it cannot be clicked multiple times |
| 236 | mAddUserView = holder.mView; |
| 237 | mAddUserView.setEnabled(false); |
Aarthi Balachander | a709600 | 2018-05-21 18:12:25 -0700 | [diff] [blame] | 238 | mAddUserRecord = userRecord; |
jovanak | a17e168 | 2018-08-09 15:02:13 -0700 | [diff] [blame] | 239 | |
| 240 | handleAddUserClicked(); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 241 | return; |
| 242 | } |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 243 | // If the user doesn't want to be a guest or add a user, switch to the user selected |
Aarthi Balachander | a709600 | 2018-05-21 18:12:25 -0700 | [diff] [blame] | 244 | notifyUserSelected(userRecord); |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 245 | mCarUserManagerHelper.switchToUser(userRecord.mInfo); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 246 | }); |
| 247 | |
| 248 | } |
| 249 | |
jovanak | a17e168 | 2018-08-09 15:02:13 -0700 | [diff] [blame] | 250 | private void handleAddUserClicked() { |
| 251 | if (mCarUserManagerHelper.isUserLimitReached()) { |
| 252 | mAddUserView.setEnabled(true); |
| 253 | showMaxUserLimitReachedDialog(); |
| 254 | } else { |
| 255 | showConfirmAddUserDialog(); |
| 256 | } |
| 257 | } |
| 258 | |
| 259 | private void showMaxUserLimitReachedDialog() { |
| 260 | AlertDialog maxUsersDialog = new Builder(mContext, R.style.Theme_Car_Dark_Dialog_Alert) |
Brad Stenning | 8d1a51c | 2018-11-20 17:34:16 -0800 | [diff] [blame] | 261 | .setTitle(R.string.user_limit_reached_title) |
| 262 | .setMessage(getResources().getQuantityString( |
| 263 | R.plurals.user_limit_reached_message, |
| 264 | mCarUserManagerHelper.getMaxSupportedRealUsers(), |
| 265 | mCarUserManagerHelper.getMaxSupportedRealUsers())) |
| 266 | .setPositiveButton(android.R.string.ok, null) |
| 267 | .create(); |
jovanak | a17e168 | 2018-08-09 15:02:13 -0700 | [diff] [blame] | 268 | // Sets window flags for the SysUI dialog |
| 269 | SystemUIDialog.applyFlags(maxUsersDialog); |
| 270 | maxUsersDialog.show(); |
| 271 | } |
| 272 | |
| 273 | private void showConfirmAddUserDialog() { |
| 274 | String message = mRes.getString(R.string.user_add_user_message_setup) |
Brad Stenning | 8d1a51c | 2018-11-20 17:34:16 -0800 | [diff] [blame] | 275 | .concat(System.getProperty("line.separator")) |
| 276 | .concat(System.getProperty("line.separator")) |
| 277 | .concat(mRes.getString(R.string.user_add_user_message_update)); |
jovanak | a17e168 | 2018-08-09 15:02:13 -0700 | [diff] [blame] | 278 | |
| 279 | AlertDialog addUserDialog = new Builder(mContext, R.style.Theme_Car_Dark_Dialog_Alert) |
Brad Stenning | 8d1a51c | 2018-11-20 17:34:16 -0800 | [diff] [blame] | 280 | .setTitle(R.string.user_add_user_title) |
| 281 | .setMessage(message) |
| 282 | .setNegativeButton(android.R.string.cancel, this) |
| 283 | .setPositiveButton(android.R.string.ok, this) |
| 284 | .setOnCancelListener(this) |
| 285 | .create(); |
jovanak | a17e168 | 2018-08-09 15:02:13 -0700 | [diff] [blame] | 286 | // Sets window flags for the SysUI dialog |
| 287 | SystemUIDialog.applyFlags(addUserDialog); |
| 288 | addUserDialog.show(); |
| 289 | } |
| 290 | |
Aarthi Balachander | a709600 | 2018-05-21 18:12:25 -0700 | [diff] [blame] | 291 | private void notifyUserSelected(UserRecord userRecord) { |
| 292 | // Notify the listener which user was selected |
| 293 | if (mUserSelectionListener != null) { |
| 294 | mUserSelectionListener.onUserSelected(userRecord); |
| 295 | } |
| 296 | } |
| 297 | |
jovanak | e4ce3cc | 2018-04-19 12:17:12 -0700 | [diff] [blame] | 298 | private Bitmap getUserRecordIcon(UserRecord userRecord) { |
| 299 | if (userRecord.mIsStartGuestSession) { |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 300 | return mCarUserManagerHelper.getGuestDefaultIcon(); |
jovanak | e4ce3cc | 2018-04-19 12:17:12 -0700 | [diff] [blame] | 301 | } |
| 302 | |
| 303 | if (userRecord.mIsAddUser) { |
| 304 | return UserIcons.convertToBitmap(mContext |
Brad Stenning | 8d1a51c | 2018-11-20 17:34:16 -0800 | [diff] [blame] | 305 | .getDrawable(R.drawable.car_add_circle_round)); |
jovanak | e4ce3cc | 2018-04-19 12:17:12 -0700 | [diff] [blame] | 306 | } |
| 307 | |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 308 | return mCarUserManagerHelper.getUserIcon(userRecord.mInfo); |
jovanak | e4ce3cc | 2018-04-19 12:17:12 -0700 | [diff] [blame] | 309 | } |
| 310 | |
Aarthi Balachander | 0e0e921 | 2018-04-19 19:25:33 -0700 | [diff] [blame] | 311 | @Override |
| 312 | public void onClick(DialogInterface dialog, int which) { |
Aarthi Balachander | 0e0e921 | 2018-04-19 19:25:33 -0700 | [diff] [blame] | 313 | if (which == BUTTON_POSITIVE) { |
Aarthi Balachander | a709600 | 2018-05-21 18:12:25 -0700 | [diff] [blame] | 314 | notifyUserSelected(mAddUserRecord); |
Aarthi Balachander | 0e0e921 | 2018-04-19 19:25:33 -0700 | [diff] [blame] | 315 | new AddNewUserTask().execute(mNewUserName); |
Aarthi Balachander | a709600 | 2018-05-21 18:12:25 -0700 | [diff] [blame] | 316 | } else if (which == BUTTON_NEGATIVE) { |
| 317 | // Enable the add button only if cancel |
| 318 | if (mAddUserView != null) { |
| 319 | mAddUserView.setEnabled(true); |
| 320 | } |
Aarthi Balachander | 0e0e921 | 2018-04-19 19:25:33 -0700 | [diff] [blame] | 321 | } |
| 322 | } |
| 323 | |
Aarthi Balachander | c6d1366 | 2018-08-13 14:49:41 -0700 | [diff] [blame] | 324 | @Override |
| 325 | public void onCancel(DialogInterface dialog) { |
| 326 | // Enable the add button again if user cancels dialog by clicking outside the dialog |
| 327 | if (mAddUserView != null) { |
| 328 | mAddUserView.setEnabled(true); |
| 329 | } |
| 330 | } |
| 331 | |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 332 | private class AddNewUserTask extends AsyncTask<String, Void, UserInfo> { |
| 333 | |
| 334 | @Override |
| 335 | protected UserInfo doInBackground(String... userNames) { |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 336 | return mCarUserManagerHelper.createNewNonAdminUser(userNames[0]); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 337 | } |
| 338 | |
| 339 | @Override |
| 340 | protected void onPreExecute() { |
| 341 | } |
| 342 | |
| 343 | @Override |
| 344 | protected void onPostExecute(UserInfo user) { |
| 345 | if (user != null) { |
jovanak | 78cacc4 | 2018-08-06 18:38:03 -0700 | [diff] [blame] | 346 | mCarUserManagerHelper.switchToUser(user); |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 347 | } |
| 348 | } |
| 349 | } |
| 350 | |
| 351 | @Override |
| 352 | public int getItemCount() { |
| 353 | return mUsers.size(); |
| 354 | } |
| 355 | |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 356 | public class UserAdapterViewHolder extends RecyclerView.ViewHolder { |
| 357 | |
| 358 | public ImageView mUserAvatarImageView; |
| 359 | public TextView mUserNameTextView; |
| 360 | public View mView; |
| 361 | |
| 362 | public UserAdapterViewHolder(View view) { |
| 363 | super(view); |
| 364 | mView = view; |
| 365 | mUserAvatarImageView = (ImageView) view.findViewById(R.id.user_avatar); |
| 366 | mUserNameTextView = (TextView) view.findViewById(R.id.user_name); |
| 367 | } |
| 368 | } |
| 369 | } |
| 370 | |
| 371 | /** |
| 372 | * Object wrapper class for the userInfo. Use it to distinguish if a profile is a |
jovanak | 0535abc | 2018-04-10 15:14:50 -0700 | [diff] [blame] | 373 | * guest profile, add user profile, or the foreground user. |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 374 | */ |
| 375 | public static final class UserRecord { |
| 376 | |
| 377 | public final UserInfo mInfo; |
jovanak | 80b4859 | 2018-04-11 17:09:53 -0700 | [diff] [blame] | 378 | public final boolean mIsStartGuestSession; |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 379 | public final boolean mIsAddUser; |
jovanak | 0535abc | 2018-04-10 15:14:50 -0700 | [diff] [blame] | 380 | public final boolean mIsForeground; |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 381 | |
jovanak | 80b4859 | 2018-04-11 17:09:53 -0700 | [diff] [blame] | 382 | public UserRecord(UserInfo userInfo, boolean isStartGuestSession, boolean isAddUser, |
jovanak | 0535abc | 2018-04-10 15:14:50 -0700 | [diff] [blame] | 383 | boolean isForeground) { |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 384 | mInfo = userInfo; |
jovanak | 80b4859 | 2018-04-11 17:09:53 -0700 | [diff] [blame] | 385 | mIsStartGuestSession = isStartGuestSession; |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 386 | mIsAddUser = isAddUser; |
jovanak | 0535abc | 2018-04-10 15:14:50 -0700 | [diff] [blame] | 387 | mIsForeground = isForeground; |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 388 | } |
| 389 | } |
| 390 | |
| 391 | /** |
| 392 | * Listener used to notify when a user has been selected |
| 393 | */ |
| 394 | interface UserSelectionListener { |
| 395 | |
| 396 | void onUserSelected(UserRecord record); |
| 397 | } |
Heemin Seog | 0d5e018 | 2019-03-13 13:49:24 -0700 | [diff] [blame^] | 398 | |
| 399 | /** |
| 400 | * A {@link RecyclerView.ItemDecoration} that will add spacing between each item in the |
| 401 | * RecyclerView that it is added to. |
| 402 | */ |
| 403 | private static class ItemSpacingDecoration extends RecyclerView.ItemDecoration { |
| 404 | private int mItemSpacing; |
| 405 | |
| 406 | private ItemSpacingDecoration(int itemSpacing) { |
| 407 | mItemSpacing = itemSpacing; |
| 408 | } |
| 409 | |
| 410 | @Override |
| 411 | public void getItemOffsets(Rect outRect, View view, RecyclerView parent, |
| 412 | RecyclerView.State state) { |
| 413 | super.getItemOffsets(outRect, view, parent, state); |
| 414 | int position = parent.getChildAdapterPosition(view); |
| 415 | |
| 416 | // Skip offset for last item except for GridLayoutManager. |
| 417 | if (position == state.getItemCount() - 1 |
| 418 | && !(parent.getLayoutManager() instanceof GridLayoutManager)) { |
| 419 | return; |
| 420 | } |
| 421 | |
| 422 | outRect.bottom = mItemSpacing; |
| 423 | } |
| 424 | } |
Aarthi Balachander | d8bf249 | 2018-03-30 11:15:59 -0700 | [diff] [blame] | 425 | } |