Home | History | Annotate | Download | only in bitmap
      1 /*
      2  * Copyright (C) 2014 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 package com.android.mail.bitmap;
     17 
     18 import android.content.res.Resources;
     19 import android.graphics.Bitmap;
     20 import android.graphics.BitmapShader;
     21 import android.graphics.Canvas;
     22 import android.graphics.Color;
     23 import android.graphics.ColorFilter;
     24 import android.graphics.Matrix;
     25 import android.graphics.Paint;
     26 import android.graphics.Rect;
     27 import android.graphics.Shader;
     28 import android.graphics.drawable.Drawable;
     29 
     30 import com.android.bitmap.BitmapCache;
     31 import com.android.bitmap.RequestKey;
     32 import com.android.bitmap.ReusableBitmap;
     33 
     34 import com.android.mail.R;
     35 import com.android.mail.bitmap.ContactResolver.ContactDrawableInterface;
     36 
     37 /**
     38  * A drawable that encapsulates all the functionality needed to display a contact image,
     39  * including request creation/cancelling and data unbinding/re-binding.
     40  * <p>
     41  * The actual contact resolving and decoding is handled by {@link ContactResolver}.
     42  * <p>
     43  * For better performance, you should define a cache with {@link #setBitmapCache(BitmapCache)}.
     44  */
     45 public abstract class AbstractAvatarDrawable extends Drawable implements ContactDrawableInterface {
     46     protected final Resources mResources;
     47 
     48     private BitmapCache mCache;
     49     private ContactResolver mContactResolver;
     50 
     51     protected ContactRequest mContactRequest;
     52     protected ReusableBitmap mBitmap;
     53 
     54     protected final float mBorderWidth;
     55     protected final Paint mBitmapPaint;
     56     protected final Paint mBorderPaint;
     57     protected final Matrix mMatrix;
     58 
     59     private int mDecodedWidth;
     60     private int mDecodedHeight;
     61 
     62     public AbstractAvatarDrawable(final Resources res) {
     63         mResources = res;
     64 
     65         mBitmapPaint = new Paint();
     66         mBitmapPaint.setAntiAlias(true);
     67         mBitmapPaint.setFilterBitmap(true);
     68         mBitmapPaint.setDither(true);
     69 
     70         mBorderWidth = res.getDimensionPixelSize(R.dimen.avatar_border_width);
     71 
     72         mBorderPaint = new Paint();
     73         mBorderPaint.setColor(Color.TRANSPARENT);
     74         mBorderPaint.setStyle(Paint.Style.STROKE);
     75         mBorderPaint.setStrokeWidth(mBorderWidth);
     76         mBorderPaint.setAntiAlias(true);
     77 
     78         mMatrix = new Matrix();
     79     }
     80 
     81     public void setBitmapCache(final BitmapCache cache) {
     82         mCache = cache;
     83     }
     84 
     85     public void setContactResolver(final ContactResolver contactResolver) {
     86         mContactResolver = contactResolver;
     87     }
     88 
     89     @Override
     90     public void draw(final Canvas canvas) {
     91         final Rect bounds = getBounds();
     92         if (!isVisible() || bounds.isEmpty()) {
     93             return;
     94         }
     95 
     96         if (mBitmap != null && mBitmap.bmp != null) {
     97             // Draw sender image.
     98             drawBitmap(mBitmap.bmp, mBitmap.getLogicalWidth(), mBitmap.getLogicalHeight(), canvas);
     99         } else {
    100             // Draw the default image
    101             drawDefaultAvatar(canvas);
    102         }
    103     }
    104 
    105     protected abstract void drawDefaultAvatar(Canvas canvas);
    106 
    107     /**
    108      * Draw the bitmap onto the canvas at the current bounds taking into account the current scale.
    109      */
    110     protected void drawBitmap(final Bitmap bitmap, final int width, final int height,
    111             final Canvas canvas) {
    112         final Rect bounds = getBounds();
    113         // Draw bitmap through shader first.
    114         final BitmapShader shader = new BitmapShader(bitmap, Shader.TileMode.CLAMP,
    115                 Shader.TileMode.CLAMP);
    116         mMatrix.reset();
    117 
    118         // Fit bitmap to bounds.
    119         final float boundsWidth = (float) bounds.width();
    120         final float boundsHeight = (float) bounds.height();
    121         final float scale = Math.max(boundsWidth / width, boundsHeight / height);
    122         mMatrix.postScale(scale, scale);
    123 
    124         // Translate bitmap to dst bounds.
    125         mMatrix.postTranslate(bounds.left, bounds.top);
    126 
    127         shader.setLocalMatrix(mMatrix);
    128         mBitmapPaint.setShader(shader);
    129         drawCircle(canvas, bounds, mBitmapPaint);
    130 
    131         // Then draw the border.
    132         final float radius = bounds.width() / 2f - mBorderWidth / 2;
    133         canvas.drawCircle(bounds.centerX(), bounds.centerY(), radius, mBorderPaint);
    134     }
    135 
    136     /**
    137      * Draws the largest circle that fits within the given <code>bounds</code>.
    138      *
    139      * @param canvas the canvas on which to draw
    140      * @param bounds the bounding box of the circle
    141      * @param paint the paint with which to draw
    142      */
    143     protected static void drawCircle(Canvas canvas, Rect bounds, Paint paint) {
    144         canvas.drawCircle(bounds.centerX(), bounds.centerY(), bounds.width() / 2, paint);
    145     }
    146 
    147     @Override
    148     public int getDecodeWidth() {
    149         return mDecodedWidth;
    150     }
    151 
    152     @Override
    153     public int getDecodeHeight() {
    154         return mDecodedHeight;
    155     }
    156 
    157     public void setDecodeDimensions(final int decodeWidth, final int decodeHeight) {
    158         mDecodedWidth = decodeWidth;
    159         mDecodedHeight = decodeHeight;
    160     }
    161 
    162     public void unbind() {
    163         setImage(null);
    164     }
    165 
    166     public void bind(final String name, final String email) {
    167         setImage(new ContactRequest(name, email));
    168     }
    169 
    170     private void setImage(final ContactRequest contactRequest) {
    171         if (mContactRequest != null && mContactRequest.equals(contactRequest)) {
    172             return;
    173         }
    174 
    175         if (mBitmap != null) {
    176             mBitmap.releaseReference();
    177             mBitmap = null;
    178         }
    179 
    180         if (mContactResolver != null) {
    181             mContactResolver.remove(mContactRequest, this);
    182         }
    183 
    184         mContactRequest = contactRequest;
    185 
    186         if (contactRequest == null) {
    187             invalidateSelf();
    188             return;
    189         }
    190 
    191         ReusableBitmap cached = null;
    192         if (mCache != null) {
    193             cached = mCache.get(contactRequest, true /* incrementRefCount */);
    194         }
    195 
    196         if (cached != null) {
    197             setBitmap(cached);
    198         } else {
    199             decode();
    200         }
    201     }
    202 
    203     private void decode() {
    204         if (mContactRequest == null) {
    205             return;
    206         }
    207         // Add to batch.
    208         mContactResolver.add(mContactRequest, this);
    209     }
    210 
    211     private void setBitmap(final ReusableBitmap bmp) {
    212         if (mBitmap != null && mBitmap != bmp) {
    213             mBitmap.releaseReference();
    214         }
    215         mBitmap = bmp;
    216         invalidateSelf();
    217     }
    218 
    219     @Override
    220     public void onDecodeComplete(RequestKey key, ReusableBitmap result) {
    221         final ContactRequest request = (ContactRequest) key;
    222         // Remove from batch.
    223         mContactResolver.remove(request, this);
    224         if (request.equals(mContactRequest)) {
    225             setBitmap(result);
    226         } else {
    227             // if the requests don't match (i.e. this request is stale), decrement the
    228             // ref count to allow the bitmap to be pooled
    229             if (result != null) {
    230                 result.releaseReference();
    231             }
    232         }
    233     }
    234 
    235     @Override
    236     public void setAlpha(int alpha) {
    237         mBitmapPaint.setAlpha(alpha);
    238     }
    239 
    240     @Override
    241     public void setColorFilter(ColorFilter cf) {
    242         mBitmapPaint.setColorFilter(cf);
    243     }
    244 
    245     @Override
    246     public int getOpacity() {
    247         return 0;
    248     }
    249 }
    250