1/* 2 * Copyright (C) 2011 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 17package com.android.contacts.detail; 18 19import android.app.Activity; 20import android.content.ActivityNotFoundException; 21import android.content.ContentValues; 22import android.content.Context; 23import android.content.Intent; 24import android.database.Cursor; 25import android.net.Uri; 26import android.provider.ContactsContract.CommonDataKinds.Photo; 27import android.provider.ContactsContract.DisplayPhoto; 28import android.provider.ContactsContract.RawContacts; 29import android.provider.MediaStore; 30import android.util.Log; 31import android.view.View; 32import android.view.View.OnClickListener; 33import android.widget.ListPopupWindow; 34import android.widget.PopupWindow.OnDismissListener; 35import android.widget.Toast; 36 37import com.android.contacts.R; 38import com.android.contacts.editor.PhotoActionPopup; 39import com.android.contacts.common.model.AccountTypeManager; 40import com.android.contacts.common.model.RawContactModifier; 41import com.android.contacts.common.model.RawContactDelta; 42import com.android.contacts.common.model.ValuesDelta; 43import com.android.contacts.common.model.account.AccountType; 44import com.android.contacts.common.model.RawContactDeltaList; 45import com.android.contacts.util.ContactPhotoUtils; 46import com.android.contacts.util.UiClosables; 47 48import java.io.FileNotFoundException; 49 50/** 51 * Handles displaying a photo selection popup for a given photo view and dealing with the results 52 * that come back. 53 */ 54public abstract class PhotoSelectionHandler implements OnClickListener { 55 56 private static final String TAG = PhotoSelectionHandler.class.getSimpleName(); 57 58 private static final int REQUEST_CODE_CAMERA_WITH_DATA = 1001; 59 private static final int REQUEST_CODE_PHOTO_PICKED_WITH_DATA = 1002; 60 private static final int REQUEST_CROP_PHOTO = 1003; 61 62 // Height and width (in pixels) to request for the photo - queried from the provider. 63 private static int mPhotoDim; 64 // Default photo dimension to use if unable to query the provider. 65 private static final int mDefaultPhotoDim = 720; 66 67 protected final Context mContext; 68 private final View mPhotoView; 69 private final int mPhotoMode; 70 private final int mPhotoPickSize; 71 private final Uri mCroppedPhotoUri; 72 private final Uri mTempPhotoUri; 73 private final RawContactDeltaList mState; 74 private final boolean mIsDirectoryContact; 75 private ListPopupWindow mPopup; 76 77 public PhotoSelectionHandler(Context context, View photoView, int photoMode, 78 boolean isDirectoryContact, RawContactDeltaList state) { 79 mContext = context; 80 mPhotoView = photoView; 81 mPhotoMode = photoMode; 82 mTempPhotoUri = ContactPhotoUtils.generateTempImageUri(context); 83 mCroppedPhotoUri = ContactPhotoUtils.generateTempCroppedImageUri(mContext); 84 mIsDirectoryContact = isDirectoryContact; 85 mState = state; 86 mPhotoPickSize = getPhotoPickSize(); 87 } 88 89 public void destroy() { 90 UiClosables.closeQuietly(mPopup); 91 } 92 93 public abstract PhotoActionListener getListener(); 94 95 @Override 96 public void onClick(View v) { 97 final PhotoActionListener listener = getListener(); 98 if (listener != null) { 99 if (getWritableEntityIndex() != -1) { 100 mPopup = PhotoActionPopup.createPopupMenu( 101 mContext, mPhotoView, listener, mPhotoMode); 102 mPopup.setOnDismissListener(new OnDismissListener() { 103 @Override 104 public void onDismiss() { 105 listener.onPhotoSelectionDismissed(); 106 } 107 }); 108 mPopup.show(); 109 } 110 } 111 } 112 113 /** 114 * Attempts to handle the given activity result. Returns whether this handler was able to 115 * process the result successfully. 116 * @param requestCode The request code. 117 * @param resultCode The result code. 118 * @param data The intent that was returned. 119 * @return Whether the handler was able to process the result. 120 */ 121 public boolean handlePhotoActivityResult(int requestCode, int resultCode, Intent data) { 122 final PhotoActionListener listener = getListener(); 123 if (resultCode == Activity.RESULT_OK) { 124 switch (requestCode) { 125 // Cropped photo was returned 126 case REQUEST_CROP_PHOTO: { 127 final Uri uri; 128 if (data != null && data.getData() != null) { 129 uri = data.getData(); 130 } else { 131 uri = mCroppedPhotoUri; 132 } 133 134 try { 135 // delete the original temporary photo if it exists 136 mContext.getContentResolver().delete(mTempPhotoUri, null, null); 137 listener.onPhotoSelected(uri); 138 return true; 139 } catch (FileNotFoundException e) { 140 return false; 141 } 142 } 143 144 // Photo was successfully taken or selected from gallery, now crop it. 145 case REQUEST_CODE_PHOTO_PICKED_WITH_DATA: 146 case REQUEST_CODE_CAMERA_WITH_DATA: 147 final Uri uri; 148 boolean isWritable = false; 149 if (data != null && data.getData() != null) { 150 uri = data.getData(); 151 } else { 152 uri = listener.getCurrentPhotoUri(); 153 isWritable = true; 154 } 155 final Uri toCrop; 156 if (isWritable) { 157 // Since this uri belongs to our file provider, we know that it is writable 158 // by us. This means that we don't have to save it into another temporary 159 // location just to be able to crop it. 160 toCrop = uri; 161 } else { 162 toCrop = mTempPhotoUri; 163 try { 164 ContactPhotoUtils.savePhotoFromUriToUri(mContext, uri, 165 toCrop, false); 166 } catch (SecurityException e) { 167 Log.d(TAG, "Did not have read-access to uri : " + uri); 168 return false; 169 } 170 } 171 172 doCropPhoto(toCrop, mCroppedPhotoUri); 173 return true; 174 } 175 } 176 return false; 177 } 178 179 /** 180 * Return the index of the first entity in the contact data that belongs to a contact-writable 181 * account, or -1 if no such entity exists. 182 */ 183 private int getWritableEntityIndex() { 184 // Directory entries are non-writable. 185 if (mIsDirectoryContact) return -1; 186 return mState.indexOfFirstWritableRawContact(mContext); 187 } 188 189 /** 190 * Return the raw-contact id of the first entity in the contact data that belongs to a 191 * contact-writable account, or -1 if no such entity exists. 192 */ 193 protected long getWritableEntityId() { 194 int index = getWritableEntityIndex(); 195 if (index == -1) return -1; 196 return mState.get(index).getValues().getId(); 197 } 198 199 /** 200 * Utility method to retrieve the entity delta for attaching the given bitmap to the contact. 201 * This will attach the photo to the first contact-writable account that provided data to the 202 * contact. It is the caller's responsibility to apply the delta. 203 * @return An entity delta list that can be applied to associate the bitmap with the contact, 204 * or null if the photo could not be parsed or none of the accounts associated with the 205 * contact are writable. 206 */ 207 public RawContactDeltaList getDeltaForAttachingPhotoToContact() { 208 // Find the first writable entity. 209 int writableEntityIndex = getWritableEntityIndex(); 210 if (writableEntityIndex != -1) { 211 // We are guaranteed to have contact data if we have a writable entity index. 212 final RawContactDelta delta = mState.get(writableEntityIndex); 213 214 // Need to find the right account so that EntityModifier knows which fields to add 215 final ContentValues entityValues = delta.getValues().getCompleteValues(); 216 final String type = entityValues.getAsString(RawContacts.ACCOUNT_TYPE); 217 final String dataSet = entityValues.getAsString(RawContacts.DATA_SET); 218 final AccountType accountType = AccountTypeManager.getInstance(mContext).getAccountType( 219 type, dataSet); 220 221 final ValuesDelta child = RawContactModifier.ensureKindExists( 222 delta, accountType, Photo.CONTENT_ITEM_TYPE); 223 child.setFromTemplate(false); 224 child.setSuperPrimary(true); 225 226 return mState; 227 } 228 return null; 229 } 230 231 /** Used by subclasses to delegate to their enclosing Activity or Fragment. */ 232 protected abstract void startPhotoActivity(Intent intent, int requestCode, Uri photoUri); 233 234 /** 235 * Sends a newly acquired photo to Gallery for cropping 236 */ 237 private void doCropPhoto(Uri inputUri, Uri outputUri) { 238 try { 239 // Launch gallery to crop the photo 240 final Intent intent = getCropImageIntent(inputUri, outputUri); 241 startPhotoActivity(intent, REQUEST_CROP_PHOTO, inputUri); 242 } catch (Exception e) { 243 Log.e(TAG, "Cannot crop image", e); 244 Toast.makeText(mContext, R.string.photoPickerNotFoundText, Toast.LENGTH_LONG).show(); 245 } 246 } 247 248 /** 249 * Should initiate an activity to take a photo using the camera. 250 * @param photoFile The file path that will be used to store the photo. This is generally 251 * what should be returned by 252 * {@link PhotoSelectionHandler.PhotoActionListener#getCurrentPhotoFile()}. 253 */ 254 private void startTakePhotoActivity(Uri photoUri) { 255 final Intent intent = getTakePhotoIntent(photoUri); 256 startPhotoActivity(intent, REQUEST_CODE_CAMERA_WITH_DATA, photoUri); 257 } 258 259 /** 260 * Should initiate an activity pick a photo from the gallery. 261 * @param photoFile The temporary file that the cropped image is written to before being 262 * stored by the content-provider. 263 * {@link PhotoSelectionHandler#handlePhotoActivityResult(int, int, Intent)}. 264 */ 265 private void startPickFromGalleryActivity(Uri photoUri) { 266 final Intent intent = getPhotoPickIntent(photoUri); 267 startPhotoActivity(intent, REQUEST_CODE_PHOTO_PICKED_WITH_DATA, photoUri); 268 } 269 270 private int getPhotoPickSize() { 271 if (mPhotoDim != 0) { 272 return mPhotoDim; 273 } 274 275 // Note that this URI is safe to call on the UI thread. 276 Cursor c = mContext.getContentResolver().query(DisplayPhoto.CONTENT_MAX_DIMENSIONS_URI, 277 new String[]{DisplayPhoto.DISPLAY_MAX_DIM}, null, null, null); 278 if (c != null) { 279 try { 280 if (c.moveToFirst()) { 281 mPhotoDim = c.getInt(0); 282 } 283 } finally { 284 c.close(); 285 } 286 } 287 return mPhotoDim != 0 ? mPhotoDim : mDefaultPhotoDim; 288 } 289 290 /** 291 * Constructs an intent for capturing a photo and storing it in a temporary output uri. 292 */ 293 private Intent getTakePhotoIntent(Uri outputUri) { 294 final Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE, null); 295 ContactPhotoUtils.addPhotoPickerExtras(intent, outputUri); 296 return intent; 297 } 298 299 /** 300 * Constructs an intent for picking a photo from Gallery, and returning the bitmap. 301 */ 302 private Intent getPhotoPickIntent(Uri outputUri) { 303 final Intent intent = new Intent(Intent.ACTION_GET_CONTENT, null); 304 intent.setType("image/*"); 305 ContactPhotoUtils.addPhotoPickerExtras(intent, outputUri); 306 return intent; 307 } 308 309 /** 310 * Constructs an intent for image cropping. 311 */ 312 private Intent getCropImageIntent(Uri inputUri, Uri outputUri) { 313 Intent intent = new Intent("com.android.camera.action.CROP"); 314 intent.setDataAndType(inputUri, "image/*"); 315 ContactPhotoUtils.addPhotoPickerExtras(intent, outputUri); 316 ContactPhotoUtils.addCropExtras(intent, mPhotoPickSize); 317 return intent; 318 } 319 320 public abstract class PhotoActionListener implements PhotoActionPopup.Listener { 321 @Override 322 public void onUseAsPrimaryChosen() { 323 // No default implementation. 324 } 325 326 @Override 327 public void onRemovePictureChosen() { 328 // No default implementation. 329 } 330 331 @Override 332 public void onTakePhotoChosen() { 333 try { 334 // Launch camera to take photo for selected contact 335 startTakePhotoActivity(mTempPhotoUri); 336 } catch (ActivityNotFoundException e) { 337 Toast.makeText( 338 mContext, R.string.photoPickerNotFoundText, Toast.LENGTH_LONG).show(); 339 } 340 } 341 342 @Override 343 public void onPickFromGalleryChosen() { 344 try { 345 // Launch picker to choose photo for selected contact 346 startPickFromGalleryActivity(mTempPhotoUri); 347 } catch (ActivityNotFoundException e) { 348 Toast.makeText( 349 mContext, R.string.photoPickerNotFoundText, Toast.LENGTH_LONG).show(); 350 } 351 } 352 353 /** 354 * Called when the user has completed selection of a photo. 355 * @throws FileNotFoundException 356 */ 357 public abstract void onPhotoSelected(Uri uri) throws FileNotFoundException; 358 359 /** 360 * Gets the current photo file that is being interacted with. It is the activity or 361 * fragment's responsibility to maintain this in saved state, since this handler instance 362 * will not survive rotation. 363 */ 364 public abstract Uri getCurrentPhotoUri(); 365 366 /** 367 * Called when the photo selection dialog is dismissed. 368 */ 369 public abstract void onPhotoSelectionDismissed(); 370 } 371} 372