| /* |
| * Copyright (C) 2014 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License |
| */ |
| |
| package com.android.systemui.statusbar.policy; |
| |
| import static android.os.UserManager.SWITCHABILITY_STATUS_OK; |
| |
| import static com.android.settingslib.RestrictedLockUtils.EnforcedAdmin; |
| import static com.android.systemui.DejankUtils.whitelistIpcs; |
| |
| import android.app.ActivityManager; |
| import android.app.Dialog; |
| import android.content.BroadcastReceiver; |
| import android.content.Context; |
| import android.content.DialogInterface; |
| import android.content.Intent; |
| import android.content.IntentFilter; |
| import android.content.pm.UserInfo; |
| import android.database.ContentObserver; |
| import android.graphics.Bitmap; |
| import android.graphics.PorterDuff.Mode; |
| import android.graphics.drawable.Drawable; |
| import android.os.AsyncTask; |
| import android.os.Handler; |
| import android.os.RemoteException; |
| import android.os.UserHandle; |
| import android.os.UserManager; |
| import android.provider.Settings; |
| import android.telephony.PhoneStateListener; |
| import android.telephony.TelephonyManager; |
| import android.util.Log; |
| import android.util.SparseArray; |
| import android.util.SparseBooleanArray; |
| import android.view.View; |
| import android.view.ViewGroup; |
| import android.widget.BaseAdapter; |
| |
| import com.android.internal.annotations.VisibleForTesting; |
| import com.android.internal.logging.nano.MetricsProto.MetricsEvent; |
| import com.android.internal.util.UserIcons; |
| import com.android.settingslib.RestrictedLockUtilsInternal; |
| import com.android.settingslib.Utils; |
| import com.android.systemui.Dumpable; |
| import com.android.systemui.GuestResumeSessionReceiver; |
| import com.android.systemui.Prefs; |
| import com.android.systemui.Prefs.Key; |
| import com.android.systemui.R; |
| import com.android.systemui.SystemUISecondaryUserService; |
| import com.android.systemui.broadcast.BroadcastDispatcher; |
| import com.android.systemui.dagger.qualifiers.Main; |
| import com.android.systemui.plugins.ActivityStarter; |
| import com.android.systemui.plugins.qs.DetailAdapter; |
| import com.android.systemui.qs.tiles.UserDetailView; |
| import com.android.systemui.statusbar.phone.SystemUIDialog; |
| |
| import java.io.FileDescriptor; |
| import java.io.PrintWriter; |
| import java.lang.ref.WeakReference; |
| import java.util.ArrayList; |
| import java.util.List; |
| |
| import javax.inject.Inject; |
| import javax.inject.Singleton; |
| |
| /** |
| * Keeps a list of all users on the device for user switching. |
| */ |
| @Singleton |
| public class UserSwitcherController implements Dumpable { |
| |
| private static final String TAG = "UserSwitcherController"; |
| private static final boolean DEBUG = false; |
| private static final String SIMPLE_USER_SWITCHER_GLOBAL_SETTING = |
| "lockscreenSimpleUserSwitcher"; |
| private static final int PAUSE_REFRESH_USERS_TIMEOUT_MS = 3000; |
| |
| private static final String PERMISSION_SELF = "com.android.systemui.permission.SELF"; |
| |
| protected final Context mContext; |
| protected final UserManager mUserManager; |
| private final ArrayList<WeakReference<BaseUserAdapter>> mAdapters = new ArrayList<>(); |
| private final GuestResumeSessionReceiver mGuestResumeSessionReceiver |
| = new GuestResumeSessionReceiver(); |
| private final KeyguardStateController mKeyguardStateController; |
| protected final Handler mHandler; |
| private final ActivityStarter mActivityStarter; |
| private final BroadcastDispatcher mBroadcastDispatcher; |
| |
| private ArrayList<UserRecord> mUsers = new ArrayList<>(); |
| private Dialog mExitGuestDialog; |
| private Dialog mAddUserDialog; |
| private int mLastNonGuestUser = UserHandle.USER_SYSTEM; |
| private boolean mResumeUserOnGuestLogout = true; |
| private boolean mSimpleUserSwitcher; |
| private boolean mAddUsersWhenLocked; |
| private boolean mPauseRefreshUsers; |
| private int mSecondaryUser = UserHandle.USER_NULL; |
| private Intent mSecondaryUserServiceIntent; |
| private SparseBooleanArray mForcePictureLoadForUserId = new SparseBooleanArray(2); |
| |
| @Inject |
| public UserSwitcherController(Context context, KeyguardStateController keyguardStateController, |
| @Main Handler handler, ActivityStarter activityStarter, |
| BroadcastDispatcher broadcastDispatcher) { |
| mContext = context; |
| mBroadcastDispatcher = broadcastDispatcher; |
| if (!UserManager.isGuestUserEphemeral()) { |
| mGuestResumeSessionReceiver.register(mBroadcastDispatcher); |
| } |
| mKeyguardStateController = keyguardStateController; |
| mHandler = handler; |
| mActivityStarter = activityStarter; |
| mUserManager = UserManager.get(context); |
| IntentFilter filter = new IntentFilter(); |
| filter.addAction(Intent.ACTION_USER_ADDED); |
| filter.addAction(Intent.ACTION_USER_REMOVED); |
| filter.addAction(Intent.ACTION_USER_INFO_CHANGED); |
| filter.addAction(Intent.ACTION_USER_SWITCHED); |
| filter.addAction(Intent.ACTION_USER_STOPPED); |
| filter.addAction(Intent.ACTION_USER_UNLOCKED); |
| mBroadcastDispatcher.registerReceiver( |
| mReceiver, filter, null /* handler */, UserHandle.SYSTEM); |
| |
| mSimpleUserSwitcher = shouldUseSimpleUserSwitcher(); |
| |
| mSecondaryUserServiceIntent = new Intent(context, SystemUISecondaryUserService.class); |
| |
| filter = new IntentFilter(); |
| mContext.registerReceiverAsUser(mReceiver, UserHandle.SYSTEM, filter, |
| PERMISSION_SELF, null /* scheduler */); |
| |
| mContext.getContentResolver().registerContentObserver( |
| Settings.Global.getUriFor(SIMPLE_USER_SWITCHER_GLOBAL_SETTING), true, |
| mSettingsObserver); |
| mContext.getContentResolver().registerContentObserver( |
| Settings.Global.getUriFor(Settings.Global.ADD_USERS_WHEN_LOCKED), true, |
| mSettingsObserver); |
| mContext.getContentResolver().registerContentObserver( |
| Settings.Global.getUriFor( |
| Settings.Global.ALLOW_USER_SWITCHING_WHEN_SYSTEM_USER_LOCKED), |
| true, mSettingsObserver); |
| // Fetch initial values. |
| mSettingsObserver.onChange(false); |
| |
| keyguardStateController.addCallback(mCallback); |
| listenForCallState(); |
| |
| refreshUsers(UserHandle.USER_NULL); |
| } |
| |
| /** |
| * Refreshes users from UserManager. |
| * |
| * The pictures are only loaded if they have not been loaded yet. |
| * |
| * @param forcePictureLoadForId forces the picture of the given user to be reloaded. |
| */ |
| @SuppressWarnings("unchecked") |
| private void refreshUsers(int forcePictureLoadForId) { |
| if (DEBUG) Log.d(TAG, "refreshUsers(forcePictureLoadForId=" + forcePictureLoadForId+")"); |
| if (forcePictureLoadForId != UserHandle.USER_NULL) { |
| mForcePictureLoadForUserId.put(forcePictureLoadForId, true); |
| } |
| |
| if (mPauseRefreshUsers) { |
| return; |
| } |
| |
| boolean forceAllUsers = mForcePictureLoadForUserId.get(UserHandle.USER_ALL); |
| SparseArray<Bitmap> bitmaps = new SparseArray<>(mUsers.size()); |
| final int N = mUsers.size(); |
| for (int i = 0; i < N; i++) { |
| UserRecord r = mUsers.get(i); |
| if (r == null || r.picture == null || r.info == null || forceAllUsers |
| || mForcePictureLoadForUserId.get(r.info.id)) { |
| continue; |
| } |
| bitmaps.put(r.info.id, r.picture); |
| } |
| mForcePictureLoadForUserId.clear(); |
| |
| final boolean addUsersWhenLocked = mAddUsersWhenLocked; |
| new AsyncTask<SparseArray<Bitmap>, Void, ArrayList<UserRecord>>() { |
| @SuppressWarnings("unchecked") |
| @Override |
| protected ArrayList<UserRecord> doInBackground(SparseArray<Bitmap>... params) { |
| final SparseArray<Bitmap> bitmaps = params[0]; |
| List<UserInfo> infos = mUserManager.getUsers(true); |
| if (infos == null) { |
| return null; |
| } |
| ArrayList<UserRecord> records = new ArrayList<>(infos.size()); |
| int currentId = ActivityManager.getCurrentUser(); |
| // Check user switchability of the foreground user since SystemUI is running in |
| // User 0 |
| boolean canSwitchUsers = mUserManager.getUserSwitchability( |
| UserHandle.of(ActivityManager.getCurrentUser())) == SWITCHABILITY_STATUS_OK; |
| UserInfo currentUserInfo = null; |
| UserRecord guestRecord = null; |
| |
| for (UserInfo info : infos) { |
| boolean isCurrent = currentId == info.id; |
| if (isCurrent) { |
| currentUserInfo = info; |
| } |
| boolean switchToEnabled = canSwitchUsers || isCurrent; |
| if (info.isEnabled()) { |
| if (info.isGuest()) { |
| // Tapping guest icon triggers remove and a user switch therefore |
| // the icon shouldn't be enabled even if the user is current |
| guestRecord = new UserRecord(info, null /* picture */, |
| true /* isGuest */, isCurrent, false /* isAddUser */, |
| false /* isRestricted */, canSwitchUsers); |
| } else if (info.supportsSwitchToByUser()) { |
| Bitmap picture = bitmaps.get(info.id); |
| if (picture == null) { |
| picture = mUserManager.getUserIcon(info.id); |
| |
| if (picture != null) { |
| int avatarSize = mContext.getResources() |
| .getDimensionPixelSize(R.dimen.max_avatar_size); |
| picture = Bitmap.createScaledBitmap( |
| picture, avatarSize, avatarSize, true); |
| } |
| } |
| int index = isCurrent ? 0 : records.size(); |
| records.add(index, new UserRecord(info, picture, false /* isGuest */, |
| isCurrent, false /* isAddUser */, false /* isRestricted */, |
| switchToEnabled)); |
| } |
| } |
| } |
| if (records.size() > 1 || guestRecord != null) { |
| Prefs.putBoolean(mContext, Key.SEEN_MULTI_USER, true); |
| } |
| |
| boolean systemCanCreateUsers = !mUserManager.hasBaseUserRestriction( |
| UserManager.DISALLOW_ADD_USER, UserHandle.SYSTEM); |
| boolean currentUserCanCreateUsers = currentUserInfo != null |
| && (currentUserInfo.isAdmin() |
| || currentUserInfo.id == UserHandle.USER_SYSTEM) |
| && systemCanCreateUsers; |
| boolean anyoneCanCreateUsers = systemCanCreateUsers && addUsersWhenLocked; |
| boolean canCreateGuest = (currentUserCanCreateUsers || anyoneCanCreateUsers) |
| && guestRecord == null; |
| boolean canCreateUser = (currentUserCanCreateUsers || anyoneCanCreateUsers) |
| && mUserManager.canAddMoreUsers(); |
| boolean createIsRestricted = !addUsersWhenLocked; |
| |
| if (guestRecord == null) { |
| if (canCreateGuest) { |
| guestRecord = new UserRecord(null /* info */, null /* picture */, |
| true /* isGuest */, false /* isCurrent */, |
| false /* isAddUser */, createIsRestricted, canSwitchUsers); |
| checkIfAddUserDisallowedByAdminOnly(guestRecord); |
| records.add(guestRecord); |
| } |
| } else { |
| int index = guestRecord.isCurrent ? 0 : records.size(); |
| records.add(index, guestRecord); |
| } |
| |
| if (canCreateUser) { |
| UserRecord addUserRecord = new UserRecord(null /* info */, null /* picture */, |
| false /* isGuest */, false /* isCurrent */, true /* isAddUser */, |
| createIsRestricted, canSwitchUsers); |
| checkIfAddUserDisallowedByAdminOnly(addUserRecord); |
| records.add(addUserRecord); |
| } |
| |
| return records; |
| } |
| |
| @Override |
| protected void onPostExecute(ArrayList<UserRecord> userRecords) { |
| if (userRecords != null) { |
| mUsers = userRecords; |
| notifyAdapters(); |
| } |
| } |
| }.execute((SparseArray) bitmaps); |
| } |
| |
| private void pauseRefreshUsers() { |
| if (!mPauseRefreshUsers) { |
| mHandler.postDelayed(mUnpauseRefreshUsers, PAUSE_REFRESH_USERS_TIMEOUT_MS); |
| mPauseRefreshUsers = true; |
| } |
| } |
| |
| private void notifyAdapters() { |
| for (int i = mAdapters.size() - 1; i >= 0; i--) { |
| BaseUserAdapter adapter = mAdapters.get(i).get(); |
| if (adapter != null) { |
| adapter.notifyDataSetChanged(); |
| } else { |
| mAdapters.remove(i); |
| } |
| } |
| } |
| |
| public boolean isSimpleUserSwitcher() { |
| return mSimpleUserSwitcher; |
| } |
| |
| public boolean useFullscreenUserSwitcher() { |
| // Use adb to override: |
| // adb shell settings put system enable_fullscreen_user_switcher 0 # Turn it off. |
| // adb shell settings put system enable_fullscreen_user_switcher 1 # Turn it on. |
| // Restart SystemUI or adb reboot. |
| final int DEFAULT = -1; |
| final int overrideUseFullscreenUserSwitcher = |
| whitelistIpcs(() -> Settings.System.getInt(mContext.getContentResolver(), |
| "enable_fullscreen_user_switcher", DEFAULT)); |
| if (overrideUseFullscreenUserSwitcher != DEFAULT) { |
| return overrideUseFullscreenUserSwitcher != 0; |
| } |
| // Otherwise default to the build setting. |
| return mContext.getResources().getBoolean(R.bool.config_enableFullscreenUserSwitcher); |
| } |
| |
| public void setResumeUserOnGuestLogout(boolean resume) { |
| mResumeUserOnGuestLogout = resume; |
| } |
| |
| public void logoutCurrentUser() { |
| int currentUser = ActivityManager.getCurrentUser(); |
| if (currentUser != UserHandle.USER_SYSTEM) { |
| pauseRefreshUsers(); |
| ActivityManager.logoutCurrentUser(); |
| } |
| } |
| |
| public void removeUserId(int userId) { |
| if (userId == UserHandle.USER_SYSTEM) { |
| Log.w(TAG, "User " + userId + " could not removed."); |
| return; |
| } |
| if (ActivityManager.getCurrentUser() == userId) { |
| switchToUserId(UserHandle.USER_SYSTEM); |
| } |
| if (mUserManager.removeUser(userId)) { |
| refreshUsers(UserHandle.USER_NULL); |
| } |
| } |
| |
| public void switchTo(UserRecord record) { |
| int id; |
| if (record.isGuest && record.info == null) { |
| // No guest user. Create one. |
| UserInfo guest = mUserManager.createGuest( |
| mContext, mContext.getString(R.string.guest_nickname)); |
| if (guest == null) { |
| // Couldn't create guest, most likely because there already exists one, we just |
| // haven't reloaded the user list yet. |
| return; |
| } |
| id = guest.id; |
| } else if (record.isAddUser) { |
| showAddUserDialog(); |
| return; |
| } else { |
| id = record.info.id; |
| } |
| |
| int currUserId = ActivityManager.getCurrentUser(); |
| if (currUserId == id) { |
| if (record.isGuest) { |
| showExitGuestDialog(id); |
| } |
| return; |
| } |
| |
| if (UserManager.isGuestUserEphemeral()) { |
| // If switching from guest, we want to bring up the guest exit dialog instead of switching |
| UserInfo currUserInfo = mUserManager.getUserInfo(currUserId); |
| if (currUserInfo != null && currUserInfo.isGuest()) { |
| showExitGuestDialog(currUserId, record.resolveId()); |
| return; |
| } |
| } |
| |
| switchToUserId(id); |
| } |
| |
| public void switchTo(int userId) { |
| final int count = mUsers.size(); |
| for (int i = 0; i < count; ++i) { |
| UserRecord record = mUsers.get(i); |
| if (record.info != null && record.info.id == userId) { |
| switchTo(record); |
| return; |
| } |
| } |
| |
| Log.e(TAG, "Couldn't switch to user, id=" + userId); |
| } |
| |
| public int getSwitchableUserCount() { |
| int count = 0; |
| final int N = mUsers.size(); |
| for (int i = 0; i < N; ++i) { |
| UserRecord record = mUsers.get(i); |
| if (record.info != null && record.info.supportsSwitchToByUser()) { |
| count++; |
| } |
| } |
| return count; |
| } |
| |
| protected void switchToUserId(int id) { |
| try { |
| pauseRefreshUsers(); |
| ActivityManager.getService().switchUser(id); |
| } catch (RemoteException e) { |
| Log.e(TAG, "Couldn't switch user.", e); |
| } |
| } |
| |
| private void showExitGuestDialog(int id) { |
| int newId = UserHandle.USER_SYSTEM; |
| if (mResumeUserOnGuestLogout && mLastNonGuestUser != UserHandle.USER_SYSTEM) { |
| UserInfo info = mUserManager.getUserInfo(mLastNonGuestUser); |
| if (info != null && info.isEnabled() && info.supportsSwitchToByUser()) { |
| newId = info.id; |
| } |
| } |
| showExitGuestDialog(id, newId); |
| } |
| |
| protected void showExitGuestDialog(int id, int targetId) { |
| if (mExitGuestDialog != null && mExitGuestDialog.isShowing()) { |
| mExitGuestDialog.cancel(); |
| } |
| mExitGuestDialog = new ExitGuestDialog(mContext, id, targetId); |
| mExitGuestDialog.show(); |
| } |
| |
| public void showAddUserDialog() { |
| if (mAddUserDialog != null && mAddUserDialog.isShowing()) { |
| mAddUserDialog.cancel(); |
| } |
| mAddUserDialog = new AddUserDialog(mContext); |
| mAddUserDialog.show(); |
| } |
| |
| protected void exitGuest(int id, int targetId) { |
| switchToUserId(targetId); |
| mUserManager.removeUser(id); |
| } |
| |
| private void listenForCallState() { |
| final TelephonyManager tele = |
| (TelephonyManager) mContext.getSystemService(Context.TELEPHONY_SERVICE); |
| if (tele != null) { |
| tele.listen(mPhoneStateListener, PhoneStateListener.LISTEN_CALL_STATE); |
| } |
| } |
| |
| private final PhoneStateListener mPhoneStateListener = new PhoneStateListener() { |
| private int mCallState; |
| |
| @Override |
| public void onCallStateChanged(int state, String incomingNumber) { |
| if (mCallState == state) return; |
| if (DEBUG) Log.v(TAG, "Call state changed: " + state); |
| mCallState = state; |
| refreshUsers(UserHandle.USER_NULL); |
| } |
| }; |
| |
| private BroadcastReceiver mReceiver = new BroadcastReceiver() { |
| @Override |
| public void onReceive(Context context, Intent intent) { |
| if (DEBUG) { |
| Log.v(TAG, "Broadcast: a=" + intent.getAction() |
| + " user=" + intent.getIntExtra(Intent.EXTRA_USER_HANDLE, -1)); |
| } |
| |
| boolean unpauseRefreshUsers = false; |
| int forcePictureLoadForId = UserHandle.USER_NULL; |
| |
| if (Intent.ACTION_USER_SWITCHED.equals(intent.getAction())) { |
| if (mExitGuestDialog != null && mExitGuestDialog.isShowing()) { |
| mExitGuestDialog.cancel(); |
| mExitGuestDialog = null; |
| } |
| |
| final int currentId = intent.getIntExtra(Intent.EXTRA_USER_HANDLE, -1); |
| final UserInfo userInfo = mUserManager.getUserInfo(currentId); |
| final int N = mUsers.size(); |
| for (int i = 0; i < N; i++) { |
| UserRecord record = mUsers.get(i); |
| if (record.info == null) continue; |
| boolean shouldBeCurrent = record.info.id == currentId; |
| if (record.isCurrent != shouldBeCurrent) { |
| mUsers.set(i, record.copyWithIsCurrent(shouldBeCurrent)); |
| } |
| if (shouldBeCurrent && !record.isGuest) { |
| mLastNonGuestUser = record.info.id; |
| } |
| if ((userInfo == null || !userInfo.isAdmin()) && record.isRestricted) { |
| // Immediately remove restricted records in case the AsyncTask is too slow. |
| mUsers.remove(i); |
| i--; |
| } |
| } |
| notifyAdapters(); |
| |
| // Disconnect from the old secondary user's service |
| if (mSecondaryUser != UserHandle.USER_NULL) { |
| context.stopServiceAsUser(mSecondaryUserServiceIntent, |
| UserHandle.of(mSecondaryUser)); |
| mSecondaryUser = UserHandle.USER_NULL; |
| } |
| // Connect to the new secondary user's service (purely to ensure that a persistent |
| // SystemUI application is created for that user) |
| if (userInfo != null && userInfo.id != UserHandle.USER_SYSTEM) { |
| context.startServiceAsUser(mSecondaryUserServiceIntent, |
| UserHandle.of(userInfo.id)); |
| mSecondaryUser = userInfo.id; |
| } |
| unpauseRefreshUsers = true; |
| } else if (Intent.ACTION_USER_INFO_CHANGED.equals(intent.getAction())) { |
| forcePictureLoadForId = intent.getIntExtra(Intent.EXTRA_USER_HANDLE, |
| UserHandle.USER_NULL); |
| } else if (Intent.ACTION_USER_UNLOCKED.equals(intent.getAction())) { |
| // Unlocking the system user may require a refresh |
| int userId = intent.getIntExtra(Intent.EXTRA_USER_HANDLE, UserHandle.USER_NULL); |
| if (userId != UserHandle.USER_SYSTEM) { |
| return; |
| } |
| } |
| refreshUsers(forcePictureLoadForId); |
| if (unpauseRefreshUsers) { |
| mUnpauseRefreshUsers.run(); |
| } |
| } |
| }; |
| |
| private final Runnable mUnpauseRefreshUsers = new Runnable() { |
| @Override |
| public void run() { |
| mHandler.removeCallbacks(this); |
| mPauseRefreshUsers = false; |
| refreshUsers(UserHandle.USER_NULL); |
| } |
| }; |
| |
| private final ContentObserver mSettingsObserver = new ContentObserver(new Handler()) { |
| public void onChange(boolean selfChange) { |
| mSimpleUserSwitcher = shouldUseSimpleUserSwitcher(); |
| mAddUsersWhenLocked = Settings.Global.getInt(mContext.getContentResolver(), |
| Settings.Global.ADD_USERS_WHEN_LOCKED, 0) != 0; |
| refreshUsers(UserHandle.USER_NULL); |
| }; |
| }; |
| |
| @Override |
| public void dump(FileDescriptor fd, PrintWriter pw, String[] args) { |
| pw.println("UserSwitcherController state:"); |
| pw.println(" mLastNonGuestUser=" + mLastNonGuestUser); |
| pw.print(" mUsers.size="); pw.println(mUsers.size()); |
| for (int i = 0; i < mUsers.size(); i++) { |
| final UserRecord u = mUsers.get(i); |
| pw.print(" "); pw.println(u.toString()); |
| } |
| pw.println("mSimpleUserSwitcher=" + mSimpleUserSwitcher); |
| } |
| |
| public String getCurrentUserName(Context context) { |
| if (mUsers.isEmpty()) return null; |
| UserRecord item = mUsers.get(0); |
| if (item == null || item.info == null) return null; |
| if (item.isGuest) return context.getString(R.string.guest_nickname); |
| return item.info.name; |
| } |
| |
| public void onDensityOrFontScaleChanged() { |
| refreshUsers(UserHandle.USER_ALL); |
| } |
| |
| @VisibleForTesting |
| public void addAdapter(WeakReference<BaseUserAdapter> adapter) { |
| mAdapters.add(adapter); |
| } |
| |
| @VisibleForTesting |
| public ArrayList<UserRecord> getUsers() { |
| return mUsers; |
| } |
| |
| public static abstract class BaseUserAdapter extends BaseAdapter { |
| |
| final UserSwitcherController mController; |
| private final KeyguardStateController mKeyguardStateController; |
| |
| protected BaseUserAdapter(UserSwitcherController controller) { |
| mController = controller; |
| mKeyguardStateController = controller.mKeyguardStateController; |
| controller.addAdapter(new WeakReference<>(this)); |
| } |
| |
| public int getUserCount() { |
| boolean secureKeyguardShowing = mKeyguardStateController.isShowing() |
| && mKeyguardStateController.isMethodSecure() |
| && !mKeyguardStateController.canDismissLockScreen(); |
| if (!secureKeyguardShowing) { |
| return mController.getUsers().size(); |
| } |
| // The lock screen is secure and showing. Filter out restricted records. |
| final int N = mController.getUsers().size(); |
| int count = 0; |
| for (int i = 0; i < N; i++) { |
| if (mController.getUsers().get(i).isGuest) continue; |
| if (mController.getUsers().get(i).isRestricted) { |
| break; |
| } else { |
| count++; |
| } |
| } |
| return count; |
| } |
| |
| @Override |
| public int getCount() { |
| boolean secureKeyguardShowing = mKeyguardStateController.isShowing() |
| && mKeyguardStateController.isMethodSecure() |
| && !mKeyguardStateController.canDismissLockScreen(); |
| if (!secureKeyguardShowing) { |
| return mController.getUsers().size(); |
| } |
| // The lock screen is secure and showing. Filter out restricted records. |
| final int N = mController.getUsers().size(); |
| int count = 0; |
| for (int i = 0; i < N; i++) { |
| if (mController.getUsers().get(i).isRestricted) { |
| break; |
| } else { |
| count++; |
| } |
| } |
| return count; |
| } |
| |
| @Override |
| public UserRecord getItem(int position) { |
| return mController.getUsers().get(position); |
| } |
| |
| @Override |
| public long getItemId(int position) { |
| return position; |
| } |
| |
| public void switchTo(UserRecord record) { |
| mController.switchTo(record); |
| } |
| |
| public String getName(Context context, UserRecord item) { |
| if (item.isGuest) { |
| if (item.isCurrent) { |
| return context.getString(R.string.guest_exit_guest); |
| } else { |
| return context.getString( |
| item.info == null ? R.string.guest_new_guest : R.string.guest_nickname); |
| } |
| } else if (item.isAddUser) { |
| return context.getString(R.string.user_add_user); |
| } else { |
| return item.info.name; |
| } |
| } |
| |
| public Drawable getDrawable(Context context, UserRecord item) { |
| if (item.isAddUser) { |
| return context.getDrawable(R.drawable.ic_add_circle_qs); |
| } |
| Drawable icon = UserIcons.getDefaultUserIcon( |
| context.getResources(), item.resolveId(), /* light= */ false); |
| if (item.isGuest) { |
| icon.setColorFilter(Utils.getColorAttrDefaultColor(context, |
| android.R.attr.colorForeground), |
| Mode.SRC_IN); |
| } |
| return icon; |
| } |
| |
| public void refresh() { |
| mController.refreshUsers(UserHandle.USER_NULL); |
| } |
| } |
| |
| private void checkIfAddUserDisallowedByAdminOnly(UserRecord record) { |
| EnforcedAdmin admin = RestrictedLockUtilsInternal.checkIfRestrictionEnforced(mContext, |
| UserManager.DISALLOW_ADD_USER, ActivityManager.getCurrentUser()); |
| if (admin != null && !RestrictedLockUtilsInternal.hasBaseUserRestriction(mContext, |
| UserManager.DISALLOW_ADD_USER, ActivityManager.getCurrentUser())) { |
| record.isDisabledByAdmin = true; |
| record.enforcedAdmin = admin; |
| } else { |
| record.isDisabledByAdmin = false; |
| record.enforcedAdmin = null; |
| } |
| } |
| |
| private boolean shouldUseSimpleUserSwitcher() { |
| int defaultSimpleUserSwitcher = mContext.getResources().getBoolean( |
| com.android.internal.R.bool.config_expandLockScreenUserSwitcher) ? 1 : 0; |
| return Settings.Global.getInt(mContext.getContentResolver(), |
| SIMPLE_USER_SWITCHER_GLOBAL_SETTING, defaultSimpleUserSwitcher) != 0; |
| } |
| |
| public void startActivity(Intent intent) { |
| mActivityStarter.startActivity(intent, true); |
| } |
| |
| public static final class UserRecord { |
| public final UserInfo info; |
| public final Bitmap picture; |
| public final boolean isGuest; |
| public final boolean isCurrent; |
| public final boolean isAddUser; |
| /** If true, the record is only visible to the owner and only when unlocked. */ |
| public final boolean isRestricted; |
| public boolean isDisabledByAdmin; |
| public EnforcedAdmin enforcedAdmin; |
| public boolean isSwitchToEnabled; |
| |
| public UserRecord(UserInfo info, Bitmap picture, boolean isGuest, boolean isCurrent, |
| boolean isAddUser, boolean isRestricted, boolean isSwitchToEnabled) { |
| this.info = info; |
| this.picture = picture; |
| this.isGuest = isGuest; |
| this.isCurrent = isCurrent; |
| this.isAddUser = isAddUser; |
| this.isRestricted = isRestricted; |
| this.isSwitchToEnabled = isSwitchToEnabled; |
| } |
| |
| public UserRecord copyWithIsCurrent(boolean _isCurrent) { |
| return new UserRecord(info, picture, isGuest, _isCurrent, isAddUser, isRestricted, |
| isSwitchToEnabled); |
| } |
| |
| public int resolveId() { |
| if (isGuest || info == null) { |
| return UserHandle.USER_NULL; |
| } |
| return info.id; |
| } |
| |
| public String toString() { |
| StringBuilder sb = new StringBuilder(); |
| sb.append("UserRecord("); |
| if (info != null) { |
| sb.append("name=\"").append(info.name).append("\" id=").append(info.id); |
| } else { |
| if (isGuest) { |
| sb.append("<add guest placeholder>"); |
| } else if (isAddUser) { |
| sb.append("<add user placeholder>"); |
| } |
| } |
| if (isGuest) sb.append(" <isGuest>"); |
| if (isAddUser) sb.append(" <isAddUser>"); |
| if (isCurrent) sb.append(" <isCurrent>"); |
| if (picture != null) sb.append(" <hasPicture>"); |
| if (isRestricted) sb.append(" <isRestricted>"); |
| if (isDisabledByAdmin) { |
| sb.append(" <isDisabledByAdmin>"); |
| sb.append(" enforcedAdmin=").append(enforcedAdmin); |
| } |
| if (isSwitchToEnabled) { |
| sb.append(" <isSwitchToEnabled>"); |
| } |
| sb.append(')'); |
| return sb.toString(); |
| } |
| } |
| |
| public final DetailAdapter userDetailAdapter = new DetailAdapter() { |
| private final Intent USER_SETTINGS_INTENT = new Intent(Settings.ACTION_USER_SETTINGS); |
| |
| @Override |
| public CharSequence getTitle() { |
| return mContext.getString(R.string.quick_settings_user_title); |
| } |
| |
| @Override |
| public View createDetailView(Context context, View convertView, ViewGroup parent) { |
| UserDetailView v; |
| if (!(convertView instanceof UserDetailView)) { |
| v = UserDetailView.inflate(context, parent, false); |
| v.createAndSetAdapter(UserSwitcherController.this); |
| } else { |
| v = (UserDetailView) convertView; |
| } |
| v.refreshAdapter(); |
| return v; |
| } |
| |
| @Override |
| public Intent getSettingsIntent() { |
| return USER_SETTINGS_INTENT; |
| } |
| |
| @Override |
| public Boolean getToggleState() { |
| return null; |
| } |
| |
| @Override |
| public void setToggleState(boolean state) { |
| } |
| |
| @Override |
| public int getMetricsCategory() { |
| return MetricsEvent.QS_USERDETAIL; |
| } |
| }; |
| |
| private final KeyguardStateController.Callback mCallback = |
| new KeyguardStateController.Callback() { |
| @Override |
| public void onKeyguardShowingChanged() { |
| |
| // When Keyguard is going away, we don't need to update our items immediately |
| // which |
| // helps making the transition faster. |
| if (!mKeyguardStateController.isShowing()) { |
| mHandler.post(UserSwitcherController.this::notifyAdapters); |
| } else { |
| notifyAdapters(); |
| } |
| } |
| }; |
| |
| private final class ExitGuestDialog extends SystemUIDialog implements |
| DialogInterface.OnClickListener { |
| |
| private final int mGuestId; |
| private final int mTargetId; |
| |
| public ExitGuestDialog(Context context, int guestId, int targetId) { |
| super(context); |
| setTitle(R.string.guest_exit_guest_dialog_title); |
| setMessage(context.getString(R.string.guest_exit_guest_dialog_message)); |
| setButton(DialogInterface.BUTTON_NEGATIVE, |
| context.getString(android.R.string.cancel), this); |
| setButton(DialogInterface.BUTTON_POSITIVE, |
| context.getString(R.string.guest_exit_guest_dialog_remove), this); |
| SystemUIDialog.setWindowOnTop(this); |
| setCanceledOnTouchOutside(false); |
| mGuestId = guestId; |
| mTargetId = targetId; |
| } |
| |
| @Override |
| public void onClick(DialogInterface dialog, int which) { |
| if (which == BUTTON_NEGATIVE) { |
| cancel(); |
| } else { |
| dismiss(); |
| exitGuest(mGuestId, mTargetId); |
| } |
| } |
| } |
| |
| private final class AddUserDialog extends SystemUIDialog implements |
| DialogInterface.OnClickListener { |
| |
| public AddUserDialog(Context context) { |
| super(context); |
| setTitle(R.string.user_add_user_title); |
| setMessage(context.getString(R.string.user_add_user_message_short)); |
| setButton(DialogInterface.BUTTON_NEGATIVE, |
| context.getString(android.R.string.cancel), this); |
| setButton(DialogInterface.BUTTON_POSITIVE, |
| context.getString(android.R.string.ok), this); |
| SystemUIDialog.setWindowOnTop(this); |
| } |
| |
| @Override |
| public void onClick(DialogInterface dialog, int which) { |
| if (which == BUTTON_NEGATIVE) { |
| cancel(); |
| } else { |
| dismiss(); |
| if (ActivityManager.isUserAMonkey()) { |
| return; |
| } |
| UserInfo user = mUserManager.createUser( |
| mContext.getString(R.string.user_new_user_name), 0 /* flags */); |
| if (user == null) { |
| // Couldn't create user, most likely because there are too many, but we haven't |
| // been able to reload the list yet. |
| return; |
| } |
| int id = user.id; |
| Bitmap icon = UserIcons.convertToBitmap(UserIcons.getDefaultUserIcon( |
| mContext.getResources(), id, /* light= */ false)); |
| mUserManager.setUserIcon(id, icon); |
| switchToUserId(id); |
| } |
| } |
| } |
| } |