/* * Copyright (C) 2006 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.contacts.activities; import android.app.Activity; import android.content.ActivityNotFoundException; import android.content.ContentResolver; import android.content.ContentValues; import android.content.Intent; import android.content.Loader; import android.content.Loader.OnLoadCompleteListener; import android.content.pm.PackageManager; import android.content.pm.ResolveInfo; import android.database.Cursor; import android.graphics.Bitmap; import android.net.Uri; import android.os.Bundle; import android.provider.ContactsContract.CommonDataKinds.Photo; import android.provider.ContactsContract.Contacts; import android.provider.ContactsContract.DisplayPhoto; import android.provider.ContactsContract.Intents; import android.provider.ContactsContract.RawContacts; import android.util.Log; import android.widget.Toast; import com.android.contacts.ContactSaveService; import com.android.contacts.ContactsActivity; import com.android.contacts.R; import com.android.contacts.common.activity.RequestPermissionsActivity; import com.android.contacts.common.model.Contact; import com.android.contacts.common.model.ContactLoader; import com.android.contacts.common.model.RawContactDelta; import com.android.contacts.common.model.RawContactDeltaList; import com.android.contacts.common.model.RawContactModifier; import com.android.contacts.common.ContactsUtils; import com.android.contacts.common.model.account.AccountType; import com.android.contacts.common.model.ValuesDelta; import com.android.contacts.common.model.account.AccountWithDataSet; import com.android.contacts.editor.ContactEditorUtils; import com.android.contacts.util.ContactPhotoUtils; import java.io.FileNotFoundException; import java.util.List; /** * Provides an external interface for other applications to attach images * to contacts. It will first present a contact picker and then run the * image that is handed to it through the cropper to make the image the proper * size and give the user a chance to use the face detector. */ public class AttachPhotoActivity extends ContactsActivity { private static final String TAG = AttachPhotoActivity.class.getSimpleName(); private static final int REQUEST_PICK_CONTACT = 1; private static final int REQUEST_CROP_PHOTO = 2; private static final int REQUEST_PICK_DEFAULT_ACCOUNT_FOR_NEW_CONTACT = 3; private static final String KEY_CONTACT_URI = "contact_uri"; private static final String KEY_TEMP_PHOTO_URI = "temp_photo_uri"; private static final String KEY_CROPPED_PHOTO_URI = "cropped_photo_uri"; private Uri mTempPhotoUri; private Uri mCroppedPhotoUri; private ContentResolver mContentResolver; // Height and width (in pixels) to request for the photo - queried from the provider. private static int mPhotoDim; // Default photo dimension to use if unable to query the provider. private static final int mDefaultPhotoDim = 720; private Uri mContactUri; @Override public void onCreate(Bundle icicle) { super.onCreate(icicle); if (RequestPermissionsActivity.startPermissionActivity(this)) { return; } if (icicle != null) { final String uri = icicle.getString(KEY_CONTACT_URI); mContactUri = (uri == null) ? null : Uri.parse(uri); mTempPhotoUri = Uri.parse(icicle.getString(KEY_TEMP_PHOTO_URI)); mCroppedPhotoUri = Uri.parse(icicle.getString(KEY_CROPPED_PHOTO_URI)); } else { mTempPhotoUri = ContactPhotoUtils.generateTempImageUri(this); mCroppedPhotoUri = ContactPhotoUtils.generateTempCroppedImageUri(this); Intent intent = new Intent(Intent.ACTION_PICK); intent.setType(Contacts.CONTENT_TYPE); intent.setPackage(getPackageName()); startActivityForResult(intent, REQUEST_PICK_CONTACT); } mContentResolver = getContentResolver(); // Load the photo dimension to request. mPhotoDim is a static class // member varible so only need to load this if this is the first time // through. if (mPhotoDim == 0) { Cursor c = mContentResolver.query(DisplayPhoto.CONTENT_MAX_DIMENSIONS_URI, new String[]{DisplayPhoto.DISPLAY_MAX_DIM}, null, null, null); if (c != null) { try { if (c.moveToFirst()) { mPhotoDim = c.getInt(0); } } finally { c.close(); } } } } @Override protected void onSaveInstanceState(Bundle outState) { super.onSaveInstanceState(outState); if (mContactUri != null) { outState.putString(KEY_CONTACT_URI, mContactUri.toString()); } if (mTempPhotoUri != null) { outState.putString(KEY_TEMP_PHOTO_URI, mTempPhotoUri.toString()); } if (mCroppedPhotoUri != null) { outState.putString(KEY_CROPPED_PHOTO_URI, mCroppedPhotoUri.toString()); } } @Override protected void onActivityResult(int requestCode, int resultCode, Intent result) { if (requestCode == REQUEST_PICK_DEFAULT_ACCOUNT_FOR_NEW_CONTACT) { // Bail if the account selector was not successful. if (resultCode != Activity.RESULT_OK) { Log.w(TAG, "account selector was not successful"); finish(); return; } // If there's an account specified, use it. if (result != null) { AccountWithDataSet account = result.getParcelableExtra( Intents.Insert.EXTRA_ACCOUNT); if (account != null) { createNewRawContact(account); return; } } // If there isn't an account specified, then the user opted to keep the contact local. createNewRawContact(null); } else if (requestCode == REQUEST_PICK_CONTACT) { if (resultCode != RESULT_OK) { finish(); return; } // A contact was picked. Launch the cropper to get face detection, the right size, etc. // TODO: get these values from constants somewhere final Intent myIntent = getIntent(); final Uri inputUri = myIntent.getData(); // Save the URI into a temporary file provider URI so that // we can add the FLAG_GRANT_WRITE_URI_PERMISSION flag to the eventual // crop intent for read-only URI's. // TODO: With b/10837468 fixed should be able to avoid this copy. if (!ContactPhotoUtils.savePhotoFromUriToUri(this, inputUri, mTempPhotoUri, false)) { finish(); return; } final Intent intent = new Intent("com.android.camera.action.CROP", mTempPhotoUri); if (myIntent.getStringExtra("mimeType") != null) { intent.setDataAndType(mTempPhotoUri, myIntent.getStringExtra("mimeType")); } ContactPhotoUtils.addPhotoPickerExtras(intent, mCroppedPhotoUri); ContactPhotoUtils.addCropExtras(intent, mPhotoDim != 0 ? mPhotoDim : mDefaultPhotoDim); if (!hasIntentHandler(intent)) { // No activity supports the crop action. So skip cropping and set the photo // without performing any cropping. mCroppedPhotoUri = mTempPhotoUri; mContactUri = result.getData(); loadContact(mContactUri, new Listener() { @Override public void onContactLoaded(Contact contact) { saveContact(contact); } }); return; } try { startActivityForResult(intent, REQUEST_CROP_PHOTO); } catch (ActivityNotFoundException ex) { Toast.makeText(this, R.string.missing_app, Toast.LENGTH_SHORT).show(); return; } mContactUri = result.getData(); } else if (requestCode == REQUEST_CROP_PHOTO) { // Delete the temporary photo from cache now that we have a cropped version. // We should do this even if the crop failed and we eventually bail getContentResolver().delete(mTempPhotoUri, null, null); if (resultCode != RESULT_OK) { finish(); return; } loadContact(mContactUri, new Listener() { @Override public void onContactLoaded(Contact contact) { saveContact(contact); } }); } } private boolean hasIntentHandler(Intent intent) { final List resolveInfo = getPackageManager() .queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY); return resolveInfo != null && resolveInfo.size() > 0; } // TODO: consider moving this to ContactLoader, especially if we keep adding similar // code elsewhere (ViewNotificationService is another case). The only concern is that, // although this is convenient, it isn't quite as robust as using LoaderManager... for // instance, the loader doesn't persist across Activity restarts. private void loadContact(Uri contactUri, final Listener listener) { final ContactLoader loader = new ContactLoader(this, contactUri, true); loader.registerListener(0, new OnLoadCompleteListener() { @Override public void onLoadComplete( Loader loader, Contact contact) { try { loader.reset(); } catch (RuntimeException e) { Log.e(TAG, "Error resetting loader", e); } listener.onContactLoaded(contact); } }); loader.startLoading(); } private interface Listener { public void onContactLoaded(Contact contact); } /** * If prerequisites have been met, attach the photo to a raw-contact and save. * The prerequisites are: * - photo has been cropped * - contact has been loaded */ private void saveContact(Contact contact) { if (contact.getRawContacts() == null) { Log.w(TAG, "No raw contacts found for contact"); finish(); return; } // Obtain the raw-contact that we will save to. RawContactDeltaList deltaList = contact.createRawContactDeltaList(); RawContactDelta raw = deltaList.getFirstWritableRawContact(this); if (raw == null) { // We can't directly insert this photo since no raw contacts exist in the contact. selectAccountAndCreateContact(); return; } saveToContact(contact, deltaList, raw); } private void saveToContact(Contact contact, RawContactDeltaList deltaList, RawContactDelta raw) { // Create a scaled, compressed bitmap to add to the entity-delta list. final int size = ContactsUtils.getThumbnailSize(this); Bitmap bitmap; try { bitmap = ContactPhotoUtils.getBitmapFromUri(this, mCroppedPhotoUri); } catch (FileNotFoundException e) { Log.w(TAG, "Could not find bitmap"); finish(); return; } if (bitmap == null) { Log.w(TAG, "Could not decode bitmap"); finish(); return; } final Bitmap scaled = Bitmap.createScaledBitmap(bitmap, size, size, false); final byte[] compressed = ContactPhotoUtils.compressBitmap(scaled); if (compressed == null) { Log.w(TAG, "could not create scaled and compressed Bitmap"); finish(); return; } // Add compressed bitmap to entity-delta... this allows us to save to // a new contact; otherwise the entity-delta-list would be empty, and // the ContactSaveService would not create the new contact, and the // full-res photo would fail to be saved to the non-existent contact. AccountType account = raw.getRawContactAccountType(this); ValuesDelta values = RawContactModifier.ensureKindExists(raw, account, Photo.CONTENT_ITEM_TYPE); if (values == null) { Log.w(TAG, "cannot attach photo to this account type"); finish(); return; } values.setPhoto(compressed); // Finally, invoke the ContactSaveService. Log.v(TAG, "all prerequisites met, about to save photo to contact"); Intent intent = ContactSaveService.createSaveContactIntent( this, deltaList, "", 0, contact.isUserProfile(), null, null, raw.getRawContactId() != null ? raw.getRawContactId() : -1, mCroppedPhotoUri ); ContactSaveService.startService(this, intent); finish(); } private void selectAccountAndCreateContact() { // If there is no default account or the accounts have changed such that we need to // prompt the user again, then launch the account prompt. final ContactEditorUtils editorUtils = ContactEditorUtils.getInstance(this); if (editorUtils.shouldShowAccountChangedNotification()) { Intent intent = new Intent(this, ContactEditorAccountsChangedActivity.class); startActivityForResult(intent, REQUEST_PICK_DEFAULT_ACCOUNT_FOR_NEW_CONTACT); } else { // Otherwise, there should be a default account. Then either create a local contact // (if default account is null) or create a contact with the specified account. AccountWithDataSet defaultAccount = editorUtils.getDefaultAccount(); createNewRawContact(defaultAccount); } } /** * Create a new writeable raw contact to store mCroppedPhotoUri. */ private void createNewRawContact(final AccountWithDataSet account) { // Reload the contact from URI instead of trying to pull the contact from a member variable, // since this function can be called after the activity stops and resumes. loadContact(mContactUri, new Listener() { @Override public void onContactLoaded(Contact contactToSave) { final RawContactDeltaList deltaList = contactToSave.createRawContactDeltaList(); final ContentValues after = new ContentValues(); after.put(RawContacts.ACCOUNT_TYPE, account != null ? account.type : null); after.put(RawContacts.ACCOUNT_NAME, account != null ? account.name : null); after.put(RawContacts.DATA_SET, account != null ? account.dataSet : null); final RawContactDelta newRawContactDelta = new RawContactDelta(ValuesDelta.fromAfter(after)); deltaList.add(newRawContactDelta); saveToContact(contactToSave, deltaList, newRawContactDelta); } }); } }