1 /* 2 * Copyright (C) 2010 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.example.android.xmladapters; 18 19 import android.content.ContentUris; 20 import android.content.Context; 21 import android.content.res.Resources; 22 import android.database.Cursor; 23 import android.graphics.Bitmap; 24 import android.graphics.BitmapFactory; 25 import android.graphics.drawable.BitmapDrawable; 26 import android.graphics.drawable.Drawable; 27 import android.net.Uri; 28 import android.provider.ContactsContract; 29 import android.view.View; 30 import android.widget.TextView; 31 32 import java.io.InputStream; 33 import java.util.HashMap; 34 35 /** 36 * This custom cursor binder is used by the adapter defined in res/xml to 37 * bind contacts photos to their respective list item. This binder simply 38 * queries a contact's photo based on the contact's id and sets the 39 * photo as a compound drawable on the TextView used to display the contact's 40 * name. 41 */ 42 public class ContactPhotoBinder extends Adapters.CursorBinder { 43 private static final int PHOTO_SIZE_DIP = 54; 44 45 private final Drawable mDefault; 46 private final HashMap<Long, Drawable> mCache; 47 private final Resources mResources; 48 private final int mPhotoSize; 49 ContactPhotoBinder(Context context, Adapters.CursorTransformation transformation)50 public ContactPhotoBinder(Context context, Adapters.CursorTransformation transformation) { 51 super(context, transformation); 52 53 mResources = mContext.getResources(); 54 // Default picture used when a contact does not provide one 55 mDefault = mResources.getDrawable(R.drawable.ic_contact_picture); 56 // Cache used to avoid re-querying contacts photos every time 57 mCache = new HashMap<Long, Drawable>(); 58 // Compute the size of the photo based on the display's density 59 mPhotoSize = (int) (PHOTO_SIZE_DIP * mResources.getDisplayMetrics().density + 0.5f); 60 } 61 62 @Override bind(View view, Cursor cursor, int columnIndex)63 public boolean bind(View view, Cursor cursor, int columnIndex) { 64 final long id = cursor.getLong(columnIndex); 65 66 // First check whether we have already cached the contact's photo 67 Drawable d = mCache.get(id); 68 69 if (d == null) { 70 // If the photo wasn't in the cache, ask the contacts provider for 71 // an input stream we can use to load the photo 72 Uri uri = ContentUris.withAppendedId(ContactsContract.Contacts.CONTENT_URI, id); 73 InputStream stream = ContactsContract.Contacts.openContactPhotoInputStream( 74 mContext.getContentResolver(), uri); 75 76 // Creates the drawable for the contact's photo or use our fallback drawable 77 if (stream != null) { 78 // decoding the bitmap could be done in a worker thread too. 79 Bitmap bitmap = BitmapFactory.decodeStream(stream); 80 d = new BitmapDrawable(mResources, bitmap); 81 } else { 82 d = mDefault; 83 } 84 85 d.setBounds(0, 0, mPhotoSize, mPhotoSize); 86 ((TextView) view).setCompoundDrawables(d, null, null, null); 87 88 // Remember the photo associated with this contact 89 mCache.put(id, d); 90 } else { 91 ((TextView) view).setCompoundDrawables(d, null, null, null); 92 } 93 94 return true; 95 } 96 } 97