Home | History | Annotate | Download | only in activities
      1 /*
      2  * Copyright (C) 2006 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.contacts.activities;
     18 
     19 import android.app.Activity;
     20 import android.content.ActivityNotFoundException;
     21 import android.content.ContentResolver;
     22 import android.content.ContentValues;
     23 import android.content.Intent;
     24 import android.content.Loader;
     25 import android.content.Loader.OnLoadCompleteListener;
     26 import android.content.pm.PackageManager;
     27 import android.content.pm.ResolveInfo;
     28 import android.database.Cursor;
     29 import android.graphics.Bitmap;
     30 import android.net.Uri;
     31 import android.os.Bundle;
     32 import android.provider.ContactsContract.CommonDataKinds.Photo;
     33 import android.provider.ContactsContract.Contacts;
     34 import android.provider.ContactsContract.DisplayPhoto;
     35 import android.provider.ContactsContract.Intents;
     36 import android.provider.ContactsContract.RawContacts;
     37 import android.util.Log;
     38 import android.widget.Toast;
     39 
     40 import com.android.contacts.ContactSaveService;
     41 import com.android.contacts.ContactsActivity;
     42 import com.android.contacts.R;
     43 import com.android.contacts.common.activity.RequestPermissionsActivity;
     44 import com.android.contacts.common.model.Contact;
     45 import com.android.contacts.common.model.ContactLoader;
     46 import com.android.contacts.common.model.RawContactDelta;
     47 import com.android.contacts.common.model.RawContactDeltaList;
     48 import com.android.contacts.common.model.RawContactModifier;
     49 import com.android.contacts.common.ContactsUtils;
     50 import com.android.contacts.common.model.account.AccountType;
     51 import com.android.contacts.common.model.ValuesDelta;
     52 import com.android.contacts.common.model.account.AccountWithDataSet;
     53 import com.android.contacts.editor.ContactEditorUtils;
     54 import com.android.contacts.util.ContactPhotoUtils;
     55 
     56 import java.io.FileNotFoundException;
     57 import java.util.List;
     58 
     59 /**
     60  * Provides an external interface for other applications to attach images
     61  * to contacts. It will first present a contact picker and then run the
     62  * image that is handed to it through the cropper to make the image the proper
     63  * size and give the user a chance to use the face detector.
     64  */
     65 public class AttachPhotoActivity extends ContactsActivity {
     66     private static final String TAG = AttachPhotoActivity.class.getSimpleName();
     67 
     68     private static final int REQUEST_PICK_CONTACT = 1;
     69     private static final int REQUEST_CROP_PHOTO = 2;
     70     private static final int REQUEST_PICK_DEFAULT_ACCOUNT_FOR_NEW_CONTACT = 3;
     71 
     72     private static final String KEY_CONTACT_URI = "contact_uri";
     73     private static final String KEY_TEMP_PHOTO_URI = "temp_photo_uri";
     74     private static final String KEY_CROPPED_PHOTO_URI = "cropped_photo_uri";
     75 
     76     private Uri mTempPhotoUri;
     77     private Uri mCroppedPhotoUri;
     78 
     79     private ContentResolver mContentResolver;
     80 
     81     // Height and width (in pixels) to request for the photo - queried from the provider.
     82     private static int mPhotoDim;
     83     // Default photo dimension to use if unable to query the provider.
     84     private static final int mDefaultPhotoDim = 720;
     85 
     86     private Uri mContactUri;
     87 
     88     @Override
     89     public void onCreate(Bundle icicle) {
     90         super.onCreate(icicle);
     91 
     92         if (RequestPermissionsActivity.startPermissionActivity(this)) {
     93             return;
     94         }
     95 
     96         if (icicle != null) {
     97             final String uri = icicle.getString(KEY_CONTACT_URI);
     98             mContactUri = (uri == null) ? null : Uri.parse(uri);
     99             mTempPhotoUri = Uri.parse(icicle.getString(KEY_TEMP_PHOTO_URI));
    100             mCroppedPhotoUri = Uri.parse(icicle.getString(KEY_CROPPED_PHOTO_URI));
    101         } else {
    102             mTempPhotoUri = ContactPhotoUtils.generateTempImageUri(this);
    103             mCroppedPhotoUri = ContactPhotoUtils.generateTempCroppedImageUri(this);
    104             Intent intent = new Intent(Intent.ACTION_PICK);
    105             intent.setType(Contacts.CONTENT_TYPE);
    106             intent.setPackage(getPackageName());
    107             startActivityForResult(intent, REQUEST_PICK_CONTACT);
    108         }
    109 
    110         mContentResolver = getContentResolver();
    111 
    112         // Load the photo dimension to request. mPhotoDim is a static class
    113         // member varible so only need to load this if this is the first time
    114         // through.
    115         if (mPhotoDim == 0) {
    116             Cursor c = mContentResolver.query(DisplayPhoto.CONTENT_MAX_DIMENSIONS_URI,
    117                     new String[]{DisplayPhoto.DISPLAY_MAX_DIM}, null, null, null);
    118             if (c != null) {
    119                 try {
    120                     if (c.moveToFirst()) {
    121                         mPhotoDim = c.getInt(0);
    122                     }
    123                 } finally {
    124                     c.close();
    125                 }
    126             }
    127         }
    128     }
    129 
    130     @Override
    131     protected void onSaveInstanceState(Bundle outState) {
    132         super.onSaveInstanceState(outState);
    133         if (mContactUri != null) {
    134             outState.putString(KEY_CONTACT_URI, mContactUri.toString());
    135         }
    136         if (mTempPhotoUri != null) {
    137             outState.putString(KEY_TEMP_PHOTO_URI, mTempPhotoUri.toString());
    138         }
    139         if (mCroppedPhotoUri != null) {
    140             outState.putString(KEY_CROPPED_PHOTO_URI, mCroppedPhotoUri.toString());
    141         }
    142     }
    143 
    144     @Override
    145     protected void onActivityResult(int requestCode, int resultCode, Intent result) {
    146         if (requestCode == REQUEST_PICK_DEFAULT_ACCOUNT_FOR_NEW_CONTACT) {
    147             // Bail if the account selector was not successful.
    148             if (resultCode != Activity.RESULT_OK) {
    149                 Log.w(TAG, "account selector was not successful");
    150                 finish();
    151                 return;
    152             }
    153             // If there's an account specified, use it.
    154             if (result != null) {
    155                 AccountWithDataSet account = result.getParcelableExtra(
    156                         Intents.Insert.EXTRA_ACCOUNT);
    157                 if (account != null) {
    158                     createNewRawContact(account);
    159                     return;
    160                 }
    161             }
    162             // If there isn't an account specified, then the user opted to keep the contact local.
    163             createNewRawContact(null);
    164         } else if (requestCode == REQUEST_PICK_CONTACT) {
    165             if (resultCode != RESULT_OK) {
    166                 finish();
    167                 return;
    168             }
    169             // A contact was picked. Launch the cropper to get face detection, the right size, etc.
    170             // TODO: get these values from constants somewhere
    171             final Intent myIntent = getIntent();
    172             final Uri inputUri = myIntent.getData();
    173 
    174 
    175             // Save the URI into a temporary file provider URI so that
    176             // we can add the FLAG_GRANT_WRITE_URI_PERMISSION flag to the eventual
    177             // crop intent for read-only URI's.
    178             // TODO: With b/10837468 fixed should be able to avoid this copy.
    179             if (!ContactPhotoUtils.savePhotoFromUriToUri(this, inputUri, mTempPhotoUri, false)) {
    180                 finish();
    181                 return;
    182             }
    183 
    184             final Intent intent = new Intent("com.android.camera.action.CROP", mTempPhotoUri);
    185             if (myIntent.getStringExtra("mimeType") != null) {
    186                 intent.setDataAndType(mTempPhotoUri, myIntent.getStringExtra("mimeType"));
    187             }
    188             ContactPhotoUtils.addPhotoPickerExtras(intent, mCroppedPhotoUri);
    189             ContactPhotoUtils.addCropExtras(intent, mPhotoDim != 0 ? mPhotoDim : mDefaultPhotoDim);
    190             if (!hasIntentHandler(intent)) {
    191                 // No activity supports the crop action. So skip cropping and set the photo
    192                 // without performing any cropping.
    193                 mCroppedPhotoUri = mTempPhotoUri;
    194                 mContactUri = result.getData();
    195                 loadContact(mContactUri, new Listener() {
    196                     @Override
    197                     public void onContactLoaded(Contact contact) {
    198                         saveContact(contact);
    199                     }
    200                 });
    201                 return;
    202             }
    203 
    204             try {
    205                 startActivityForResult(intent, REQUEST_CROP_PHOTO);
    206             } catch (ActivityNotFoundException ex) {
    207                 Toast.makeText(this, R.string.missing_app, Toast.LENGTH_SHORT).show();
    208                 return;
    209             }
    210 
    211             mContactUri = result.getData();
    212 
    213         } else if (requestCode == REQUEST_CROP_PHOTO) {
    214             // Delete the temporary photo from cache now that we have a cropped version.
    215             // We should do this even if the crop failed and we eventually bail
    216             getContentResolver().delete(mTempPhotoUri, null, null);
    217             if (resultCode != RESULT_OK) {
    218                 finish();
    219                 return;
    220             }
    221             loadContact(mContactUri, new Listener() {
    222                 @Override
    223                 public void onContactLoaded(Contact contact) {
    224                     saveContact(contact);
    225                 }
    226             });
    227         }
    228     }
    229 
    230     private boolean hasIntentHandler(Intent intent) {
    231         final List<ResolveInfo> resolveInfo = getPackageManager()
    232                 .queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
    233         return resolveInfo != null && resolveInfo.size() > 0;
    234     }
    235 
    236     // TODO: consider moving this to ContactLoader, especially if we keep adding similar
    237     // code elsewhere (ViewNotificationService is another case).  The only concern is that,
    238     // although this is convenient, it isn't quite as robust as using LoaderManager... for
    239     // instance, the loader doesn't persist across Activity restarts.
    240     private void loadContact(Uri contactUri, final Listener listener) {
    241         final ContactLoader loader = new ContactLoader(this, contactUri, true);
    242         loader.registerListener(0, new OnLoadCompleteListener<Contact>() {
    243             @Override
    244             public void onLoadComplete(
    245                     Loader<Contact> loader, Contact contact) {
    246                 try {
    247                     loader.reset();
    248                 }
    249                 catch (RuntimeException e) {
    250                     Log.e(TAG, "Error resetting loader", e);
    251                 }
    252                 listener.onContactLoaded(contact);
    253             }
    254         });
    255         loader.startLoading();
    256     }
    257 
    258     private interface Listener {
    259         public void onContactLoaded(Contact contact);
    260     }
    261 
    262     /**
    263      * If prerequisites have been met, attach the photo to a raw-contact and save.
    264      * The prerequisites are:
    265      * - photo has been cropped
    266      * - contact has been loaded
    267      */
    268     private void saveContact(Contact contact) {
    269 
    270         if (contact.getRawContacts() == null) {
    271             Log.w(TAG, "No raw contacts found for contact");
    272             finish();
    273             return;
    274         }
    275 
    276         // Obtain the raw-contact that we will save to.
    277         RawContactDeltaList deltaList = contact.createRawContactDeltaList();
    278         RawContactDelta raw = deltaList.getFirstWritableRawContact(this);
    279         if (raw == null) {
    280             // We can't directly insert this photo since no raw contacts exist in the contact.
    281             selectAccountAndCreateContact();
    282             return;
    283         }
    284 
    285         saveToContact(contact, deltaList, raw);
    286     }
    287 
    288     private void saveToContact(Contact contact, RawContactDeltaList deltaList,
    289             RawContactDelta raw) {
    290 
    291         // Create a scaled, compressed bitmap to add to the entity-delta list.
    292         final int size = ContactsUtils.getThumbnailSize(this);
    293         Bitmap bitmap;
    294         try {
    295             bitmap = ContactPhotoUtils.getBitmapFromUri(this, mCroppedPhotoUri);
    296         } catch (FileNotFoundException e) {
    297             Log.w(TAG, "Could not find bitmap");
    298             finish();
    299             return;
    300         }
    301         if (bitmap == null) {
    302             Log.w(TAG, "Could not decode bitmap");
    303             finish();
    304             return;
    305         }
    306 
    307         final Bitmap scaled = Bitmap.createScaledBitmap(bitmap, size, size, false);
    308         final byte[] compressed = ContactPhotoUtils.compressBitmap(scaled);
    309         if (compressed == null) {
    310             Log.w(TAG, "could not create scaled and compressed Bitmap");
    311             finish();
    312             return;
    313         }
    314 
    315         // Add compressed bitmap to entity-delta... this allows us to save to
    316         // a new contact; otherwise the entity-delta-list would be empty, and
    317         // the ContactSaveService would not create the new contact, and the
    318         // full-res photo would fail to be saved to the non-existent contact.
    319         AccountType account = raw.getRawContactAccountType(this);
    320         ValuesDelta values =
    321                 RawContactModifier.ensureKindExists(raw, account, Photo.CONTENT_ITEM_TYPE);
    322         if (values == null) {
    323             Log.w(TAG, "cannot attach photo to this account type");
    324             finish();
    325             return;
    326         }
    327         values.setPhoto(compressed);
    328 
    329         // Finally, invoke the ContactSaveService.
    330         Log.v(TAG, "all prerequisites met, about to save photo to contact");
    331         Intent intent = ContactSaveService.createSaveContactIntent(
    332                 this,
    333                 deltaList,
    334                 "", 0,
    335                 contact.isUserProfile(),
    336                 null, null,
    337                 raw.getRawContactId() != null ? raw.getRawContactId() : -1,
    338                 mCroppedPhotoUri
    339         );
    340         startService(intent);
    341         finish();
    342     }
    343 
    344     private void selectAccountAndCreateContact() {
    345         // If there is no default account or the accounts have changed such that we need to
    346         // prompt the user again, then launch the account prompt.
    347         final ContactEditorUtils editorUtils = ContactEditorUtils.getInstance(this);
    348         if (editorUtils.shouldShowAccountChangedNotification()) {
    349             Intent intent = new Intent(this, ContactEditorAccountsChangedActivity.class);
    350             startActivityForResult(intent, REQUEST_PICK_DEFAULT_ACCOUNT_FOR_NEW_CONTACT);
    351         } else {
    352             // Otherwise, there should be a default account. Then either create a local contact
    353             // (if default account is null) or create a contact with the specified account.
    354             AccountWithDataSet defaultAccount = editorUtils.getDefaultAccount();
    355             createNewRawContact(defaultAccount);
    356         }
    357     }
    358 
    359     /**
    360      * Create a new writeable raw contact to store mCroppedPhotoUri.
    361      */
    362     private void createNewRawContact(final AccountWithDataSet account) {
    363         // Reload the contact from URI instead of trying to pull the contact from a member variable,
    364         // since this function can be called after the activity stops and resumes.
    365         loadContact(mContactUri, new Listener() {
    366             @Override
    367             public void onContactLoaded(Contact contactToSave) {
    368                 final RawContactDeltaList deltaList = contactToSave.createRawContactDeltaList();
    369                 final ContentValues after = new ContentValues();
    370                 after.put(RawContacts.ACCOUNT_TYPE, account != null ? account.type : null);
    371                 after.put(RawContacts.ACCOUNT_NAME, account != null ? account.name : null);
    372                 after.put(RawContacts.DATA_SET, account != null ? account.dataSet : null);
    373 
    374                 final RawContactDelta newRawContactDelta
    375                         = new RawContactDelta(ValuesDelta.fromAfter(after));
    376                 deltaList.add(newRawContactDelta);
    377                 saveToContact(contactToSave, deltaList, newRawContactDelta);
    378             }
    379         });
    380     }
    381 }
    382