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