Loading packages/CarSystemUI/src/com/android/systemui/car/userswitcher/FullScreenUserSwitcherViewController.java +15 −21 Original line number Diff line number Diff line Loading @@ -18,6 +18,8 @@ package com.android.systemui.car.userswitcher; import android.animation.Animator; import android.animation.AnimatorListenerAdapter; import android.car.Car; import android.car.user.CarUserManager; import android.content.Context; import android.content.res.Resources; import android.view.View; Loading @@ -25,6 +27,7 @@ import android.view.View; import androidx.recyclerview.widget.GridLayoutManager; import com.android.systemui.R; import com.android.systemui.car.CarServiceProvider; import com.android.systemui.car.window.OverlayViewController; import com.android.systemui.car.window.OverlayViewGlobalStateController; import com.android.systemui.dagger.qualifiers.Main; Loading @@ -39,7 +42,9 @@ import javax.inject.Singleton; public class FullScreenUserSwitcherViewController extends OverlayViewController { private final Context mContext; private final Resources mResources; private final CarServiceProvider mCarServiceProvider; private final int mShortAnimationDuration; private CarUserManager mCarUserManager; private UserGridRecyclerView mUserGridView; private UserGridRecyclerView.UserSelectionListener mUserSelectionListener; Loading @@ -47,10 +52,16 @@ public class FullScreenUserSwitcherViewController extends OverlayViewController public FullScreenUserSwitcherViewController( Context context, @Main Resources resources, CarServiceProvider carServiceProvider, OverlayViewGlobalStateController overlayViewGlobalStateController) { super(R.id.fullscreen_user_switcher_stub, overlayViewGlobalStateController); mContext = context; mResources = resources; mCarServiceProvider = carServiceProvider; mCarServiceProvider.addListener(car -> { mCarUserManager = (CarUserManager) car.getCarManager(Car.CAR_USER_SERVICE); registerCarUserManagerIfPossible(); }); mShortAnimationDuration = mResources.getInteger(android.R.integer.config_shortAnimTime); } Loading @@ -63,6 +74,7 @@ public class FullScreenUserSwitcherViewController extends OverlayViewController mUserGridView.setLayoutManager(layoutManager); mUserGridView.buildAdapter(); mUserGridView.setUserSelectionListener(mUserSelectionListener); registerCarUserManagerIfPossible(); } @Override Loading Loading @@ -90,18 +102,6 @@ public class FullScreenUserSwitcherViewController extends OverlayViewController } /** * Invalidate underlying view. */ void invalidate() { if (getLayout() == null) { // layout hasn't been inflated. return; } getLayout().invalidate(); } /** * Set {@link UserGridRecyclerView.UserSelectionListener}. */ Loading @@ -110,15 +110,9 @@ public class FullScreenUserSwitcherViewController extends OverlayViewController mUserSelectionListener = userGridSelectionListener; } /** * Returns {@code true} when layout is visible. */ boolean isVisible() { if (getLayout() == null) { // layout hasn't been inflated. return false; private void registerCarUserManagerIfPossible() { if (mUserGridView != null && mCarUserManager != null) { mUserGridView.setCarUserManager(mCarUserManager); } return getLayout().getVisibility() == View.VISIBLE; } } packages/CarSystemUI/src/com/android/systemui/car/userswitcher/UserGridRecyclerView.java +98 −9 Original line number Diff line number Diff line Loading @@ -24,11 +24,15 @@ import static android.view.WindowInsets.Type.statusBars; import android.annotation.IntDef; import android.annotation.Nullable; import android.annotation.UserIdInt; import android.app.ActivityManager; import android.app.AlertDialog; import android.app.AlertDialog.Builder; import android.app.Dialog; import android.car.userlib.CarUserManagerHelper; import android.car.user.CarUserManager; import android.car.user.UserCreationResult; import android.car.user.UserSwitchResult; import android.car.userlib.UserHelper; import android.content.BroadcastReceiver; import android.content.Context; import android.content.DialogInterface; Loading @@ -40,7 +44,9 @@ import android.graphics.Rect; import android.os.AsyncTask; import android.os.UserHandle; import android.os.UserManager; import android.sysprop.CarProperties; import android.util.AttributeSet; import android.util.Log; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; Loading @@ -54,6 +60,7 @@ import androidx.core.graphics.drawable.RoundedBitmapDrawableFactory; import androidx.recyclerview.widget.GridLayoutManager; import androidx.recyclerview.widget.RecyclerView; import com.android.internal.infra.AndroidFuture; import com.android.internal.util.UserIcons; import com.android.systemui.R; Loading @@ -61,6 +68,7 @@ import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.util.ArrayList; import java.util.List; import java.util.concurrent.TimeUnit; import java.util.stream.Collectors; /** Loading @@ -68,9 +76,12 @@ import java.util.stream.Collectors; * One of the uses of this is for the lock screen in auto. */ public class UserGridRecyclerView extends RecyclerView { private static final String TAG = UserGridRecyclerView.class.getSimpleName(); private static final int TIMEOUT_MS = CarProperties.user_hal_timeout().orElse(5_000) + 500; private UserSelectionListener mUserSelectionListener; private UserAdapter mAdapter; private CarUserManagerHelper mCarUserManagerHelper; private CarUserManager mCarUserManager; private UserManager mUserManager; private Context mContext; private UserIconProvider mUserIconProvider; Loading @@ -85,7 +96,6 @@ public class UserGridRecyclerView extends RecyclerView { public UserGridRecyclerView(Context context, AttributeSet attrs) { super(context, attrs); mContext = context; mCarUserManagerHelper = new CarUserManagerHelper(mContext); mUserManager = UserManager.get(mContext); mUserIconProvider = new UserIconProvider(); Loading Loading @@ -184,6 +194,11 @@ public class UserGridRecyclerView extends RecyclerView { mUserSelectionListener = userSelectionListener; } /** Sets a {@link CarUserManager}. */ public void setCarUserManager(CarUserManager carUserManager) { mCarUserManager = carUserManager; } private void onUsersUpdate() { mAdapter.clearUsers(); mAdapter.updateUsers(createUserRecords(getUsersForUserGrid())); Loading Loading @@ -273,7 +288,9 @@ public class UserGridRecyclerView extends RecyclerView { notifyUserSelected(userRecord); UserInfo guest = createNewOrFindExistingGuest(mContext); if (guest != null) { mCarUserManagerHelper.switchToUser(guest); if (!switchUser(guest.id)) { Log.e(TAG, "Failed to switch to guest user: " + guest.id); } } break; case UserRecord.ADD_USER: Loading @@ -289,7 +306,9 @@ public class UserGridRecyclerView extends RecyclerView { // If the user doesn't want to be a guest or add a user, switch to the user // selected notifyUserSelected(userRecord); mCarUserManagerHelper.switchToUser(userRecord.mInfo); if (!switchUser(userRecord.mInfo.id)) { Log.e(TAG, "Failed to switch users: " + userRecord.mInfo.id); } } }); Loading Loading @@ -430,8 +449,9 @@ public class UserGridRecyclerView extends RecyclerView { */ @Nullable public UserInfo createNewOrFindExistingGuest(Context context) { AndroidFuture<UserCreationResult> future = mCarUserManager.createGuest(mGuestName); // CreateGuest will return null if a guest already exists. UserInfo newGuest = mUserManager.createGuest(context, mGuestName); UserInfo newGuest = getUserInfo(future); if (newGuest != null) { new UserIconProvider().assignDefaultIcon( mUserManager, context.getResources(), newGuest); Loading @@ -444,7 +464,6 @@ public class UserGridRecyclerView extends RecyclerView { @Override public void onClick(DialogInterface dialog, int which) { if (which == BUTTON_POSITIVE) { notifyUserSelected(mAddUserRecord); new AddNewUserTask().execute(mNewUserName); } else if (which == BUTTON_NEGATIVE) { // Enable the add button only if cancel Loading @@ -462,11 +481,77 @@ public class UserGridRecyclerView extends RecyclerView { } } @Nullable private UserInfo getUserInfo(AndroidFuture<UserCreationResult> future) { UserCreationResult userCreationResult; try { userCreationResult = future.get(TIMEOUT_MS, TimeUnit.MILLISECONDS); } catch (Exception e) { Log.w(TAG, "Could not create user.", e); return null; } if (userCreationResult == null) { Log.w(TAG, "Timed out while creating user: " + TIMEOUT_MS + "ms"); return null; } if (!userCreationResult.isSuccess() || userCreationResult.getUser() == null) { Log.w(TAG, "Could not create user: " + userCreationResult); return null; } return userCreationResult.getUser(); } private boolean switchUser(@UserIdInt int userId) { AndroidFuture<UserSwitchResult> userSwitchResultFuture = mCarUserManager.switchUser(userId); UserSwitchResult userSwitchResult; try { userSwitchResult = userSwitchResultFuture.get(TIMEOUT_MS, TimeUnit.MILLISECONDS); } catch (Exception e) { Log.w(TAG, "Could not switch user.", e); return false; } if (userSwitchResult == null) { Log.w(TAG, "Timed out while switching user: " + TIMEOUT_MS + "ms"); return false; } if (!userSwitchResult.isSuccess()) { Log.w(TAG, "Could not switch user: " + userSwitchResult); return false; } return true; } // TODO(b/161539497): Replace AsyncTask with standard {@link java.util.concurrent} code. private class AddNewUserTask extends AsyncTask<String, Void, UserInfo> { @Override protected UserInfo doInBackground(String... userNames) { return mCarUserManagerHelper.createNewNonAdminUser(userNames[0]); AndroidFuture<UserCreationResult> future = mCarUserManager.createUser(userNames[0], /* flags= */ 0); try { UserInfo user = getUserInfo(future); if (user != null) { UserHelper.setDefaultNonAdminRestrictions(mContext, user, /* enable= */ true); UserHelper.assignDefaultIcon(mContext, user); mAddUserRecord = new UserRecord(user, UserRecord.ADD_USER); return user; } else { Log.e(TAG, "Failed to create user in the background"); return user; } } catch (Exception e) { if (e instanceof InterruptedException) { Thread.currentThread().interrupt(); } Log.e(TAG, "Error creating new user: ", e); } return null; } @Override Loading @@ -476,7 +561,11 @@ public class UserGridRecyclerView extends RecyclerView { @Override protected void onPostExecute(UserInfo user) { if (user != null) { mCarUserManagerHelper.switchToUser(user); notifyUserSelected(mAddUserRecord); mAddUserView.setEnabled(true); if (!switchUser(user.id)) { Log.e(TAG, "Failed to switch to new user: " + user.id); } } } } Loading Loading
packages/CarSystemUI/src/com/android/systemui/car/userswitcher/FullScreenUserSwitcherViewController.java +15 −21 Original line number Diff line number Diff line Loading @@ -18,6 +18,8 @@ package com.android.systemui.car.userswitcher; import android.animation.Animator; import android.animation.AnimatorListenerAdapter; import android.car.Car; import android.car.user.CarUserManager; import android.content.Context; import android.content.res.Resources; import android.view.View; Loading @@ -25,6 +27,7 @@ import android.view.View; import androidx.recyclerview.widget.GridLayoutManager; import com.android.systemui.R; import com.android.systemui.car.CarServiceProvider; import com.android.systemui.car.window.OverlayViewController; import com.android.systemui.car.window.OverlayViewGlobalStateController; import com.android.systemui.dagger.qualifiers.Main; Loading @@ -39,7 +42,9 @@ import javax.inject.Singleton; public class FullScreenUserSwitcherViewController extends OverlayViewController { private final Context mContext; private final Resources mResources; private final CarServiceProvider mCarServiceProvider; private final int mShortAnimationDuration; private CarUserManager mCarUserManager; private UserGridRecyclerView mUserGridView; private UserGridRecyclerView.UserSelectionListener mUserSelectionListener; Loading @@ -47,10 +52,16 @@ public class FullScreenUserSwitcherViewController extends OverlayViewController public FullScreenUserSwitcherViewController( Context context, @Main Resources resources, CarServiceProvider carServiceProvider, OverlayViewGlobalStateController overlayViewGlobalStateController) { super(R.id.fullscreen_user_switcher_stub, overlayViewGlobalStateController); mContext = context; mResources = resources; mCarServiceProvider = carServiceProvider; mCarServiceProvider.addListener(car -> { mCarUserManager = (CarUserManager) car.getCarManager(Car.CAR_USER_SERVICE); registerCarUserManagerIfPossible(); }); mShortAnimationDuration = mResources.getInteger(android.R.integer.config_shortAnimTime); } Loading @@ -63,6 +74,7 @@ public class FullScreenUserSwitcherViewController extends OverlayViewController mUserGridView.setLayoutManager(layoutManager); mUserGridView.buildAdapter(); mUserGridView.setUserSelectionListener(mUserSelectionListener); registerCarUserManagerIfPossible(); } @Override Loading Loading @@ -90,18 +102,6 @@ public class FullScreenUserSwitcherViewController extends OverlayViewController } /** * Invalidate underlying view. */ void invalidate() { if (getLayout() == null) { // layout hasn't been inflated. return; } getLayout().invalidate(); } /** * Set {@link UserGridRecyclerView.UserSelectionListener}. */ Loading @@ -110,15 +110,9 @@ public class FullScreenUserSwitcherViewController extends OverlayViewController mUserSelectionListener = userGridSelectionListener; } /** * Returns {@code true} when layout is visible. */ boolean isVisible() { if (getLayout() == null) { // layout hasn't been inflated. return false; private void registerCarUserManagerIfPossible() { if (mUserGridView != null && mCarUserManager != null) { mUserGridView.setCarUserManager(mCarUserManager); } return getLayout().getVisibility() == View.VISIBLE; } }
packages/CarSystemUI/src/com/android/systemui/car/userswitcher/UserGridRecyclerView.java +98 −9 Original line number Diff line number Diff line Loading @@ -24,11 +24,15 @@ import static android.view.WindowInsets.Type.statusBars; import android.annotation.IntDef; import android.annotation.Nullable; import android.annotation.UserIdInt; import android.app.ActivityManager; import android.app.AlertDialog; import android.app.AlertDialog.Builder; import android.app.Dialog; import android.car.userlib.CarUserManagerHelper; import android.car.user.CarUserManager; import android.car.user.UserCreationResult; import android.car.user.UserSwitchResult; import android.car.userlib.UserHelper; import android.content.BroadcastReceiver; import android.content.Context; import android.content.DialogInterface; Loading @@ -40,7 +44,9 @@ import android.graphics.Rect; import android.os.AsyncTask; import android.os.UserHandle; import android.os.UserManager; import android.sysprop.CarProperties; import android.util.AttributeSet; import android.util.Log; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; Loading @@ -54,6 +60,7 @@ import androidx.core.graphics.drawable.RoundedBitmapDrawableFactory; import androidx.recyclerview.widget.GridLayoutManager; import androidx.recyclerview.widget.RecyclerView; import com.android.internal.infra.AndroidFuture; import com.android.internal.util.UserIcons; import com.android.systemui.R; Loading @@ -61,6 +68,7 @@ import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.util.ArrayList; import java.util.List; import java.util.concurrent.TimeUnit; import java.util.stream.Collectors; /** Loading @@ -68,9 +76,12 @@ import java.util.stream.Collectors; * One of the uses of this is for the lock screen in auto. */ public class UserGridRecyclerView extends RecyclerView { private static final String TAG = UserGridRecyclerView.class.getSimpleName(); private static final int TIMEOUT_MS = CarProperties.user_hal_timeout().orElse(5_000) + 500; private UserSelectionListener mUserSelectionListener; private UserAdapter mAdapter; private CarUserManagerHelper mCarUserManagerHelper; private CarUserManager mCarUserManager; private UserManager mUserManager; private Context mContext; private UserIconProvider mUserIconProvider; Loading @@ -85,7 +96,6 @@ public class UserGridRecyclerView extends RecyclerView { public UserGridRecyclerView(Context context, AttributeSet attrs) { super(context, attrs); mContext = context; mCarUserManagerHelper = new CarUserManagerHelper(mContext); mUserManager = UserManager.get(mContext); mUserIconProvider = new UserIconProvider(); Loading Loading @@ -184,6 +194,11 @@ public class UserGridRecyclerView extends RecyclerView { mUserSelectionListener = userSelectionListener; } /** Sets a {@link CarUserManager}. */ public void setCarUserManager(CarUserManager carUserManager) { mCarUserManager = carUserManager; } private void onUsersUpdate() { mAdapter.clearUsers(); mAdapter.updateUsers(createUserRecords(getUsersForUserGrid())); Loading Loading @@ -273,7 +288,9 @@ public class UserGridRecyclerView extends RecyclerView { notifyUserSelected(userRecord); UserInfo guest = createNewOrFindExistingGuest(mContext); if (guest != null) { mCarUserManagerHelper.switchToUser(guest); if (!switchUser(guest.id)) { Log.e(TAG, "Failed to switch to guest user: " + guest.id); } } break; case UserRecord.ADD_USER: Loading @@ -289,7 +306,9 @@ public class UserGridRecyclerView extends RecyclerView { // If the user doesn't want to be a guest or add a user, switch to the user // selected notifyUserSelected(userRecord); mCarUserManagerHelper.switchToUser(userRecord.mInfo); if (!switchUser(userRecord.mInfo.id)) { Log.e(TAG, "Failed to switch users: " + userRecord.mInfo.id); } } }); Loading Loading @@ -430,8 +449,9 @@ public class UserGridRecyclerView extends RecyclerView { */ @Nullable public UserInfo createNewOrFindExistingGuest(Context context) { AndroidFuture<UserCreationResult> future = mCarUserManager.createGuest(mGuestName); // CreateGuest will return null if a guest already exists. UserInfo newGuest = mUserManager.createGuest(context, mGuestName); UserInfo newGuest = getUserInfo(future); if (newGuest != null) { new UserIconProvider().assignDefaultIcon( mUserManager, context.getResources(), newGuest); Loading @@ -444,7 +464,6 @@ public class UserGridRecyclerView extends RecyclerView { @Override public void onClick(DialogInterface dialog, int which) { if (which == BUTTON_POSITIVE) { notifyUserSelected(mAddUserRecord); new AddNewUserTask().execute(mNewUserName); } else if (which == BUTTON_NEGATIVE) { // Enable the add button only if cancel Loading @@ -462,11 +481,77 @@ public class UserGridRecyclerView extends RecyclerView { } } @Nullable private UserInfo getUserInfo(AndroidFuture<UserCreationResult> future) { UserCreationResult userCreationResult; try { userCreationResult = future.get(TIMEOUT_MS, TimeUnit.MILLISECONDS); } catch (Exception e) { Log.w(TAG, "Could not create user.", e); return null; } if (userCreationResult == null) { Log.w(TAG, "Timed out while creating user: " + TIMEOUT_MS + "ms"); return null; } if (!userCreationResult.isSuccess() || userCreationResult.getUser() == null) { Log.w(TAG, "Could not create user: " + userCreationResult); return null; } return userCreationResult.getUser(); } private boolean switchUser(@UserIdInt int userId) { AndroidFuture<UserSwitchResult> userSwitchResultFuture = mCarUserManager.switchUser(userId); UserSwitchResult userSwitchResult; try { userSwitchResult = userSwitchResultFuture.get(TIMEOUT_MS, TimeUnit.MILLISECONDS); } catch (Exception e) { Log.w(TAG, "Could not switch user.", e); return false; } if (userSwitchResult == null) { Log.w(TAG, "Timed out while switching user: " + TIMEOUT_MS + "ms"); return false; } if (!userSwitchResult.isSuccess()) { Log.w(TAG, "Could not switch user: " + userSwitchResult); return false; } return true; } // TODO(b/161539497): Replace AsyncTask with standard {@link java.util.concurrent} code. private class AddNewUserTask extends AsyncTask<String, Void, UserInfo> { @Override protected UserInfo doInBackground(String... userNames) { return mCarUserManagerHelper.createNewNonAdminUser(userNames[0]); AndroidFuture<UserCreationResult> future = mCarUserManager.createUser(userNames[0], /* flags= */ 0); try { UserInfo user = getUserInfo(future); if (user != null) { UserHelper.setDefaultNonAdminRestrictions(mContext, user, /* enable= */ true); UserHelper.assignDefaultIcon(mContext, user); mAddUserRecord = new UserRecord(user, UserRecord.ADD_USER); return user; } else { Log.e(TAG, "Failed to create user in the background"); return user; } } catch (Exception e) { if (e instanceof InterruptedException) { Thread.currentThread().interrupt(); } Log.e(TAG, "Error creating new user: ", e); } return null; } @Override Loading @@ -476,7 +561,11 @@ public class UserGridRecyclerView extends RecyclerView { @Override protected void onPostExecute(UserInfo user) { if (user != null) { mCarUserManagerHelper.switchToUser(user); notifyUserSelected(mAddUserRecord); mAddUserView.setEnabled(true); if (!switchUser(user.id)) { Log.e(TAG, "Failed to switch to new user: " + user.id); } } } } Loading