1 /* 2 * Copyright (C) 2013 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 17 package com.android.settingslib.users; 18 19 import android.app.Activity; 20 import android.app.Dialog; 21 import android.content.Context; 22 import android.content.Intent; 23 import android.graphics.Bitmap; 24 import android.graphics.drawable.Drawable; 25 import android.os.Bundle; 26 import android.os.UserHandle; 27 import android.os.UserManager; 28 import android.view.LayoutInflater; 29 import android.view.View; 30 import android.view.WindowManager; 31 import android.widget.EditText; 32 import android.widget.ImageView; 33 34 import androidx.annotation.Nullable; 35 import androidx.annotation.VisibleForTesting; 36 37 import com.android.internal.util.UserIcons; 38 import com.android.settingslib.R; 39 import com.android.settingslib.RestrictedLockUtils; 40 import com.android.settingslib.RestrictedLockUtilsInternal; 41 import com.android.settingslib.drawable.CircleFramedDrawable; 42 import com.android.settingslib.utils.CustomDialogHelper; 43 44 import java.io.File; 45 import java.util.function.BiConsumer; 46 47 /** 48 * This class encapsulates a Dialog for editing the user nickname and photo. 49 */ 50 public class EditUserInfoController { 51 52 private static final String KEY_AWAITING_RESULT = "awaiting_result"; 53 private static final String KEY_SAVED_PHOTO = "pending_photo"; 54 55 private Dialog mEditUserInfoDialog; 56 private Bitmap mSavedPhoto; 57 private Drawable mSavedDrawable; 58 private EditUserPhotoController mEditUserPhotoController; 59 private boolean mWaitingForActivityResult = false; 60 private final String mFileAuthority; 61 EditUserInfoController(String fileAuthority)62 public EditUserInfoController(String fileAuthority) { 63 mFileAuthority = fileAuthority; 64 } 65 clear()66 private void clear() { 67 if (mEditUserPhotoController != null) { 68 mEditUserPhotoController.removeNewUserPhotoBitmapFile(); 69 } 70 mEditUserInfoDialog = null; 71 mSavedPhoto = null; 72 mSavedDrawable = null; 73 } 74 75 /** 76 * This should be called when the container activity/fragment got re-initialized from a 77 * previously saved state. 78 */ onRestoreInstanceState(Bundle icicle)79 public void onRestoreInstanceState(Bundle icicle) { 80 String pendingPhoto = icicle.getString(KEY_SAVED_PHOTO); 81 if (pendingPhoto != null) { 82 mSavedPhoto = EditUserPhotoController.loadNewUserPhotoBitmap(new File(pendingPhoto)); 83 } 84 mWaitingForActivityResult = icicle.getBoolean(KEY_AWAITING_RESULT, false); 85 } 86 87 /** 88 * Should be called from the container activity/fragment when it's onSaveInstanceState is 89 * called. 90 */ onSaveInstanceState(Bundle outState)91 public void onSaveInstanceState(Bundle outState) { 92 if (mEditUserInfoDialog != null && mEditUserPhotoController != null) { 93 // Bitmap cannot be stored into bundle because it may exceed parcel limit 94 // Store it in a temporary file instead 95 File file = mEditUserPhotoController.saveNewUserPhotoBitmap(); 96 if (file != null) { 97 outState.putString(KEY_SAVED_PHOTO, file.getPath()); 98 } 99 } 100 outState.putBoolean(KEY_AWAITING_RESULT, mWaitingForActivityResult); 101 } 102 103 /** 104 * Should be called from the container activity/fragment when an activity has started for 105 * take/choose/crop photo actions. 106 */ startingActivityForResult()107 public void startingActivityForResult() { 108 mWaitingForActivityResult = true; 109 } 110 111 /** 112 * Should be called from the container activity/fragment after it receives a result from 113 * take/choose/crop photo activity. 114 */ onActivityResult(int requestCode, int resultCode, Intent data)115 public void onActivityResult(int requestCode, int resultCode, Intent data) { 116 mWaitingForActivityResult = false; 117 118 if (mEditUserPhotoController != null && mEditUserInfoDialog != null) { 119 mEditUserPhotoController.onActivityResult(requestCode, resultCode, data); 120 } 121 } 122 123 /** 124 * Creates a user edit dialog with option to change the user's name and photo. 125 * 126 * @param activityStarter - ActivityStarter is called with appropriate intents and request 127 * codes to take photo/choose photo/crop photo. 128 */ createDialog(Activity activity, ActivityStarter activityStarter, @Nullable Drawable oldUserIcon, String defaultUserName, BiConsumer<String, Drawable> successCallback, Runnable cancelCallback)129 public Dialog createDialog(Activity activity, ActivityStarter activityStarter, 130 @Nullable Drawable oldUserIcon, String defaultUserName, 131 BiConsumer<String, Drawable> successCallback, Runnable cancelCallback) { 132 LayoutInflater inflater = LayoutInflater.from(activity); 133 View content = inflater.inflate(R.layout.edit_user_info_dialog_content, null); 134 135 EditText userNameView = content.findViewById(R.id.user_name); 136 userNameView.setText(defaultUserName); 137 138 ImageView userPhotoView = content.findViewById(R.id.user_photo); 139 140 // if oldUserIcon param is null then we use a default gray user icon 141 Drawable defaultUserIcon = oldUserIcon != null ? oldUserIcon : UserIcons.getDefaultUserIcon( 142 activity.getResources(), UserHandle.USER_NULL, false); 143 // in case a new photo was selected and the activity got recreated we have to load the image 144 Drawable userIcon = getUserIcon(activity, defaultUserIcon); 145 userPhotoView.setImageDrawable(userIcon); 146 147 if (isChangePhotoRestrictedByBase(activity)) { 148 // some users can't change their photos so we need to remove the suggestive icon 149 content.findViewById(R.id.add_a_photo_icon).setVisibility(View.GONE); 150 } else { 151 RestrictedLockUtils.EnforcedAdmin adminRestriction = 152 getChangePhotoAdminRestriction(activity); 153 if (adminRestriction != null) { 154 userPhotoView.setOnClickListener(view -> 155 RestrictedLockUtils.sendShowAdminSupportDetailsIntent( 156 activity, adminRestriction)); 157 } else { 158 mEditUserPhotoController = createEditUserPhotoController(activity, activityStarter, 159 userPhotoView); 160 } 161 } 162 mEditUserInfoDialog = buildDialog(activity, content, userNameView, oldUserIcon, 163 defaultUserName, successCallback, cancelCallback); 164 165 // Make sure the IME is up. 166 mEditUserInfoDialog.getWindow() 167 .setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_VISIBLE); 168 169 return mEditUserInfoDialog; 170 } 171 getUserIcon(Activity activity, Drawable defaultUserIcon)172 private Drawable getUserIcon(Activity activity, Drawable defaultUserIcon) { 173 if (mSavedPhoto != null) { 174 mSavedDrawable = CircleFramedDrawable.getInstance(activity, mSavedPhoto); 175 return mSavedDrawable; 176 } 177 return defaultUserIcon; 178 } 179 buildDialog(Activity activity, View content, EditText userNameView, @Nullable Drawable oldUserIcon, String defaultUserName, BiConsumer<String, Drawable> successCallback, Runnable cancelCallback)180 private Dialog buildDialog(Activity activity, View content, EditText userNameView, 181 @Nullable Drawable oldUserIcon, String defaultUserName, 182 BiConsumer<String, Drawable> successCallback, Runnable cancelCallback) { 183 CustomDialogHelper dialogHelper = new CustomDialogHelper(activity); 184 dialogHelper 185 .setTitle(R.string.user_info_settings_title) 186 .addCustomView(content) 187 .setPositiveButton(R.string.okay, view -> { 188 Drawable newUserIcon = mEditUserPhotoController != null 189 ? mEditUserPhotoController.getNewUserPhotoDrawable() 190 : null; 191 Drawable userIcon = newUserIcon != null 192 ? newUserIcon 193 : oldUserIcon; 194 195 String newName = userNameView.getText().toString().trim(); 196 String userName = !newName.isEmpty() ? newName : defaultUserName; 197 198 clear(); 199 if (successCallback != null) { 200 successCallback.accept(userName, userIcon); 201 } 202 dialogHelper.getDialog().dismiss(); 203 }) 204 .setBackButton(R.string.cancel, view -> { 205 clear(); 206 if (cancelCallback != null) { 207 cancelCallback.run(); 208 } 209 dialogHelper.getDialog().dismiss(); 210 }); 211 dialogHelper.getDialog().setOnCancelListener(dialog -> { 212 clear(); 213 if (cancelCallback != null) { 214 cancelCallback.run(); 215 } 216 dialogHelper.getDialog().dismiss(); 217 }); 218 return dialogHelper.getDialog(); 219 } 220 221 @VisibleForTesting isChangePhotoRestrictedByBase(Context context)222 boolean isChangePhotoRestrictedByBase(Context context) { 223 return RestrictedLockUtilsInternal.hasBaseUserRestriction( 224 context, UserManager.DISALLOW_SET_USER_ICON, UserHandle.myUserId()); 225 } 226 227 @VisibleForTesting getChangePhotoAdminRestriction(Context context)228 RestrictedLockUtils.EnforcedAdmin getChangePhotoAdminRestriction(Context context) { 229 return RestrictedLockUtilsInternal.checkIfRestrictionEnforced( 230 context, UserManager.DISALLOW_SET_USER_ICON, UserHandle.myUserId()); 231 } 232 233 @VisibleForTesting createEditUserPhotoController(Activity activity, ActivityStarter activityStarter, ImageView userPhotoView)234 EditUserPhotoController createEditUserPhotoController(Activity activity, 235 ActivityStarter activityStarter, ImageView userPhotoView) { 236 return new EditUserPhotoController(activity, activityStarter, userPhotoView, 237 mSavedPhoto, mSavedDrawable, mFileAuthority, false); 238 } 239 } 240